VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/misc/once.cpp@ 20606

Last change on this file since 20606 was 19896, checked in by vboxsync, 15 years ago

IPRT: Added a RTOnceReset method (needed it for a testcase).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.9 KB
Line 
1/* $Id: once.cpp 19896 2009-05-21 21:35:42Z vboxsync $ */
2/** @file
3 * IPRT - Execute Once.
4 */
5
6/*
7 * Copyright (C) 2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31/*******************************************************************************
32* Header Files *
33*******************************************************************************/
34#include <iprt/once.h>
35#include <iprt/semaphore.h>
36#include <iprt/thread.h>
37#include <iprt/err.h>
38#include <iprt/assert.h>
39#include <iprt/asm.h>
40
41
42
43RTDECL(int) RTOnce(PRTONCE pOnce, PFNRTONCE pfnOnce, void *pvUser1, void *pvUser2)
44{
45 /*
46 * Validate input (strict builds only).
47 */
48 AssertPtr(pOnce);
49 AssertPtr(pfnOnce);
50
51 /*
52 * Deal with the 'initialized' case first
53 */
54 int32_t iState = ASMAtomicUoReadS32(&pOnce->iState);
55 if (RT_LIKELY(iState == 2))
56 return ASMAtomicUoReadS32(&pOnce->rc);
57 AssertReturn(iState == -1 || iState == 1, VERR_INTERNAL_ERROR);
58
59 /*
60 * Do we initialize it?
61 */
62 if (iState == -1)
63 {
64 RTSEMEVENTMULTI hEventMulti;
65 int rc = RTSemEventMultiCreate(&hEventMulti);
66 if (RT_FAILURE(rc))
67 hEventMulti = NIL_RTSEMEVENTMULTI;
68
69 if (ASMAtomicCmpXchgS32(&pOnce->iState, 1, -1))
70 {
71 ASMAtomicWriteHandle(&pOnce->hEventMulti, hEventMulti);
72 ASMAtomicIncS32(&pOnce->cEventRefs);
73
74 /* do the execute once stuff. */
75 int32_t rcOnce = pfnOnce(pvUser1, pvUser2);
76
77 /* set the return code, change the state and signal any waiters. */
78 ASMAtomicWriteS32(&pOnce->rc, rcOnce);
79 ASMAtomicWriteS32(&pOnce->iState, 2);
80 if (hEventMulti != NIL_RTSEMEVENTMULTI)
81 RTSemEventMultiSignal(hEventMulti);
82
83 /* last guy destroys the semaphore. */
84 if (ASMAtomicDecS32(&pOnce->cEventRefs) == 0)
85 ASMAtomicWriteSize(&pOnce->hEventMulti, NIL_RTSEMEVENTMULTI);
86 else
87 hEventMulti = NIL_RTSEMEVENTMULTI;
88 }
89 if (hEventMulti != NIL_RTSEMEVENTMULTI)
90 RTSemEventMultiDestroy(hEventMulti);
91 }
92
93 /*
94 * Wait for it to finish initializing.
95 */
96 if (ASMAtomicReadS32(&pOnce->iState) == 1)
97 {
98 int i = 0;
99 while (ASMAtomicReadS32(&pOnce->iState) == 1)
100 {
101 bool fYieldSleep = true;
102
103 /*
104 * Take care not to increment the counter if it's 0, that indicates
105 * that RTONCE::hEventMulti isn't valid either because it's not set
106 * yet, or because it's being destroyed.
107 */
108 int32_t cEventRefs = ASMAtomicUoReadS32(&pOnce->cEventRefs);
109 while ( cEventRefs > 0
110 && !ASMAtomicCmpXchgS32(&pOnce->cEventRefs, cEventRefs + 1, cEventRefs))
111 cEventRefs = ASMAtomicUoReadS32(&pOnce->cEventRefs);
112 if (cEventRefs > 1)
113 {
114 /*
115 * The hEventMulti might be NIL for two reasons, see above in
116 * the init code, if it isn't valid just do the yield/sleep thing.
117 */
118 RTSEMEVENTMULTI hEventMulti;
119 ASMAtomicUoReadSize(&pOnce->hEventMulti, &hEventMulti);
120 if (hEventMulti != NIL_RTSEMEVENTMULTI)
121 {
122 fYieldSleep = false;
123 RTSemEventMultiWait(hEventMulti, RT_INDEFINITE_WAIT);
124 }
125
126 /*
127 * Last thread cleans up.
128 */
129 if (ASMAtomicDecS32(&pOnce->cEventRefs) == 0)
130 {
131 ASMAtomicXchgHandle(&pOnce->hEventMulti, NIL_RTSEMEVENTMULTI, &hEventMulti);
132 if (hEventMulti != NIL_RTSEMEVENTMULTI)
133 RTSemEventMultiDestroy(hEventMulti);
134 }
135 }
136
137 /*
138 * If we didn't block, yield or sleep for a bit.
139 *
140 * The sleep is essential to prevent higher priority threads from spinning wildly
141 * and preventing a lower priority thread from completing the pfnOnce operation
142 * in a timely manner.
143 */
144 if (fYieldSleep)
145 {
146 if (ASMAtomicReadS32(&pOnce->iState) != 1)
147 break;
148 if (!(++i % 8) )
149 RTThreadSleep(1);
150 else
151 RTThreadYield();
152 }
153 }
154 }
155
156 /*
157 * Finally, return the status code from the execute once function.
158 */
159 return ASMAtomicUoReadS32(&pOnce->rc);
160}
161
162
163RTDECL(void) RTOnceReset(PRTONCE pOnce)
164{
165 /* Cannot be done while busy! */
166 AssertPtr(pOnce);
167 Assert(pOnce->hEventMulti == NIL_RTSEMEVENTMULTI);
168 Assert(pOnce->iState != 1);
169
170 /* Do the same as RTONCE_INITIALIZER does. */
171 ASMAtomicWriteS32(&pOnce->rc, VERR_INTERNAL_ERROR);
172 ASMAtomicWriteS32(&pOnce->iState, -1);
173}
174
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