VirtualBox

source: vbox/trunk/include/iprt/semaphore.h@ 25475

Last change on this file since 25475 was 25467, checked in by vboxsync, 15 years ago

IPRT,PDMCritSect: More lock validation.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.5 KB
Line 
1/** @file
2 * IPRT - Semaphore.
3 */
4
5/*
6 * Copyright (C) 2006-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_semaphore_h
31#define ___iprt_semaphore_h
32
33#include <iprt/cdefs.h>
34#include <iprt/types.h>
35
36
37RT_C_DECLS_BEGIN
38
39/** @defgroup grp_rt_sems RTSem - Semaphores
40 *
41 * This module implements all kinds of event and mutex semaphores; in addition
42 * to these, IPRT implements "critical sections", which are fast recursive
43 * mutexes (see @ref grp_rt_critsect ). C++ users may find @ref grp_rt_lock
44 * interesting.
45 *
46 * @ingroup grp_rt
47 * @{
48 */
49
50
51/** @defgroup grp_rt_sems_event RTSemEvent - Single Release Event Semaphores
52 *
53 * Event semaphores can be used for inter-thread communication when one thread
54 * wants to notify another thread that something happened. A thread can block
55 * ("wait") on an event semaphore until it is signalled by another thread; see
56 * RTSemEventCreate, RTSemEventSignal and RTSemEventWait.
57 *
58 * @{ */
59
60/**
61 * Create a event semaphore.
62 *
63 * @returns iprt status code.
64 * @param pEventSem Where to store the event semaphore handle.
65 */
66RTDECL(int) RTSemEventCreate(PRTSEMEVENT pEventSem);
67
68/**
69 * Destroy an event semaphore.
70 *
71 * @returns iprt status code.
72 * @param EventSem Handle of the
73 */
74RTDECL(int) RTSemEventDestroy(RTSEMEVENT EventSem);
75
76/**
77 * Signal an event semaphore.
78 *
79 * The event semaphore will be signaled and automatically reset after exactly
80 * one thread have successfully returned from RTSemEventWait() after
81 * waiting/polling on that semaphore.
82 *
83 * @returns iprt status code.
84 * @param EventSem The event semaphore to signal.
85 */
86RTDECL(int) RTSemEventSignal(RTSEMEVENT EventSem);
87
88/**
89 * Wait for the event semaphore to be signaled, resume on interruption.
90 *
91 * This function will resume if the wait is interrupted by an async system event
92 * (like a unix signal) or similar.
93 *
94 * @returns iprt status code.
95 * Will not return VERR_INTERRUPTED.
96 * @param EventSem The event semaphore to wait on.
97 * @param cMillies Number of milliseconds to wait.
98 */
99RTDECL(int) RTSemEventWait(RTSEMEVENT EventSem, unsigned cMillies);
100
101/**
102 * Wait for the event semaphore to be signaled, return on interruption.
103 *
104 * This function will not resume the wait if interrupted.
105 *
106 * @returns iprt status code.
107 * @param EventSem The event semaphore to wait on.
108 * @param cMillies Number of milliseconds to wait.
109 */
110RTDECL(int) RTSemEventWaitNoResume(RTSEMEVENT EventSem, unsigned cMillies);
111
112/** @} */
113
114
115/** @defgroup grp_rt_sems_event_multi RTSemEventMulti - Multiple Release Event Semaphores
116 *
117 * A variant of @ref grp_rt_sems_event where all threads will be unblocked when
118 * signalling the semaphore.
119 *
120 * @{ */
121
122/**
123 * Create a event multi semaphore.
124 *
125 * @returns iprt status code.
126 * @param pEventMultiSem Where to store the event multi semaphore handle.
127 */
128RTDECL(int) RTSemEventMultiCreate(PRTSEMEVENTMULTI pEventMultiSem);
129
130/**
131 * Destroy an event multi semaphore.
132 *
133 * @returns iprt status code.
134 * @param EventMultiSem The event multi sempahore to destroy.
135 */
136RTDECL(int) RTSemEventMultiDestroy(RTSEMEVENTMULTI EventMultiSem);
137
138/**
139 * Signal an event multi semaphore.
140 *
141 * @returns iprt status code.
142 * @param EventMultiSem The event multi semaphore to signal.
143 */
144RTDECL(int) RTSemEventMultiSignal(RTSEMEVENTMULTI EventMultiSem);
145
146/**
147 * Resets an event multi semaphore to non-signaled state.
148 *
149 * @returns iprt status code.
150 * @param EventMultiSem The event multi semaphore to reset.
151 */
152RTDECL(int) RTSemEventMultiReset(RTSEMEVENTMULTI EventMultiSem);
153
154/**
155 * Wait for the event multi semaphore to be signaled, resume on interruption.
156 *
157 * This function will resume if the wait is interrupted by an async
158 * system event (like a unix signal) or similar.
159 *
160 * @returns iprt status code.
161 * Will not return VERR_INTERRUPTED.
162 * @param EventMultiSem The event multi semaphore to wait on.
163 * @param cMillies Number of milliseconds to wait.
164 */
165RTDECL(int) RTSemEventMultiWait(RTSEMEVENTMULTI EventMultiSem, unsigned cMillies);
166
167
168/**
169 * Wait for the event multi semaphore to be signaled, return on interruption.
170 *
171 * This function will not resume the wait if interrupted.
172 *
173 * @returns iprt status code.
174 * @param EventMultiSem The event multi semaphore to wait on.
175 * @param cMillies Number of milliseconds to wait.
176 */
177RTDECL(int) RTSemEventMultiWaitNoResume(RTSEMEVENTMULTI EventMultiSem, unsigned cMillies);
178
179/** @} */
180
181
182/** @defgroup grp_rt_sems_mutex RTMutex - Mutex semaphores.
183 *
184 * Mutex semaphores protect a section of code or data to which access must be
185 * exclusive. Only one thread can hold access to a critical section at one
186 * time. See RTSemMutexCreate, RTSemMutexRequest and RTSemMutexRelease.
187 *
188 * @remarks These are less efficient than "fast mutexes" and "critical
189 * sections", which IPRT implements as well; see @ref
190 * grp_rt_sems_fast_mutex and @ref grp_rt_critsect .
191 *
192 * @{ */
193
194/**
195 * Create a mutex semaphore.
196 *
197 * @returns iprt status code.
198 * @param pMutexSem Where to store the mutex semaphore handle.
199 */
200RTDECL(int) RTSemMutexCreate(PRTSEMMUTEX pMutexSem);
201
202/**
203 * Destroy a mutex semaphore.
204 *
205 * @returns iprt status code.
206 * @param MutexSem The mutex semaphore to destroy.
207 */
208RTDECL(int) RTSemMutexDestroy(RTSEMMUTEX MutexSem);
209
210/**
211 * Request ownership of a mutex semaphore, resume on interruption.
212 *
213 * This function will resume if the wait is interrupted by an async
214 * system event (like a unix signal) or similar.
215 *
216 * The same thread may request a mutex semaphore multiple times,
217 * a nested counter is kept to make sure it's released on the right
218 * RTSemMutexRelease() call.
219 *
220 * @returns iprt status code.
221 * Will not return VERR_INTERRUPTED.
222 * @param MutexSem The mutex semaphore to request ownership over.
223 * @param cMillies The number of milliseconds to wait.
224 */
225RTDECL(int) RTSemMutexRequest(RTSEMMUTEX MutexSem, unsigned cMillies);
226
227/**
228 * Request ownership of a mutex semaphore, return on interruption.
229 *
230 * This function will not resume the wait if interrupted.
231 *
232 * The same thread may request a mutex semaphore multiple times,
233 * a nested counter is kept to make sure it's released on the right
234 * RTSemMutexRelease() call.
235 *
236 * @returns iprt status code.
237 * @param MutexSem The mutex semaphore to request ownership over.
238 * @param cMillies The number of milliseconds to wait.
239 */
240RTDECL(int) RTSemMutexRequestNoResume(RTSEMMUTEX MutexSem, unsigned cMillies);
241
242/**
243 * Debug version of RTSemMutexRequest that tracks the location.
244 *
245 * @returns iprt status code.
246 * Will not return VERR_INTERRUPTED.
247 * @param MutexSem The mutex semaphore to request ownership over.
248 * @param cMillies The number of milliseconds to wait.
249 * @param uId Some kind of locking location ID. Typically a
250 * return address up the stack. Optional (0).
251 * @param pszFile The file where the lock is being acquired from.
252 * Optional.
253 * @param iLine The line number in that file. Optional (0).
254 * @param pszFunction The functionn where the lock is being acquired
255 * from. Optional.
256 */
257RTDECL(int) RTSemMutexRequestDebug(RTSEMMUTEX MutexSem, unsigned cMillies, RTHCUINTPTR uId, RT_SRC_POS_DECL);
258
259/**
260 * Debug version of RTSemMutexRequestNoResume that tracks the location.
261 *
262 * @returns iprt status code.
263 * @param MutexSem The mutex semaphore to request ownership over.
264 * @param cMillies The number of milliseconds to wait.
265 * @param uId Some kind of locking location ID. Typically a
266 * return address up the stack. Optional (0).
267 * @param pszFile The file where the lock is being acquired from.
268 * Optional.
269 * @param iLine The line number in that file. Optional (0).
270 * @param pszFunction The functionn where the lock is being acquired
271 * from. Optional.
272 */
273RTDECL(int) RTSemMutexRequestNoResumeDebug(RTSEMMUTEX MutexSem, unsigned cMillies, RTHCUINTPTR uId, RT_SRC_POS_DECL);
274
275/**
276 * Release the ownership of a mutex semaphore.
277 *
278 * @returns iprt status code.
279 * @param MutexSem The mutex to release the ownership of.
280 * It goes without saying the the calling thread must own it.
281 */
282RTDECL(int) RTSemMutexRelease(RTSEMMUTEX MutexSem);
283
284/* Strict build: Remap the two request calls to the debug versions. */
285#ifdef RT_STRICT
286# ifdef ___iprt_asm_h
287# define RTSemMutexRequest(pCritSect, cMillies) RTSemMutexRequestDebug((pCritSect), (cMillies), (uintptr_t)ASMReturnAddress(), RT_SRC_POS)
288# define RTSemMutexRequestNoResume(pCritSect, cMillies) RTSemMutexRequestNoResumeDebug((pCritSect), (cMillies), (uintptr_t)ASMReturnAddress(), RT_SRC_POS)
289# else
290# define RTSemMutexRequest(pCritSect, cMillies) RTSemMutexRequestDebug((pCritSect), (cMillies), 0, RT_SRC_POS)
291# define RTSemMutexRequestNoResume(pCritSect, cMillies) RTSemMutexRequestNoResumeDebug((pCritSect), (cMillies), 0, RT_SRC_POS)
292# endif
293#endif
294
295/** @} */
296
297
298/** @defgroup grp_rt_sems_fast_mutex RTSemFastMutex - Fast Mutex Semaphores
299 *
300 * Fast mutexes work like regular mutexes in that they allow only a single
301 * thread access to a critical piece of code or data. As opposed to mutexes,
302 * they require no syscall if the fast mutex is not held (like critical
303 * sections). Unlike critical sections however, they are *not* recursive.
304 *
305 * @{ */
306
307/**
308 * Create a fast mutex semaphore.
309 *
310 * @returns iprt status code.
311 * @param pMutexSem Where to store the mutex semaphore handle.
312 *
313 * @remarks Fast mutex semaphores are not recursive.
314 */
315RTDECL(int) RTSemFastMutexCreate(PRTSEMFASTMUTEX pMutexSem);
316
317/**
318 * Destroy a fast mutex semaphore.
319 *
320 * @returns iprt status code.
321 * @param MutexSem The mutex semaphore to destroy.
322 */
323RTDECL(int) RTSemFastMutexDestroy(RTSEMFASTMUTEX MutexSem);
324
325/**
326 * Request ownership of a fast mutex semaphore.
327 *
328 * The same thread may request a mutex semaphore multiple times,
329 * a nested counter is kept to make sure it's released on the right
330 * RTSemMutexRelease() call.
331 *
332 * @returns iprt status code.
333 * @param MutexSem The mutex semaphore to request ownership over.
334 */
335RTDECL(int) RTSemFastMutexRequest(RTSEMFASTMUTEX MutexSem);
336
337/**
338 * Release the ownership of a fast mutex semaphore.
339 *
340 * @returns iprt status code.
341 * @param MutexSem The mutex to release the ownership of.
342 * It goes without saying the the calling thread must own it.
343 */
344RTDECL(int) RTSemFastMutexRelease(RTSEMFASTMUTEX MutexSem);
345
346/** @} */
347
348
349/** @defgroup grp_rt_sems_spin_mutex RTSemSpinMutex - Spinning Mutex Semaphores
350 *
351 * A very adaptive variant of mutex semaphore that is tailored for the ring-0
352 * logger.
353 *
354 * @{ */
355
356/**
357 * Creates a spinning mutex semaphore.
358 *
359 * @returns iprt status code.
360 * @retval VERR_INVALID_PARAMETER on invalid flags.
361 * @retval VERR_NO_MEMORY if out of memory for the semaphore structure and
362 * handle.
363 *
364 * @param phSpinMtx Where to return the handle to the create semaphore.
365 * @param fFlags Flags, see RTSEMSPINMUTEX_FLAGS_XXX.
366 */
367RTDECL(int) RTSemSpinMutexCreate(PRTSEMSPINMUTEX phSpinMtx, uint32_t fFlags);
368
369/** @name RTSemSpinMutexCreate flags.
370 * @{ */
371/** Always take the semaphore in a IRQ safe way.
372 * (In plain words: always disable interrupts.) */
373#define RTSEMSPINMUTEX_FLAGS_IRQ_SAFE RT_BIT_32(0)
374/** Mask of valid flags. */
375#define RTSEMSPINMUTEX_FLAGS_VALID_MASK UINT32_C(0x00000001)
376/** @} */
377
378/**
379 * Destroys a spinning mutex semaphore.
380 *
381 * @returns iprt status code.
382 * @retval VERR_INVALID_HANDLE (or crash) if the handle is invalid. (NIL will
383 * not cause this status.)
384 *
385 * @param hSpinMtx The semaphore handle. NIL_RTSEMSPINMUTEX is ignored
386 * quietly (VINF_SUCCESS).
387 */
388RTDECL(int) RTSemSpinMutexDestroy(RTSEMSPINMUTEX hSpinMtx);
389
390/**
391 * Request the spinning mutex semaphore.
392 *
393 * This may block if the context we're called in allows this. If not it will
394 * spin. If called in an interrupt context, we will only spin if the current
395 * owner isn't interrupted. Also, on some systems it is not always possible to
396 * wake up blocking threads in all contexts, so, which will either be indicated
397 * by returning VERR_SEM_BAD_CONTEXT or by temporarily switching the semaphore
398 * into pure spinlock state.
399 *
400 * Preemption will be disabled upon return. IRQs may also be disabled.
401 *
402 * @returns iprt status code.
403 * @retval VERR_SEM_BAD_CONTEXT if the context it's called in isn't suitable
404 * for releasing it if someone is sleeping on it.
405 * @retval VERR_SEM_DESTROYED if destroyed.
406 * @retval VERR_SEM_NESTED if held by the caller. Asserted.
407 * @retval VERR_INVALID_HANDLE if the handle is invalid. Asserted
408 *
409 * @param hSpinMtx The semaphore handle.
410 */
411RTDECL(int) RTSemSpinMutexRequest(RTSEMSPINMUTEX hSpinMtx);
412
413/**
414 * Like RTSemSpinMutexRequest but it won't block or spin if the semaphore is
415 * held by someone else.
416 *
417 * @returns iprt status code.
418 * @retval VERR_SEM_BUSY if held by someone else.
419 * @retval VERR_SEM_DESTROYED if destroyed.
420 * @retval VERR_SEM_NESTED if held by the caller. Asserted.
421 * @retval VERR_INVALID_HANDLE if the handle is invalid. Asserted
422 *
423 * @param hSpinMtx The semaphore handle.
424 */
425RTDECL(int) RTSemSpinMutexTryRequest(RTSEMSPINMUTEX hSpinMtx);
426
427/**
428 * Releases the semaphore previously acquired by RTSemSpinMutexRequest or
429 * RTSemSpinMutexTryRequest.
430 *
431 * @returns iprt status code.
432 * @retval VERR_SEM_DESTROYED if destroyed.
433 * @retval VERR_NOT_OWNER if not owner. Asserted.
434 * @retval VERR_INVALID_HANDLE if the handle is invalid. Asserted.
435 *
436 * @param hSpinMtx The semaphore handle.
437 */
438RTDECL(int) RTSemSpinMutexRelease(RTSEMSPINMUTEX hSpinMtx);
439
440/** @} */
441
442
443/** @defgroup grp_rt_sem_rw RTSemRW - Read / Write Semaphores
444 *
445 * Read/write semaphores are a fancier version of mutexes in that they grant
446 * read access to the protected data to several threads at the same time but
447 * allow only one writer at a time. This can make code scale better at the
448 * expense of slightly more overhead in mutex management.
449 *
450 * @{ */
451
452/**
453 * Creates a read/write semaphore.
454 *
455 * @returns iprt status code.
456 * @param pRWSem Where to store the handle to the created RW semaphore.
457 */
458RTDECL(int) RTSemRWCreate(PRTSEMRW pRWSem);
459
460/**
461 * Destroys a read/write semaphore.
462 *
463 * @returns iprt status code.
464 * @param RWSem The Read/Write semaphore to destroy.
465 */
466RTDECL(int) RTSemRWDestroy(RTSEMRW RWSem);
467
468/**
469 * Request read access to a read/write semaphore, resume on interruption
470 *
471 * @returns iprt status code.
472 * @retval VINF_SUCCESS on success.
473 * @retval VERR_INTERRUPT if the wait was interrupted.
474 * @retval VERR_INVALID_HANDLE if RWSem is invalid.
475 *
476 * @param RWSem The Read/Write semaphore to request read access to.
477 * @param cMillies The number of milliseconds to wait.
478 */
479RTDECL(int) RTSemRWRequestRead(RTSEMRW RWSem, unsigned cMillies);
480
481/**
482 * Request read access to a read/write semaphore, return on interruption
483 *
484 * @returns iprt status code.
485 * @retval VINF_SUCCESS on success.
486 * @retval VERR_INTERRUPT if the wait was interrupted.
487 * @retval VERR_INVALID_HANDLE if RWSem is invalid.
488 *
489 * @param RWSem The Read/Write semaphore to request read access to.
490 * @param cMillies The number of milliseconds to wait.
491 */
492RTDECL(int) RTSemRWRequestReadNoResume(RTSEMRW RWSem, unsigned cMillies);
493
494/**
495 * Release read access to a read/write semaphore.
496 *
497 * @returns iprt status code.
498 * @param RWSem The Read/Write sempahore to release read access to.
499 * Goes without saying that caller must have read access to the sem.
500 */
501RTDECL(int) RTSemRWReleaseRead(RTSEMRW RWSem);
502
503/**
504 * Request write access to a read/write semaphore, resume on interruption.
505 *
506 * @returns iprt status code.
507 * @retval VINF_SUCCESS on success.
508 * @retval VERR_DEADLOCK if the caller owned the read lock.
509 * @retval VERR_INVALID_HANDLE if RWSem is invalid.
510 *
511 * @param RWSem The Read/Write semaphore to request write access to.
512 * @param cMillies The number of milliseconds to wait.
513 */
514RTDECL(int) RTSemRWRequestWrite(RTSEMRW RWSem, unsigned cMillies);
515
516/**
517 * Request write access to a read/write semaphore, return on interruption.
518 *
519 * @returns iprt status code.
520 * @retval VINF_SUCCESS on success.
521 * @retval VERR_INTERRUPT if the wait was interrupted.
522 * @retval VERR_DEADLOCK if the caller owned the read lock.
523 * @retval VERR_INVALID_HANDLE if RWSem is invalid.
524 *
525 * @param RWSem The Read/Write semaphore to request write access to.
526 * @param cMillies The number of milliseconds to wait.
527 */
528RTDECL(int) RTSemRWRequestWriteNoResume(RTSEMRW RWSem, unsigned cMillies);
529
530/**
531 * Debug version of RTSemRWRequestWrite that tracks the location.
532 *
533 * @returns IPRT status code, see RTSemRWRequestWrite.
534 * @param RWSem The Read/Write semaphore to request write access
535 * to.
536 * @param cMillies The number of milliseconds to wait.
537 * @param uId Some kind of locking location ID. Typically a
538 * return address up the stack. Optional (0).
539 * @param pszFile The file where the lock is being acquired from.
540 * Optional.
541 * @param iLine The line number in that file. Optional (0).
542 * @param pszFunction The functionn where the lock is being acquired
543 * from. Optional.
544 */
545RTDECL(int) RTSemRWRequestWriteDebug(RTSEMRW MutexSem, unsigned cMillies, RTHCUINTPTR uId, RT_SRC_POS_DECL);
546
547/**
548 * Debug version of RTSemRWRequestWriteNoResume that tracks the location.
549 *
550 * @returns IPRT status code, see RTSemRWRequestWriteNoResume.
551 * @param RWSem The Read/Write semaphore to request write access
552 * to.
553 * @param cMillies The number of milliseconds to wait.
554 * @param uId Some kind of locking location ID. Typically a
555 * return address up the stack. Optional (0).
556 * @param pszFile The file where the lock is being acquired from.
557 * Optional.
558 * @param iLine The line number in that file. Optional (0).
559 * @param pszFunction The functionn where the lock is being acquired
560 * from. Optional.
561 */
562RTDECL(int) RTSemRWRequestWriteNoResumeDebug(RTSEMRW RWSem, unsigned cMillies, RTHCUINTPTR uId, RT_SRC_POS_DECL);
563
564/**
565 * Release write access to a read/write semaphore.
566 *
567 * @returns iprt status code.
568 * @param RWSem The Read/Write sempahore to release read access to.
569 * Goes without saying that caller must have write access to the sem.
570 */
571RTDECL(int) RTSemRWReleaseWrite(RTSEMRW RWSem);
572
573/**
574 * Checks if the caller is the exclusive semaphore owner.
575 *
576 * @returns true / false accoringly.
577 * @param RWSem The Read/Write semaphore in question.
578 */
579RTDECL(bool) RTSemRWIsWriteOwner(RTSEMRW RWSem);
580
581/**
582 * Gets the write recursion count.
583 *
584 * @returns The write recursion count (0 if bad semaphore handle).
585 * @param RWSem The Read/Write semaphore in question.
586 */
587RTDECL(uint32_t) RTSemRWGetWriteRecursion(RTSEMRW RWSem);
588
589/**
590 * Gets the read recursion count of the current writer.
591 *
592 * @returns The read recursion count (0 if bad semaphore handle).
593 * @param RWSem The Read/Write semaphore in question.
594 */
595RTDECL(uint32_t) RTSemRWGetWriterReadRecursion(RTSEMRW RWSem);
596
597/** @} */
598
599
600/** @defgroup grp_rt_sems_pingpong RTSemPingPong - Ping-Pong Construct
601 *
602 * Serialization of a two way communication.
603 *
604 * @{ */
605
606/**
607 * Ping-pong speaker
608 */
609typedef enum RTPINGPONGSPEAKER
610{
611 /** Not initialized. */
612 RTPINGPONGSPEAKER_UNINITIALIZE = 0,
613 /** Ping is speaking, Pong is waiting. */
614 RTPINGPONGSPEAKER_PING,
615 /** Pong is signaled, Ping is waiting. */
616 RTPINGPONGSPEAKER_PONG_SIGNALED,
617 /** Pong is speaking, Ping is waiting. */
618 RTPINGPONGSPEAKER_PONG,
619 /** Ping is signaled, Pong is waiting. */
620 RTPINGPONGSPEAKER_PING_SIGNALED,
621 /** Hack to ensure that it's at least 32-bits wide. */
622 RTPINGPONGSPEAKER_HACK = 0x7fffffff
623} RTPINGPONGSPEAKER;
624
625/**
626 * Ping-Pong construct.
627 *
628 * Two threads, one saying Ping and the other saying Pong. The construct
629 * makes sure they don't speak out of turn and that they can wait and poll
630 * on the conversation.
631 */
632typedef struct RTPINGPONG
633{
634 /** The semaphore the Ping thread waits on. */
635 RTSEMEVENT Ping;
636 /** The semaphore the Pong thread waits on. */
637 RTSEMEVENT Pong;
638 /** The current speaker. */
639 volatile RTPINGPONGSPEAKER enmSpeaker;
640#if HC_ARCH_BITS == 64
641 /** Padding the structure to become a multiple of sizeof(RTHCPTR). */
642 uint32_t u32Padding;
643#endif
644} RTPINGPONG;
645/** Pointer to Ping-Pong construct. */
646typedef RTPINGPONG *PRTPINGPONG;
647
648/**
649 * Init a Ping-Pong construct.
650 *
651 * @returns iprt status code.
652 * @param pPP Pointer to the ping-pong structure which needs initialization.
653 */
654RTDECL(int) RTSemPingPongInit(PRTPINGPONG pPP);
655
656/**
657 * Deletes a Ping-Pong construct.
658 *
659 * @returns iprt status code.
660 * @param pPP Pointer to the ping-pong structure which is to be destroyed.
661 * (I.e. put into uninitialized state.)
662 */
663RTDECL(int) RTSemPingPongDelete(PRTPINGPONG pPP);
664
665/**
666 * Signals the pong thread in a ping-pong construct. (I.e. sends ping.)
667 * This is called by the ping thread.
668 *
669 * @returns iprt status code.
670 * @param pPP Pointer to the ping-pong structure to ping.
671 */
672RTDECL(int) RTSemPing(PRTPINGPONG pPP);
673
674/**
675 * Signals the ping thread in a ping-pong construct. (I.e. sends pong.)
676 * This is called by the pong thread.
677 *
678 * @returns iprt status code.
679 * @param pPP Pointer to the ping-pong structure to pong.
680 */
681RTDECL(int) RTSemPong(PRTPINGPONG pPP);
682
683/**
684 * Wait function for the ping thread.
685 *
686 * @returns iprt status code.
687 * Will not return VERR_INTERRUPTED.
688 * @param pPP Pointer to the ping-pong structure to wait on.
689 * @param cMillies Number of milliseconds to wait.
690 */
691RTDECL(int) RTSemPingWait(PRTPINGPONG pPP, unsigned cMillies);
692
693/**
694 * Wait function for the pong thread.
695 *
696 * @returns iprt status code.
697 * Will not return VERR_INTERRUPTED.
698 * @param pPP Pointer to the ping-pong structure to wait on.
699 * @param cMillies Number of milliseconds to wait.
700 */
701RTDECL(int) RTSemPongWait(PRTPINGPONG pPP, unsigned cMillies);
702
703
704/**
705 * Checks if the pong thread is speaking.
706 *
707 * @returns true / false.
708 * @param pPP Pointer to the ping-pong structure.
709 * @remark This is NOT the same as !RTSemPongIsSpeaker().
710 */
711DECLINLINE(bool) RTSemPingIsSpeaker(PRTPINGPONG pPP)
712{
713 RTPINGPONGSPEAKER enmSpeaker = pPP->enmSpeaker;
714 return enmSpeaker == RTPINGPONGSPEAKER_PING;
715}
716
717
718/**
719 * Checks if the pong thread is speaking.
720 *
721 * @returns true / false.
722 * @param pPP Pointer to the ping-pong structure.
723 * @remark This is NOT the same as !RTSemPingIsSpeaker().
724 */
725DECLINLINE(bool) RTSemPongIsSpeaker(PRTPINGPONG pPP)
726{
727 RTPINGPONGSPEAKER enmSpeaker = pPP->enmSpeaker;
728 return enmSpeaker == RTPINGPONGSPEAKER_PONG;
729}
730
731
732/**
733 * Checks whether the ping thread should wait.
734 *
735 * @returns true / false.
736 * @param pPP Pointer to the ping-pong structure.
737 * @remark This is NOT the same as !RTSemPongShouldWait().
738 */
739DECLINLINE(bool) RTSemPingShouldWait(PRTPINGPONG pPP)
740{
741 RTPINGPONGSPEAKER enmSpeaker = pPP->enmSpeaker;
742 return enmSpeaker == RTPINGPONGSPEAKER_PONG
743 || enmSpeaker == RTPINGPONGSPEAKER_PONG_SIGNALED
744 || enmSpeaker == RTPINGPONGSPEAKER_PING_SIGNALED;
745}
746
747
748/**
749 * Checks whether the pong thread should wait.
750 *
751 * @returns true / false.
752 * @param pPP Pointer to the ping-pong structure.
753 * @remark This is NOT the same as !RTSemPingShouldWait().
754 */
755DECLINLINE(bool) RTSemPongShouldWait(PRTPINGPONG pPP)
756{
757 RTPINGPONGSPEAKER enmSpeaker = pPP->enmSpeaker;
758 return enmSpeaker == RTPINGPONGSPEAKER_PING
759 || enmSpeaker == RTPINGPONGSPEAKER_PING_SIGNALED
760 || enmSpeaker == RTPINGPONGSPEAKER_PONG_SIGNALED;
761}
762
763/** @} */
764
765
766/** @defgroup grp_rt_sems_xroads RTSemXRoads - Crossroads
767 *
768 * The crossroads semaphore is intended to prevent two classes of incompatible
769 * events from occuring simultaneously, like south/north bound traffic and
770 * west/east bound traffic at a 4-way junction.
771 *
772 * @remarks In order to simplify the implementation, the current flow is always
773 * given priority. So, it won't work at all well when busy!
774 *
775 * @remarks "XRoads" is used as a name because it is briefer than "crossroads"
776 * and it slightly stresses that is a 4 way crossing to the users of
777 * American English.
778 * @{
779 */
780
781/**
782 * Creates a crossroads semaphore.
783 *
784 * @returns IPRT status code.
785 *
786 * @param phXRoads Where to return the handle to the newly created
787 * crossroads semaphore.
788 */
789RTDECL(int) RTSemXRoadsCreate(PRTSEMXROADS phXRoads);
790
791/**
792 * Destroys a crossroads semaphore.
793 *
794 * @returns IPRT status code.
795 *
796 * @param hXRoads Handle to the crossroads semaphore that is to be
797 * destroyed. NIL_RTSEMXROADS is quitetly ignored
798 * (VINF_SUCCESS).
799 */
800RTDECL(int) RTSemXRoadsDestroy(RTSEMXROADS hXRoads);
801
802/**
803 * Enter the crossroads from the south or north.
804 *
805 * (Coupled with RTSemXRoadsNSLeave.)
806 *
807 * @returns IPRT status code.
808 * @param hXRoads Handle to the crossroads semaphore.
809 */
810RTDECL(int) RTSemXRoadsNSEnter(RTSEMXROADS hXRoads);
811
812/**
813 * Leave the crossroads to the north or south.
814 *
815 * (Coupled with RTSemXRoadsNSEnter.)
816 *
817 * @returns IPRT status code.
818 * @param hXRoads Handle to the crossroads semaphore.
819 */
820RTDECL(int) RTSemXRoadsNSLeave(RTSEMXROADS hXRoads);
821
822/**
823 * Leave the crossroads from the east or west.
824 *
825 * (Coupled with RTSemXRoadsEWLeave.)
826 *
827 * @returns IPRT status code.
828 * @param hXRoads Handle to the crossroads semaphore.
829 */
830RTDECL(int) RTSemXRoadsEWEnter(RTSEMXROADS hXRoads);
831
832/**
833 * Leave the crossroads to the west or east.
834 *
835 * (Coupled with RTSemXRoadsEWEnter.)
836 *
837 * @returns IPRT status code.
838 * @param hXRoads Handle to the crossroads semaphore.
839 */
840RTDECL(int) RTSemXRoadsEWLeave(RTSEMXROADS hXRoads);
841
842/** @} */
843
844/** @} */
845
846RT_C_DECLS_END
847
848#endif
849
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette