mirror of
https://github.com/postgres/postgres.git
synced 2025-11-28 11:44:57 +03:00
Teach DSM registry to retry entry initialization if needed.
If DSM registry entry initialization fails, backends could try to
use an uninitialized DSM segment, DSA, or dshash table (since the
entry is still added to the registry). To fix, restructure the
code so that the registry retries initialization as needed. This
commit also modifies pg_get_dsm_registry_allocations() to leave out
partially-initialized entries, as they shouldn't have any allocated
memory.
DSM registry entry initialization shouldn't fail often in practice,
but retrying was deemed better than leaving entries in a
permanently failed state (as was done by commit 1165a933aa, which
has since been reverted).
Suggested-by: Robert Haas <robertmhaas@gmail.com>
Reviewed-by: Robert Haas <robertmhaas@gmail.com>
Discussion: https://postgr.es/m/E1vJHUk-006I7r-37%40gemulon.postgresql.org
Backpatch-through: 17
This commit is contained in:
@@ -155,9 +155,10 @@ init_dsm_registry(void)
|
|||||||
{
|
{
|
||||||
/* Initialize dynamic shared hash table for registry. */
|
/* Initialize dynamic shared hash table for registry. */
|
||||||
dsm_registry_dsa = dsa_create(LWTRANCHE_DSM_REGISTRY_DSA);
|
dsm_registry_dsa = dsa_create(LWTRANCHE_DSM_REGISTRY_DSA);
|
||||||
|
dsm_registry_table = dshash_create(dsm_registry_dsa, &dsh_params, NULL);
|
||||||
|
|
||||||
dsa_pin(dsm_registry_dsa);
|
dsa_pin(dsm_registry_dsa);
|
||||||
dsa_pin_mapping(dsm_registry_dsa);
|
dsa_pin_mapping(dsm_registry_dsa);
|
||||||
dsm_registry_table = dshash_create(dsm_registry_dsa, &dsh_params, NULL);
|
|
||||||
|
|
||||||
/* Store handles in shared memory for other backends to use. */
|
/* Store handles in shared memory for other backends to use. */
|
||||||
DSMRegistryCtx->dsah = dsa_get_handle(dsm_registry_dsa);
|
DSMRegistryCtx->dsah = dsa_get_handle(dsm_registry_dsa);
|
||||||
@@ -188,6 +189,8 @@ GetNamedDSMSegment(const char *name, size_t size,
|
|||||||
DSMRegistryEntry *entry;
|
DSMRegistryEntry *entry;
|
||||||
MemoryContext oldcontext;
|
MemoryContext oldcontext;
|
||||||
void *ret;
|
void *ret;
|
||||||
|
NamedDSMState *state;
|
||||||
|
dsm_segment *seg;
|
||||||
|
|
||||||
Assert(found);
|
Assert(found);
|
||||||
|
|
||||||
@@ -210,36 +213,36 @@ GetNamedDSMSegment(const char *name, size_t size,
|
|||||||
init_dsm_registry();
|
init_dsm_registry();
|
||||||
|
|
||||||
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
||||||
|
state = &entry->dsm;
|
||||||
if (!(*found))
|
if (!(*found))
|
||||||
{
|
{
|
||||||
NamedDSMState *state = &entry->dsm;
|
|
||||||
dsm_segment *seg;
|
|
||||||
|
|
||||||
entry->type = DSMR_ENTRY_TYPE_DSM;
|
entry->type = DSMR_ENTRY_TYPE_DSM;
|
||||||
|
state->handle = DSM_HANDLE_INVALID;
|
||||||
/* Initialize the segment. */
|
|
||||||
seg = dsm_create(size, 0);
|
|
||||||
|
|
||||||
dsm_pin_segment(seg);
|
|
||||||
dsm_pin_mapping(seg);
|
|
||||||
state->handle = dsm_segment_handle(seg);
|
|
||||||
state->size = size;
|
state->size = size;
|
||||||
ret = dsm_segment_address(seg);
|
|
||||||
|
|
||||||
if (init_callback)
|
|
||||||
(*init_callback) (ret);
|
|
||||||
}
|
}
|
||||||
else if (entry->type != DSMR_ENTRY_TYPE_DSM)
|
else if (entry->type != DSMR_ENTRY_TYPE_DSM)
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errmsg("requested DSM segment does not match type of existing entry")));
|
(errmsg("requested DSM segment does not match type of existing entry")));
|
||||||
else if (entry->dsm.size != size)
|
else if (state->size != size)
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errmsg("requested DSM segment size does not match size of existing segment")));
|
(errmsg("requested DSM segment size does not match size of existing segment")));
|
||||||
|
|
||||||
|
if (state->handle == DSM_HANDLE_INVALID)
|
||||||
|
{
|
||||||
|
*found = false;
|
||||||
|
|
||||||
|
/* Initialize the segment. */
|
||||||
|
seg = dsm_create(size, 0);
|
||||||
|
|
||||||
|
if (init_callback)
|
||||||
|
(*init_callback) (dsm_segment_address(seg));
|
||||||
|
|
||||||
|
dsm_pin_segment(seg);
|
||||||
|
dsm_pin_mapping(seg);
|
||||||
|
state->handle = dsm_segment_handle(seg);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
NamedDSMState *state = &entry->dsm;
|
|
||||||
dsm_segment *seg;
|
|
||||||
|
|
||||||
/* If the existing segment is not already attached, attach it now. */
|
/* If the existing segment is not already attached, attach it now. */
|
||||||
seg = dsm_find_mapping(state->handle);
|
seg = dsm_find_mapping(state->handle);
|
||||||
if (seg == NULL)
|
if (seg == NULL)
|
||||||
@@ -250,10 +253,9 @@ GetNamedDSMSegment(const char *name, size_t size,
|
|||||||
|
|
||||||
dsm_pin_mapping(seg);
|
dsm_pin_mapping(seg);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = dsm_segment_address(seg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = dsm_segment_address(seg);
|
||||||
dshash_release_lock(dsm_registry_table, entry);
|
dshash_release_lock(dsm_registry_table, entry);
|
||||||
MemoryContextSwitchTo(oldcontext);
|
MemoryContextSwitchTo(oldcontext);
|
||||||
|
|
||||||
@@ -274,6 +276,7 @@ GetNamedDSA(const char *name, bool *found)
|
|||||||
DSMRegistryEntry *entry;
|
DSMRegistryEntry *entry;
|
||||||
MemoryContext oldcontext;
|
MemoryContext oldcontext;
|
||||||
dsa_area *ret;
|
dsa_area *ret;
|
||||||
|
NamedDSAState *state;
|
||||||
|
|
||||||
Assert(found);
|
Assert(found);
|
||||||
|
|
||||||
@@ -292,14 +295,28 @@ GetNamedDSA(const char *name, bool *found)
|
|||||||
init_dsm_registry();
|
init_dsm_registry();
|
||||||
|
|
||||||
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
||||||
|
state = &entry->dsa;
|
||||||
if (!(*found))
|
if (!(*found))
|
||||||
{
|
{
|
||||||
NamedDSAState *state = &entry->dsa;
|
|
||||||
|
|
||||||
entry->type = DSMR_ENTRY_TYPE_DSA;
|
entry->type = DSMR_ENTRY_TYPE_DSA;
|
||||||
|
state->handle = DSA_HANDLE_INVALID;
|
||||||
|
state->tranche = -1;
|
||||||
|
}
|
||||||
|
else if (entry->type != DSMR_ENTRY_TYPE_DSA)
|
||||||
|
ereport(ERROR,
|
||||||
|
(errmsg("requested DSA does not match type of existing entry")));
|
||||||
|
|
||||||
|
if (state->tranche == -1)
|
||||||
|
{
|
||||||
|
*found = false;
|
||||||
|
|
||||||
/* Initialize the LWLock tranche for the DSA. */
|
/* Initialize the LWLock tranche for the DSA. */
|
||||||
state->tranche = LWLockNewTrancheId(name);
|
state->tranche = LWLockNewTrancheId(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state->handle == DSA_HANDLE_INVALID)
|
||||||
|
{
|
||||||
|
*found = false;
|
||||||
|
|
||||||
/* Initialize the DSA. */
|
/* Initialize the DSA. */
|
||||||
ret = dsa_create(state->tranche);
|
ret = dsa_create(state->tranche);
|
||||||
@@ -309,17 +326,11 @@ GetNamedDSA(const char *name, bool *found)
|
|||||||
/* Store handle for other backends to use. */
|
/* Store handle for other backends to use. */
|
||||||
state->handle = dsa_get_handle(ret);
|
state->handle = dsa_get_handle(ret);
|
||||||
}
|
}
|
||||||
else if (entry->type != DSMR_ENTRY_TYPE_DSA)
|
else if (dsa_is_attached(state->handle))
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errmsg("requested DSA does not match type of existing entry")));
|
(errmsg("requested DSA already attached to current process")));
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
NamedDSAState *state = &entry->dsa;
|
|
||||||
|
|
||||||
if (dsa_is_attached(state->handle))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errmsg("requested DSA already attached to current process")));
|
|
||||||
|
|
||||||
/* Attach to existing DSA. */
|
/* Attach to existing DSA. */
|
||||||
ret = dsa_attach(state->handle);
|
ret = dsa_attach(state->handle);
|
||||||
dsa_pin_mapping(ret);
|
dsa_pin_mapping(ret);
|
||||||
@@ -346,6 +357,7 @@ GetNamedDSHash(const char *name, const dshash_parameters *params, bool *found)
|
|||||||
DSMRegistryEntry *entry;
|
DSMRegistryEntry *entry;
|
||||||
MemoryContext oldcontext;
|
MemoryContext oldcontext;
|
||||||
dshash_table *ret;
|
dshash_table *ret;
|
||||||
|
NamedDSHState *dsh_state;
|
||||||
|
|
||||||
Assert(params);
|
Assert(params);
|
||||||
Assert(found);
|
Assert(found);
|
||||||
@@ -365,45 +377,57 @@ GetNamedDSHash(const char *name, const dshash_parameters *params, bool *found)
|
|||||||
init_dsm_registry();
|
init_dsm_registry();
|
||||||
|
|
||||||
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
entry = dshash_find_or_insert(dsm_registry_table, name, found);
|
||||||
|
dsh_state = &entry->dsh;
|
||||||
if (!(*found))
|
if (!(*found))
|
||||||
{
|
{
|
||||||
NamedDSHState *dsh_state = &entry->dsh;
|
|
||||||
dshash_parameters params_copy;
|
|
||||||
dsa_area *dsa;
|
|
||||||
|
|
||||||
entry->type = DSMR_ENTRY_TYPE_DSH;
|
entry->type = DSMR_ENTRY_TYPE_DSH;
|
||||||
|
dsh_state->dsa_handle = DSA_HANDLE_INVALID;
|
||||||
|
dsh_state->dsh_handle = DSHASH_HANDLE_INVALID;
|
||||||
|
dsh_state->tranche = -1;
|
||||||
|
}
|
||||||
|
else if (entry->type != DSMR_ENTRY_TYPE_DSH)
|
||||||
|
ereport(ERROR,
|
||||||
|
(errmsg("requested DSHash does not match type of existing entry")));
|
||||||
|
|
||||||
|
if (dsh_state->tranche == -1)
|
||||||
|
{
|
||||||
|
*found = false;
|
||||||
|
|
||||||
/* Initialize the LWLock tranche for the hash table. */
|
/* Initialize the LWLock tranche for the hash table. */
|
||||||
dsh_state->tranche = LWLockNewTrancheId(name);
|
dsh_state->tranche = LWLockNewTrancheId(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dsh_state->dsa_handle == DSA_HANDLE_INVALID)
|
||||||
|
{
|
||||||
|
dshash_parameters params_copy;
|
||||||
|
dsa_area *dsa;
|
||||||
|
|
||||||
|
*found = false;
|
||||||
|
|
||||||
/* Initialize the DSA for the hash table. */
|
/* Initialize the DSA for the hash table. */
|
||||||
dsa = dsa_create(dsh_state->tranche);
|
dsa = dsa_create(dsh_state->tranche);
|
||||||
dsa_pin(dsa);
|
|
||||||
dsa_pin_mapping(dsa);
|
|
||||||
|
|
||||||
/* Initialize the dshash table. */
|
/* Initialize the dshash table. */
|
||||||
memcpy(¶ms_copy, params, sizeof(dshash_parameters));
|
memcpy(¶ms_copy, params, sizeof(dshash_parameters));
|
||||||
params_copy.tranche_id = dsh_state->tranche;
|
params_copy.tranche_id = dsh_state->tranche;
|
||||||
ret = dshash_create(dsa, ¶ms_copy, NULL);
|
ret = dshash_create(dsa, ¶ms_copy, NULL);
|
||||||
|
|
||||||
|
dsa_pin(dsa);
|
||||||
|
dsa_pin_mapping(dsa);
|
||||||
|
|
||||||
/* Store handles for other backends to use. */
|
/* Store handles for other backends to use. */
|
||||||
dsh_state->dsa_handle = dsa_get_handle(dsa);
|
dsh_state->dsa_handle = dsa_get_handle(dsa);
|
||||||
dsh_state->dsh_handle = dshash_get_hash_table_handle(ret);
|
dsh_state->dsh_handle = dshash_get_hash_table_handle(ret);
|
||||||
}
|
}
|
||||||
else if (entry->type != DSMR_ENTRY_TYPE_DSH)
|
else if (dsa_is_attached(dsh_state->dsa_handle))
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errmsg("requested DSHash does not match type of existing entry")));
|
(errmsg("requested DSHash already attached to current process")));
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
NamedDSHState *dsh_state = &entry->dsh;
|
|
||||||
dsa_area *dsa;
|
dsa_area *dsa;
|
||||||
|
|
||||||
/* XXX: Should we verify params matches what table was created with? */
|
/* XXX: Should we verify params matches what table was created with? */
|
||||||
|
|
||||||
if (dsa_is_attached(dsh_state->dsa_handle))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errmsg("requested DSHash already attached to current process")));
|
|
||||||
|
|
||||||
/* Attach to existing DSA for the hash table. */
|
/* Attach to existing DSA for the hash table. */
|
||||||
dsa = dsa_attach(dsh_state->dsa_handle);
|
dsa = dsa_attach(dsh_state->dsa_handle);
|
||||||
dsa_pin_mapping(dsa);
|
dsa_pin_mapping(dsa);
|
||||||
@@ -439,6 +463,17 @@ pg_get_dsm_registry_allocations(PG_FUNCTION_ARGS)
|
|||||||
Datum vals[3];
|
Datum vals[3];
|
||||||
bool nulls[3] = {0};
|
bool nulls[3] = {0};
|
||||||
|
|
||||||
|
/* Do not show partially-initialized entries. */
|
||||||
|
if (entry->type == DSMR_ENTRY_TYPE_DSM &&
|
||||||
|
entry->dsm.handle == DSM_HANDLE_INVALID)
|
||||||
|
continue;
|
||||||
|
if (entry->type == DSMR_ENTRY_TYPE_DSA &&
|
||||||
|
entry->dsa.handle == DSA_HANDLE_INVALID)
|
||||||
|
continue;
|
||||||
|
if (entry->type == DSMR_ENTRY_TYPE_DSH &&
|
||||||
|
entry->dsh.dsa_handle == DSA_HANDLE_INVALID)
|
||||||
|
continue;
|
||||||
|
|
||||||
vals[0] = CStringGetTextDatum(entry->name);
|
vals[0] = CStringGetTextDatum(entry->name);
|
||||||
vals[1] = CStringGetTextDatum(DSMREntryTypeNames[entry->type]);
|
vals[1] = CStringGetTextDatum(DSMREntryTypeNames[entry->type]);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user