diff options
author | Kaloian Manassiev <kaloian.manassiev@mongodb.com> | 2014-09-08 14:31:23 -0400 |
---|---|---|
committer | Kaloian Manassiev <kaloian.manassiev@mongodb.com> | 2014-09-10 16:19:11 -0400 |
commit | 2bae1279afcab72bd20967d65c8e350f8c2cffb9 (patch) | |
tree | cf8deaee460be24419d8140927f2c5c7130d8e09 /src/mongo | |
parent | d71566a55ebbe76ee7f40fd3f2a2a8cd486b0d16 (diff) | |
download | mongo-2bae1279afcab72bd20967d65c8e350f8c2cffb9.tar.gz |
SERVER-14668 Fix the lock manager granting logic
Diffstat (limited to 'src/mongo')
-rw-r--r-- | src/mongo/db/concurrency/d_concurrency_test.cpp | 14 | ||||
-rw-r--r-- | src/mongo/db/concurrency/lock_mgr_new.cpp | 417 | ||||
-rw-r--r-- | src/mongo/db/concurrency/lock_mgr_new.h | 64 | ||||
-rw-r--r-- | src/mongo/db/concurrency/lock_mgr_new_test.cpp | 26 |
4 files changed, 331 insertions, 190 deletions
diff --git a/src/mongo/db/concurrency/d_concurrency_test.cpp b/src/mongo/db/concurrency/d_concurrency_test.cpp index 19a2bbead66..f29dbbc3c28 100644 --- a/src/mongo/db/concurrency/d_concurrency_test.cpp +++ b/src/mongo/db/concurrency/d_concurrency_test.cpp @@ -71,12 +71,22 @@ namespace mongo { ls.assertAtLeastReadLocked("db2"); } - TEST(DConcurrency, MultipleDBLocks) { + TEST(DConcurrency, MultipleWriteDBLocksOnSameThread) { + LockState ls; + + Lock::DBWrite r1(&ls, "db1"); + Lock::DBWrite r2(&ls, "db1"); + + ASSERT(ls.isWriteLocked("db1")); + } + + TEST(DConcurrency, MultipleConflictingDBLocksOnSameThread) { LockState ls; Lock::DBWrite r1(&ls, "db1"); Lock::DBRead r2(&ls, "db1"); - ls.assertWriteLocked("db1"); + ASSERT(ls.isWriteLocked("db1")); } + } // namespace mongo diff --git a/src/mongo/db/concurrency/lock_mgr_new.cpp b/src/mongo/db/concurrency/lock_mgr_new.cpp index 6c4fc94ebea..c0410883ab1 100644 --- a/src/mongo/db/concurrency/lock_mgr_new.cpp +++ b/src/mongo/db/concurrency/lock_mgr_new.cpp @@ -107,41 +107,56 @@ namespace newlm { } LockResult LockManager::lock(const ResourceId& resId, LockRequest* request, LockMode mode) { + dassert(mode > MODE_NONE); + + // Fast path for acquiring the same lock multiple times in modes, which are already covered + // by the current mode. It is safe to do this without locking, because 1) all calls for the + // same lock request must be done on the same thread and 2) if there are lock requests + // hanging off a given LockHead, then this lock will never disappear. + if ((LockConflictsTable[request->mode] | LockConflictsTable[mode]) == + LockConflictsTable[request->mode]) { + request->recursiveCount++; + return LOCK_OK; + } + + // TODO: For the time being we do not need conversions between unrelated lock modes (i.e., + // modes which both add and remove to the conflicts set), so these are not implemented yet + // (e.g., S -> IX). + invariant((LockConflictsTable[request->mode] | LockConflictsTable[mode]) == + LockConflictsTable[mode]); + LockBucket* bucket = _getBucket(resId); scoped_spinlock scopedLock(bucket->mutex); - LockHeadMap::iterator it = bucket->data.find(resId); + LockHead* lock; + LockHeadMap::iterator it = bucket->data.find(resId); if (it == bucket->data.end()) { // Lock is free (not on the map) invariant(request->status == LockRequest::STATUS_NEW); - request->prev = NULL; - request->next = NULL; - request->status = LockRequest::STATUS_GRANTED; - request->mode = mode; - request->convertMode = MODE_NONE; - request->recursiveCount = 1; - - LockHead* lock = new LockHead(resId); - lock->grantedQueue = request; - lock->grantedModes = modeMask(mode); - lock->conflictQueueBegin = NULL; - lock->conflictQueueEnd = NULL; - + lock = new LockHead(resId); bucket->data.insert(LockHeadPair(resId, lock)); - return LOCK_OK; } + else { + // Lock is not free + lock = it->second; - // Lock not free case (still under the spin-lock for the bucket) - LockHead* lock = it->second; + invariant(lock->grantedQueue != NULL); + invariant(lock->grantedModes != 0); + } - invariant(lock->grantedQueue != NULL); - invariant(lock->grantedModes != 0); + request->recursiveCount++; if (request->status == LockRequest::STATUS_NEW) { + invariant(request->recursiveCount == 1); + // New lock request if (conflicts(mode, lock->grantedModes)) { + request->status = LockRequest::STATUS_WAITING; + request->mode = mode; + request->convertMode = MODE_NONE; + // Put it on the wait queue. This is the place where various policies could be // applied for where in the wait queue does a request go. if (lock->conflictQueueBegin == NULL) { @@ -149,10 +164,6 @@ namespace newlm { request->prev = NULL; request->next = NULL; - request->status = LockRequest::STATUS_WAITING; - request->mode = mode; - request->convertMode = MODE_NONE; - request->recursiveCount = 1; lock->conflictQueueBegin = request; lock->conflictQueueEnd = request; @@ -162,15 +173,13 @@ namespace newlm { request->prev = lock->conflictQueueEnd; request->next = NULL; - request->status = LockRequest::STATUS_WAITING; - request->mode = mode; - request->convertMode = MODE_NONE; - request->recursiveCount = 1; lock->conflictQueueEnd->next = request; lock->conflictQueueEnd = request; } + lock->changeRequestedModeCount(mode, LockHead::Increment); + return LOCK_WAITING; } else { // No conflict, new request @@ -179,44 +188,77 @@ namespace newlm { request->status = LockRequest::STATUS_GRANTED; request->mode = mode; request->convertMode = MODE_NONE; - request->recursiveCount = 1; if (lock->grantedQueue != NULL) { lock->grantedQueue->prev = request; } lock->grantedQueue = request; - lock->grantedModes |= modeMask(mode); + + lock->changeGrantedModeCount(mode, LockHead::Increment); return LOCK_OK; } } else { - // Relock or conversion. To keep it simple, cannot request a conversion while a lock - // is waiting or pending conversion. + // If we are here, we already hold the lock in some mode. In order to keep it simple, + // we do not allow requesting a conversion while a lock is already waiting or pending + // conversion, hence the assertion below. invariant(request->status == LockRequest::STATUS_GRANTED); + invariant(request->recursiveCount > 1); + invariant(request->mode != mode); - if (conflicts(mode, lock->grantedModes)) { + // Construct granted mask without our current mode, so that it is not counted as + // conflicting + uint32_t grantedModesWithoutCurrentRequest = 0; + + // We start the counting at 1 below, because LockModesCount also includes MODE_NONE + // at position 0, which can never be acquired/granted. + for (uint32_t i = 1; i < LockModesCount; i++) { + const uint32_t currentRequestHolds = + (request->mode == static_cast<LockMode>(i) ? 1 : 0); + + if (lock->grantedCounts[i] > currentRequestHolds) { + grantedModesWithoutCurrentRequest |= modeMask(static_cast<LockMode>(i)); + } + } + + // This check favours conversion requests over pending requests. For example: + // + // T1 requests lock L in IS + // T2 requests lock L in X + // T1 then upgrades L from IS -> S + // + // Because the check does not look into the requested modes bitmap, it will grant L + // to T1 in S mode, instead of block, which would otherwise cause deadlock. + if (conflicts(mode, grantedModesWithoutCurrentRequest)) { request->status = LockRequest::STATUS_CONVERTING; request->convertMode = mode; - request->recursiveCount++; - // This will update the grantedModes on the lock - _recalcAndGrant(lock, NULL, request); + lock->changeGrantedModeCount(request->convertMode, LockHead::Increment); - return (request->status == LockRequest::STATUS_GRANTED ? LOCK_OK : LOCK_WAITING); + return LOCK_WAITING; } else { // No conflict, existing request - request->mode = std::max(request->mode, mode); - request->recursiveCount++; + lock->changeGrantedModeCount(mode, LockHead::Increment); + lock->changeGrantedModeCount(request->mode, LockHead::Decrement); + request->mode = mode; - lock->grantedModes |= modeMask(mode); return LOCK_OK; } } } - void LockManager::unlock(LockRequest* request) { + bool LockManager::unlock(LockRequest* request) { + // Fast path for decrementing multiple references of the same lock. It is safe to do this + // without locking, because 1) all calls for the same lock request must be done on the same + // thread and 2) if there are lock requests hanging of a given LockHead, then this lock + // will never disappear. + request->recursiveCount--; + if ((request->status == LockRequest::STATUS_GRANTED) && (request->recursiveCount > 0)) { + return false; + } + LockBucket* bucket = _getBucket(request->resourceId); scoped_spinlock scopedLock(bucket->mutex); @@ -231,14 +273,31 @@ namespace newlm { invariant(lock->grantedQueue != NULL); invariant(lock->grantedModes != 0); - request->recursiveCount--; - - bool recalcGrantedModes = false; + bool grantedModesChanged = false; if (request->status == LockRequest::STATUS_WAITING) { // This cancels a pending lock request invariant(request->recursiveCount == 0); - recalcGrantedModes = true; + + // Remove from the conflict queue + if (request->prev != NULL) { + request->prev->next = request->next; + } + else { + lock->conflictQueueBegin = request->next; + } + + if (request->next != NULL) { + request->next->prev = request->prev; + } + else { + lock->conflictQueueEnd = request->prev; + } + + request->prev = NULL; + request->next = NULL; + + lock->changeRequestedModeCount(request->mode, LockHead::Decrement); } else if (request->status == LockRequest::STATUS_CONVERTING) { // This cancels a pending convert request @@ -247,35 +306,54 @@ namespace newlm { // Lock only goes from GRANTED to CONVERTING, so cancelling the conversion request // brings it back to the previous granted mode. request->status = LockRequest::STATUS_GRANTED; - request->convertMode = MODE_NONE; - recalcGrantedModes = true; + lock->changeGrantedModeCount(request->convertMode, LockHead::Decrement); + grantedModesChanged = true; + + request->convertMode = MODE_NONE; } - else { + else if (request->status == LockRequest::STATUS_GRANTED) { // This releases an existing lock - invariant(request->status == LockRequest::STATUS_GRANTED); - invariant(request->recursiveCount >= 0); + invariant(request->recursiveCount == 0); - if (request->recursiveCount == 0) { - recalcGrantedModes = true; + // Remove from the granted list + if (request->prev != NULL) { + request->prev->next = request->next; + } + else { + lock->grantedQueue = request->next; + } + + if (request->next != NULL) { + request->next->prev = request->prev; } - } - if (!recalcGrantedModes) return; + lock->changeGrantedModeCount(request->mode, LockHead::Decrement); + grantedModesChanged = true; + } + else { + // Invalid request status + invariant(false); + } - _recalcAndGrant(lock, request, NULL); + // Granted modes did not change, so nothing to be done + if (grantedModesChanged) { + _onLockModeChanged(lock); - // If some locks have been granted then there should be something on the grantedQueue and - // vice versa (sanity check that either one or the other is true). - invariant((lock->grantedModes == 0) ^ (lock->grantedQueue != NULL)); + // If some locks have been granted then there should be something on the grantedQueue and + // vice versa (sanity check that either one or the other is true). + invariant((lock->grantedModes == 0) ^ (lock->grantedQueue != NULL)); - // This lock is no longer in use - if (lock->grantedModes == 0) { - bucket->data.erase(it); + // This lock is no longer in use + if (lock->grantedModes == 0) { + bucket->data.erase(it); - // TODO: As an optimization, we could keep a cache of pre-allocated LockHead objects - delete lock; + // TODO: As an optimization, we could keep a cache of pre-allocated LockHead objects + delete lock; + } } + + return (request->recursiveCount == 0); } void LockManager::downgrade(LockRequest* request, LockMode newMode) { @@ -301,160 +379,101 @@ namespace newlm { invariant(lock->grantedQueue != NULL); invariant(lock->grantedModes != 0); + lock->changeGrantedModeCount(newMode, LockHead::Increment); + lock->changeGrantedModeCount(request->mode, LockHead::Decrement); request->mode = newMode; - _recalcAndGrant(it->second, NULL, NULL); + _onLockModeChanged(it->second); } void LockManager::setNoCheckForLeakedLocksTestOnly(bool newValue) { _noCheckForLeakedLocksTestOnly = newValue; } - void LockManager::_recalcAndGrant(LockHead* lock, - LockRequest* unlocked, - LockRequest* converting) { - - // Either unlocked or converting must be specified, but not both. There are fixed callers - // of this method, so dassert. - dassert(!unlocked || !converting); - - // Resets the granted modes mask - lock->grantedModes = 0; - - uint32_t convertModes = 0; - - // First iterate through all owners of the Lock in order to determine the existing granted - // mode. The way this loop is implemented would mean linear complexity of the removal for - // each removal, which might turn out to be expensive. There are ways to fix it throgh - // keeping counts per lock-type, but those will be done once we have confirmed correctness. + void LockManager::_onLockModeChanged(LockHead* lock) { + // Unblock any converting requests (because conversions are still counted as granted and + // are on the granted queue). for (LockRequest* iter = lock->grantedQueue; iter != NULL; iter = iter->next) { - - // If the recursiveCount of this request is zero, this should be the lock that was just - // unlocked by the LockManager::unlock call. There should be exactly one for the unlock - // case and none for the recalc on downgrade case (where changed != NULL). - if (iter->recursiveCount == 0) { - invariant(iter == unlocked); - - // If recursive count is zero, this request must be removed from the granted list - // and it does not participate in the granted mode anymore. - if (iter->prev != NULL) { - iter->prev->next = iter->next; - } - else { - lock->grantedQueue = iter->next; - } - - if (iter->next != NULL) { - iter->next->prev = iter->prev; - } - - continue; - } - // Conversion requests are going in a separate queue if (iter->status == LockRequest::STATUS_CONVERTING) { invariant(iter->convertMode != 0); - convertModes |= iter->convertMode; - continue; - } + // Construct granted mask without our current mode, so that it is not accounted as + // a conflict + uint32_t grantedModesWithoutCurrentRequest = 0; - invariant(iter->convertMode == 0); + // We start the counting at 1 below, because LockModesCount also includes + // MODE_NONE at position 0, which can never be acquired/granted. + for (uint32_t i = 1; i < LockModesCount; i++) { + const uint32_t currentRequestHolds = + (iter->mode == static_cast<LockMode>(i) ? 1 : 0); - // All the others, just add them to the mask - lock->grantedModes |= modeMask(iter->mode); - } + const uint32_t currentRequestWaits = + (iter->convertMode == static_cast<LockMode>(i) ? 1 : 0); + + // We cannot both hold and wait on the same lock mode + invariant(currentRequestHolds + currentRequestWaits <= 1); - // Potentially grant any conversion requests by doing another pass on the granted queue. We - // cannot do this pass together with the loop above, because there could be a conversion - // request which would conflict with an already granted request later in the queue, but we - // will see it first. E.g., consider this: - // - // GrantedQueue -> (S converts to X) -> (S) -> (S) -> NULL - // - // Now, if the first S is converted to X, since the grantedModes on the lock are reset to - // zero, it will be granted, because there is no conflict yet, which is a bug. - if (convertModes) { - for (LockRequest* iter = lock->grantedQueue; iter != NULL; iter = iter->next) { - // Grant any conversion requests, which are compatible with the lock mask which has - // been granted so far. - if ((iter->status == LockRequest::STATUS_CONVERTING) && - !conflicts(iter->convertMode, lock->grantedModes)) { - - iter->status = LockRequest::STATUS_GRANTED; - - // Locks are in strictly increasing strictness, so the convert mode would be - // higher than the existing mode - iter->mode = std::max(iter->mode, iter->convertMode); - iter->convertMode = MODE_NONE; - - lock->grantedModes |= modeMask(iter->mode); - - // The caller can infer that the lock was granted from the new mode and from - // the status changing to granted. - if (iter != converting) { - iter->notify->notify(lock->resourceId, LOCK_OK); + if (lock->grantedCounts[i] > (currentRequestHolds + currentRequestWaits)) { + grantedModesWithoutCurrentRequest |= modeMask(static_cast<LockMode>(i)); } } - } - // Throw in all the convert modes so that no new requests would be granted if there is - // a pending conflicting conversion request - lock->grantedModes |= convertModes; + if (!conflicts(iter->convertMode, grantedModesWithoutCurrentRequest)) { + lock->changeGrantedModeCount(iter->convertMode, LockHead::Increment); + lock->changeGrantedModeCount(iter->mode, LockHead::Decrement); + iter->mode = iter->convertMode; + + iter->notify->notify(lock->resourceId, LOCK_OK); + } + } + else { + invariant(iter->status == LockRequest::STATUS_GRANTED); + } } - // Grant all other requests next + // Grant any conflicting requests, which might now be unblocked LockRequest* iterNext = NULL; for (LockRequest* iter = lock->conflictQueueBegin; iter != NULL; iter = iterNext) { - invariant(iter != converting); invariant(iter->status == LockRequest::STATUS_WAITING); // Store the actual next pointer, because we muck with the iter below and move it to // the granted queue. iterNext = iter->next; - const bool cancelWaiting = (iter == unlocked); - const bool conflict = conflicts(iter->mode, lock->grantedModes); + if (conflicts(iter->mode, lock->grantedModes)) continue; - // Remove eligible entries from the conflict queue (cancelled we will just drop, - // non-conflicting we will add on the granted queue). - if (cancelWaiting || !conflict) { - // This is equivalent to the expression cancelWaiting => iter->recursiveCount == 0 - // i.e., if cancelling a waiting request, the recursive count should be zero. - invariant(!cancelWaiting || iter->recursiveCount == 0); - - if (iter->prev != NULL) { - iter->prev->next = iter->next; - } - else { - lock->conflictQueueBegin = iter->next; - } - - if (iter->next != NULL) { - iter->next->prev = iter->prev; - } - else { - lock->conflictQueueEnd = iter->prev; - } + if (iter->prev != NULL) { + iter->prev->next = iter->next; + } + else { + lock->conflictQueueBegin = iter->next; + } - // iter is detached at this point - iter->next = NULL; - iter->prev = NULL; + if (iter->next != NULL) { + iter->next->prev = iter->prev; + } + else { + lock->conflictQueueEnd = iter->prev; } - // This breaks the FIFO order and may potentially starve conflicters. Here is where the - // fairness policy should come into play. - if (cancelWaiting || conflict) continue; + // iter is detached from the granted queue at this point + iter->next = NULL; + iter->prev = NULL; - iter->next = lock->grantedQueue; iter->status = LockRequest::STATUS_GRANTED; + iter->next = lock->grantedQueue; + if (lock->grantedQueue != NULL) { lock->grantedQueue->prev = iter; } + lock->grantedQueue = iter; - lock->grantedModes |= modeMask(iter->mode); + + lock->changeGrantedModeCount(iter->mode, LockHead::Increment); + lock->changeRequestedModeCount(iter->mode, LockHead::Decrement); iter->notify->notify(lock->resourceId, LOCK_OK); } @@ -548,8 +567,15 @@ namespace newlm { // LockHead::LockHead(const ResourceId& resId) - : resourceId(resId) { - + : resourceId(resId), + grantedQueue(NULL), + grantedModes(0), + conflictQueueBegin(NULL), + conflictQueueEnd(NULL), + conflictModes(0) { + + memset(grantedCounts, 0, sizeof(grantedCounts)); + memset(conflictCounts, 0, sizeof(conflictCounts)); } LockHead::~LockHead() { @@ -557,6 +583,43 @@ namespace newlm { invariant(grantedModes == 0); invariant(conflictQueueBegin == NULL); invariant(conflictQueueEnd == NULL); + invariant(conflictModes == 0); + } + + void LockHead::changeGrantedModeCount(LockMode mode, ChangeModeCountAction action) { + if (action == Increment) { + invariant(grantedCounts[mode] >= 0); + if (++grantedCounts[mode] == 1) { + invariant((grantedModes & modeMask(mode)) == 0); + grantedModes |= modeMask(mode); + } + } + else { + invariant(action == Decrement); + invariant(grantedCounts[mode] >= 1); + if (--grantedCounts[mode] == 0) { + invariant((grantedModes & modeMask(mode)) == modeMask(mode)); + grantedModes &= ~modeMask(mode); + } + } + } + + void LockHead::changeRequestedModeCount(LockMode mode, ChangeModeCountAction action) { + if (action == Increment) { + invariant(conflictCounts[mode] >= 0); + if (++conflictCounts[mode] == 1) { + invariant((conflictModes & modeMask(mode)) == 0); + conflictModes |= modeMask(mode); + } + } + else { + invariant(action == Decrement); + invariant(conflictCounts[mode] >= 1); + if (--conflictCounts[mode] == 0) { + invariant((conflictModes & modeMask(mode)) == modeMask(mode)); + conflictModes &= ~modeMask(mode); + } + } } diff --git a/src/mongo/db/concurrency/lock_mgr_new.h b/src/mongo/db/concurrency/lock_mgr_new.h index b4bcd951a02..c37ae8b0763 100644 --- a/src/mongo/db/concurrency/lock_mgr_new.h +++ b/src/mongo/db/concurrency/lock_mgr_new.h @@ -61,6 +61,14 @@ namespace newlm { MODE_X = 4, }; + enum { + // Counts the entries in the LockMode enumeration. Used for array size allocations, etc. + LockModesCount = 5, + }; + + // To ensure lock modes are not added without updating the counts + BOOST_STATIC_ASSERT(LockModesCount == MODE_X + 1); + /** * Return values for the locking functions of the lock manager. @@ -283,22 +291,58 @@ namespace newlm { * Not thread-safe and should only be accessed under the LockManager's bucket lock. */ struct LockHead { + LockHead(const ResourceId& resId); ~LockHead(); + // Increments the granted/requested counts and maintains the grantedModes/requestedModes + // masks respectively. The count argument can only be -1 or 1, because we always change the + // modes one at a time. + enum ChangeModeCountAction { + Increment = 1, Decrement = -1 + }; + + void changeGrantedModeCount(LockMode mode, ChangeModeCountAction action); + void changeRequestedModeCount(LockMode mode, ChangeModeCountAction count); + + + // Id of the resource which this lock protects const ResourceId resourceId; + + // + // Granted queue + // + // The head of the doubly-linked list of granted or converting requests LockRequest* grantedQueue; - // Bit-mask of the maximum of the granted + converting modes on the granted queue. + // Counts the grants and coversion counts for each of the supported lock modes. These + // counts should exactly match the aggregated granted modes on the granted list. + uint32_t grantedCounts[LockModesCount]; + + // Bit-mask of the granted + converting modes on the granted queue. Maintained in lock-step + // with the grantedCounts array. uint32_t grantedModes; + + // + // Conflict queue + // + // Doubly-linked list of requests, which have not been granted yet because they conflict // with the set of granted modes. The reason to have both begin and end pointers is to make // the FIFO scheduling easier (queue at begin and take from the end). LockRequest* conflictQueueBegin; LockRequest* conflictQueueEnd; + + // Counts the conflicting requests for each of the lock modes. These counts should exactly + // match the aggregated requested modes on the conflicts list. + uint32_t conflictCounts[LockModesCount]; + + // Bit-mask of the requested modes on the conflict queue. Maintained in lock-step with the + // conflictCounts array. + uint32_t conflictModes; }; @@ -347,9 +391,12 @@ namespace newlm { * * This method always succeeds and never blocks. * - * Calling unlock more times than lock was called for the same LockRequest is not valid. + * @param request A previously locked request. Calling unlock more times than lock was + * called for the same LockRequest is an error. + * + * @return true if this is the last reference for the request; false otherwise */ - void unlock(LockRequest* request); + bool unlock(LockRequest* request); /** * Downgrades the mode in which an already granted request is held, without changing the @@ -395,18 +442,13 @@ namespace newlm { /** * Should be invoked when the state of a lock changes in a way, which could potentially - * allow blocked requests to proceed. + * allow other blocked requests to proceed. * * MUST be called under the lock bucket's spin lock. * - * @param lock Lock whose grant state should be recalculated - * @param unlocked Which request from those on the lock was just unlocked - * @param converting Which request from those on the lock was just converted - * - * Only one of unlocked or converting parameters can be non-NULL (both can be left NULL - * though). + * @param lock Lock whose grant state should be recalculated. */ - void _recalcAndGrant(LockHead* lock, LockRequest* unlocked, LockRequest* converting); + void _onLockModeChanged(LockHead* lock); unsigned _numLockBuckets; LockBucket* _lockBuckets; diff --git a/src/mongo/db/concurrency/lock_mgr_new_test.cpp b/src/mongo/db/concurrency/lock_mgr_new_test.cpp index a089484a5c7..a7dd19bc981 100644 --- a/src/mongo/db/concurrency/lock_mgr_new_test.cpp +++ b/src/mongo/db/concurrency/lock_mgr_new_test.cpp @@ -532,6 +532,32 @@ namespace newlm { lockMgr.unlock(&request[1]); } + TEST(LockManager, Upgrade) { + LockManager lockMgr; + const ResourceId resId(RESOURCE_COLLECTION, std::string("TestDB.collection")); + + LockState locker1; + TrackingLockGrantNotification notify1; + LockRequest request1; + request1.initNew(resId, &locker1, ¬ify1); + ASSERT(LOCK_OK == lockMgr.lock(resId, &request1, MODE_IS)); + + LockState locker2; + TrackingLockGrantNotification notify2; + LockRequest request2; + request2.initNew(resId, &locker2, ¬ify2); + ASSERT(LOCK_OK == lockMgr.lock(resId, &request2, MODE_S)); + ASSERT(request2.recursiveCount == 1); + + // Upgrade the IS lock to X + ASSERT(LOCK_WAITING == lockMgr.lock(resId, &request1, MODE_IX)); + + ASSERT(!lockMgr.unlock(&request1)); + ASSERT(lockMgr.unlock(&request1)); + + ASSERT(lockMgr.unlock(&request2)); + } + TEST(LockManager, Downgrade) { LockManager lockMgr; const ResourceId resId(RESOURCE_COLLECTION, std::string("TestDB.collection")); |