LCOV - code coverage report
Current view: top level - src/backend/utils/fmgr - dfmgr.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 134 198 67.7 %
Date: 2025-01-18 04:15:08 Functions: 11 13 84.6 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * dfmgr.c
       4             :  *    Dynamic function manager code.
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/utils/fmgr/dfmgr.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include <sys/stat.h>
      18             : 
      19             : #ifndef WIN32
      20             : #include <dlfcn.h>
      21             : #endif                          /* !WIN32 */
      22             : 
      23             : #include "fmgr.h"
      24             : #include "lib/stringinfo.h"
      25             : #include "miscadmin.h"
      26             : #include "storage/fd.h"
      27             : #include "storage/shmem.h"
      28             : #include "utils/hsearch.h"
      29             : 
      30             : 
      31             : /* signature for PostgreSQL-specific library init function */
      32             : typedef void (*PG_init_t) (void);
      33             : 
      34             : /* hashtable entry for rendezvous variables */
      35             : typedef struct
      36             : {
      37             :     char        varName[NAMEDATALEN];   /* hash key (must be first) */
      38             :     void       *varValue;
      39             : } rendezvousHashEntry;
      40             : 
      41             : /*
      42             :  * List of dynamically loaded files (kept in malloc'd memory).
      43             :  */
      44             : 
      45             : typedef struct df_files
      46             : {
      47             :     struct df_files *next;      /* List link */
      48             :     dev_t       device;         /* Device file is on */
      49             : #ifndef WIN32                   /* ensures we never again depend on this under
      50             :                                  * win32 */
      51             :     ino_t       inode;          /* Inode number of file */
      52             : #endif
      53             :     void       *handle;         /* a handle for pg_dl* functions */
      54             :     char        filename[FLEXIBLE_ARRAY_MEMBER];    /* Full pathname of file */
      55             : } DynamicFileList;
      56             : 
      57             : static DynamicFileList *file_list = NULL;
      58             : static DynamicFileList *file_tail = NULL;
      59             : 
      60             : /* stat() call under Win32 returns an st_ino field, but it has no meaning */
      61             : #ifndef WIN32
      62             : #define SAME_INODE(A,B) ((A).st_ino == (B).inode && (A).st_dev == (B).device)
      63             : #else
      64             : #define SAME_INODE(A,B) false
      65             : #endif
      66             : 
      67             : char       *Dynamic_library_path;
      68             : 
      69             : static void *internal_load_library(const char *libname);
      70             : static void incompatible_module_error(const char *libname,
      71             :                                       const Pg_magic_struct *module_magic_data) pg_attribute_noreturn();
      72             : static char *expand_dynamic_library_name(const char *name);
      73             : static void check_restricted_library_name(const char *name);
      74             : static char *substitute_libpath_macro(const char *name);
      75             : static char *find_in_dynamic_libpath(const char *basename);
      76             : 
      77             : /* Magic structure that module needs to match to be accepted */
      78             : static const Pg_magic_struct magic_data = PG_MODULE_MAGIC_DATA;
      79             : 
      80             : 
      81             : /*
      82             :  * Load the specified dynamic-link library file, and look for a function
      83             :  * named funcname in it.
      84             :  *
      85             :  * If the function is not found, we raise an error if signalNotFound is true,
      86             :  * else return NULL.  Note that errors in loading the library
      87             :  * will provoke ereport() regardless of signalNotFound.
      88             :  *
      89             :  * If filehandle is not NULL, then *filehandle will be set to a handle
      90             :  * identifying the library file.  The filehandle can be used with
      91             :  * lookup_external_function to lookup additional functions in the same file
      92             :  * at less cost than repeating load_external_function.
      93             :  */
      94             : void *
      95       14456 : load_external_function(const char *filename, const char *funcname,
      96             :                        bool signalNotFound, void **filehandle)
      97             : {
      98             :     char       *fullname;
      99             :     void       *lib_handle;
     100             :     void       *retval;
     101             : 
     102             :     /* Expand the possibly-abbreviated filename to an exact path name */
     103       14456 :     fullname = expand_dynamic_library_name(filename);
     104             : 
     105             :     /* Load the shared library, unless we already did */
     106       14456 :     lib_handle = internal_load_library(fullname);
     107             : 
     108             :     /* Return handle if caller wants it */
     109       14448 :     if (filehandle)
     110       11824 :         *filehandle = lib_handle;
     111             : 
     112             :     /* Look up the function within the library. */
     113       14448 :     retval = dlsym(lib_handle, funcname);
     114             : 
     115       14448 :     if (retval == NULL && signalNotFound)
     116           6 :         ereport(ERROR,
     117             :                 (errcode(ERRCODE_UNDEFINED_FUNCTION),
     118             :                  errmsg("could not find function \"%s\" in file \"%s\"",
     119             :                         funcname, fullname)));
     120             : 
     121       14442 :     pfree(fullname);
     122       14442 :     return retval;
     123             : }
     124             : 
     125             : /*
     126             :  * This function loads a shlib file without looking up any particular
     127             :  * function in it.  If the same shlib has previously been loaded,
     128             :  * we do not load it again.
     129             :  *
     130             :  * When 'restricted' is true, only libraries in the presumed-secure
     131             :  * directory $libdir/plugins may be referenced.
     132             :  */
     133             : void
     134        2374 : load_file(const char *filename, bool restricted)
     135             : {
     136             :     char       *fullname;
     137             : 
     138             :     /* Apply security restriction if requested */
     139        2374 :     if (restricted)
     140           0 :         check_restricted_library_name(filename);
     141             : 
     142             :     /* Expand the possibly-abbreviated filename to an exact path name */
     143        2374 :     fullname = expand_dynamic_library_name(filename);
     144             : 
     145             :     /* Load the shared library, unless we already did */
     146        2374 :     (void) internal_load_library(fullname);
     147             : 
     148        2374 :     pfree(fullname);
     149        2374 : }
     150             : 
     151             : /*
     152             :  * Lookup a function whose library file is already loaded.
     153             :  * Return NULL if not found.
     154             :  */
     155             : void *
     156       11818 : lookup_external_function(void *filehandle, const char *funcname)
     157             : {
     158       11818 :     return dlsym(filehandle, funcname);
     159             : }
     160             : 
     161             : 
     162             : /*
     163             :  * Load the specified dynamic-link library file, unless it already is
     164             :  * loaded.  Return the pg_dl* handle for the file.
     165             :  *
     166             :  * Note: libname is expected to be an exact name for the library file.
     167             :  *
     168             :  * NB: There is presently no way to unload a dynamically loaded file.  We might
     169             :  * add one someday if we can convince ourselves we have safe protocols for un-
     170             :  * hooking from hook function pointers, releasing custom GUC variables, and
     171             :  * perhaps other things that are definitely unsafe currently.
     172             :  */
     173             : static void *
     174       21718 : internal_load_library(const char *libname)
     175             : {
     176             :     DynamicFileList *file_scanner;
     177             :     PGModuleMagicFunction magic_func;
     178             :     char       *load_error;
     179             :     struct stat stat_buf;
     180             :     PG_init_t   PG_init;
     181             : 
     182             :     /*
     183             :      * Scan the list of loaded FILES to see if the file has been loaded.
     184             :      */
     185       33106 :     for (file_scanner = file_list;
     186       22212 :          file_scanner != NULL &&
     187       22212 :          strcmp(libname, file_scanner->filename) != 0;
     188       11388 :          file_scanner = file_scanner->next)
     189             :         ;
     190             : 
     191       21718 :     if (file_scanner == NULL)
     192             :     {
     193             :         /*
     194             :          * Check for same files - different paths (ie, symlink or link)
     195             :          */
     196       10894 :         if (stat(libname, &stat_buf) == -1)
     197           8 :             ereport(ERROR,
     198             :                     (errcode_for_file_access(),
     199             :                      errmsg("could not access file \"%s\": %m",
     200             :                             libname)));
     201             : 
     202       17346 :         for (file_scanner = file_list;
     203        6460 :              file_scanner != NULL &&
     204        6460 :              !SAME_INODE(stat_buf, *file_scanner);
     205        6460 :              file_scanner = file_scanner->next)
     206             :             ;
     207             :     }
     208             : 
     209       21710 :     if (file_scanner == NULL)
     210             :     {
     211             :         /*
     212             :          * File not loaded yet.
     213             :          */
     214             :         file_scanner = (DynamicFileList *)
     215       10886 :             malloc(offsetof(DynamicFileList, filename) + strlen(libname) + 1);
     216       10886 :         if (file_scanner == NULL)
     217           0 :             ereport(ERROR,
     218             :                     (errcode(ERRCODE_OUT_OF_MEMORY),
     219             :                      errmsg("out of memory")));
     220             : 
     221       54430 :         MemSet(file_scanner, 0, offsetof(DynamicFileList, filename));
     222       10886 :         strcpy(file_scanner->filename, libname);
     223       10886 :         file_scanner->device = stat_buf.st_dev;
     224             : #ifndef WIN32
     225       10886 :         file_scanner->inode = stat_buf.st_ino;
     226             : #endif
     227       10886 :         file_scanner->next = NULL;
     228             : 
     229       10886 :         file_scanner->handle = dlopen(file_scanner->filename, RTLD_NOW | RTLD_GLOBAL);
     230       10886 :         if (file_scanner->handle == NULL)
     231             :         {
     232           0 :             load_error = dlerror();
     233           0 :             free(file_scanner);
     234             :             /* errcode_for_file_access might not be appropriate here? */
     235           0 :             ereport(ERROR,
     236             :                     (errcode_for_file_access(),
     237             :                      errmsg("could not load library \"%s\": %s",
     238             :                             libname, load_error)));
     239             :         }
     240             : 
     241             :         /* Check the magic function to determine compatibility */
     242       10886 :         magic_func = (PGModuleMagicFunction)
     243       10886 :             dlsym(file_scanner->handle, PG_MAGIC_FUNCTION_NAME_STRING);
     244       10886 :         if (magic_func)
     245             :         {
     246       10886 :             const Pg_magic_struct *magic_data_ptr = (*magic_func) ();
     247             : 
     248       10886 :             if (magic_data_ptr->len != magic_data.len ||
     249       10886 :                 memcmp(magic_data_ptr, &magic_data, magic_data.len) != 0)
     250             :             {
     251             :                 /* copy data block before unlinking library */
     252           0 :                 Pg_magic_struct module_magic_data = *magic_data_ptr;
     253             : 
     254             :                 /* try to close library */
     255           0 :                 dlclose(file_scanner->handle);
     256           0 :                 free(file_scanner);
     257             : 
     258             :                 /* issue suitable complaint */
     259           0 :                 incompatible_module_error(libname, &module_magic_data);
     260             :             }
     261             :         }
     262             :         else
     263             :         {
     264             :             /* try to close library */
     265           0 :             dlclose(file_scanner->handle);
     266           0 :             free(file_scanner);
     267             :             /* complain */
     268           0 :             ereport(ERROR,
     269             :                     (errmsg("incompatible library \"%s\": missing magic block",
     270             :                             libname),
     271             :                      errhint("Extension libraries are required to use the PG_MODULE_MAGIC macro.")));
     272             :         }
     273             : 
     274             :         /*
     275             :          * If the library has a _PG_init() function, call it.
     276             :          */
     277       10886 :         PG_init = (PG_init_t) dlsym(file_scanner->handle, "_PG_init");
     278       10886 :         if (PG_init)
     279        5880 :             (*PG_init) ();
     280             : 
     281             :         /* OK to link it into list */
     282       10886 :         if (file_list == NULL)
     283        7412 :             file_list = file_scanner;
     284             :         else
     285        3474 :             file_tail->next = file_scanner;
     286       10886 :         file_tail = file_scanner;
     287             :     }
     288             : 
     289       21710 :     return file_scanner->handle;
     290             : }
     291             : 
     292             : /*
     293             :  * Report a suitable error for an incompatible magic block.
     294             :  */
     295             : static void
     296           0 : incompatible_module_error(const char *libname,
     297             :                           const Pg_magic_struct *module_magic_data)
     298             : {
     299             :     StringInfoData details;
     300             : 
     301             :     /*
     302             :      * If the version doesn't match, just report that, because the rest of the
     303             :      * block might not even have the fields we expect.
     304             :      */
     305           0 :     if (magic_data.version != module_magic_data->version)
     306             :     {
     307             :         char        library_version[32];
     308             : 
     309           0 :         if (module_magic_data->version >= 1000)
     310           0 :             snprintf(library_version, sizeof(library_version), "%d",
     311           0 :                      module_magic_data->version / 100);
     312             :         else
     313           0 :             snprintf(library_version, sizeof(library_version), "%d.%d",
     314           0 :                      module_magic_data->version / 100,
     315           0 :                      module_magic_data->version % 100);
     316           0 :         ereport(ERROR,
     317             :                 (errmsg("incompatible library \"%s\": version mismatch",
     318             :                         libname),
     319             :                  errdetail("Server is version %d, library is version %s.",
     320             :                            magic_data.version / 100, library_version)));
     321             :     }
     322             : 
     323             :     /*
     324             :      * Similarly, if the ABI extra field doesn't match, error out.  Other
     325             :      * fields below might also mismatch, but that isn't useful information if
     326             :      * you're using the wrong product altogether.
     327             :      */
     328           0 :     if (strcmp(module_magic_data->abi_extra, magic_data.abi_extra) != 0)
     329             :     {
     330           0 :         ereport(ERROR,
     331             :                 (errmsg("incompatible library \"%s\": ABI mismatch",
     332             :                         libname),
     333             :                  errdetail("Server has ABI \"%s\", library has \"%s\".",
     334             :                            magic_data.abi_extra,
     335             :                            module_magic_data->abi_extra)));
     336             :     }
     337             : 
     338             :     /*
     339             :      * Otherwise, spell out which fields don't agree.
     340             :      *
     341             :      * XXX this code has to be adjusted any time the set of fields in a magic
     342             :      * block change!
     343             :      */
     344           0 :     initStringInfo(&details);
     345             : 
     346           0 :     if (module_magic_data->funcmaxargs != magic_data.funcmaxargs)
     347             :     {
     348           0 :         if (details.len)
     349           0 :             appendStringInfoChar(&details, '\n');
     350           0 :         appendStringInfo(&details,
     351             :         /* translator: %s is a variable name and %d its values */
     352           0 :                          _("Server has %s = %d, library has %d."),
     353             :                          "FUNC_MAX_ARGS", magic_data.funcmaxargs,
     354             :                          module_magic_data->funcmaxargs);
     355             :     }
     356           0 :     if (module_magic_data->indexmaxkeys != magic_data.indexmaxkeys)
     357             :     {
     358           0 :         if (details.len)
     359           0 :             appendStringInfoChar(&details, '\n');
     360           0 :         appendStringInfo(&details,
     361             :         /* translator: %s is a variable name and %d its values */
     362           0 :                          _("Server has %s = %d, library has %d."),
     363             :                          "INDEX_MAX_KEYS", magic_data.indexmaxkeys,
     364             :                          module_magic_data->indexmaxkeys);
     365             :     }
     366           0 :     if (module_magic_data->namedatalen != magic_data.namedatalen)
     367             :     {
     368           0 :         if (details.len)
     369           0 :             appendStringInfoChar(&details, '\n');
     370           0 :         appendStringInfo(&details,
     371             :         /* translator: %s is a variable name and %d its values */
     372           0 :                          _("Server has %s = %d, library has %d."),
     373             :                          "NAMEDATALEN", magic_data.namedatalen,
     374             :                          module_magic_data->namedatalen);
     375             :     }
     376           0 :     if (module_magic_data->float8byval != magic_data.float8byval)
     377             :     {
     378           0 :         if (details.len)
     379           0 :             appendStringInfoChar(&details, '\n');
     380           0 :         appendStringInfo(&details,
     381             :         /* translator: %s is a variable name and %d its values */
     382           0 :                          _("Server has %s = %s, library has %s."),
     383           0 :                          "FLOAT8PASSBYVAL", magic_data.float8byval ? "true" : "false",
     384           0 :                          module_magic_data->float8byval ? "true" : "false");
     385             :     }
     386             : 
     387           0 :     if (details.len == 0)
     388           0 :         appendStringInfoString(&details,
     389           0 :                                _("Magic block has unexpected length or padding difference."));
     390             : 
     391           0 :     ereport(ERROR,
     392             :             (errmsg("incompatible library \"%s\": magic block mismatch",
     393             :                     libname),
     394             :              errdetail_internal("%s", details.data)));
     395             : }
     396             : 
     397             : 
     398             : /*
     399             :  * If name contains a slash, check if the file exists, if so return
     400             :  * the name.  Else (no slash) try to expand using search path (see
     401             :  * find_in_dynamic_libpath below); if that works, return the fully
     402             :  * expanded file name.  If the previous failed, append DLSUFFIX and
     403             :  * try again.  If all fails, just return the original name.
     404             :  *
     405             :  * The result will always be freshly palloc'd.
     406             :  */
     407             : static char *
     408       16830 : expand_dynamic_library_name(const char *name)
     409             : {
     410             :     bool        have_slash;
     411             :     char       *new;
     412             :     char       *full;
     413             : 
     414             :     Assert(name);
     415             : 
     416       16830 :     have_slash = (first_dir_separator(name) != NULL);
     417             : 
     418       16830 :     if (!have_slash)
     419             :     {
     420        4370 :         full = find_in_dynamic_libpath(name);
     421        4370 :         if (full)
     422           0 :             return full;
     423             :     }
     424             :     else
     425             :     {
     426       12460 :         full = substitute_libpath_macro(name);
     427       12460 :         if (pg_file_exists(full))
     428         918 :             return full;
     429       11542 :         pfree(full);
     430             :     }
     431             : 
     432       15912 :     new = psprintf("%s%s", name, DLSUFFIX);
     433             : 
     434       15912 :     if (!have_slash)
     435             :     {
     436        4370 :         full = find_in_dynamic_libpath(new);
     437        4370 :         pfree(new);
     438        4370 :         if (full)
     439        4362 :             return full;
     440             :     }
     441             :     else
     442             :     {
     443       11542 :         full = substitute_libpath_macro(new);
     444       11542 :         pfree(new);
     445       11542 :         if (pg_file_exists(full))
     446       11542 :             return full;
     447           0 :         pfree(full);
     448             :     }
     449             : 
     450             :     /*
     451             :      * If we can't find the file, just return the string as-is. The ensuing
     452             :      * load attempt will fail and report a suitable message.
     453             :      */
     454           8 :     return pstrdup(name);
     455             : }
     456             : 
     457             : /*
     458             :  * Check a restricted library name.  It must begin with "$libdir/plugins/"
     459             :  * and there must not be any directory separators after that (this is
     460             :  * sufficient to prevent ".." style attacks).
     461             :  */
     462             : static void
     463           0 : check_restricted_library_name(const char *name)
     464             : {
     465           0 :     if (strncmp(name, "$libdir/plugins/", 16) != 0 ||
     466           0 :         first_dir_separator(name + 16) != NULL)
     467           0 :         ereport(ERROR,
     468             :                 (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
     469             :                  errmsg("access to library \"%s\" is not allowed",
     470             :                         name)));
     471           0 : }
     472             : 
     473             : /*
     474             :  * Substitute for any macros appearing in the given string.
     475             :  * Result is always freshly palloc'd.
     476             :  */
     477             : static char *
     478       32742 : substitute_libpath_macro(const char *name)
     479             : {
     480             :     const char *sep_ptr;
     481             : 
     482             :     Assert(name != NULL);
     483             : 
     484             :     /* Currently, we only recognize $libdir at the start of the string */
     485       32742 :     if (name[0] != '$')
     486         918 :         return pstrdup(name);
     487             : 
     488       31824 :     if ((sep_ptr = first_dir_separator(name)) == NULL)
     489        8740 :         sep_ptr = name + strlen(name);
     490             : 
     491       31824 :     if (strlen("$libdir") != sep_ptr - name ||
     492       31824 :         strncmp(name, "$libdir", strlen("$libdir")) != 0)
     493           0 :         ereport(ERROR,
     494             :                 (errcode(ERRCODE_INVALID_NAME),
     495             :                  errmsg("invalid macro name in dynamic library path: %s",
     496             :                         name)));
     497             : 
     498       31824 :     return psprintf("%s%s", pkglib_path, sep_ptr);
     499             : }
     500             : 
     501             : 
     502             : /*
     503             :  * Search for a file called 'basename' in the colon-separated search
     504             :  * path Dynamic_library_path.  If the file is found, the full file name
     505             :  * is returned in freshly palloc'd memory.  If the file is not found,
     506             :  * return NULL.
     507             :  */
     508             : static char *
     509        8740 : find_in_dynamic_libpath(const char *basename)
     510             : {
     511             :     const char *p;
     512             :     size_t      baselen;
     513             : 
     514             :     Assert(basename != NULL);
     515             :     Assert(first_dir_separator(basename) == NULL);
     516             :     Assert(Dynamic_library_path != NULL);
     517             : 
     518        8740 :     p = Dynamic_library_path;
     519        8740 :     if (strlen(p) == 0)
     520           0 :         return NULL;
     521             : 
     522        8740 :     baselen = strlen(basename);
     523             : 
     524             :     for (;;)
     525           0 :     {
     526             :         size_t      len;
     527             :         char       *piece;
     528             :         char       *mangled;
     529             :         char       *full;
     530             : 
     531        8740 :         piece = first_path_var_separator(p);
     532        8740 :         if (piece == p)
     533           0 :             ereport(ERROR,
     534             :                     (errcode(ERRCODE_INVALID_NAME),
     535             :                      errmsg("zero-length component in parameter \"dynamic_library_path\"")));
     536             : 
     537        8740 :         if (piece == NULL)
     538        8740 :             len = strlen(p);
     539             :         else
     540           0 :             len = piece - p;
     541             : 
     542        8740 :         piece = palloc(len + 1);
     543        8740 :         strlcpy(piece, p, len + 1);
     544             : 
     545        8740 :         mangled = substitute_libpath_macro(piece);
     546        8740 :         pfree(piece);
     547             : 
     548        8740 :         canonicalize_path(mangled);
     549             : 
     550             :         /* only absolute paths */
     551        8740 :         if (!is_absolute_path(mangled))
     552           0 :             ereport(ERROR,
     553             :                     (errcode(ERRCODE_INVALID_NAME),
     554             :                      errmsg("component in parameter \"dynamic_library_path\" is not an absolute path")));
     555             : 
     556        8740 :         full = palloc(strlen(mangled) + 1 + baselen + 1);
     557        8740 :         sprintf(full, "%s/%s", mangled, basename);
     558        8740 :         pfree(mangled);
     559             : 
     560        8740 :         elog(DEBUG3, "find_in_dynamic_libpath: trying \"%s\"", full);
     561             : 
     562        8740 :         if (pg_file_exists(full))
     563        4362 :             return full;
     564             : 
     565        4378 :         pfree(full);
     566             : 
     567        4378 :         if (p[len] == '\0')
     568        4378 :             break;
     569             :         else
     570           0 :             p += len + 1;
     571             :     }
     572             : 
     573        4378 :     return NULL;
     574             : }
     575             : 
     576             : 
     577             : /*
     578             :  * Find (or create) a rendezvous variable that one dynamically
     579             :  * loaded library can use to meet up with another.
     580             :  *
     581             :  * On the first call of this function for a particular varName,
     582             :  * a "rendezvous variable" is created with the given name.
     583             :  * The value of the variable is a void pointer (initially set to NULL).
     584             :  * Subsequent calls with the same varName just return the address of
     585             :  * the existing variable.  Once created, a rendezvous variable lasts
     586             :  * for the life of the process.
     587             :  *
     588             :  * Dynamically loaded libraries can use rendezvous variables
     589             :  * to find each other and share information: they just need to agree
     590             :  * on the variable name and the data it will point to.
     591             :  */
     592             : void      **
     593        3548 : find_rendezvous_variable(const char *varName)
     594             : {
     595             :     static HTAB *rendezvousHash = NULL;
     596             : 
     597             :     rendezvousHashEntry *hentry;
     598             :     bool        found;
     599             : 
     600             :     /* Create a hashtable if we haven't already done so in this process */
     601        3548 :     if (rendezvousHash == NULL)
     602             :     {
     603             :         HASHCTL     ctl;
     604             : 
     605        3542 :         ctl.keysize = NAMEDATALEN;
     606        3542 :         ctl.entrysize = sizeof(rendezvousHashEntry);
     607        3542 :         rendezvousHash = hash_create("Rendezvous variable hash",
     608             :                                      16,
     609             :                                      &ctl,
     610             :                                      HASH_ELEM | HASH_STRINGS);
     611             :     }
     612             : 
     613             :     /* Find or create the hashtable entry for this varName */
     614        3548 :     hentry = (rendezvousHashEntry *) hash_search(rendezvousHash,
     615             :                                                  varName,
     616             :                                                  HASH_ENTER,
     617             :                                                  &found);
     618             : 
     619             :     /* Initialize to NULL if first time */
     620        3548 :     if (!found)
     621        3548 :         hentry->varValue = NULL;
     622             : 
     623        3548 :     return &hentry->varValue;
     624             : }
     625             : 
     626             : /*
     627             :  * Estimate the amount of space needed to serialize the list of libraries
     628             :  * we have loaded.
     629             :  */
     630             : Size
     631         892 : EstimateLibraryStateSpace(void)
     632             : {
     633             :     DynamicFileList *file_scanner;
     634         892 :     Size        size = 1;
     635             : 
     636        2208 :     for (file_scanner = file_list;
     637             :          file_scanner != NULL;
     638        1316 :          file_scanner = file_scanner->next)
     639        1316 :         size = add_size(size, strlen(file_scanner->filename) + 1);
     640             : 
     641         892 :     return size;
     642             : }
     643             : 
     644             : /*
     645             :  * Serialize the list of libraries we have loaded to a chunk of memory.
     646             :  */
     647             : void
     648         892 : SerializeLibraryState(Size maxsize, char *start_address)
     649             : {
     650             :     DynamicFileList *file_scanner;
     651             : 
     652        2208 :     for (file_scanner = file_list;
     653             :          file_scanner != NULL;
     654        1316 :          file_scanner = file_scanner->next)
     655             :     {
     656             :         Size        len;
     657             : 
     658        1316 :         len = strlcpy(start_address, file_scanner->filename, maxsize) + 1;
     659             :         Assert(len < maxsize);
     660        1316 :         maxsize -= len;
     661        1316 :         start_address += len;
     662             :     }
     663         892 :     start_address[0] = '\0';
     664         892 : }
     665             : 
     666             : /*
     667             :  * Load every library the serializing backend had loaded.
     668             :  */
     669             : void
     670        2712 : RestoreLibraryState(char *start_address)
     671             : {
     672        7600 :     while (*start_address != '\0')
     673             :     {
     674        4888 :         internal_load_library(start_address);
     675        4888 :         start_address += strlen(start_address) + 1;
     676             :     }
     677        2712 : }

Generated by: LCOV version 1.14