Commit 27838981 authored by Alvaro Herrera's avatar Alvaro Herrera

Relax lock level for setting PGPROC->statusFlags

We don't actually need a lock to set PGPROC->statusFlags itself; what we
do need is a shared lock on either XidGenLock or ProcArrayLock in order to
ensure MyProc->pgxactoff keeps still while we modify the mirror array in
ProcGlobal->statusFlags.  Some places were using an exclusive lock for
that, which is excessive.  Relax those to use shared lock only.

procarray.c has a couple of places with somewhat brittle assumptions
about PGPROC changes: ProcArrayEndTransaction uses only shared lock, so
it's permissible to change MyProc only.  On the other hand,
ProcArrayEndTransactionInternal also changes other procs, so it must
hold exclusive lock.  Add asserts to ensure those assumptions continue
to hold.

Author: Álvaro Herrera <alvherre@alvh.no-ip.org>
Reviewed-by: default avatarMichael Paquier <michael@paquier.xyz>
Discussion: https://postgr.es/m/20201117155501.GA13805@alvherre.pgsql
parent 2cccb627
...@@ -1741,7 +1741,7 @@ vacuum_rel(Oid relid, RangeVar *relation, VacuumParams *params) ...@@ -1741,7 +1741,7 @@ vacuum_rel(Oid relid, RangeVar *relation, VacuumParams *params)
* MyProc->xid/xmin, otherwise GetOldestNonRemovableTransactionId() * MyProc->xid/xmin, otherwise GetOldestNonRemovableTransactionId()
* might appear to go backwards, which is probably Not Good. * might appear to go backwards, which is probably Not Good.
*/ */
LWLockAcquire(ProcArrayLock, LW_EXCLUSIVE); LWLockAcquire(ProcArrayLock, LW_SHARED);
MyProc->statusFlags |= PROC_IN_VACUUM; MyProc->statusFlags |= PROC_IN_VACUUM;
if (params->is_wraparound) if (params->is_wraparound)
MyProc->statusFlags |= PROC_VACUUM_FOR_WRAPAROUND; MyProc->statusFlags |= PROC_VACUUM_FOR_WRAPAROUND;
......
...@@ -181,7 +181,7 @@ StartupDecodingContext(List *output_plugin_options, ...@@ -181,7 +181,7 @@ StartupDecodingContext(List *output_plugin_options,
*/ */
if (!IsTransactionOrTransactionBlock()) if (!IsTransactionOrTransactionBlock())
{ {
LWLockAcquire(ProcArrayLock, LW_EXCLUSIVE); LWLockAcquire(ProcArrayLock, LW_SHARED);
MyProc->statusFlags |= PROC_IN_LOGICAL_DECODING; MyProc->statusFlags |= PROC_IN_LOGICAL_DECODING;
ProcGlobal->statusFlags[MyProc->pgxactoff] = MyProc->statusFlags; ProcGlobal->statusFlags[MyProc->pgxactoff] = MyProc->statusFlags;
LWLockRelease(ProcArrayLock); LWLockRelease(ProcArrayLock);
......
...@@ -527,7 +527,7 @@ ReplicationSlotRelease(void) ...@@ -527,7 +527,7 @@ ReplicationSlotRelease(void)
MyReplicationSlot = NULL; MyReplicationSlot = NULL;
/* might not have been set when we've been a plain slot */ /* might not have been set when we've been a plain slot */
LWLockAcquire(ProcArrayLock, LW_EXCLUSIVE); LWLockAcquire(ProcArrayLock, LW_SHARED);
MyProc->statusFlags &= ~PROC_IN_LOGICAL_DECODING; MyProc->statusFlags &= ~PROC_IN_LOGICAL_DECODING;
ProcGlobal->statusFlags[MyProc->pgxactoff] = MyProc->statusFlags; ProcGlobal->statusFlags[MyProc->pgxactoff] = MyProc->statusFlags;
LWLockRelease(ProcArrayLock); LWLockRelease(ProcArrayLock);
......
...@@ -662,6 +662,8 @@ ProcArrayEndTransaction(PGPROC *proc, TransactionId latestXid) ...@@ -662,6 +662,8 @@ ProcArrayEndTransaction(PGPROC *proc, TransactionId latestXid)
/* avoid unnecessarily dirtying shared cachelines */ /* avoid unnecessarily dirtying shared cachelines */
if (proc->statusFlags & PROC_VACUUM_STATE_MASK) if (proc->statusFlags & PROC_VACUUM_STATE_MASK)
{ {
/* Only safe to change my own flags with just share lock */
Assert(proc == MyProc);
Assert(!LWLockHeldByMe(ProcArrayLock)); Assert(!LWLockHeldByMe(ProcArrayLock));
LWLockAcquire(ProcArrayLock, LW_SHARED); LWLockAcquire(ProcArrayLock, LW_SHARED);
Assert(proc->statusFlags == ProcGlobal->statusFlags[proc->pgxactoff]); Assert(proc->statusFlags == ProcGlobal->statusFlags[proc->pgxactoff]);
...@@ -682,7 +684,11 @@ ProcArrayEndTransactionInternal(PGPROC *proc, TransactionId latestXid) ...@@ -682,7 +684,11 @@ ProcArrayEndTransactionInternal(PGPROC *proc, TransactionId latestXid)
{ {
size_t pgxactoff = proc->pgxactoff; size_t pgxactoff = proc->pgxactoff;
Assert(LWLockHeldByMe(ProcArrayLock)); /*
* Note: we need exclusive lock here because we're going to
* change other processes' PGPROC entries.
*/
Assert(LWLockHeldByMeInMode(ProcArrayLock, LW_EXCLUSIVE));
Assert(TransactionIdIsValid(ProcGlobal->xids[pgxactoff])); Assert(TransactionIdIsValid(ProcGlobal->xids[pgxactoff]));
Assert(ProcGlobal->xids[pgxactoff] == proc->xid); Assert(ProcGlobal->xids[pgxactoff] == proc->xid);
......
...@@ -623,7 +623,7 @@ FindLockCycleRecurseMember(PGPROC *checkProc, ...@@ -623,7 +623,7 @@ FindLockCycleRecurseMember(PGPROC *checkProc,
* because that flag is set at process start and never * because that flag is set at process start and never
* reset. There is logic elsewhere to avoid canceling an * reset. There is logic elsewhere to avoid canceling an
* autovacuum that is working to prevent XID wraparound * autovacuum that is working to prevent XID wraparound
* problems (which needs to read a different vacuumFlag * problems (which needs to read a different statusFlags
* bit), but we don't do that here to avoid grabbing * bit), but we don't do that here to avoid grabbing
* ProcArrayLock. * ProcArrayLock.
*/ */
......
...@@ -98,6 +98,11 @@ typedef enum ...@@ -98,6 +98,11 @@ typedef enum
* The semaphore and lock-activity fields in a prepared-xact PGPROC are unused, * The semaphore and lock-activity fields in a prepared-xact PGPROC are unused,
* but its myProcLocks[] lists are valid. * but its myProcLocks[] lists are valid.
* *
* We allow many fields of this struct to be accessed without locks, such as
* statusFlags or delayChkpt. However, keep in mind that writing mirrored ones
* (see below) requires holding ProcArrayLock or XidGenLock in at least shared
* mode, so that pgxactoff does not change concurrently.
*
* Mirrored fields: * Mirrored fields:
* *
* Some fields in PGPROC (see "mirrored in ..." comment) are mirrored into an * Some fields in PGPROC (see "mirrored in ..." comment) are mirrored into an
......
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