VirtualBox

source: vbox/trunk/src/VBox/Additions/haiku/include/lock.h@ 106945

Last change on this file since 106945 was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.6 KB
Line 
1/* $Id: lock.h 106061 2024-09-16 14:03:52Z vboxsync $ */
2/** @file
3 * Lock.h - Haiku, private locking internals.
4 */
5
6/*
7 * Copyright (C) 2012-2024 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * SPDX-License-Identifier: GPL-3.0-only
26 */
27
28/*
29 * This code is based on:
30 *
31 * VirtualBox Guest Additions for Haiku.
32 *
33 * Copyright 2008-2010, Ingo Weinhold, ingo_weinhold@gmx.de.
34 * Copyright 2002-2009, Axel Dörfler, axeld@pinc-software.de.
35 * Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
36 * Distributed under the terms of the MIT License.
37 */
38
39/** @todo r=ramshankar: Eventually this file should be shipped by Haiku and
40 * should be removed from the VBox tree. */
41
42#ifndef GA_INCLUDED_HAIKU_lock_h
43#define GA_INCLUDED_HAIKU_lock_h
44#ifndef RT_WITHOUT_PRAGMA_ONCE
45# pragma once
46#endif
47
48#include <OS.h>
49
50
51struct mutex_waiter;
52
53typedef struct mutex {
54 const char* name;
55 struct mutex_waiter* waiters;
56#if KDEBUG
57 thread_id holder;
58#else
59 int32 count;
60 uint16 ignore_unlock_count;
61#endif
62 uint8 flags;
63} mutex;
64
65#define MUTEX_FLAG_CLONE_NAME 0x1
66
67
68typedef struct recursive_lock {
69 mutex lock;
70#if !KDEBUG
71 thread_id holder;
72#endif
73 int recursion;
74} recursive_lock;
75
76
77struct rw_lock_waiter;
78
79typedef struct rw_lock {
80 const char* name;
81 struct rw_lock_waiter* waiters;
82 thread_id holder;
83 vint32 count;
84 int32 owner_count;
85 int16 active_readers;
86 // Only > 0 while a writer is waiting: number
87 // of active readers when the first waiting
88 // writer started waiting.
89 int16 pending_readers;
90 // Number of readers that have already
91 // incremented "count", but have not yet started
92 // to wait at the time the last writer unlocked.
93 uint32 flags;
94} rw_lock;
95
96#define RW_LOCK_WRITER_COUNT_BASE 0x10000
97
98#define RW_LOCK_FLAG_CLONE_NAME 0x1
99
100
101#if KDEBUG
102# define KDEBUG_RW_LOCK_DEBUG 0
103 // Define to 1 if you want to use ASSERT_READ_LOCKED_RW_LOCK().
104 // The rw_lock will just behave like a recursive locker then.
105# define ASSERT_LOCKED_RECURSIVE(r) \
106 { ASSERT(find_thread(NULL) == (r)->lock.holder); }
107# define ASSERT_LOCKED_MUTEX(m) { ASSERT(find_thread(NULL) == (m)->holder); }
108# define ASSERT_WRITE_LOCKED_RW_LOCK(l) \
109 { ASSERT(find_thread(NULL) == (l)->holder); }
110# if KDEBUG_RW_LOCK_DEBUG
111# define ASSERT_READ_LOCKED_RW_LOCK(l) \
112 { ASSERT(find_thread(NULL) == (l)->holder); }
113# else
114# define ASSERT_READ_LOCKED_RW_LOCK(l) do {} while (false)
115# endif
116#else
117# define ASSERT_LOCKED_RECURSIVE(r) do {} while (false)
118# define ASSERT_LOCKED_MUTEX(m) do {} while (false)
119# define ASSERT_WRITE_LOCKED_RW_LOCK(m) do {} while (false)
120# define ASSERT_READ_LOCKED_RW_LOCK(l) do {} while (false)
121#endif
122
123
124// static initializers
125#if KDEBUG
126# define MUTEX_INITIALIZER(name) { name, NULL, -1, 0 }
127# define RECURSIVE_LOCK_INITIALIZER(name) { MUTEX_INITIALIZER(name), 0 }
128#else
129# define MUTEX_INITIALIZER(name) { name, NULL, 0, 0, 0 }
130# define RECURSIVE_LOCK_INITIALIZER(name) { MUTEX_INITIALIZER(name), -1, 0 }
131#endif
132
133#define RW_LOCK_INITIALIZER(name) { name, NULL, -1, 0, 0, 0 }
134
135
136#if KDEBUG
137# define RECURSIVE_LOCK_HOLDER(recursiveLock) ((recursiveLock)->lock.holder)
138#else
139# define RECURSIVE_LOCK_HOLDER(recursiveLock) ((recursiveLock)->holder)
140#endif
141
142
143#ifdef __cplusplus
144extern "C" {
145#endif
146
147extern void recursive_lock_init(recursive_lock *lock, const char *name);
148 // name is *not* cloned nor freed in recursive_lock_destroy()
149extern void recursive_lock_init_etc(recursive_lock *lock, const char *name,
150 uint32 flags);
151extern void recursive_lock_destroy(recursive_lock *lock);
152extern status_t recursive_lock_lock(recursive_lock *lock);
153extern status_t recursive_lock_trylock(recursive_lock *lock);
154extern void recursive_lock_unlock(recursive_lock *lock);
155extern int32 recursive_lock_get_recursion(recursive_lock *lock);
156
157extern void rw_lock_init(rw_lock* lock, const char* name);
158 // name is *not* cloned nor freed in rw_lock_destroy()
159extern void rw_lock_init_etc(rw_lock* lock, const char* name, uint32 flags);
160extern void rw_lock_destroy(rw_lock* lock);
161extern status_t rw_lock_write_lock(rw_lock* lock);
162
163extern void mutex_init(mutex* lock, const char* name);
164 // name is *not* cloned nor freed in mutex_destroy()
165extern void mutex_init_etc(mutex* lock, const char* name, uint32 flags);
166extern void mutex_destroy(mutex* lock);
167extern status_t mutex_switch_lock(mutex* from, mutex* to);
168 // Unlocks "from" and locks "to" such that unlocking and starting to wait
169 // for the lock is atomically. I.e. if "from" guards the object "to" belongs
170 // to, the operation is safe as long as "from" is held while destroying
171 // "to".
172extern status_t mutex_switch_from_read_lock(rw_lock* from, mutex* to);
173 // Like mutex_switch_lock(), just for a switching from a read-locked
174 // rw_lock.
175
176
177// implementation private:
178
179extern status_t _rw_lock_read_lock(rw_lock* lock);
180extern status_t _rw_lock_read_lock_with_timeout(rw_lock* lock,
181 uint32 timeoutFlags, bigtime_t timeout);
182extern void _rw_lock_read_unlock(rw_lock* lock, bool threadsLocked);
183extern void _rw_lock_write_unlock(rw_lock* lock, bool threadsLocked);
184
185extern status_t _mutex_lock(mutex* lock, bool threadsLocked);
186extern void _mutex_unlock(mutex* lock, bool threadsLocked);
187extern status_t _mutex_trylock(mutex* lock);
188extern status_t _mutex_lock_with_timeout(mutex* lock, uint32 timeoutFlags,
189 bigtime_t timeout);
190
191
192static inline status_t
193rw_lock_read_lock(rw_lock* lock)
194{
195#if KDEBUG_RW_LOCK_DEBUG
196 return rw_lock_write_lock(lock);
197#else
198 int32 oldCount = atomic_add(&lock->count, 1);
199 if (oldCount >= RW_LOCK_WRITER_COUNT_BASE)
200 return _rw_lock_read_lock(lock);
201 return B_OK;
202#endif
203}
204
205
206static inline status_t
207rw_lock_read_lock_with_timeout(rw_lock* lock, uint32 timeoutFlags,
208 bigtime_t timeout)
209{
210#if KDEBUG_RW_LOCK_DEBUG
211 return mutex_lock_with_timeout(lock, timeoutFlags, timeout);
212#else
213 int32 oldCount = atomic_add(&lock->count, 1);
214 if (oldCount >= RW_LOCK_WRITER_COUNT_BASE)
215 return _rw_lock_read_lock_with_timeout(lock, timeoutFlags, timeout);
216 return B_OK;
217#endif
218}
219
220
221static inline void
222rw_lock_read_unlock(rw_lock* lock)
223{
224#if KDEBUG_RW_LOCK_DEBUG
225 rw_lock_write_unlock(lock);
226#else
227 int32 oldCount = atomic_add(&lock->count, -1);
228 if (oldCount >= RW_LOCK_WRITER_COUNT_BASE)
229 _rw_lock_read_unlock(lock, false);
230#endif
231}
232
233
234static inline void
235rw_lock_write_unlock(rw_lock* lock)
236{
237 _rw_lock_write_unlock(lock, false);
238}
239
240
241static inline status_t
242mutex_lock(mutex* lock)
243{
244#if KDEBUG
245 return _mutex_lock(lock, false);
246#else
247 if (atomic_add(&lock->count, -1) < 0)
248 return _mutex_lock(lock, false);
249 return B_OK;
250#endif
251}
252
253
254static inline status_t
255mutex_lock_threads_locked(mutex* lock)
256{
257#if KDEBUG
258 return _mutex_lock(lock, true);
259#else
260 if (atomic_add(&lock->count, -1) < 0)
261 return _mutex_lock(lock, true);
262 return B_OK;
263#endif
264}
265
266
267static inline status_t
268mutex_trylock(mutex* lock)
269{
270#if KDEBUG
271 return _mutex_trylock(lock);
272#else
273 if (atomic_test_and_set(&lock->count, -1, 0) != 0)
274 return B_WOULD_BLOCK;
275 return B_OK;
276#endif
277}
278
279
280static inline status_t
281mutex_lock_with_timeout(mutex* lock, uint32 timeoutFlags, bigtime_t timeout)
282{
283#if KDEBUG
284 return _mutex_lock_with_timeout(lock, timeoutFlags, timeout);
285#else
286 if (atomic_add(&lock->count, -1) < 0)
287 return _mutex_lock_with_timeout(lock, timeoutFlags, timeout);
288 return B_OK;
289#endif
290}
291
292
293static inline void
294mutex_unlock(mutex* lock)
295{
296#if !KDEBUG
297 if (atomic_add(&lock->count, 1) < -1)
298#endif
299 _mutex_unlock(lock, false);
300}
301
302
303static inline void
304mutex_transfer_lock(mutex* lock, thread_id thread)
305{
306#if KDEBUG
307 lock->holder = thread;
308#endif
309}
310
311
312extern void lock_debug_init();
313
314#ifdef __cplusplus
315}
316#endif
317
318#endif /* !GA_INCLUDED_HAIKU_lock_h */
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