VirtualBox

source: vbox/trunk/src/VBox/Main/AutoLock.cpp@ 25786

Last change on this file since 25786 was 25748, checked in by vboxsync, 15 years ago

iprt/cdefs,*: Use RT_LOCK_STRICT and RT_LOCK_STRICT_ORDER for controlling deadlock detection and lock order validation. Currently both are disabled by default, but it's possible to add VBOX_WITH_STRICT_LOCKS=1 to LocalConfig.kmk to enable it all.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 28.6 KB
Line 
1/** @file
2 *
3 * Automatic locks, implementation
4 */
5
6/*
7 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#include "AutoLock.h"
23
24#include "Logging.h"
25
26#include <iprt/cdefs.h>
27#include <iprt/critsect.h>
28#include <iprt/thread.h>
29#include <iprt/semaphore.h>
30
31#include <iprt/err.h>
32#include <iprt/assert.h>
33
34#if defined(DEBUG)
35# include <iprt/asm.h> // for ASMReturnAddress
36#endif
37
38#include <iprt/string.h>
39#include <iprt/path.h>
40
41#include <VBox/com/string.h>
42
43#include <vector>
44#include <list>
45
46namespace util
47{
48
49////////////////////////////////////////////////////////////////////////////////
50//
51// Per-thread stacks for locking validation
52//
53////////////////////////////////////////////////////////////////////////////////
54
55#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
56// index used for allocating thread-local storage for locking stack
57RTTLS LockHandle::s_lockingStackTlsIndex = NIL_RTTLS;
58
59/**
60 * One item on the LockingStack. One of these gets pushed on the
61 * stack for each lock operation and popped for each unlock.
62 */
63struct LockStackItem
64{
65 LockStackItem(LockHandle *pLock_,
66 const char *pcszFile_,
67 unsigned uLine_,
68 const char *pcszFunction_)
69 : pLock(pLock_),
70 pcszFile(pcszFile_),
71 uLine(uLine_),
72 pcszFunction(pcszFunction_)
73 {
74 pcszFile = RTPathFilename(pcszFile_);
75 }
76
77 LockHandle *pLock;
78
79 // information about where the lock occured (passed down from the AutoLock classes)
80 const char *pcszFile;
81 unsigned uLine;
82 const char *pcszFunction;
83};
84
85typedef std::list<LockStackItem> LockHandlesList;
86
87/**
88 * LockingStack class. One of these gets created for each thread
89 * that calls lock/unlock methods, and a pointer to this is
90 * stored in thread-local storage.
91 */
92struct LockingStack
93{
94 LockingStack()
95 : threadSelf(RTThreadSelf()),
96 pcszThreadName(NULL),
97 c(0)
98 {
99 threadSelf = RTThreadSelf();
100 pcszThreadName = RTThreadGetName(threadSelf);
101 }
102
103 RTTHREAD threadSelf;
104 const char *pcszThreadName;
105
106 LockHandlesList ll;
107 // first item (front) is newest, last item (back) is oldest; I'd do it
108 // the other way round but there is no implementation for erase(reverse_iterator)
109 // which I'd need otherwise
110 size_t c;
111};
112
113/**
114 * Global helper that looks up the LockingStack structure for the
115 * current thread in thread-local storage, or creates one on the
116 * thread's first call.
117 */
118LockingStack* getThreadLocalLockingStack()
119{
120 // very first call in this process: allocate the TLS variable
121 if (LockHandle::s_lockingStackTlsIndex == NIL_RTTLS)
122 {
123 LockHandle::s_lockingStackTlsIndex = RTTlsAlloc();
124 Assert(LockHandle::s_lockingStackTlsIndex != NIL_RTTLS);
125 }
126
127 // get pointer to thread-local locking stack
128 LockingStack *pStack = (LockingStack*)RTTlsGet(LockHandle::s_lockingStackTlsIndex);
129 if (!pStack)
130 {
131 // first call on this thread:
132 pStack = new LockingStack;
133 RTTlsSet(LockHandle::s_lockingStackTlsIndex, pStack);
134 }
135
136 return pStack;
137}
138
139void dumpThreadLocalLockingStack(LockingStack *pStack)
140{
141 uint32_t c = 0;
142 for (LockHandlesList::iterator it = pStack->ll.begin();
143 it != pStack->ll.end();
144 ++it)
145 {
146 LockStackItem lsi = *it;
147 LogFlow(("LOCKVAL: lock %d under top is [%s] locked by %s (%s:%u)\n", c, lsi.pLock->describe(), lsi.pcszFunction, lsi.pcszFile, lsi.uLine));
148 ++c;
149 }
150}
151
152#endif
153
154////////////////////////////////////////////////////////////////////////////////
155//
156// LockHandle
157//
158////////////////////////////////////////////////////////////////////////////////
159
160#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
161
162/**
163 * If the lock validator is enabled, this gets called from the
164 * lock methods to push a LockStackItem onto the thread-local
165 * locking stack for tracing.
166 */
167void LockHandle::validateLock(LOCKVAL_SRC_POS_DECL)
168{
169 // put "this" on locking stack
170 LockingStack *pStack = getThreadLocalLockingStack();
171 LockStackItem lsi(this, RT_SRC_POS_ARGS);
172 pStack->ll.push_front(lsi);
173 ++pStack->c;
174
175 LogFlow(("LOCKVAL [%s]: lock from %s (%s:%u), new count: %RI32\n", describe(), lsi.pcszFunction, lsi.pcszFile, lsi.uLine, (uint32_t)pStack->c));
176}
177
178/**
179 * If the lock validator is enabled, this gets called from the
180 * unlock methods to validate the unlock request. This pops the
181 * LockStackItem from the thread-local locking stack.
182 */
183void LockHandle::validateUnlock()
184{
185 // pop "this" from locking stack
186 LockingStack *pStack = getThreadLocalLockingStack();
187
188 AssertMsg(pStack->c == pStack->ll.size(), ("Locking size mismatch"));
189 AssertMsg(pStack->c > 0, ("Locking stack is empty when it should have current LockHandle on top"));
190
191 // validate that "this" is the top item on the stack
192 LockStackItem &lsiTop = pStack->ll.front();
193 if (lsiTop.pLock != this)
194 {
195 // "this" was not the last to be locked on this thread;
196 // see if it's somewhere deep under the locks
197 dumpThreadLocalLockingStack(pStack);
198
199 bool fFound;
200 uint32_t c = 0;
201 for (LockHandlesList::iterator it = pStack->ll.begin();
202 it != pStack->ll.end();
203 ++it)
204 {
205 LockStackItem &lsiThis = *it;
206 if (lsiThis.pLock == this)
207 {
208 LogFlow(("LOCKVAL [%s]: unlock, stack item was %d items under the top item, corresponsing lock was at %s (%s:%u)\n", describe(), c, lsiThis.pcszFunction, lsiThis.pcszFile, lsiThis.uLine));
209 pStack->ll.erase(it);
210 fFound = true;
211 break;
212 }
213 ++c;
214 }
215
216 if (!fFound)
217 {
218 LogFlow(("LOCKVAL [%s]: unlock, stack item not found!\n", describe()));
219 AssertMsgFailed(("Locking stack does not contain current LockHandle at all\n"));
220 }
221 }
222 else
223 {
224 pStack->ll.pop_front();
225 LogFlow(("LOCKVAL [%s]: unlock, stack item was on top, old count: %RI32\n", describe(), (uint32_t)pStack->c));
226 }
227
228 --pStack->c;
229}
230
231#endif // VBOX_WITH_DEBUG_LOCK_VALIDATOR
232
233////////////////////////////////////////////////////////////////////////////////
234//
235// RWLockHandle
236//
237////////////////////////////////////////////////////////////////////////////////
238
239struct RWLockHandle::Data
240{
241 Data()
242 { }
243
244 RTSEMRW sem;
245
246#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
247 com::Utf8Str strDescription;
248#endif
249};
250
251RWLockHandle::RWLockHandle()
252{
253 m = new Data();
254/** @todo Lock order validation: There are two options here:
255 * -# Use one RTLOCKVALCLASS (share it with WriteLockHandle) and then use
256 * sub-classes (integers) to deal with lock ordering.
257 * -# Use different classes (RTLOCKVALCLASS) for each object and only use
258 * sub-classes to deal with things like the media tree.
259 */
260
261 int vrc = RTSemRWCreateEx(&m->sem, 0 /*fFlags*/, NIL_RTLOCKVALCLASS, RTLOCKVAL_SUB_CLASS_NONE, NULL);
262 AssertRC(vrc);
263
264#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
265 m->strDescription = com::Utf8StrFmt("r/w %RCv", this);
266#endif
267}
268
269/*virtual*/ RWLockHandle::~RWLockHandle()
270{
271 RTSemRWDestroy(m->sem);
272 delete m;
273}
274
275/*virtual*/ bool RWLockHandle::isWriteLockOnCurrentThread() const
276{
277 return RTSemRWIsWriteOwner(m->sem);
278}
279
280/*virtual*/ void RWLockHandle::lockWrite(LOCKVAL_SRC_POS_DECL)
281{
282#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
283 validateLock(LOCKVAL_SRC_POS_ARGS);
284#endif
285#if defined(RT_LOCK_STRICT) && defined(VBOX_WITH_DEBUG_LOCK_VALIDATOR)
286 int vrc = RTSemRWRequestWriteDebug(m->sem, RT_INDEFINITE_WAIT, (uintptr_t)ASMReturnAddress(), RT_SRC_POS_ARGS);
287#elif defined(RT_LOCK_STRICT)
288 int vrc = RTSemRWRequestWriteDebug(m->sem, RT_INDEFINITE_WAIT, (uintptr_t)ASMReturnAddress(), RT_SRC_POS);
289#else
290 int vrc = RTSemRWRequestWrite(m->sem, RT_INDEFINITE_WAIT);
291#endif
292 AssertRC(vrc);
293}
294
295/*virtual*/ void RWLockHandle::unlockWrite()
296{
297#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
298 validateUnlock();
299#endif
300 int vrc = RTSemRWReleaseWrite(m->sem);
301 AssertRC(vrc);
302
303}
304
305/*virtual*/ void RWLockHandle::lockRead(LOCKVAL_SRC_POS_DECL)
306{
307#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
308 validateLock(LOCKVAL_SRC_POS_ARGS);
309#endif
310#if defined(RT_LOCK_STRICT) && defined(VBOX_WITH_DEBUG_LOCK_VALIDATOR)
311 int vrc = RTSemRWRequestReadDebug(m->sem, RT_INDEFINITE_WAIT, (uintptr_t)ASMReturnAddress(), RT_SRC_POS_ARGS);
312#elif defined(RT_LOCK_STRICT)
313 int vrc = RTSemRWRequestReadDebug(m->sem, RT_INDEFINITE_WAIT, (uintptr_t)ASMReturnAddress(), RT_SRC_POS);
314#else
315 int vrc = RTSemRWRequestRead(m->sem, RT_INDEFINITE_WAIT);
316#endif
317 AssertRC(vrc);
318}
319
320/*virtual*/ void RWLockHandle::unlockRead()
321{
322#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
323 validateUnlock();
324#endif
325 int vrc = RTSemRWReleaseRead(m->sem);
326 AssertRC(vrc);
327}
328
329/*virtual*/ uint32_t RWLockHandle::writeLockLevel() const
330{
331 /* Note! This does not include read recursions done by the writer! */
332 return RTSemRWGetWriteRecursion(m->sem);
333}
334
335#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
336/*virtual*/ const char* RWLockHandle::describe() const
337{
338 return m->strDescription.c_str();
339}
340#endif
341
342////////////////////////////////////////////////////////////////////////////////
343//
344// WriteLockHandle
345//
346////////////////////////////////////////////////////////////////////////////////
347
348struct WriteLockHandle::Data
349{
350 Data()
351 { }
352
353 mutable RTCRITSECT sem;
354
355#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
356 com::Utf8Str strDescription;
357#endif
358};
359
360WriteLockHandle::WriteLockHandle()
361{
362 m = new Data;
363 /** @todo see todo in RWLockHandle::RWLockHandle(). */
364 int vrc = RTCritSectInitEx(&m->sem, 0/*fFlags*/, NIL_RTLOCKVALCLASS, RTLOCKVAL_SUB_CLASS_NONE, NULL);
365 AssertRC(vrc);
366
367#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
368 m->strDescription = com::Utf8StrFmt("crit %RCv", this);
369#endif
370}
371
372WriteLockHandle::~WriteLockHandle()
373{
374 RTCritSectDelete(&m->sem);
375 delete m;
376}
377
378/*virtual*/ bool WriteLockHandle::isWriteLockOnCurrentThread() const
379{
380 return RTCritSectIsOwner(&m->sem);
381}
382
383/*virtual*/ void WriteLockHandle::lockWrite(LOCKVAL_SRC_POS_DECL)
384{
385#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
386 validateLock(LOCKVAL_SRC_POS_ARGS);
387#endif
388
389#if defined(RT_LOCK_STRICT) && defined(VBOX_WITH_DEBUG_LOCK_VALIDATOR)
390 RTCritSectEnterDebug(&m->sem, (uintptr_t)ASMReturnAddress(), RT_SRC_POS_ARGS);
391#elif defined(RT_LOCK_STRICT)
392 RTCritSectEnterDebug(&m->sem, (uintptr_t)ASMReturnAddress(),
393 "return address >>>", 0, __PRETTY_FUNCTION__);
394#else
395 RTCritSectEnter(&m->sem);
396#endif
397}
398
399/*virtual*/ void WriteLockHandle::unlockWrite()
400{
401#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
402 validateUnlock();
403#endif
404
405 RTCritSectLeave(&m->sem);
406}
407
408/*virtual*/ void WriteLockHandle::lockRead(LOCKVAL_SRC_POS_DECL)
409{
410 lockWrite(LOCKVAL_SRC_POS_ARGS);
411}
412
413/*virtual*/ void WriteLockHandle::unlockRead()
414{
415 unlockWrite();
416}
417
418/*virtual*/ uint32_t WriteLockHandle::writeLockLevel() const
419{
420 return RTCritSectGetRecursion(&m->sem);
421}
422
423#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
424/*virtual*/ const char* WriteLockHandle::describe() const
425{
426 return m->strDescription.c_str();
427}
428#endif
429
430////////////////////////////////////////////////////////////////////////////////
431//
432// AutoLockBase
433//
434////////////////////////////////////////////////////////////////////////////////
435
436typedef std::vector<LockHandle*> HandlesVector;
437typedef std::vector<uint32_t> CountsVector;
438
439struct AutoLockBase::Data
440{
441 Data(size_t cHandles
442#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
443 , const char *pcszFile_,
444 unsigned uLine_,
445 const char *pcszFunction_
446#endif
447 )
448 : fIsLocked(false),
449 aHandles(cHandles), // size of array
450 acUnlockedInLeave(cHandles)
451#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
452 , pcszFile(pcszFile_),
453 uLine(uLine_),
454 pcszFunction(pcszFunction_)
455#endif
456 {
457 for (uint32_t i = 0; i < cHandles; ++i)
458 {
459 acUnlockedInLeave[i] = 0;
460 aHandles[i] = NULL;
461 }
462 }
463
464 bool fIsLocked; // if true, then all items in aHandles are locked by this AutoLock and
465 // need to be unlocked in the destructor
466 HandlesVector aHandles; // array (vector) of LockHandle instances; in the case of AutoWriteLock
467 // and AutoReadLock, there will only be one item on the list; with the
468 // AutoMulti* derivatives, there will be multiple
469 CountsVector acUnlockedInLeave; // for each lock handle, how many times the handle was unlocked in leave(); otherwise 0
470
471#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
472 // information about where the lock occured (passed down from the AutoLock classes)
473 const char *pcszFile;
474 unsigned uLine;
475 const char *pcszFunction;
476#endif
477};
478
479AutoLockBase::AutoLockBase(uint32_t cHandles
480 COMMA_LOCKVAL_SRC_POS_DECL)
481{
482 m = new Data(cHandles
483 COMMA_LOCKVAL_SRC_POS_ARGS);
484}
485
486AutoLockBase::AutoLockBase(uint32_t cHandles,
487 LockHandle *pHandle
488 COMMA_LOCKVAL_SRC_POS_DECL)
489{
490 Assert(cHandles == 1);
491 m = new Data(1
492 COMMA_LOCKVAL_SRC_POS_ARGS);
493 m->aHandles[0] = pHandle;
494}
495
496AutoLockBase::~AutoLockBase()
497{
498 delete m;
499}
500
501/**
502 * Requests ownership of all contained lock handles by calling
503 * the pure virtual callLockImpl() function on each of them,
504 * which must be implemented by the descendant class; in the
505 * implementation, AutoWriteLock will request a write lock
506 * whereas AutoReadLock will request a read lock.
507 *
508 * Does *not* modify the lock counts in the member variables.
509 */
510void AutoLockBase::callLockOnAllHandles()
511{
512 for (HandlesVector::iterator it = m->aHandles.begin();
513 it != m->aHandles.end();
514 ++it)
515 {
516 LockHandle *pHandle = *it;
517 if (pHandle)
518 // call virtual function implemented in AutoWriteLock or AutoReadLock
519 this->callLockImpl(*pHandle);
520 }
521}
522
523/**
524 * Releases ownership of all contained lock handles by calling
525 * the pure virtual callUnlockImpl() function on each of them,
526 * which must be implemented by the descendant class; in the
527 * implementation, AutoWriteLock will release a write lock
528 * whereas AutoReadLock will release a read lock.
529 *
530 * Does *not* modify the lock counts in the member variables.
531 */
532void AutoLockBase::callUnlockOnAllHandles()
533{
534 // unlock in reverse order!
535 for (HandlesVector::reverse_iterator it = m->aHandles.rbegin();
536 it != m->aHandles.rend();
537 ++it)
538 {
539 LockHandle *pHandle = *it;
540 if (pHandle)
541 // call virtual function implemented in AutoWriteLock or AutoReadLock
542 this->callUnlockImpl(*pHandle);
543 }
544}
545
546/**
547 * Destructor implementation that can also be called explicitly, if required.
548 * Restores the exact state before the AutoLock was created; that is, unlocks
549 * all contained semaphores and might actually lock them again if leave()
550 * was called during the AutoLock's lifetime.
551 */
552void AutoLockBase::cleanup()
553{
554 bool fAnyUnlockedInLeave = false;
555
556 uint32_t i = 0;
557 for (HandlesVector::iterator it = m->aHandles.begin();
558 it != m->aHandles.end();
559 ++it)
560 {
561 LockHandle *pHandle = *it;
562 if (pHandle)
563 {
564 if (m->acUnlockedInLeave[i])
565 {
566 // there was a leave() before the destruction: then restore the
567 // lock level that might have been set by locks other than our own
568 if (m->fIsLocked)
569 {
570 --m->acUnlockedInLeave[i];
571 fAnyUnlockedInLeave = true;
572 }
573 for (; m->acUnlockedInLeave[i]; --m->acUnlockedInLeave[i])
574 callLockImpl(*pHandle);
575 }
576 }
577 ++i;
578 }
579
580 if (m->fIsLocked && !fAnyUnlockedInLeave)
581 callUnlockOnAllHandles();
582}
583
584/**
585 * Requests ownership of all contained semaphores. Public method that can
586 * only be called once and that also gets called by the AutoLock constructors.
587 */
588void AutoLockBase::acquire()
589{
590 AssertMsg(!m->fIsLocked, ("m->fIsLocked is true, attempting to lock twice!"));
591 callLockOnAllHandles();
592 m->fIsLocked = true;
593}
594
595/**
596 * Releases ownership of all contained semaphores. Public method.
597 */
598void AutoLockBase::release()
599{
600 AssertMsg(m->fIsLocked, ("m->fIsLocked is false, cannot release!"));
601 callUnlockOnAllHandles();
602 m->fIsLocked = false;
603}
604
605#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
606void AutoLockBase::dumpStack(const char *pcszMessage, RT_SRC_POS_DECL)
607{
608 LockingStack *pStack = getThreadLocalLockingStack();
609 LogFlow(("LOCKVAL DUMPSTACK at %s (%s:%u)\nLOCKVAL DUMPSTACK %s\n", pszFunction, pszFile, iLine, pcszMessage));
610 dumpThreadLocalLockingStack(pStack);
611}
612#endif
613
614////////////////////////////////////////////////////////////////////////////////
615//
616// AutoReadLock
617//
618////////////////////////////////////////////////////////////////////////////////
619
620/**
621 * Release all read locks acquired by this instance through the #lock()
622 * call and destroys the instance.
623 *
624 * Note that if there there are nested #lock() calls without the
625 * corresponding number of #unlock() calls when the destructor is called, it
626 * will assert. This is because having an unbalanced number of nested locks
627 * is a program logic error which must be fixed.
628 */
629/*virtual*/ AutoReadLock::~AutoReadLock()
630{
631 LockHandle *pHandle = m->aHandles[0];
632
633 if (pHandle)
634 {
635 if (m->fIsLocked)
636 callUnlockImpl(*pHandle);
637 }
638}
639
640/**
641 * Implementation of the pure virtual declared in AutoLockBase.
642 * This gets called by AutoLockBase.acquire() to actually request
643 * the semaphore; in the AutoReadLock implementation, we request
644 * the semaphore in read mode.
645 */
646/*virtual*/ void AutoReadLock::callLockImpl(LockHandle &l)
647{
648#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
649 l.lockRead(m->pcszFile, m->uLine, m->pcszFunction);
650#else
651 l.lockRead();
652#endif
653}
654
655/**
656 * Implementation of the pure virtual declared in AutoLockBase.
657 * This gets called by AutoLockBase.release() to actually release
658 * the semaphore; in the AutoReadLock implementation, we release
659 * the semaphore in read mode.
660 */
661/*virtual*/ void AutoReadLock::callUnlockImpl(LockHandle &l)
662{
663 l.unlockRead();
664}
665
666////////////////////////////////////////////////////////////////////////////////
667//
668// AutoWriteLockBase
669//
670////////////////////////////////////////////////////////////////////////////////
671
672/**
673 * Implementation of the pure virtual declared in AutoLockBase.
674 * This gets called by AutoLockBase.acquire() to actually request
675 * the semaphore; in the AutoWriteLock implementation, we request
676 * the semaphore in write mode.
677 */
678/*virtual*/ void AutoWriteLockBase::callLockImpl(LockHandle &l)
679{
680#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
681 l.lockWrite(m->pcszFile, m->uLine, m->pcszFunction);
682#else
683 l.lockWrite();
684#endif
685}
686
687/**
688 * Implementation of the pure virtual declared in AutoLockBase.
689 * This gets called by AutoLockBase.release() to actually release
690 * the semaphore; in the AutoWriteLock implementation, we release
691 * the semaphore in write mode.
692 */
693/*virtual*/ void AutoWriteLockBase::callUnlockImpl(LockHandle &l)
694{
695 l.unlockWrite();
696}
697
698/**
699 * Causes the current thread to completely release the write lock to make
700 * the managed semaphore immediately available for locking by other threads.
701 *
702 * This implies that all nested write locks on the semaphore will be
703 * released, even those that were acquired through the calls to #lock()
704 * methods of all other AutoWriteLock/AutoReadLock instances managing the
705 * <b>same</b> read/write semaphore.
706 *
707 * After calling this method, the only method you are allowed to call is
708 * #enter(). It will acquire the write lock again and restore the same
709 * level of nesting as it had before calling #leave().
710 *
711 * If this instance is destroyed without calling #enter(), the destructor
712 * will try to restore the write lock level that existed when #leave() was
713 * called minus the number of nested #lock() calls made on this instance
714 * itself. This is done to preserve lock levels of other
715 * AutoWriteLock/AutoReadLock instances managing the same semaphore (if
716 * any). Tiis also means that the destructor may indefinitely block if a
717 * write or a read lock is owned by some other thread by that time.
718 */
719void AutoWriteLockBase::leave()
720{
721 AssertMsg(m->fIsLocked, ("m->fIsLocked is false, cannot leave()!"));
722
723 // unlock in reverse order!
724 uint32_t i = m->aHandles.size();
725 for (HandlesVector::reverse_iterator it = m->aHandles.rbegin();
726 it != m->aHandles.rend();
727 ++it)
728 {
729 --i; // array index is zero based, decrement with every loop since we iterate backwards
730 LockHandle *pHandle = *it;
731 if (pHandle)
732 {
733 AssertMsg(m->acUnlockedInLeave[i] == 0, ("m->cUnlockedInLeave[%d] is %d, must be 0! Called leave() twice?", i, m->acUnlockedInLeave[i]));
734 m->acUnlockedInLeave[i] = pHandle->writeLockLevel();
735 AssertMsg(m->acUnlockedInLeave[i] >= 1, ("m->cUnlockedInLeave[%d] is %d, must be >=1!", i, m->acUnlockedInLeave[i]));
736
737#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
738 LogFlowFunc(("LOCKVAL: will unlock handle %d [%s] %d times\n", i, pHandle->describe(), m->acUnlockedInLeave[i]));
739#endif
740
741 for (uint32_t left = m->acUnlockedInLeave[i];
742 left;
743 --left)
744 callUnlockImpl(*pHandle);
745 }
746 }
747}
748
749/**
750 * Causes the current thread to restore the write lock level after the
751 * #leave() call. This call will indefinitely block if another thread has
752 * successfully acquired a write or a read lock on the same semaphore in
753 * between.
754 */
755void AutoWriteLockBase::enter()
756{
757 AssertMsg(m->fIsLocked, ("m->fIsLocked is false, cannot enter()!"));
758
759 uint32_t i = 0;
760 for (HandlesVector::iterator it = m->aHandles.begin();
761 it != m->aHandles.end();
762 ++it)
763 {
764 LockHandle *pHandle = *it;
765 if (pHandle)
766 {
767 AssertMsg(m->acUnlockedInLeave[i] != 0, ("m->cUnlockedInLeave[%d] is 0! enter() without leave()?", i));
768
769#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
770 LogFlowFunc(("LOCKVAL: will lock handle %d [%s] %d times\n", i, pHandle->describe(), m->acUnlockedInLeave[i]));
771#endif
772
773 for (; m->acUnlockedInLeave[i]; --m->acUnlockedInLeave[i])
774 callLockImpl(*pHandle);
775 }
776 ++i;
777 }
778}
779
780/**
781 * Same as #leave() but checks if the current thread actally owns the lock
782 * and only proceeds in this case. As a result, as opposed to #leave(),
783 * doesn't assert when called with no lock being held.
784 */
785void AutoWriteLockBase::maybeLeave()
786{
787 // unlock in reverse order!
788 uint32_t i = m->aHandles.size();
789 for (HandlesVector::reverse_iterator it = m->aHandles.rbegin();
790 it != m->aHandles.rend();
791 ++it)
792 {
793 --i; // array index is zero based, decrement with every loop since we iterate backwards
794 LockHandle *pHandle = *it;
795 if (pHandle)
796 {
797 if (pHandle->isWriteLockOnCurrentThread())
798 {
799 m->acUnlockedInLeave[i] = pHandle->writeLockLevel();
800 AssertMsg(m->acUnlockedInLeave[i] >= 1, ("m->cUnlockedInLeave[%d] is %d, must be >=1!", i, m->acUnlockedInLeave[i]));
801
802#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
803 LogFlowFunc(("LOCKVAL: will unlock handle %d [%s] %d times\n", i, pHandle->describe(), m->acUnlockedInLeave[i]));
804#endif
805
806 for (uint32_t left = m->acUnlockedInLeave[i];
807 left;
808 --left)
809 callUnlockImpl(*pHandle);
810 }
811 }
812 ++i;
813 }
814}
815
816/**
817 * Same as #enter() but checks if the current thread actally owns the lock
818 * and only proceeds if not. As a result, as opposed to #enter(), doesn't
819 * assert when called with the lock already being held.
820 */
821void AutoWriteLockBase::maybeEnter()
822{
823 uint32_t i = 0;
824 for (HandlesVector::iterator it = m->aHandles.begin();
825 it != m->aHandles.end();
826 ++it)
827 {
828 LockHandle *pHandle = *it;
829 if (pHandle)
830 {
831 if (!pHandle->isWriteLockOnCurrentThread())
832 {
833#ifdef VBOX_WITH_DEBUG_LOCK_VALIDATOR
834 LogFlowFunc(("LOCKVAL: will lock handle %d [%s] %d times\n", i, pHandle->describe(), m->acUnlockedInLeave[i]));
835#endif
836
837 for (; m->acUnlockedInLeave[i]; --m->acUnlockedInLeave[i])
838 callLockImpl(*pHandle);
839 }
840 }
841 ++i;
842 }
843}
844
845////////////////////////////////////////////////////////////////////////////////
846//
847// AutoWriteLock
848//
849////////////////////////////////////////////////////////////////////////////////
850
851/**
852 * Attaches another handle to this auto lock instance.
853 *
854 * The previous object's lock is completely released before the new one is
855 * acquired. The lock level of the new handle will be the same. This
856 * also means that if the lock was not acquired at all before #attach(), it
857 * will not be acquired on the new handle too.
858 *
859 * @param aHandle New handle to attach.
860 */
861void AutoWriteLock::attach(LockHandle *aHandle)
862{
863 LockHandle *pHandle = m->aHandles[0];
864
865 /* detect simple self-reattachment */
866 if (pHandle != aHandle)
867 {
868 bool fWasLocked = m->fIsLocked;
869
870 cleanup();
871
872 m->aHandles[0] = aHandle;
873 m->fIsLocked = fWasLocked;
874
875 if (aHandle)
876 if (fWasLocked)
877 callLockImpl(*aHandle);
878 }
879}
880
881/**
882 * Returns @c true if the current thread holds a write lock on the managed
883 * read/write semaphore. Returns @c false if the managed semaphore is @c
884 * NULL.
885 *
886 * @note Intended for debugging only.
887 */
888bool AutoWriteLock::isWriteLockOnCurrentThread() const
889{
890 return m->aHandles[0] ? m->aHandles[0]->isWriteLockOnCurrentThread() : false;
891}
892
893 /**
894 * Returns the current write lock level of the managed smaphore. The lock
895 * level determines the number of nested #lock() calls on the given
896 * semaphore handle. Returns @c 0 if the managed semaphore is @c
897 * NULL.
898 *
899 * Note that this call is valid only when the current thread owns a write
900 * lock on the given semaphore handle and will assert otherwise.
901 *
902 * @note Intended for debugging only.
903 */
904uint32_t AutoWriteLock::writeLockLevel() const
905{
906 return m->aHandles[0] ? m->aHandles[0]->writeLockLevel() : 0;
907}
908
909////////////////////////////////////////////////////////////////////////////////
910//
911// AutoMultiWriteLock*
912//
913////////////////////////////////////////////////////////////////////////////////
914
915AutoMultiWriteLock2::AutoMultiWriteLock2(Lockable *pl1,
916 Lockable *pl2
917 COMMA_LOCKVAL_SRC_POS_DECL)
918 : AutoWriteLockBase(2
919 COMMA_LOCKVAL_SRC_POS_ARGS)
920{
921 if (pl1)
922 m->aHandles[0] = pl1->lockHandle();
923 if (pl2)
924 m->aHandles[1] = pl2->lockHandle();
925 acquire();
926}
927
928AutoMultiWriteLock2::AutoMultiWriteLock2(LockHandle *pl1,
929 LockHandle *pl2
930 COMMA_LOCKVAL_SRC_POS_DECL)
931 : AutoWriteLockBase(2
932 COMMA_LOCKVAL_SRC_POS_ARGS)
933{
934 m->aHandles[0] = pl1;
935 m->aHandles[1] = pl2;
936 acquire();
937}
938
939AutoMultiWriteLock3::AutoMultiWriteLock3(Lockable *pl1,
940 Lockable *pl2,
941 Lockable *pl3
942 COMMA_LOCKVAL_SRC_POS_DECL)
943 : AutoWriteLockBase(3
944 COMMA_LOCKVAL_SRC_POS_ARGS)
945{
946 if (pl1)
947 m->aHandles[0] = pl1->lockHandle();
948 if (pl2)
949 m->aHandles[1] = pl2->lockHandle();
950 if (pl3)
951 m->aHandles[2] = pl3->lockHandle();
952 acquire();
953}
954
955AutoMultiWriteLock3::AutoMultiWriteLock3(LockHandle *pl1,
956 LockHandle *pl2,
957 LockHandle *pl3
958 COMMA_LOCKVAL_SRC_POS_DECL)
959 : AutoWriteLockBase(3
960 COMMA_LOCKVAL_SRC_POS_ARGS)
961{
962 m->aHandles[0] = pl1;
963 m->aHandles[1] = pl2;
964 m->aHandles[2] = pl3;
965 acquire();
966}
967
968} /* namespace util */
969/* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette