mirror of
https://github.com/postgres/postgres.git
synced 2025-05-11 05:41:32 +03:00
Commit ecd9e9f0b fixed the problem in the wrong place, causing unwanted side-effects on the behavior of GetNextTempTableSpace(). Instead, let's make SharedFileSetInit() responsible for subbing in the value of MyDatabaseTableSpace when the default tablespace is called for. The convention about what is in the tempTableSpaces[] array is evidently insufficiently documented, so try to improve that. It also looks like SharedFileSetInit() is doing the wrong thing in the case where temp_tablespaces is empty. It was hard-wiring use of the pg_default tablespace, but it seems like using MyDatabaseTableSpace is more consistent with what happens for other temp files. Back-patch the reversion of PrepareTempTablespaces()'s behavior to 9.5, as ecd9e9f0b was. The changes in SharedFileSetInit() go back to v11 where that was introduced. (Note there is net zero code change before v11 from these two patch sets, so nothing to release-note.) Magnus Hagander and Tom Lane Discussion: https://postgr.es/m/CABUevExg5YEsOvqMxrjoNvb3ApVyH+9jggWGKwTDFyFCVWczGQ@mail.gmail.com
261 lines
7.0 KiB
C
261 lines
7.0 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* sharedfileset.c
|
|
* Shared temporary file management.
|
|
*
|
|
* Portions Copyright (c) 1996-2018, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
* IDENTIFICATION
|
|
* src/backend/storage/file/sharedfileset.c
|
|
*
|
|
* SharefFileSets provide a temporary namespace (think directory) so that
|
|
* files can be discovered by name, and a shared ownership semantics so that
|
|
* shared files survive until the last user detaches.
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "postgres.h"
|
|
|
|
#include "access/hash.h"
|
|
#include "catalog/pg_tablespace.h"
|
|
#include "commands/tablespace.h"
|
|
#include "miscadmin.h"
|
|
#include "storage/dsm.h"
|
|
#include "storage/sharedfileset.h"
|
|
#include "utils/builtins.h"
|
|
|
|
static void SharedFileSetOnDetach(dsm_segment *segment, Datum datum);
|
|
static void SharedFileSetPath(char *path, SharedFileSet *fileset, Oid tablespace);
|
|
static void SharedFilePath(char *path, SharedFileSet *fileset, const char *name);
|
|
static Oid ChooseTablespace(const SharedFileSet *fileset, const char *name);
|
|
|
|
/*
|
|
* Initialize a space for temporary files that can be opened for read-only
|
|
* access by other backends. Other backends must attach to it before
|
|
* accessing it. Associate this SharedFileSet with 'seg'. Any contained
|
|
* files will be deleted when the last backend detaches.
|
|
*
|
|
* Files will be distributed over the tablespaces configured in
|
|
* temp_tablespaces.
|
|
*
|
|
* Under the covers the set is one or more directories which will eventually
|
|
* be deleted when there are no backends attached.
|
|
*/
|
|
void
|
|
SharedFileSetInit(SharedFileSet *fileset, dsm_segment *seg)
|
|
{
|
|
static uint32 counter = 0;
|
|
|
|
SpinLockInit(&fileset->mutex);
|
|
fileset->refcnt = 1;
|
|
fileset->creator_pid = MyProcPid;
|
|
fileset->number = counter;
|
|
counter = (counter + 1) % INT_MAX;
|
|
|
|
/* Capture the tablespace OIDs so that all backends agree on them. */
|
|
PrepareTempTablespaces();
|
|
fileset->ntablespaces =
|
|
GetTempTablespaces(&fileset->tablespaces[0],
|
|
lengthof(fileset->tablespaces));
|
|
if (fileset->ntablespaces == 0)
|
|
{
|
|
/* If the GUC is empty, use current database's default tablespace */
|
|
fileset->tablespaces[0] = MyDatabaseTableSpace;
|
|
fileset->ntablespaces = 1;
|
|
}
|
|
else
|
|
{
|
|
int i;
|
|
|
|
/*
|
|
* An entry of InvalidOid means use the default tablespace for the
|
|
* current database. Replace that now, to be sure that all users of
|
|
* the SharedFileSet agree on what to do.
|
|
*/
|
|
for (i = 0; i < fileset->ntablespaces; i++)
|
|
{
|
|
if (fileset->tablespaces[i] == InvalidOid)
|
|
fileset->tablespaces[i] = MyDatabaseTableSpace;
|
|
}
|
|
}
|
|
|
|
/* Register our cleanup callback. */
|
|
on_dsm_detach(seg, SharedFileSetOnDetach, PointerGetDatum(fileset));
|
|
}
|
|
|
|
/*
|
|
* Attach to a set of directories that was created with SharedFileSetInit.
|
|
*/
|
|
void
|
|
SharedFileSetAttach(SharedFileSet *fileset, dsm_segment *seg)
|
|
{
|
|
bool success;
|
|
|
|
SpinLockAcquire(&fileset->mutex);
|
|
if (fileset->refcnt == 0)
|
|
success = false;
|
|
else
|
|
{
|
|
++fileset->refcnt;
|
|
success = true;
|
|
}
|
|
SpinLockRelease(&fileset->mutex);
|
|
|
|
if (!success)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
|
|
errmsg("could not attach to a SharedFileSet that is already destroyed")));
|
|
|
|
/* Register our cleanup callback. */
|
|
on_dsm_detach(seg, SharedFileSetOnDetach, PointerGetDatum(fileset));
|
|
}
|
|
|
|
/*
|
|
* Create a new file in the given set.
|
|
*/
|
|
File
|
|
SharedFileSetCreate(SharedFileSet *fileset, const char *name)
|
|
{
|
|
char path[MAXPGPATH];
|
|
File file;
|
|
|
|
SharedFilePath(path, fileset, name);
|
|
file = PathNameCreateTemporaryFile(path, false);
|
|
|
|
/* If we failed, see if we need to create the directory on demand. */
|
|
if (file <= 0)
|
|
{
|
|
char tempdirpath[MAXPGPATH];
|
|
char filesetpath[MAXPGPATH];
|
|
Oid tablespace = ChooseTablespace(fileset, name);
|
|
|
|
TempTablespacePath(tempdirpath, tablespace);
|
|
SharedFileSetPath(filesetpath, fileset, tablespace);
|
|
PathNameCreateTemporaryDir(tempdirpath, filesetpath);
|
|
file = PathNameCreateTemporaryFile(path, true);
|
|
}
|
|
|
|
return file;
|
|
}
|
|
|
|
/*
|
|
* Open a file that was created with SharedFileSetCreate(), possibly in
|
|
* another backend.
|
|
*/
|
|
File
|
|
SharedFileSetOpen(SharedFileSet *fileset, const char *name)
|
|
{
|
|
char path[MAXPGPATH];
|
|
File file;
|
|
|
|
SharedFilePath(path, fileset, name);
|
|
file = PathNameOpenTemporaryFile(path);
|
|
|
|
return file;
|
|
}
|
|
|
|
/*
|
|
* Delete a file that was created with PathNameCreateShared().
|
|
* Return true if the file existed, false if didn't.
|
|
*/
|
|
bool
|
|
SharedFileSetDelete(SharedFileSet *fileset, const char *name,
|
|
bool error_on_failure)
|
|
{
|
|
char path[MAXPGPATH];
|
|
|
|
SharedFilePath(path, fileset, name);
|
|
|
|
return PathNameDeleteTemporaryFile(path, error_on_failure);
|
|
}
|
|
|
|
/*
|
|
* Delete all files in the set.
|
|
*/
|
|
void
|
|
SharedFileSetDeleteAll(SharedFileSet *fileset)
|
|
{
|
|
char dirpath[MAXPGPATH];
|
|
int i;
|
|
|
|
/*
|
|
* Delete the directory we created in each tablespace. Doesn't fail
|
|
* because we use this in error cleanup paths, but can generate LOG
|
|
* message on IO error.
|
|
*/
|
|
for (i = 0; i < fileset->ntablespaces; ++i)
|
|
{
|
|
SharedFileSetPath(dirpath, fileset, fileset->tablespaces[i]);
|
|
PathNameDeleteTemporaryDir(dirpath);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback function that will be invoked when this backend detaches from a
|
|
* DSM segment holding a SharedFileSet that it has created or attached to. If
|
|
* we are the last to detach, then try to remove the directories and
|
|
* everything in them. We can't raise an error on failures, because this runs
|
|
* in error cleanup paths.
|
|
*/
|
|
static void
|
|
SharedFileSetOnDetach(dsm_segment *segment, Datum datum)
|
|
{
|
|
bool unlink_all = false;
|
|
SharedFileSet *fileset = (SharedFileSet *) DatumGetPointer(datum);
|
|
|
|
SpinLockAcquire(&fileset->mutex);
|
|
Assert(fileset->refcnt > 0);
|
|
if (--fileset->refcnt == 0)
|
|
unlink_all = true;
|
|
SpinLockRelease(&fileset->mutex);
|
|
|
|
/*
|
|
* If we are the last to detach, we delete the directory in all
|
|
* tablespaces. Note that we are still actually attached for the rest of
|
|
* this function so we can safely access its data.
|
|
*/
|
|
if (unlink_all)
|
|
SharedFileSetDeleteAll(fileset);
|
|
}
|
|
|
|
/*
|
|
* Build the path for the directory holding the files backing a SharedFileSet
|
|
* in a given tablespace.
|
|
*/
|
|
static void
|
|
SharedFileSetPath(char *path, SharedFileSet *fileset, Oid tablespace)
|
|
{
|
|
char tempdirpath[MAXPGPATH];
|
|
|
|
TempTablespacePath(tempdirpath, tablespace);
|
|
snprintf(path, MAXPGPATH, "%s/%s%lu.%u.sharedfileset",
|
|
tempdirpath, PG_TEMP_FILE_PREFIX,
|
|
(unsigned long) fileset->creator_pid, fileset->number);
|
|
}
|
|
|
|
/*
|
|
* Sorting hat to determine which tablespace a given shared temporary file
|
|
* belongs in.
|
|
*/
|
|
static Oid
|
|
ChooseTablespace(const SharedFileSet *fileset, const char *name)
|
|
{
|
|
uint32 hash = hash_any((const unsigned char *) name, strlen(name));
|
|
|
|
return fileset->tablespaces[hash % fileset->ntablespaces];
|
|
}
|
|
|
|
/*
|
|
* Compute the full path of a file in a SharedFileSet.
|
|
*/
|
|
static void
|
|
SharedFilePath(char *path, SharedFileSet *fileset, const char *name)
|
|
{
|
|
char dirpath[MAXPGPATH];
|
|
|
|
SharedFileSetPath(dirpath, fileset, ChooseTablespace(fileset, name));
|
|
snprintf(path, MAXPGPATH, "%s/%s", dirpath, name);
|
|
}
|