VirtualBox

source: vbox/trunk/include/iprt/cpp/lock.h@ 26196

Last change on this file since 26196 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
File size: 4.2 KB
Line 
1/** @file
2 * IPRT - RTLock Classes for Scope-based Locking.
3 */
4
5/*
6 * Copyright (C) 2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
26 * Clara, CA 95054 USA or visit http://www.sun.com if you need
27 * additional information or have any questions.
28 */
29
30#ifndef ___iprt_cpp_lock_h
31#define ___iprt_cpp_lock_h
32
33#include <iprt/critsect.h>
34#ifdef RT_LOCK_STRICT
35# include <iprt/lockvalidator.h>
36#endif
37
38RT_C_DECLS_BEGIN
39
40/** @defgroup grp_rt_lock RTLock - Scope-based Locking (C++).
41 * @ingroup grp_rt
42 * @{
43 */
44
45class RTLock;
46
47/**
48 * The mutex lock.
49 *
50 * This is used as a object data member if the intention is to lock
51 * a single object. This can also be used statically, initialized in
52 * a global variable, for class wide purposes.
53 *
54 * This is best used together with RTLock.
55 */
56class RTLockMtx
57{
58 friend class RTLock;
59
60 private:
61 RTCRITSECT mMtx;
62
63 public:
64 RTLockMtx()
65 {
66#ifdef RT_LOCK_STRICT_ORDER
67 RTCritSectInitEx(&mMtx, 0 /*fFlags*/,
68 RTLockValidatorClassCreateUnique(RT_SRC_POS, NULL),
69 RTLOCKVAL_SUB_CLASS_NONE, NULL);
70#else
71 RTCritSectInit(&mMtx);
72#endif
73 }
74
75 /** Use to when creating locks that belongs in the same "class". */
76 RTLockMtx(RT_SRC_POS_DECL, uint32_t uSubClass = RTLOCKVAL_SUB_CLASS_NONE)
77 {
78#ifdef RT_LOCK_STRICT_ORDER
79 RTCritSectInitEx(&mMtx, 0 /*fFlags*/,
80 RTLockValidatorClassForSrcPos(RT_SRC_POS_ARGS, NULL),
81 uSubClass, NULL);
82#else
83 RTCritSectInit(&mMtx);
84 RT_SRC_POS_NOREF();
85#endif
86 }
87
88 ~RTLockMtx()
89 {
90 RTCritSectDelete(&mMtx);
91 }
92
93 // lock() and unlock() are private so that only
94 // friend RTLock can access them
95 private:
96 inline void lock()
97 {
98 RTCritSectEnter(&mMtx);
99 }
100
101 inline void unlock()
102 {
103 RTCritSectLeave(&mMtx);
104 }
105};
106
107
108/**
109 * The stack object for automatic locking and unlocking.
110 *
111 * This is a helper class for automatic locks, to simplify
112 * requesting a RTLockMtx and to not forget releasing it.
113 * To request a RTLockMtx, simply create an instance of RTLock
114 * on the stack and pass the mutex to it:
115 *
116 * @code
117 extern RTLockMtx gMtx; // wherever this is
118 ...
119 if (...)
120 {
121 RTLock lock(gMtx);
122 ... // do stuff
123 // when lock goes out of scope, destructor releases the mutex
124 }
125 @endcode
126 *
127 * You can also explicitly release the mutex by calling RTLock::release().
128 * This might be helpful if the lock doesn't go out of scope early enough
129 * for your mutex to be released.
130 */
131class RTLock
132{
133 private:
134 RTLockMtx &mMtx;
135 bool mfLocked;
136
137 public:
138 RTLock(RTLockMtx &aMtx)
139 : mMtx(aMtx)
140 {
141 mMtx.lock();
142 mfLocked = true;
143 }
144
145 ~RTLock()
146 {
147 if (mfLocked)
148 mMtx.unlock();
149 }
150
151 inline void release()
152 {
153 if (mfLocked)
154 {
155 mMtx.unlock();
156 mfLocked = false;
157 }
158 }
159};
160
161
162/** @} */
163
164RT_C_DECLS_END
165
166#endif
167
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