From d32e8387bac15ad353029c259e832f78d456c7a8 Mon Sep 17 00:00:00 2001 From: Tom Lane Date: Wed, 27 Nov 2013 18:10:03 -0500 Subject: [PATCH] Fix stale-pointer problem in fast-path locking logic. When acquiring a lock in fast-path mode, we must reset the locallock object's lock and proclock fields to NULL. They are not necessarily that way to start with, because the locallock could be left over from a failed lock acquisition attempt earlier in the transaction. Failure to do this led to all sorts of interesting misbehaviors when LockRelease tried to clean up no-longer-related lock and proclock objects in shared memory. Per report from Dan Wood. In passing, modify LockRelease to elog not just Assert if it doesn't find lock and proclock objects for a formerly fast-path lock, matching the code in FastPathGetRelationLockEntry and LockRefindAndRelease. This isn't a bug but it will help in diagnosing any future bugs in this area. Also, modify FastPathTransferRelationLocks and FastPathGetRelationLockEntry to break out of their loops over the fastpath array once they've found the sole matching entry. This was inconsistently done in some search loops and not others. Improve assorted related comments, too. Back-patch to 9.2 where the fast-path mechanism was introduced. --- src/backend/storage/lmgr/lock.c | 83 ++++++++++++++++++++++----------- src/include/storage/lock.h | 20 ++++++-- 2 files changed, 73 insertions(+), 30 deletions(-) diff --git a/src/backend/storage/lmgr/lock.c b/src/backend/storage/lmgr/lock.c index 273c722302..df68c99f44 100644 --- a/src/backend/storage/lmgr/lock.c +++ b/src/backend/storage/lmgr/lock.c @@ -788,14 +788,16 @@ LockAcquireExtended(const LOCKTAG *locktag, } /* - * Emit a WAL record if acquisition of this lock needs to be replayed in a - * standby server. Only AccessExclusiveLocks can conflict with lock types - * that read-only transactions can acquire in a standby server. + * Prepare to emit a WAL record if acquisition of this lock needs to be + * replayed in a standby server. * - * Make sure this definition matches the one in - * GetRunningTransactionLocks(). + * Here we prepare to log; after lock is acquired we'll issue log record. + * This arrangement simplifies error recovery in case the preparation step + * fails. * - * First we prepare to log, then after lock acquired we issue log record. + * Only AccessExclusiveLocks can conflict with lock types that read-only + * transactions can acquire in a standby server. Make sure this definition + * matches the one in GetRunningTransactionLocks(). */ if (lockmode >= AccessExclusiveLock && locktag->locktag_type == LOCKTAG_RELATION && @@ -816,8 +818,8 @@ LockAcquireExtended(const LOCKTAG *locktag, * lock type on a relation we have already locked using the fast-path, but * for now we don't worry about that case either. */ - if (EligibleForRelationFastPath(locktag, lockmode) - && FastPathLocalUseCount < FP_LOCK_SLOTS_PER_BACKEND) + if (EligibleForRelationFastPath(locktag, lockmode) && + FastPathLocalUseCount < FP_LOCK_SLOTS_PER_BACKEND) { uint32 fasthashcode = FastPathStrongLockHashPartition(hashcode); bool acquired; @@ -837,6 +839,13 @@ LockAcquireExtended(const LOCKTAG *locktag, LWLockRelease(MyProc->backendLock); if (acquired) { + /* + * The locallock might contain stale pointers to some old shared + * objects; we MUST reset these to null before considering the + * lock to be acquired via fast-path. + */ + locallock->lock = NULL; + locallock->proclock = NULL; GrantLockLocal(locallock, owner); return LOCKACQUIRE_OK; } @@ -877,7 +886,13 @@ LockAcquireExtended(const LOCKTAG *locktag, LWLockAcquire(partitionLock, LW_EXCLUSIVE); /* - * Find or create a proclock entry with this tag + * Find or create lock and proclock entries with this tag + * + * Note: if the locallock object already existed, it might have a pointer + * to the lock already ... but we should not assume that that pointer is + * valid, since a lock object with zero hold and request counts can go + * away anytime. So we have to use SetupLockInTable() to recompute the + * lock and proclock pointers, even if they're already set. */ proclock = SetupLockInTable(lockMethodTable, MyProc, locktag, hashcode, lockmode); @@ -1010,7 +1025,7 @@ LockAcquireExtended(const LOCKTAG *locktag, LWLockRelease(partitionLock); /* - * Emit a WAL record if acquisition of this lock need to be replayed in a + * Emit a WAL record if acquisition of this lock needs to be replayed in a * standby server. */ if (log_lock) @@ -1049,11 +1064,6 @@ SetupLockInTable(LockMethod lockMethodTable, PGPROC *proc, /* * Find or create a lock with this tag. - * - * Note: if the locallock object already existed, it might have a pointer - * to the lock already ... but we probably should not assume that that - * pointer is valid, since a lock object with no locks can go away - * anytime. */ lock = (LOCK *) hash_search_with_hash_value(LockMethodLockHash, (const void *) locktag, @@ -1822,8 +1832,8 @@ LockRelease(const LOCKTAG *locktag, LOCKMODE lockmode, bool sessionLock) return TRUE; /* Attempt fast release of any lock eligible for the fast path. */ - if (EligibleForRelationFastPath(locktag, lockmode) - && FastPathLocalUseCount > 0) + if (EligibleForRelationFastPath(locktag, lockmode) && + FastPathLocalUseCount > 0) { bool released; @@ -1853,30 +1863,33 @@ LockRelease(const LOCKTAG *locktag, LOCKMODE lockmode, bool sessionLock) * Normally, we don't need to re-find the lock or proclock, since we kept * their addresses in the locallock table, and they couldn't have been * removed while we were holding a lock on them. But it's possible that - * the locks have been moved to the main hash table by another backend, in - * which case we might need to go look them up after all. + * the lock was taken fast-path and has since been moved to the main hash + * table by another backend, in which case we will need to look up the + * objects here. We assume the lock field is NULL if so. */ lock = locallock->lock; if (!lock) { PROCLOCKTAG proclocktag; - bool found; Assert(EligibleForRelationFastPath(locktag, lockmode)); lock = (LOCK *) hash_search_with_hash_value(LockMethodLockHash, (const void *) locktag, locallock->hashcode, HASH_FIND, - &found); - Assert(found && lock != NULL); + NULL); + if (!lock) + elog(ERROR, "failed to re-find shared lock object"); locallock->lock = lock; proclocktag.myLock = lock; proclocktag.myProc = MyProc; locallock->proclock = (PROCLOCK *) hash_search(LockMethodProcLockHash, (void *) &proclocktag, - HASH_FIND, &found); - Assert(found); + HASH_FIND, + NULL); + if (!locallock->proclock) + elog(ERROR, "failed to re-find shared proclock object"); } LOCK_PRINT("LockRelease: found", lock, lockmode); proclock = locallock->proclock; @@ -1957,7 +1970,8 @@ LockReleaseAll(LOCKMETHODID lockmethodid, bool allLocks) * entries, then we scan the process's proclocks and get rid of those. We * do this separately because we may have multiple locallock entries * pointing to the same proclock, and we daren't end up with any dangling - * pointers. + * pointers. Fast-path locks are cleaned up during the locallock table + * scan, though. */ hash_seq_init(&status, LockMethodLocalHash); @@ -2012,7 +2026,7 @@ LockReleaseAll(LOCKMETHODID lockmethodid, bool allLocks) /* * If the lock or proclock pointers are NULL, this lock was taken via - * the relation fast-path. + * the relation fast-path (and is not known to have been transferred). */ if (locallock->proclock == NULL || locallock->lock == NULL) { @@ -2026,7 +2040,10 @@ LockReleaseAll(LOCKMETHODID lockmethodid, bool allLocks) /* * If we don't currently hold the LWLock that protects our * fast-path data structures, we must acquire it before attempting - * to release the lock via the fast-path. + * to release the lock via the fast-path. We will continue to + * hold the LWLock until we're done scanning the locallock table, + * unless we hit a transferred fast-path lock. (XXX is this + * really such a good idea? There could be a lot of entries ...) */ if (!have_fast_path_lwlock) { @@ -2071,6 +2088,7 @@ LockReleaseAll(LOCKMETHODID lockmethodid, bool allLocks) RemoveLocalLock(locallock); } + /* Done with the fast-path data structures */ if (have_fast_path_lwlock) LWLockRelease(MyProc->backendLock); @@ -2422,6 +2440,7 @@ FastPathUnGrantRelationLock(Oid relid, LOCKMODE lockmode) Assert(!result); FAST_PATH_CLEAR_LOCKMODE(MyProc, f, lockmode); result = true; + /* we continue iterating so as to update FastPathLocalUseCount */ } if (FAST_PATH_GET_BITS(MyProc, f) != 0) ++FastPathLocalUseCount; @@ -2507,6 +2526,9 @@ FastPathTransferRelationLocks(LockMethod lockMethodTable, const LOCKTAG *locktag FAST_PATH_CLEAR_LOCKMODE(proc, f, lockmode); } LWLockRelease(partitionLock); + + /* No need to examine remaining slots. */ + break; } LWLockRelease(proc->backendLock); } @@ -2517,6 +2539,8 @@ FastPathTransferRelationLocks(LockMethod lockMethodTable, const LOCKTAG *locktag * FastPathGetLockEntry * Return the PROCLOCK for a lock originally taken via the fast-path, * transferring it to the primary lock table if necessary. + * + * Note: caller takes care of updating the locallock object. */ static PROCLOCK * FastPathGetRelationLockEntry(LOCALLOCK *locallock) @@ -2560,6 +2584,9 @@ FastPathGetRelationLockEntry(LOCALLOCK *locallock) FAST_PATH_CLEAR_LOCKMODE(MyProc, f, lockmode); LWLockRelease(partitionLock); + + /* No need to examine remaining slots. */ + break; } LWLockRelease(MyProc->backendLock); @@ -2732,6 +2759,8 @@ GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode) */ if (VirtualTransactionIdIsValid(vxid)) vxids[count++] = vxid; + + /* No need to examine remaining slots. */ break; } diff --git a/src/include/storage/lock.h b/src/include/storage/lock.h index 9642a19821..7a31d3363c 100644 --- a/src/include/storage/lock.h +++ b/src/include/storage/lock.h @@ -379,6 +379,20 @@ typedef struct PROCLOCK * shared memory. We also track the number of lock acquisitions per * ResourceOwner, so that we can release just those locks belonging to a * particular ResourceOwner. + * + * When holding a lock taken "normally", the lock and proclock fields always + * point to the associated objects in shared memory. However, if we acquired + * the lock via the fast-path mechanism, the lock and proclock fields are set + * to NULL, since there probably aren't any such objects in shared memory. + * (If the lock later gets promoted to normal representation, we may eventually + * update our locallock's lock/proclock fields after finding the shared + * objects.) + * + * Caution: a locallock object can be left over from a failed lock acquisition + * attempt. In this case its lock/proclock fields are untrustworthy, since + * the shared lock object is neither held nor awaited, and hence is available + * to be reclaimed. If nLocks > 0 then these pointers must either be valid or + * NULL, but when nLocks == 0 they should be considered garbage. */ typedef struct LOCALLOCKTAG { @@ -404,13 +418,13 @@ typedef struct LOCALLOCK LOCALLOCKTAG tag; /* unique identifier of locallock entry */ /* data */ - LOCK *lock; /* associated LOCK object in shared mem */ - PROCLOCK *proclock; /* associated PROCLOCK object in shmem */ + LOCK *lock; /* associated LOCK object, if any */ + PROCLOCK *proclock; /* associated PROCLOCK object, if any */ uint32 hashcode; /* copy of LOCKTAG's hash value */ int64 nLocks; /* total number of times lock is held */ int numLockOwners; /* # of relevant ResourceOwners */ int maxLockOwners; /* allocated size of array */ - bool holdsStrongLockCount; /* bumped FastPathStrongRelatonLocks? */ + bool holdsStrongLockCount; /* bumped FastPathStrongRelationLocks */ LOCALLOCKOWNER *lockOwners; /* dynamically resizable array */ } LOCALLOCK; -- 2.40.0