VirtualBox

source: vbox/trunk/include/VBox/com/AutoLock.h@ 41247

Last change on this file since 41247 was 40257, checked in by vboxsync, 13 years ago

Main/Medium: rework locking scheme to solve lock order violations and long GUI start up time caused by too much locking
Main/all: Remove the enter and leave methods from write locks, they cause hard to find locking problems. Better solve them explicitly.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.5 KB
Line 
1/** @file
2 *
3 * Automatic locks, implementation
4 */
5
6/*
7 * Copyright (C) 2006-2012 Oracle Corporation
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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27#ifndef ____H_AUTOLOCK
28#define ____H_AUTOLOCK
29
30#include <iprt/types.h>
31
32// macros for automatic lock validation; these will amount to nothing
33// unless lock validation is enabled for the runtime
34#if defined(RT_LOCK_STRICT)
35# define VBOX_WITH_MAIN_LOCK_VALIDATION
36# define COMMA_LOCKVAL_SRC_POS , RT_SRC_POS
37# define LOCKVAL_SRC_POS_DECL RT_SRC_POS_DECL
38# define COMMA_LOCKVAL_SRC_POS_DECL , RT_SRC_POS_DECL
39# define LOCKVAL_SRC_POS_ARGS RT_SRC_POS_ARGS
40# define COMMA_LOCKVAL_SRC_POS_ARGS , RT_SRC_POS_ARGS
41#else
42# define COMMA_LOCKVAL_SRC_POS
43# define LOCKVAL_SRC_POS_DECL
44# define COMMA_LOCKVAL_SRC_POS_DECL
45# define LOCKVAL_SRC_POS_ARGS
46# define COMMA_LOCKVAL_SRC_POS_ARGS
47#endif
48
49namespace util
50{
51
52////////////////////////////////////////////////////////////////////////////////
53//
54// Order classes for lock validation
55//
56////////////////////////////////////////////////////////////////////////////////
57
58/**
59 * IPRT now has a sophisticated system of run-time locking classes to validate
60 * locking order. Since the Main code is handled by simpler minds, we want
61 * compile-time constants for simplicity, and we'll look up the run-time classes
62 * in AutoLock.cpp transparently. These are passed to the constructors of the
63 * LockHandle classes.
64 */
65enum VBoxLockingClass
66{
67 LOCKCLASS_NONE = 0,
68 LOCKCLASS_WEBSERVICE = 1, // highest order: webservice locks
69 LOCKCLASS_VIRTUALBOXOBJECT = 2, // highest order within Main itself: VirtualBox object lock
70 LOCKCLASS_HOSTOBJECT = 3, // Host object lock
71 LOCKCLASS_LISTOFMACHINES = 4, // list of machines in VirtualBox object
72 LOCKCLASS_MACHINEOBJECT = 5, // Machine object lock
73 LOCKCLASS_SNAPSHOTOBJECT = 6, // snapshot object locks
74 // (the snapshots tree, including the child pointers in Snapshot,
75 // is protected by the normal Machine object lock)
76 LOCKCLASS_LISTOFMEDIA = 7, // list of media (hard disks, DVDs, floppies) in VirtualBox object
77 LOCKCLASS_LISTOFOTHEROBJECTS = 8, // any other list of objects
78 LOCKCLASS_MEDIUMQUERY = 9, // lock used to protect Machine::queryInfo
79 LOCKCLASS_OTHEROBJECT = 10, // any regular object member variable lock
80 LOCKCLASS_USBLIST = 11, // temporary hack to avoid having to clean up the USB filters
81 // too much @todo r=dj get rid of this!
82 LOCKCLASS_PROGRESSLIST = 12, // list of progress objects in VirtualBox; no other object lock
83 // may be held after this!
84 LOCKCLASS_OBJECTSTATE = 13 // object state lock (handled by AutoCaller classes)
85};
86
87void InitAutoLockSystem();
88
89/**
90 * Check whether the current thread holds any locks in the given class
91 *
92 * @return true if any such locks are held, false otherwise. If the lock
93 * validator is not compiled in, always returns false.
94 * @param lockClass Which lock class to check.
95 */
96bool AutoLockHoldsLocksInClass(VBoxLockingClass lockClass);
97
98////////////////////////////////////////////////////////////////////////////////
99//
100// LockHandle and friends
101//
102////////////////////////////////////////////////////////////////////////////////
103
104/**
105 * Abstract base class for semaphore handles (RWLockHandle and WriteLockHandle).
106 * Don't use this directly, but this implements lock validation for them.
107 */
108class LockHandle
109{
110public:
111 LockHandle()
112 {}
113
114 virtual ~LockHandle()
115 {}
116
117 /**
118 * Returns @c true if the current thread holds a write lock on this
119 * read/write semaphore. Intended for debugging only.
120 */
121 virtual bool isWriteLockOnCurrentThread() const = 0;
122
123 /**
124 * Returns the current write lock level of this semaphore. The lock level
125 * determines the number of nested #lock() calls on the given semaphore
126 * handle.
127 *
128 * Note that this call is valid only when the current thread owns a write
129 * lock on the given semaphore handle and will assert otherwise.
130 */
131 virtual uint32_t writeLockLevel() const = 0;
132
133 virtual void lockWrite(LOCKVAL_SRC_POS_DECL) = 0;
134 virtual void unlockWrite() = 0;
135 virtual void lockRead(LOCKVAL_SRC_POS_DECL) = 0;
136 virtual void unlockRead() = 0;
137
138#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
139 virtual const char* describe() const = 0;
140#endif
141
142private:
143 // prohibit copy + assignment
144 LockHandle(const LockHandle&);
145 LockHandle& operator=(const LockHandle&);
146};
147
148/**
149 * Full-featured read/write semaphore handle implementation.
150 *
151 * This is an auxiliary base class for classes that need full-featured
152 * read/write locking as described in the AutoWriteLock class documentation.
153 * Instances of classes inherited from this class can be passed as arguments to
154 * the AutoWriteLock and AutoReadLock constructors.
155 */
156class RWLockHandle : public LockHandle
157{
158public:
159 RWLockHandle(VBoxLockingClass lockClass);
160 virtual ~RWLockHandle();
161
162 virtual bool isWriteLockOnCurrentThread() const;
163
164 virtual void lockWrite(LOCKVAL_SRC_POS_DECL);
165 virtual void unlockWrite();
166 virtual void lockRead(LOCKVAL_SRC_POS_DECL);
167 virtual void unlockRead();
168
169 virtual uint32_t writeLockLevel() const;
170
171#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
172 virtual const char* describe() const;
173#endif
174
175private:
176 struct Data;
177 Data *m;
178};
179
180/**
181 * Write-only semaphore handle implementation.
182 *
183 * This is an auxiliary base class for classes that need write-only (exclusive)
184 * locking and do not need read (shared) locking. This implementation uses a
185 * cheap and fast critical section for both lockWrite() and lockRead() methods
186 * which makes a lockRead() call fully equivalent to the lockWrite() call and
187 * therefore makes it pointless to use instahces of this class with
188 * AutoReadLock instances -- shared locking will not be possible anyway and
189 * any call to lock() will block if there are lock owners on other threads.
190 *
191 * Use with care only when absolutely sure that shared locks are not necessary.
192 */
193class WriteLockHandle : public LockHandle
194{
195public:
196 WriteLockHandle(VBoxLockingClass lockClass);
197 virtual ~WriteLockHandle();
198 virtual bool isWriteLockOnCurrentThread() const;
199
200 virtual void lockWrite(LOCKVAL_SRC_POS_DECL);
201 virtual void unlockWrite();
202 virtual void lockRead(LOCKVAL_SRC_POS_DECL);
203 virtual void unlockRead();
204 virtual uint32_t writeLockLevel() const;
205
206#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
207 virtual const char* describe() const;
208#endif
209
210private:
211 struct Data;
212 Data *m;
213};
214
215////////////////////////////////////////////////////////////////////////////////
216//
217// Lockable
218//
219////////////////////////////////////////////////////////////////////////////////
220
221/**
222 * Lockable interface.
223 *
224 * This is an abstract base for classes that need read/write locking. Unlike
225 * RWLockHandle and other classes that makes the read/write semaphore a part of
226 * class data, this class allows subclasses to decide which semaphore handle to
227 * use.
228 */
229class Lockable
230{
231public:
232
233 /**
234 * Returns a pointer to a LockHandle used by AutoWriteLock/AutoReadLock
235 * for locking. Subclasses are allowed to return @c NULL -- in this case,
236 * the AutoWriteLock/AutoReadLock object constructed using an instance of
237 * such subclass will simply turn into no-op.
238 */
239 virtual LockHandle *lockHandle() const = 0;
240
241 /**
242 * Equivalent to <tt>#lockHandle()->isWriteLockOnCurrentThread()</tt>.
243 * Returns @c false if lockHandle() returns @c NULL.
244 */
245 bool isWriteLockOnCurrentThread()
246 {
247 LockHandle *h = lockHandle();
248 return h ? h->isWriteLockOnCurrentThread() : false;
249 }
250};
251
252////////////////////////////////////////////////////////////////////////////////
253//
254// AutoLockBase
255//
256////////////////////////////////////////////////////////////////////////////////
257
258/**
259 * Abstract base class for all autolocks.
260 *
261 * This cannot be used directly. Use AutoReadLock or AutoWriteLock or AutoMultiWriteLock2/3
262 * which directly and indirectly derive from this.
263 *
264 * In the implementation, the instance data contains a list of lock handles.
265 * The class provides some utility functions to help locking and unlocking
266 * them.
267 */
268
269class AutoLockBase
270{
271protected:
272 AutoLockBase(uint32_t cHandles
273 COMMA_LOCKVAL_SRC_POS_DECL);
274 AutoLockBase(uint32_t cHandles,
275 LockHandle *pHandle
276 COMMA_LOCKVAL_SRC_POS_DECL);
277 virtual ~AutoLockBase();
278
279 struct Data;
280 Data *m;
281
282 virtual void callLockImpl(LockHandle &l) = 0;
283 virtual void callUnlockImpl(LockHandle &l) = 0;
284
285 void callLockOnAllHandles();
286 void callUnlockOnAllHandles();
287
288 void cleanup();
289
290public:
291 void acquire();
292 void release();
293
294private:
295 // prohibit copy + assignment
296 AutoLockBase(const AutoLockBase&);
297 AutoLockBase& operator=(const AutoLockBase&);
298};
299
300////////////////////////////////////////////////////////////////////////////////
301//
302// AutoReadLock
303//
304////////////////////////////////////////////////////////////////////////////////
305
306/**
307 * Automatic read lock. Use this with a RWLockHandle to request a read/write
308 * semaphore in read mode. You can also use this with a WriteLockHandle but
309 * that makes little sense since they treat read mode like write mode.
310 *
311 * If constructed with a RWLockHandle or an instance of Lockable (which in
312 * practice means any VirtualBoxBase derivative), it autoamtically requests
313 * the lock in read mode and releases the read lock in the destructor.
314 */
315class AutoReadLock : public AutoLockBase
316{
317public:
318
319 /**
320 * Constructs a null instance that does not manage any read/write
321 * semaphore.
322 *
323 * Note that all method calls on a null instance are no-ops. This allows to
324 * have the code where lock protection can be selected (or omitted) at
325 * runtime.
326 */
327 AutoReadLock(LOCKVAL_SRC_POS_DECL)
328 : AutoLockBase(1,
329 NULL
330 COMMA_LOCKVAL_SRC_POS_ARGS)
331 { }
332
333 /**
334 * Constructs a new instance that will start managing the given read/write
335 * semaphore by requesting a read lock.
336 */
337 AutoReadLock(LockHandle *aHandle
338 COMMA_LOCKVAL_SRC_POS_DECL)
339 : AutoLockBase(1,
340 aHandle
341 COMMA_LOCKVAL_SRC_POS_ARGS)
342 {
343 acquire();
344 }
345
346 /**
347 * Constructs a new instance that will start managing the given read/write
348 * semaphore by requesting a read lock.
349 */
350 AutoReadLock(LockHandle &aHandle
351 COMMA_LOCKVAL_SRC_POS_DECL)
352 : AutoLockBase(1,
353 &aHandle
354 COMMA_LOCKVAL_SRC_POS_ARGS)
355 {
356 acquire();
357 }
358
359 /**
360 * Constructs a new instance that will start managing the given read/write
361 * semaphore by requesting a read lock.
362 */
363 AutoReadLock(const Lockable &aLockable
364 COMMA_LOCKVAL_SRC_POS_DECL)
365 : AutoLockBase(1,
366 aLockable.lockHandle()
367 COMMA_LOCKVAL_SRC_POS_ARGS)
368 {
369 acquire();
370 }
371
372 /**
373 * Constructs a new instance that will start managing the given read/write
374 * semaphore by requesting a read lock.
375 */
376 AutoReadLock(const Lockable *aLockable
377 COMMA_LOCKVAL_SRC_POS_DECL)
378 : AutoLockBase(1,
379 aLockable ? aLockable->lockHandle() : NULL
380 COMMA_LOCKVAL_SRC_POS_ARGS)
381 {
382 acquire();
383 }
384
385 virtual ~AutoReadLock();
386
387 virtual void callLockImpl(LockHandle &l);
388 virtual void callUnlockImpl(LockHandle &l);
389};
390
391////////////////////////////////////////////////////////////////////////////////
392//
393// AutoWriteLockBase
394//
395////////////////////////////////////////////////////////////////////////////////
396
397/**
398 * Base class for all auto write locks.
399 *
400 * This cannot be used directly. Use AutoWriteLock or AutoMultiWriteLock2/3
401 * which derive from this.
402 *
403 * It has some utility methods for subclasses.
404 */
405class AutoWriteLockBase : public AutoLockBase
406{
407protected:
408 AutoWriteLockBase(uint32_t cHandles
409 COMMA_LOCKVAL_SRC_POS_DECL)
410 : AutoLockBase(cHandles
411 COMMA_LOCKVAL_SRC_POS_ARGS)
412 { }
413
414 AutoWriteLockBase(uint32_t cHandles,
415 LockHandle *pHandle
416 COMMA_LOCKVAL_SRC_POS_DECL)
417 : AutoLockBase(cHandles,
418 pHandle
419 COMMA_LOCKVAL_SRC_POS_ARGS)
420 { }
421
422 virtual ~AutoWriteLockBase()
423 { }
424
425 virtual void callLockImpl(LockHandle &l);
426 virtual void callUnlockImpl(LockHandle &l);
427};
428
429////////////////////////////////////////////////////////////////////////////////
430//
431// AutoWriteLock
432//
433////////////////////////////////////////////////////////////////////////////////
434
435/**
436 * Automatic write lock. Use this with a RWLockHandle to request a read/write
437 * semaphore in write mode. There can only ever be one writer of a read/write
438 * semaphore: while the lock is held in write mode, no other writer or reader
439 * can request the semaphore and will block.
440 *
441 * If constructed with a RWLockHandle or an instance of Lockable (which in
442 * practice means any VirtualBoxBase derivative), it autoamtically requests
443 * the lock in write mode and releases the write lock in the destructor.
444 *
445 * When used with a WriteLockHandle, it requests the semaphore contained therein
446 * exclusively.
447 */
448class AutoWriteLock : public AutoWriteLockBase
449{
450public:
451
452 /**
453 * Constructs a null instance that does not manage any read/write
454 * semaphore.
455 *
456 * Note that all method calls on a null instance are no-ops. This allows to
457 * have the code where lock protection can be selected (or omitted) at
458 * runtime.
459 */
460 AutoWriteLock(LOCKVAL_SRC_POS_DECL)
461 : AutoWriteLockBase(1,
462 NULL
463 COMMA_LOCKVAL_SRC_POS_ARGS)
464 { }
465
466 /**
467 * Constructs a new instance that will start managing the given read/write
468 * semaphore by requesting a write lock.
469 */
470 AutoWriteLock(LockHandle *aHandle
471 COMMA_LOCKVAL_SRC_POS_DECL)
472 : AutoWriteLockBase(1,
473 aHandle
474 COMMA_LOCKVAL_SRC_POS_ARGS)
475 {
476 acquire();
477 }
478
479 /**
480 * Constructs a new instance that will start managing the given read/write
481 * semaphore by requesting a write lock.
482 */
483 AutoWriteLock(LockHandle &aHandle
484 COMMA_LOCKVAL_SRC_POS_DECL)
485 : AutoWriteLockBase(1,
486 &aHandle
487 COMMA_LOCKVAL_SRC_POS_ARGS)
488 {
489 acquire();
490 }
491
492 /**
493 * Constructs a new instance that will start managing the given read/write
494 * semaphore by requesting a write lock.
495 */
496 AutoWriteLock(const Lockable &aLockable
497 COMMA_LOCKVAL_SRC_POS_DECL)
498 : AutoWriteLockBase(1,
499 aLockable.lockHandle()
500 COMMA_LOCKVAL_SRC_POS_ARGS)
501 {
502 acquire();
503 }
504
505 /**
506 * Constructs a new instance that will start managing the given read/write
507 * semaphore by requesting a write lock.
508 */
509 AutoWriteLock(const Lockable *aLockable
510 COMMA_LOCKVAL_SRC_POS_DECL)
511 : AutoWriteLockBase(1,
512 aLockable ? aLockable->lockHandle() : NULL
513 COMMA_LOCKVAL_SRC_POS_ARGS)
514 {
515 acquire();
516 }
517
518 /**
519 * Constructs a new instance that will start managing the given read/write
520 * semaphore by requesting a write lock.
521 */
522 AutoWriteLock(uint32_t cHandles,
523 LockHandle** pHandles
524 COMMA_LOCKVAL_SRC_POS_DECL);
525
526 /**
527 * Release all write locks acquired by this instance through the #lock()
528 * call and destroys the instance.
529 *
530 * Note that if there there are nested #lock() calls without the
531 * corresponding number of #unlock() calls when the destructor is called, it
532 * will assert. This is because having an unbalanced number of nested locks
533 * is a program logic error which must be fixed.
534 */
535 virtual ~AutoWriteLock()
536 {
537 cleanup();
538 }
539
540 void attach(LockHandle *aHandle);
541
542 /** @see attach (LockHandle *) */
543 void attach(LockHandle &aHandle)
544 {
545 attach(&aHandle);
546 }
547
548 /** @see attach (LockHandle *) */
549 void attach(const Lockable &aLockable)
550 {
551 attach(aLockable.lockHandle());
552 }
553
554 /** @see attach (LockHandle *) */
555 void attach(const Lockable *aLockable)
556 {
557 attach(aLockable ? aLockable->lockHandle() : NULL);
558 }
559
560 bool isWriteLockOnCurrentThread() const;
561 uint32_t writeLockLevel() const;
562};
563
564////////////////////////////////////////////////////////////////////////////////
565//
566// AutoMultiWriteLock*
567//
568////////////////////////////////////////////////////////////////////////////////
569
570/**
571 * A multi-write-lock containing two other write locks.
572 *
573 */
574class AutoMultiWriteLock2 : public AutoWriteLockBase
575{
576public:
577 AutoMultiWriteLock2(Lockable *pl1,
578 Lockable *pl2
579 COMMA_LOCKVAL_SRC_POS_DECL);
580 AutoMultiWriteLock2(LockHandle *pl1,
581 LockHandle *pl2
582 COMMA_LOCKVAL_SRC_POS_DECL);
583
584 virtual ~AutoMultiWriteLock2()
585 {
586 cleanup();
587 }
588};
589
590/**
591 * A multi-write-lock containing three other write locks.
592 *
593 */
594class AutoMultiWriteLock3 : public AutoWriteLockBase
595{
596public:
597 AutoMultiWriteLock3(Lockable *pl1,
598 Lockable *pl2,
599 Lockable *pl3
600 COMMA_LOCKVAL_SRC_POS_DECL);
601 AutoMultiWriteLock3(LockHandle *pl1,
602 LockHandle *pl2,
603 LockHandle *pl3
604 COMMA_LOCKVAL_SRC_POS_DECL);
605
606 virtual ~AutoMultiWriteLock3()
607 {
608 cleanup();
609 }
610};
611
612/**
613 * A multi-write-lock containing four other write locks.
614 *
615 */
616class AutoMultiWriteLock4 : public AutoWriteLockBase
617{
618public:
619 AutoMultiWriteLock4(Lockable *pl1,
620 Lockable *pl2,
621 Lockable *pl3,
622 Lockable *pl4
623 COMMA_LOCKVAL_SRC_POS_DECL);
624 AutoMultiWriteLock4(LockHandle *pl1,
625 LockHandle *pl2,
626 LockHandle *pl3,
627 LockHandle *pl4
628 COMMA_LOCKVAL_SRC_POS_DECL);
629
630 virtual ~AutoMultiWriteLock4()
631 {
632 cleanup();
633 }
634};
635
636} /* namespace util */
637
638#endif // ____H_AUTOLOCK
639
640/* 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