VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/semeventmulti-posix.cpp@ 34079

Last change on this file since 34079 was 33540, checked in by vboxsync, 14 years ago

*: spelling fixes, thanks Timeless!

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 22.9 KB
Line 
1/* $Id: semeventmulti-posix.cpp 33540 2010-10-28 09:27:05Z vboxsync $ */
2/** @file
3 * IPRT - Multiple Release Event Semaphore, POSIX.
4 */
5
6/*
7 * Copyright (C) 2006-2007 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/*******************************************************************************
28* Header Files *
29*******************************************************************************/
30#include <iprt/semaphore.h>
31#include "internal/iprt.h"
32
33#include <iprt/asm.h>
34#include <iprt/assert.h>
35#include <iprt/err.h>
36#include <iprt/lockvalidator.h>
37#include <iprt/mem.h>
38#include <iprt/time.h>
39
40#include "internal/strict.h"
41
42#include <errno.h>
43#include <pthread.h>
44#include <unistd.h>
45#include <sys/time.h>
46
47
48/*******************************************************************************
49* Defined Constants And Macros *
50*******************************************************************************/
51/** @def IPRT_HAVE_PTHREAD_CONDATTR_SETCLOCK
52 * Set if the platform implements pthread_condattr_setclock().
53 * Enables the use of the monotonic clock for waiting on condition variables. */
54#ifndef IPRT_HAVE_PTHREAD_CONDATTR_SETCLOCK
55/* Linux detection */
56# if defined(RT_OS_LINUX) && defined(__USE_XOPEN2K)
57# include <features.h>
58# if __GLIBC_PREREQ(2,6) /** @todo figure the exact version where this was added */
59# define IPRT_HAVE_PTHREAD_CONDATTR_SETCLOCK
60# endif
61# endif
62/** @todo check other platforms */
63#endif
64
65
66/*******************************************************************************
67* Structures and Typedefs *
68*******************************************************************************/
69/** Posix internal representation of a Mutex Multi semaphore.
70 * The POSIX implementation uses a mutex and a condition variable to implement
71 * the automatic reset event semaphore semantics. */
72struct RTSEMEVENTMULTIINTERNAL
73{
74 /** pthread condition. */
75 pthread_cond_t Cond;
76 /** pthread mutex which protects the condition and the event state. */
77 pthread_mutex_t Mutex;
78 /** The state of the semaphore.
79 * This is operated while owning mutex and using atomic updating. */
80 volatile uint32_t u32State;
81 /** Number of waiters. */
82 volatile uint32_t cWaiters;
83#ifdef RTSEMEVENTMULTI_STRICT
84 /** Signallers. */
85 RTLOCKVALRECSHRD Signallers;
86 /** Indicates that lock validation should be performed. */
87 bool volatile fEverHadSignallers;
88#endif
89 /** Set if we're using the monotonic clock. */
90 bool fMonotonicClock;
91};
92
93/** The values of the u32State variable in RTSEMEVENTMULTIINTERNAL.
94 * @{ */
95/** The object isn't initialized. */
96#define EVENTMULTI_STATE_UNINITIALIZED 0
97/** The semaphore is signaled. */
98#define EVENTMULTI_STATE_SIGNALED 0xff00ff00
99/** The semaphore is not signaled. */
100#define EVENTMULTI_STATE_NOT_SIGNALED 0x00ff00ff
101/** @} */
102
103
104
105RTDECL(int) RTSemEventMultiCreate(PRTSEMEVENTMULTI phEventMultiSem)
106{
107 return RTSemEventMultiCreateEx(phEventMultiSem, 0 /*fFlags*/, NIL_RTLOCKVALCLASS, NULL);
108}
109
110
111RTDECL(int) RTSemEventMultiCreateEx(PRTSEMEVENTMULTI phEventMultiSem, uint32_t fFlags, RTLOCKVALCLASS hClass,
112 const char *pszNameFmt, ...)
113{
114 AssertReturn(!(fFlags & ~RTSEMEVENTMULTI_FLAGS_NO_LOCK_VAL), VERR_INVALID_PARAMETER);
115
116 /*
117 * Allocate semaphore handle.
118 */
119 int rc;
120 struct RTSEMEVENTMULTIINTERNAL *pThis = (struct RTSEMEVENTMULTIINTERNAL *)RTMemAlloc(sizeof(struct RTSEMEVENTMULTIINTERNAL));
121 if (pThis)
122 {
123 /*
124 * Create the condition variable.
125 */
126 pthread_condattr_t CondAttr;
127 rc = pthread_condattr_init(&CondAttr);
128 if (!rc)
129 {
130#if defined(CLOCK_MONOTONIC) && defined(IPRT_HAVE_PTHREAD_CONDATTR_SETCLOCK)
131 /* ASSUMES RTTimeSystemNanoTS() == RTTimeNanoTS() == clock_gettime(CLOCK_MONOTONIC). */
132 rc = pthread_condattr_setclock(&CondAttr, CLOCK_MONOTONIC);
133 pThis->fMonotonicClock = rc == 0;
134#else
135 pThis->fMonotonicClock = false;
136#endif
137 rc = pthread_cond_init(&pThis->Cond, &CondAttr);
138 if (!rc)
139 {
140 /*
141 * Create the semaphore.
142 */
143 pthread_mutexattr_t MutexAttr;
144 rc = pthread_mutexattr_init(&MutexAttr);
145 if (!rc)
146 {
147 rc = pthread_mutex_init(&pThis->Mutex, &MutexAttr);
148 if (!rc)
149 {
150 pthread_mutexattr_destroy(&MutexAttr);
151 pthread_condattr_destroy(&CondAttr);
152
153 ASMAtomicXchgU32(&pThis->u32State, EVENTMULTI_STATE_NOT_SIGNALED);
154 ASMAtomicXchgU32(&pThis->cWaiters, 0);
155#ifdef RTSEMEVENTMULTI_STRICT
156 if (!pszNameFmt)
157 {
158 static uint32_t volatile s_iSemEventMultiAnon = 0;
159 RTLockValidatorRecSharedInit(&pThis->Signallers, hClass, RTLOCKVAL_SUB_CLASS_ANY, pThis,
160 true /*fSignaller*/, !(fFlags & RTSEMEVENTMULTI_FLAGS_NO_LOCK_VAL),
161 "RTSemEventMulti-%u", ASMAtomicIncU32(&s_iSemEventMultiAnon) - 1);
162 }
163 else
164 {
165 va_list va;
166 va_start(va, pszNameFmt);
167 RTLockValidatorRecSharedInitV(&pThis->Signallers, hClass, RTLOCKVAL_SUB_CLASS_ANY, pThis,
168 true /*fSignaller*/, !(fFlags & RTSEMEVENTMULTI_FLAGS_NO_LOCK_VAL),
169 pszNameFmt, va);
170 va_end(va);
171 }
172 pThis->fEverHadSignallers = false;
173#endif
174
175 *phEventMultiSem = pThis;
176 return VINF_SUCCESS;
177 }
178
179 pthread_mutexattr_destroy(&MutexAttr);
180 }
181 pthread_cond_destroy(&pThis->Cond);
182 }
183 pthread_condattr_destroy(&CondAttr);
184 }
185
186 rc = RTErrConvertFromErrno(rc);
187 RTMemFree(pThis);
188 }
189 else
190 rc = VERR_NO_MEMORY;
191
192 return rc;
193
194}
195
196
197RTDECL(int) RTSemEventMultiDestroy(RTSEMEVENTMULTI hEventMultiSem)
198{
199 /*
200 * Validate handle.
201 */
202 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
203 if (pThis == NIL_RTSEMEVENTMULTI)
204 return VINF_SUCCESS;
205 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
206 uint32_t u32 = pThis->u32State;
207 AssertReturn(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED, VERR_INVALID_HANDLE);
208
209 /*
210 * Abort all waiters forcing them to return failure.
211 */
212 int rc;
213 for (int i = 30; i > 0; i--)
214 {
215 ASMAtomicXchgU32(&pThis->u32State, EVENTMULTI_STATE_UNINITIALIZED);
216 rc = pthread_cond_destroy(&pThis->Cond);
217 if (rc != EBUSY)
218 break;
219 pthread_cond_broadcast(&pThis->Cond);
220 usleep(1000);
221 }
222 if (rc)
223 {
224 AssertMsgFailed(("Failed to destroy event sem %p, rc=%d.\n", hEventMultiSem, rc));
225 return RTErrConvertFromErrno(rc);
226 }
227
228 /*
229 * Destroy the semaphore
230 * If it's busy we'll wait a bit to give the threads a chance to be scheduled.
231 */
232 for (int i = 30; i > 0; i--)
233 {
234 rc = pthread_mutex_destroy(&pThis->Mutex);
235 if (rc != EBUSY)
236 break;
237 usleep(1000);
238 }
239 if (rc)
240 {
241 AssertMsgFailed(("Failed to destroy event sem %p, rc=%d. (mutex)\n", hEventMultiSem, rc));
242 return RTErrConvertFromErrno(rc);
243 }
244
245 /*
246 * Free the semaphore memory and be gone.
247 */
248#ifdef RTSEMEVENTMULTI_STRICT
249 RTLockValidatorRecSharedDelete(&pThis->Signallers);
250#endif
251 RTMemFree(pThis);
252 return VINF_SUCCESS;
253}
254
255
256RTDECL(int) RTSemEventMultiSignal(RTSEMEVENTMULTI hEventMultiSem)
257{
258 /*
259 * Validate input.
260 */
261 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
262 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
263 uint32_t u32 = pThis->u32State;
264 AssertReturn(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED, VERR_INVALID_HANDLE);
265
266#ifdef RTSEMEVENTMULTI_STRICT
267 if (pThis->fEverHadSignallers)
268 {
269 int rc9 = RTLockValidatorRecSharedCheckSignaller(&pThis->Signallers, NIL_RTTHREAD);
270 if (RT_FAILURE(rc9))
271 return rc9;
272 }
273#endif
274
275 /*
276 * Lock the mutex semaphore.
277 */
278 int rc = pthread_mutex_lock(&pThis->Mutex);
279 if (rc)
280 {
281 AssertMsgFailed(("Failed to lock event sem %p, rc=%d.\n", hEventMultiSem, rc));
282 return RTErrConvertFromErrno(rc);
283 }
284
285 /*
286 * Check the state.
287 */
288 if (pThis->u32State == EVENTMULTI_STATE_NOT_SIGNALED)
289 {
290 ASMAtomicXchgU32(&pThis->u32State, EVENTMULTI_STATE_SIGNALED);
291 rc = pthread_cond_broadcast(&pThis->Cond);
292 AssertMsg(!rc, ("Failed to signal event sem %p, rc=%d.\n", hEventMultiSem, rc));
293 }
294 else if (pThis->u32State == EVENTMULTI_STATE_SIGNALED)
295 {
296 rc = pthread_cond_broadcast(&pThis->Cond); /* give'm another kick... */
297 AssertMsg(!rc, ("Failed to signal event sem %p, rc=%d. (2)\n", hEventMultiSem, rc));
298 }
299 else
300 rc = VERR_SEM_DESTROYED;
301
302 /*
303 * Release the mutex and return.
304 */
305 int rc2 = pthread_mutex_unlock(&pThis->Mutex);
306 AssertMsg(!rc2, ("Failed to unlock event sem %p, rc=%d.\n", hEventMultiSem, rc));
307 if (rc)
308 return RTErrConvertFromErrno(rc);
309 if (rc2)
310 return RTErrConvertFromErrno(rc2);
311
312 return VINF_SUCCESS;
313}
314
315
316RTDECL(int) RTSemEventMultiReset(RTSEMEVENTMULTI hEventMultiSem)
317{
318 /*
319 * Validate input.
320 */
321 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
322 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
323 uint32_t u32 = pThis->u32State;
324 AssertReturn(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED, VERR_INVALID_HANDLE);
325
326 /*
327 * Lock the mutex semaphore.
328 */
329 int rc = pthread_mutex_lock(&pThis->Mutex);
330 if (rc)
331 {
332 AssertMsgFailed(("Failed to lock event multi sem %p, rc=%d.\n", hEventMultiSem, rc));
333 return RTErrConvertFromErrno(rc);
334 }
335
336 /*
337 * Check the state.
338 */
339 if (pThis->u32State == EVENTMULTI_STATE_SIGNALED)
340 ASMAtomicXchgU32(&pThis->u32State, EVENTMULTI_STATE_NOT_SIGNALED);
341 else if (pThis->u32State != EVENTMULTI_STATE_NOT_SIGNALED)
342 rc = VERR_SEM_DESTROYED;
343
344 /*
345 * Release the mutex and return.
346 */
347 rc = pthread_mutex_unlock(&pThis->Mutex);
348 if (rc)
349 {
350 AssertMsgFailed(("Failed to unlock event multi sem %p, rc=%d.\n", hEventMultiSem, rc));
351 return RTErrConvertFromErrno(rc);
352 }
353
354 return VINF_SUCCESS;
355
356}
357
358
359/**
360 * Handle polling (timeout already expired at the time of the call).
361 *
362 * @returns VINF_SUCCESS, VERR_TIMEOUT, VERR_SEM_DESTROYED.
363 * @param pThis The semaphore.
364 */
365DECLINLINE(int) rtSemEventMultiPosixWaitPoll(struct RTSEMEVENTMULTIINTERNAL *pThis)
366{
367 int rc = pthread_mutex_lock(&pThis->Mutex);
368 AssertMsgReturn(!rc, ("Failed to lock event multi sem %p, rc=%d.\n", pThis, rc), RTErrConvertFromErrno(rc));
369
370 uint32_t const u32State = pThis->u32State;
371
372 rc = pthread_mutex_unlock(&pThis->Mutex);
373 AssertMsg(!rc, ("Failed to unlock event multi sem %p, rc=%d.\n", pThis, rc)); NOREF(rc);
374
375 return u32State == EVENTMULTI_STATE_SIGNALED
376 ? VINF_SUCCESS
377 : u32State != EVENTMULTI_STATE_UNINITIALIZED
378 ? VERR_TIMEOUT
379 : VERR_SEM_DESTROYED;
380}
381
382
383
384/**
385 * Implements the indefinite wait.
386 *
387 * @returns See RTSemEventMultiWaitEx.
388 * @param pThis The semaphore.
389 * @param fFlags See RTSemEventMultiWaitEx.
390 * @param pSrcPos The source position, can be NULL.
391 */
392static int rtSemEventMultiPosixWaitIndefinite(struct RTSEMEVENTMULTIINTERNAL *pThis, uint32_t fFlags, PCRTLOCKVALSRCPOS pSrcPos)
393{
394 /* take mutex */
395 int rc = pthread_mutex_lock(&pThis->Mutex);
396 AssertMsgReturn(!rc, ("Failed to lock event multi sem %p, rc=%d.\n", pThis, rc), RTErrConvertFromErrno(rc));
397 ASMAtomicIncU32(&pThis->cWaiters);
398
399 for (;;)
400 {
401 /* check state. */
402 uint32_t const u32State = pThis->u32State;
403 if (u32State != EVENTMULTI_STATE_NOT_SIGNALED)
404 {
405 ASMAtomicDecU32(&pThis->cWaiters);
406 rc = pthread_mutex_unlock(&pThis->Mutex);
407 AssertMsg(!rc, ("Failed to unlock event multi sem %p, rc=%d.\n", pThis, rc));
408 return u32State == EVENTMULTI_STATE_SIGNALED
409 ? VINF_SUCCESS
410 : VERR_SEM_DESTROYED;
411 }
412
413 /* wait */
414#ifdef RTSEMEVENTMULTI_STRICT
415 RTTHREAD hThreadSelf = RTThreadSelfAutoAdopt();
416 if (pThis->fEverHadSignallers)
417 {
418 rc = RTLockValidatorRecSharedCheckBlocking(&pThis->Signallers, hThreadSelf, pSrcPos, false,
419 RT_INDEFINITE_WAIT, RTTHREADSTATE_EVENT_MULTI, true);
420 if (RT_FAILURE(rc))
421 {
422 ASMAtomicDecU32(&pThis->cWaiters);
423 pthread_mutex_unlock(&pThis->Mutex);
424 return rc;
425 }
426 }
427#else
428 RTTHREAD hThreadSelf = RTThreadSelf();
429#endif
430 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_EVENT_MULTI, true);
431 rc = pthread_cond_wait(&pThis->Cond, &pThis->Mutex);
432 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_EVENT_MULTI);
433 if (RT_UNLIKELY(rc))
434 {
435 AssertMsgFailed(("Failed to wait on event multi sem %p, rc=%d.\n", pThis, rc));
436 ASMAtomicDecU32(&pThis->cWaiters);
437 int rc2 = pthread_mutex_unlock(&pThis->Mutex);
438 AssertMsg(!rc2, ("Failed to unlock event multi sem %p, rc=%d.\n", pThis, rc2)); NOREF(rc2);
439 return RTErrConvertFromErrno(rc);
440 }
441 }
442}
443
444
445/**
446 * Implements the timed wait.
447 *
448 * @returns See RTSemEventMultiWaitEx
449 * @param pThis The semaphore.
450 * @param fFlags See RTSemEventMultiWaitEx.
451 * @param uTimeout See RTSemEventMultiWaitEx.
452 * @param pSrcPos The source position, can be NULL.
453 */
454static int rtSemEventMultiPosixWaitTimed(struct RTSEMEVENTMULTIINTERNAL *pThis, uint32_t fFlags, uint64_t uTimeout,
455 PCRTLOCKVALSRCPOS pSrcPos)
456{
457 /*
458 * Convert uTimeout to a relative value in nano seconds.
459 */
460 if (fFlags & RTSEMWAIT_FLAGS_MILLISECS)
461 uTimeout = uTimeout < UINT64_MAX / UINT32_C(1000000) * UINT32_C(1000000)
462 ? uTimeout * UINT32_C(1000000)
463 : UINT64_MAX;
464 if (uTimeout == UINT64_MAX) /* unofficial way of indicating an indefinite wait */
465 return rtSemEventMultiPosixWaitIndefinite(pThis, fFlags, pSrcPos);
466
467 uint64_t uAbsTimeout = uTimeout;
468 if (fFlags & RTSEMWAIT_FLAGS_ABSOLUTE)
469 {
470 uint64_t u64Now = RTTimeSystemNanoTS();
471 uTimeout = uTimeout > u64Now ? uTimeout - u64Now : 0;
472 }
473
474 if (uTimeout == 0)
475 return rtSemEventMultiPosixWaitPoll(pThis);
476
477 /*
478 * Get current time and calc end of deadline relative to real time.
479 */
480 struct timespec ts = {0,0};
481 if (!pThis->fMonotonicClock)
482 {
483#ifdef RT_OS_DARWIN
484 struct timeval tv = {0,0};
485 gettimeofday(&tv, NULL);
486 ts.tv_sec = tv.tv_sec;
487 ts.tv_nsec = tv.tv_usec * 1000;
488#else
489 clock_gettime(CLOCK_REALTIME, &ts);
490#endif
491 struct timespec tsAdd;
492 tsAdd.tv_nsec = uTimeout % UINT32_C(1000000000);
493 tsAdd.tv_sec = uTimeout / UINT32_C(1000000000);
494 if ( sizeof(ts.tv_sec) < sizeof(uint64_t)
495 && ( uTimeout > UINT64_C(1000000000) * UINT32_MAX
496 || (uint64_t)ts.tv_sec + tsAdd.tv_sec >= UINT32_MAX) )
497 return rtSemEventMultiPosixWaitIndefinite(pThis, fFlags, pSrcPos);
498
499 ts.tv_sec += tsAdd.tv_sec;
500 ts.tv_nsec += tsAdd.tv_nsec;
501 if (ts.tv_nsec >= 1000000000)
502 {
503 ts.tv_nsec -= 1000000000;
504 ts.tv_sec++;
505 }
506 /* Note! No need to complete uAbsTimeout for RTSEMWAIT_FLAGS_RELATIVE in this path. */
507 }
508 else
509 {
510 /* ASSUMES RTTimeSystemNanoTS() == RTTimeNanoTS() == clock_gettime(CLOCK_MONOTONIC). */
511 if (fFlags & RTSEMWAIT_FLAGS_RELATIVE)
512 uAbsTimeout += RTTimeSystemNanoTS();
513 if ( sizeof(ts.tv_sec) < sizeof(uint64_t)
514 && uAbsTimeout > UINT64_C(1000000000) * UINT32_MAX)
515 return rtSemEventMultiPosixWaitIndefinite(pThis, fFlags, pSrcPos);
516 ts.tv_nsec = uAbsTimeout % UINT32_C(1000000000);
517 ts.tv_sec = uAbsTimeout / UINT32_C(1000000000);
518 }
519
520 /*
521 * To business!
522 */
523 /* take mutex */
524 int rc = pthread_mutex_lock(&pThis->Mutex);
525 AssertMsgReturn(rc == 0, ("rc=%d pThis=%p\n", rc, pThis), RTErrConvertFromErrno(rc)); NOREF(rc);
526 ASMAtomicIncU32(&pThis->cWaiters);
527
528 for (;;)
529 {
530 /* check state. */
531 uint32_t const u32State = pThis->u32State;
532 if (u32State != EVENTMULTI_STATE_NOT_SIGNALED)
533 {
534 ASMAtomicDecU32(&pThis->cWaiters);
535 rc = pthread_mutex_unlock(&pThis->Mutex);
536 AssertMsg(!rc, ("Failed to unlock event multi sem %p, rc=%d.\n", pThis, rc));
537 return u32State == EVENTMULTI_STATE_SIGNALED
538 ? VINF_SUCCESS
539 : VERR_SEM_DESTROYED;
540 }
541
542 /* wait */
543#ifdef RTSEMEVENTMULTI_STRICT
544 RTTHREAD hThreadSelf = RTThreadSelfAutoAdopt();
545 if (pThis->fEverHadSignallers)
546 {
547 rc = RTLockValidatorRecSharedCheckBlocking(&pThis->Signallers, hThreadSelf, pSrcPos, false,
548 uTimeout / UINT32_C(1000000), RTTHREADSTATE_EVENT_MULTI, true);
549 if (RT_FAILURE(rc))
550 {
551 ASMAtomicDecU32(&pThis->cWaiters);
552 pthread_mutex_unlock(&pThis->Mutex);
553 return rc;
554 }
555 }
556#else
557 RTTHREAD hThreadSelf = RTThreadSelf();
558#endif
559 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_EVENT_MULTI, true);
560 rc = pthread_cond_timedwait(&pThis->Cond, &pThis->Mutex, &ts);
561 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_EVENT_MULTI);
562 if ( rc
563 && ( rc != EINTR /* according to SuS this function shall not return EINTR, but linux man page says differently. */
564 || (fFlags & RTSEMWAIT_FLAGS_NORESUME)) )
565 {
566 AssertMsg(rc == ETIMEDOUT, ("Failed to wait on event multi sem %p, rc=%d.\n", pThis, rc));
567 ASMAtomicDecU32(&pThis->cWaiters);
568 int rc2 = pthread_mutex_unlock(&pThis->Mutex);
569 AssertMsg(!rc2, ("Failed to unlock event multi sem %p, rc=%d.\n", pThis, rc2)); NOREF(rc2);
570 return RTErrConvertFromErrno(rc);
571 }
572
573 /* check the absolute deadline. */
574 }
575}
576
577
578DECLINLINE(int) rtSemEventMultiPosixWait(RTSEMEVENTMULTI hEventMultiSem, uint32_t fFlags, uint64_t uTimeout,
579 PCRTLOCKVALSRCPOS pSrcPos)
580{
581 /*
582 * Validate input.
583 */
584 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
585 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
586 uint32_t u32 = pThis->u32State;
587 AssertReturn(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED, VERR_INVALID_HANDLE);
588 AssertReturn(RTSEMWAIT_FLAGS_ARE_VALID(fFlags), VERR_INVALID_PARAMETER);
589
590 /*
591 * Optimize the case where the event is signalled.
592 */
593 if (ASMAtomicUoReadU32(&pThis->u32State) == EVENTMULTI_STATE_SIGNALED)
594 {
595 int rc = rtSemEventMultiPosixWaitPoll(pThis);
596 if (RT_LIKELY(rc != VERR_TIMEOUT))
597 return rc;
598 }
599
600 /*
601 * Indefinite or timed wait?
602 */
603 if (fFlags & RTSEMWAIT_FLAGS_INDEFINITE)
604 return rtSemEventMultiPosixWaitIndefinite(pThis, fFlags, pSrcPos);
605 return rtSemEventMultiPosixWaitTimed(pThis, fFlags, uTimeout, pSrcPos);
606}
607
608
609#undef RTSemEventMultiWaitEx
610RTDECL(int) RTSemEventMultiWaitEx(RTSEMEVENTMULTI hEventMultiSem, uint32_t fFlags, uint64_t uTimeout)
611{
612#ifndef RTSEMEVENT_STRICT
613 return rtSemEventMultiPosixWait(hEventMultiSem, fFlags, uTimeout, NULL);
614#else
615 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
616 return rtSemEventMultiPosixWait(hEventMultiSem, fFlags, uTimeout, &SrcPos);
617#endif
618}
619
620
621RTDECL(int) RTSemEventMultiWaitExDebug(RTSEMEVENTMULTI hEventMultiSem, uint32_t fFlags, uint64_t uTimeout,
622 RTHCUINTPTR uId, RT_SRC_POS_DECL)
623{
624 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
625 return rtSemEventMultiPosixWait(hEventMultiSem, fFlags, uTimeout, &SrcPos);
626}
627
628
629RTDECL(void) RTSemEventMultiSetSignaller(RTSEMEVENTMULTI hEventMultiSem, RTTHREAD hThread)
630{
631#ifdef RTSEMEVENTMULTI_STRICT
632 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
633 AssertPtrReturnVoid(pThis);
634 uint32_t u32 = pThis->u32State;
635 AssertReturnVoid(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED);
636
637 ASMAtomicWriteBool(&pThis->fEverHadSignallers, true);
638 RTLockValidatorRecSharedResetOwner(&pThis->Signallers, hThread, NULL);
639#endif
640}
641
642
643RTDECL(void) RTSemEventMultiAddSignaller(RTSEMEVENTMULTI hEventMultiSem, RTTHREAD hThread)
644{
645#ifdef RTSEMEVENTMULTI_STRICT
646 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
647 AssertPtrReturnVoid(pThis);
648 uint32_t u32 = pThis->u32State;
649 AssertReturnVoid(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED);
650
651 ASMAtomicWriteBool(&pThis->fEverHadSignallers, true);
652 RTLockValidatorRecSharedAddOwner(&pThis->Signallers, hThread, NULL);
653#endif
654}
655
656
657RTDECL(void) RTSemEventMultiRemoveSignaller(RTSEMEVENTMULTI hEventMultiSem, RTTHREAD hThread)
658{
659#ifdef RTSEMEVENTMULTI_STRICT
660 struct RTSEMEVENTMULTIINTERNAL *pThis = hEventMultiSem;
661 AssertPtrReturnVoid(pThis);
662 uint32_t u32 = pThis->u32State;
663 AssertReturnVoid(u32 == EVENTMULTI_STATE_NOT_SIGNALED || u32 == EVENTMULTI_STATE_SIGNALED);
664
665 RTLockValidatorRecSharedRemoveOwner(&pThis->Signallers, hThread);
666#endif
667}
668
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