Line data Source code
1 : /*------------------------------------------------------------------------- 2 : * 3 : * dsm_registry.c 4 : * Functions for interfacing with the dynamic shared memory registry. 5 : * 6 : * This provides a way for libraries to use shared memory without needing 7 : * to request it at startup time via a shmem_request_hook. The registry 8 : * stores dynamic shared memory (DSM) segment handles keyed by a 9 : * library-specified string. 10 : * 11 : * The registry is accessed by calling GetNamedDSMSegment(). If a segment 12 : * with the provided name does not yet exist, it is created and initialized 13 : * with the provided init_callback callback function. Otherwise, 14 : * GetNamedDSMSegment() simply ensures that the segment is attached to the 15 : * current backend. This function guarantees that only one backend 16 : * initializes the segment and that all other backends just attach it. 17 : * 18 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group 19 : * Portions Copyright (c) 1994, Regents of the University of California 20 : * 21 : * IDENTIFICATION 22 : * src/backend/storage/ipc/dsm_registry.c 23 : * 24 : *------------------------------------------------------------------------- 25 : */ 26 : 27 : #include "postgres.h" 28 : 29 : #include "lib/dshash.h" 30 : #include "storage/dsm_registry.h" 31 : #include "storage/lwlock.h" 32 : #include "storage/shmem.h" 33 : #include "utils/memutils.h" 34 : 35 : typedef struct DSMRegistryCtxStruct 36 : { 37 : dsa_handle dsah; 38 : dshash_table_handle dshh; 39 : } DSMRegistryCtxStruct; 40 : 41 : static DSMRegistryCtxStruct *DSMRegistryCtx; 42 : 43 : typedef struct DSMRegistryEntry 44 : { 45 : char name[64]; 46 : dsm_handle handle; 47 : size_t size; 48 : } DSMRegistryEntry; 49 : 50 : static const dshash_parameters dsh_params = { 51 : offsetof(DSMRegistryEntry, handle), 52 : sizeof(DSMRegistryEntry), 53 : dshash_strcmp, 54 : dshash_strhash, 55 : dshash_strcpy, 56 : LWTRANCHE_DSM_REGISTRY_HASH 57 : }; 58 : 59 : static dsa_area *dsm_registry_dsa; 60 : static dshash_table *dsm_registry_table; 61 : 62 : Size 63 5484 : DSMRegistryShmemSize(void) 64 : { 65 5484 : return MAXALIGN(sizeof(DSMRegistryCtxStruct)); 66 : } 67 : 68 : void 69 1918 : DSMRegistryShmemInit(void) 70 : { 71 : bool found; 72 : 73 1918 : DSMRegistryCtx = (DSMRegistryCtxStruct *) 74 1918 : ShmemInitStruct("DSM Registry Data", 75 : DSMRegistryShmemSize(), 76 : &found); 77 : 78 1918 : if (!found) 79 : { 80 1918 : DSMRegistryCtx->dsah = DSA_HANDLE_INVALID; 81 1918 : DSMRegistryCtx->dshh = DSHASH_HANDLE_INVALID; 82 : } 83 1918 : } 84 : 85 : /* 86 : * Initialize or attach to the dynamic shared hash table that stores the DSM 87 : * registry entries, if not already done. This must be called before accessing 88 : * the table. 89 : */ 90 : static void 91 48 : init_dsm_registry(void) 92 : { 93 : /* Quick exit if we already did this. */ 94 48 : if (dsm_registry_table) 95 0 : return; 96 : 97 : /* Otherwise, use a lock to ensure only one process creates the table. */ 98 48 : LWLockAcquire(DSMRegistryLock, LW_EXCLUSIVE); 99 : 100 48 : if (DSMRegistryCtx->dshh == DSHASH_HANDLE_INVALID) 101 : { 102 : /* Initialize dynamic shared hash table for registry. */ 103 20 : dsm_registry_dsa = dsa_create(LWTRANCHE_DSM_REGISTRY_DSA); 104 20 : dsa_pin(dsm_registry_dsa); 105 20 : dsa_pin_mapping(dsm_registry_dsa); 106 20 : dsm_registry_table = dshash_create(dsm_registry_dsa, &dsh_params, NULL); 107 : 108 : /* Store handles in shared memory for other backends to use. */ 109 20 : DSMRegistryCtx->dsah = dsa_get_handle(dsm_registry_dsa); 110 20 : DSMRegistryCtx->dshh = dshash_get_hash_table_handle(dsm_registry_table); 111 : } 112 : else 113 : { 114 : /* Attach to existing dynamic shared hash table. */ 115 28 : dsm_registry_dsa = dsa_attach(DSMRegistryCtx->dsah); 116 28 : dsa_pin_mapping(dsm_registry_dsa); 117 28 : dsm_registry_table = dshash_attach(dsm_registry_dsa, &dsh_params, 118 28 : DSMRegistryCtx->dshh, NULL); 119 : } 120 : 121 48 : LWLockRelease(DSMRegistryLock); 122 : } 123 : 124 : /* 125 : * Initialize or attach a named DSM segment. 126 : * 127 : * This routine returns the address of the segment. init_callback is called to 128 : * initialize the segment when it is first created. 129 : */ 130 : void * 131 48 : GetNamedDSMSegment(const char *name, size_t size, 132 : void (*init_callback) (void *ptr), bool *found) 133 : { 134 : DSMRegistryEntry *entry; 135 : MemoryContext oldcontext; 136 : void *ret; 137 : 138 : Assert(found); 139 : 140 48 : if (!name || *name == '\0') 141 0 : ereport(ERROR, 142 : (errmsg("DSM segment name cannot be empty"))); 143 : 144 48 : if (strlen(name) >= offsetof(DSMRegistryEntry, handle)) 145 0 : ereport(ERROR, 146 : (errmsg("DSM segment name too long"))); 147 : 148 48 : if (size == 0) 149 0 : ereport(ERROR, 150 : (errmsg("DSM segment size must be nonzero"))); 151 : 152 : /* Be sure any local memory allocated by DSM/DSA routines is persistent. */ 153 48 : oldcontext = MemoryContextSwitchTo(TopMemoryContext); 154 : 155 : /* Connect to the registry. */ 156 48 : init_dsm_registry(); 157 : 158 48 : entry = dshash_find_or_insert(dsm_registry_table, name, found); 159 48 : if (!(*found)) 160 : { 161 : /* Initialize the segment. */ 162 20 : dsm_segment *seg = dsm_create(size, 0); 163 : 164 20 : dsm_pin_segment(seg); 165 20 : dsm_pin_mapping(seg); 166 20 : entry->handle = dsm_segment_handle(seg); 167 20 : entry->size = size; 168 20 : ret = dsm_segment_address(seg); 169 : 170 20 : if (init_callback) 171 20 : (*init_callback) (ret); 172 : } 173 28 : else if (entry->size != size) 174 : { 175 0 : ereport(ERROR, 176 : (errmsg("requested DSM segment size does not match size of " 177 : "existing segment"))); 178 : } 179 : else 180 : { 181 28 : dsm_segment *seg = dsm_find_mapping(entry->handle); 182 : 183 : /* If the existing segment is not already attached, attach it now. */ 184 28 : if (seg == NULL) 185 : { 186 28 : seg = dsm_attach(entry->handle); 187 28 : if (seg == NULL) 188 0 : elog(ERROR, "could not map dynamic shared memory segment"); 189 : 190 28 : dsm_pin_mapping(seg); 191 : } 192 : 193 28 : ret = dsm_segment_address(seg); 194 : } 195 : 196 48 : dshash_release_lock(dsm_registry_table, entry); 197 48 : MemoryContextSwitchTo(oldcontext); 198 : 199 48 : return ret; 200 : }