VirtualBox

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

Last change on this file since 41760 was 41528, checked in by vboxsync, 13 years ago

Main/HostUSBDevice(all platforms)+USBProxyService: redo USB locking, fixes major regression, added lots of assertions to catch locking flaws early, whitespace cleanup
Main/Machine: small USB locking fix to be consistent with the remaining code
Main/Host+glue/AutoLock: replace USB list lock by host lock, small numbering cleanup
Main/Console: redo USB locking, do less in USB callbacks/EMT (addresses long standing todo items), eliminate unsafe iterator parameters

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