Commit 80484049 authored by Tom Lane's avatar Tom Lane

Fix bogus initialization of replication origin shared memory state.

The previous coding zeroed out offsetof(ReplicationStateCtl, states)
more bytes than it was entitled to, as a consequence of starting the
zeroing from the wrong pointer (or, if you prefer, using the wrong
calculation of how much to zero).

It's unsurprising that this has not caused any reported problems,
since it can be expected that the newly-allocated block is at the end
of what we've used in shared memory, and we always make the shmem
block substantially bigger than minimally necessary.  Nonetheless,
this is wrong and it could bite us someday; plus it's a dangerous
model for somebody to copy.

This dates back to the introduction of this code (commit 5aa23504),
so back-patch to all supported branches.
parent 36ac359d
...@@ -144,7 +144,9 @@ typedef struct ReplicationStateOnDisk ...@@ -144,7 +144,9 @@ typedef struct ReplicationStateOnDisk
typedef struct ReplicationStateCtl typedef struct ReplicationStateCtl
{ {
/* Tranche to use for per-origin LWLocks */
int tranche_id; int tranche_id;
/* Array of length max_replication_slots */
ReplicationState states[FLEXIBLE_ARRAY_MEMBER]; ReplicationState states[FLEXIBLE_ARRAY_MEMBER];
} ReplicationStateCtl; } ReplicationStateCtl;
...@@ -161,6 +163,10 @@ TimestampTz replorigin_session_origin_timestamp = 0; ...@@ -161,6 +163,10 @@ TimestampTz replorigin_session_origin_timestamp = 0;
* max_replication_slots? * max_replication_slots?
*/ */
static ReplicationState *replication_states; static ReplicationState *replication_states;
/*
* Actual shared memory block (replication_states[] is now part of this).
*/
static ReplicationStateCtl *replication_states_ctl; static ReplicationStateCtl *replication_states_ctl;
/* /*
...@@ -476,7 +482,7 @@ ReplicationOriginShmemSize(void) ...@@ -476,7 +482,7 @@ ReplicationOriginShmemSize(void)
/* /*
* XXX: max_replication_slots is arguably the wrong thing to use, as here * XXX: max_replication_slots is arguably the wrong thing to use, as here
* we keep the replay state of *remote* transactions. But for now it seems * we keep the replay state of *remote* transactions. But for now it seems
* sufficient to reuse it, lest we introduce a separate GUC. * sufficient to reuse it, rather than introduce a separate GUC.
*/ */
if (max_replication_slots == 0) if (max_replication_slots == 0)
return size; return size;
...@@ -506,9 +512,9 @@ ReplicationOriginShmemInit(void) ...@@ -506,9 +512,9 @@ ReplicationOriginShmemInit(void)
{ {
int i; int i;
replication_states_ctl->tranche_id = LWTRANCHE_REPLICATION_ORIGIN_STATE; MemSet(replication_states_ctl, 0, ReplicationOriginShmemSize());
MemSet(replication_states, 0, ReplicationOriginShmemSize()); replication_states_ctl->tranche_id = LWTRANCHE_REPLICATION_ORIGIN_STATE;
for (i = 0; i < max_replication_slots; i++) for (i = 0; i < max_replication_slots; i++)
{ {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment