LCOV - code coverage report
Current view: top level - src/backend/storage/file - fileset.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 45 48 93.8 %
Date: 2025-01-18 03:14:54 Functions: 8 8 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * fileset.c
       4             :  *    Management of named temporary files.
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  * IDENTIFICATION
      10             :  *    src/backend/storage/file/fileset.c
      11             :  *
      12             :  * FileSets provide a temporary namespace (think directory) so that files can
      13             :  * be discovered by name.
      14             :  *
      15             :  * FileSets can be used by backends when the temporary files need to be
      16             :  * opened/closed multiple times and the underlying files need to survive across
      17             :  * transactions.
      18             :  *
      19             :  *-------------------------------------------------------------------------
      20             :  */
      21             : 
      22             : #include "postgres.h"
      23             : 
      24             : #include <limits.h>
      25             : 
      26             : #include "commands/tablespace.h"
      27             : #include "common/file_utils.h"
      28             : #include "common/hashfn.h"
      29             : #include "miscadmin.h"
      30             : #include "storage/fileset.h"
      31             : 
      32             : static void FileSetPath(char *path, FileSet *fileset, Oid tablespace);
      33             : static void FilePath(char *path, FileSet *fileset, const char *name);
      34             : static Oid  ChooseTablespace(const FileSet *fileset, const char *name);
      35             : 
      36             : /*
      37             :  * Initialize a space for temporary files. This API can be used by shared
      38             :  * fileset as well as if the temporary files are used only by single backend
      39             :  * but the files need to be opened and closed multiple times and also the
      40             :  * underlying files need to survive across transactions.
      41             :  *
      42             :  * The callers are expected to explicitly remove such files by using
      43             :  * FileSetDelete/FileSetDeleteAll.
      44             :  *
      45             :  * Files will be distributed over the tablespaces configured in
      46             :  * temp_tablespaces.
      47             :  *
      48             :  * Under the covers the set is one or more directories which will eventually
      49             :  * be deleted.
      50             :  */
      51             : void
      52         370 : FileSetInit(FileSet *fileset)
      53             : {
      54             :     static uint32 counter = 0;
      55             : 
      56         370 :     fileset->creator_pid = MyProcPid;
      57         370 :     fileset->number = counter;
      58         370 :     counter = (counter + 1) % INT_MAX;
      59             : 
      60             :     /* Capture the tablespace OIDs so that all backends agree on them. */
      61         370 :     PrepareTempTablespaces();
      62         370 :     fileset->ntablespaces =
      63         370 :         GetTempTablespaces(&fileset->tablespaces[0],
      64             :                            lengthof(fileset->tablespaces));
      65         370 :     if (fileset->ntablespaces == 0)
      66             :     {
      67             :         /* If the GUC is empty, use current database's default tablespace */
      68         370 :         fileset->tablespaces[0] = MyDatabaseTableSpace;
      69         370 :         fileset->ntablespaces = 1;
      70             :     }
      71             :     else
      72             :     {
      73             :         int         i;
      74             : 
      75             :         /*
      76             :          * An entry of InvalidOid means use the default tablespace for the
      77             :          * current database.  Replace that now, to be sure that all users of
      78             :          * the FileSet agree on what to do.
      79             :          */
      80           0 :         for (i = 0; i < fileset->ntablespaces; i++)
      81             :         {
      82           0 :             if (fileset->tablespaces[i] == InvalidOid)
      83           0 :                 fileset->tablespaces[i] = MyDatabaseTableSpace;
      84             :         }
      85             :     }
      86         370 : }
      87             : 
      88             : /*
      89             :  * Create a new file in the given set.
      90             :  */
      91             : File
      92        2366 : FileSetCreate(FileSet *fileset, const char *name)
      93             : {
      94             :     char        path[MAXPGPATH];
      95             :     File        file;
      96             : 
      97        2366 :     FilePath(path, fileset, name);
      98        2366 :     file = PathNameCreateTemporaryFile(path, false);
      99             : 
     100             :     /* If we failed, see if we need to create the directory on demand. */
     101        2366 :     if (file <= 0)
     102             :     {
     103             :         char        tempdirpath[MAXPGPATH];
     104             :         char        filesetpath[MAXPGPATH];
     105         350 :         Oid         tablespace = ChooseTablespace(fileset, name);
     106             : 
     107         350 :         TempTablespacePath(tempdirpath, tablespace);
     108         350 :         FileSetPath(filesetpath, fileset, tablespace);
     109         350 :         PathNameCreateTemporaryDir(tempdirpath, filesetpath);
     110         350 :         file = PathNameCreateTemporaryFile(path, true);
     111             :     }
     112             : 
     113        2366 :     return file;
     114             : }
     115             : 
     116             : /*
     117             :  * Open a file that was created with FileSetCreate() */
     118             : File
     119        7070 : FileSetOpen(FileSet *fileset, const char *name, int mode)
     120             : {
     121             :     char        path[MAXPGPATH];
     122             :     File        file;
     123             : 
     124        7070 :     FilePath(path, fileset, name);
     125        7070 :     file = PathNameOpenTemporaryFile(path, mode);
     126             : 
     127        7070 :     return file;
     128             : }
     129             : 
     130             : /*
     131             :  * Delete a file that was created with FileSetCreate().
     132             :  *
     133             :  * Return true if the file existed, false if didn't.
     134             :  */
     135             : bool
     136        3148 : FileSetDelete(FileSet *fileset, const char *name,
     137             :               bool error_on_failure)
     138             : {
     139             :     char        path[MAXPGPATH];
     140             : 
     141        3148 :     FilePath(path, fileset, name);
     142             : 
     143        3148 :     return PathNameDeleteTemporaryFile(path, error_on_failure);
     144             : }
     145             : 
     146             : /*
     147             :  * Delete all files in the set.
     148             :  */
     149             : void
     150         418 : FileSetDeleteAll(FileSet *fileset)
     151             : {
     152             :     char        dirpath[MAXPGPATH];
     153             :     int         i;
     154             : 
     155             :     /*
     156             :      * Delete the directory we created in each tablespace.  Doesn't fail
     157             :      * because we use this in error cleanup paths, but can generate LOG
     158             :      * message on IO error.
     159             :      */
     160         836 :     for (i = 0; i < fileset->ntablespaces; ++i)
     161             :     {
     162         418 :         FileSetPath(dirpath, fileset, fileset->tablespaces[i]);
     163         418 :         PathNameDeleteTemporaryDir(dirpath);
     164             :     }
     165         418 : }
     166             : 
     167             : /*
     168             :  * Build the path for the directory holding the files backing a FileSet in a
     169             :  * given tablespace.
     170             :  */
     171             : static void
     172       13352 : FileSetPath(char *path, FileSet *fileset, Oid tablespace)
     173             : {
     174             :     char        tempdirpath[MAXPGPATH];
     175             : 
     176       13352 :     TempTablespacePath(tempdirpath, tablespace);
     177       13352 :     snprintf(path, MAXPGPATH, "%s/%s%lu.%u.fileset",
     178             :              tempdirpath, PG_TEMP_FILE_PREFIX,
     179       13352 :              (unsigned long) fileset->creator_pid, fileset->number);
     180       13352 : }
     181             : 
     182             : /*
     183             :  * Sorting has to determine which tablespace a given temporary file belongs in.
     184             :  */
     185             : static Oid
     186       12934 : ChooseTablespace(const FileSet *fileset, const char *name)
     187             : {
     188       12934 :     uint32      hash = hash_any((const unsigned char *) name, strlen(name));
     189             : 
     190       12934 :     return fileset->tablespaces[hash % fileset->ntablespaces];
     191             : }
     192             : 
     193             : /*
     194             :  * Compute the full path of a file in a FileSet.
     195             :  */
     196             : static void
     197       12584 : FilePath(char *path, FileSet *fileset, const char *name)
     198             : {
     199             :     char        dirpath[MAXPGPATH];
     200             : 
     201       12584 :     FileSetPath(dirpath, fileset, ChooseTablespace(fileset, name));
     202       12584 :     snprintf(path, MAXPGPATH, "%s/%s", dirpath, name);
     203       12584 : }

Generated by: LCOV version 1.14