VirtualBox

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

Last change on this file since 92384 was 90828, checked in by vboxsync, 3 years ago

Main: bugref:1909: Added API localization

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