VirtualBox

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

Last change on this file since 46852 was 43893, checked in by vboxsync, 12 years ago

Runtime/once: fixed assertion

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.3 KB
Line 
1/* $Id: once.cpp 43893 2012-11-16 09:30:27Z vboxsync $ */
2/** @file
3 * IPRT - Execute Once.
4 */
5
6/*
7 * Copyright (C) 2007-2012 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/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <iprt/once.h>
32#include "internal/iprt.h"
33
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/critsect.h>
37#include <iprt/err.h>
38#include <iprt/initterm.h>
39#include <iprt/semaphore.h>
40#include <iprt/thread.h>
41
42
43/*******************************************************************************
44* Global Variables *
45*******************************************************************************/
46#ifdef IN_RING3
47
48/** For initializing the clean-up list code. */
49static RTONCE g_OnceCleanUp = RTONCE_INITIALIZER;
50/** Critical section protecting the clean-up list. */
51static RTCRITSECT g_CleanUpCritSect;
52/** The clean-up list. */
53static RTLISTANCHOR g_CleanUpList;
54
55
56/** @callback_method_impl{FNRTTERMCALLBACK} */
57static DECLCALLBACK(void) rtOnceTermCallback(RTTERMREASON enmReason, int32_t iStatus, void *pvUser)
58{
59 bool const fLazyCleanUpOk = RTTERMREASON_IS_LAZY_CLEANUP_OK(enmReason);
60 RTCritSectEnter(&g_CleanUpCritSect); /* Potentially dangerous. */
61
62 PRTONCE pCur, pPrev;
63 RTListForEachReverseSafe(&g_CleanUpList, pCur, pPrev, RTONCE, CleanUpNode)
64 {
65 /*
66 * Mostly reset it before doing the callback.
67 *
68 * Should probably introduce some new states here, but I'm not sure
69 * it's really worth it at this point.
70 */
71 PFNRTONCECLEANUP pfnCleanUp = pCur->pfnCleanUp;
72 void *pvUserCleanUp = pCur->pvUser;
73 pCur->pvUser = NULL;
74 pCur->pfnCleanUp = NULL;
75 ASMAtomicWriteS32(&pCur->rc, VERR_WRONG_ORDER);
76
77 pfnCleanUp(pvUserCleanUp, fLazyCleanUpOk);
78
79 /*
80 * Reset the reset of the state if we're being unloaded or smth.
81 */
82 if (!fLazyCleanUpOk)
83 {
84 ASMAtomicWriteS32(&pCur->rc, VERR_INTERNAL_ERROR);
85 ASMAtomicWriteS32(&pCur->iState, RTONCESTATE_UNINITIALIZED);
86 }
87 }
88
89 RTCritSectLeave(&g_CleanUpCritSect);
90 NOREF(pvUser); NOREF(enmReason); NOREF(iStatus);
91}
92
93
94
95/**
96 * Initializes the globals (using RTOnce).
97 *
98 * @returns IPRT status code
99 * @param pvUser Unused.
100 */
101static DECLCALLBACK(int32_t) rtOnceInitCleanUp(void *pvUser)
102{
103 NOREF(pvUser);
104 RTListInit(&g_CleanUpList);
105 int rc = RTCritSectInit(&g_CleanUpCritSect);
106 if (RT_SUCCESS(rc))
107 {
108 rc = RTTermRegisterCallback(rtOnceTermCallback, NULL);
109 if (RT_SUCCESS(rc))
110 return rc;
111
112 RTCritSectDelete(&g_CleanUpCritSect);
113 }
114 return rc;
115}
116
117#endif /* IN_RING3 */
118
119
120
121/**
122 * The state loop of the other threads.
123 *
124 * @returns VINF_SUCCESS when everything went smoothly. IPRT status code if we
125 * encountered trouble.
126 * @param pOnce The execute once structure.
127 * @param phEvtM Where to store the semaphore handle so the caller
128 * can do the cleaning up for us.
129 */
130static int rtOnceOtherThread(PRTONCE pOnce, PRTSEMEVENTMULTI phEvtM)
131{
132 uint32_t cYields = 0;
133 for (;;)
134 {
135 int32_t iState = ASMAtomicReadS32(&pOnce->iState);
136 switch (iState)
137 {
138 /*
139 * No semaphore, try create one.
140 */
141 case RTONCESTATE_BUSY_NO_SEM:
142 if (ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_BUSY_CREATING_SEM, RTONCESTATE_BUSY_NO_SEM))
143 {
144 int rc = RTSemEventMultiCreate(phEvtM);
145 if (RT_SUCCESS(rc))
146 {
147 ASMAtomicWriteHandle(&pOnce->hEventMulti, *phEvtM);
148 int32_t cRefs = ASMAtomicIncS32(&pOnce->cEventRefs); Assert(cRefs == 1); NOREF(cRefs);
149
150 if (!ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_BUSY_HAVE_SEM, RTONCESTATE_BUSY_CREATING_SEM))
151 {
152 /* Too slow. */
153 AssertReturn(ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE, RTONCESTATE_DONE_CREATING_SEM)
154 , VERR_INTERNAL_ERROR_5);
155
156 ASMAtomicWriteHandle(&pOnce->hEventMulti, NIL_RTSEMEVENTMULTI);
157 cRefs = ASMAtomicDecS32(&pOnce->cEventRefs); Assert(cRefs == 0);
158
159 RTSemEventMultiDestroy(*phEvtM);
160 *phEvtM = NIL_RTSEMEVENTMULTI;
161 }
162 }
163 else
164 {
165 AssertReturn( ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_BUSY_SPIN, RTONCESTATE_BUSY_CREATING_SEM)
166 || ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE, RTONCESTATE_DONE_CREATING_SEM)
167 , VERR_INTERNAL_ERROR_4);
168 *phEvtM = NIL_RTSEMEVENTMULTI;
169 }
170 }
171 break;
172
173 /*
174 * This isn't nice, but it's the easy way out.
175 */
176 case RTONCESTATE_BUSY_CREATING_SEM:
177 case RTONCESTATE_BUSY_SPIN:
178 cYields++;
179 if (!(++cYields % 8))
180 RTThreadSleep(1);
181 else
182 RTThreadYield();
183 break;
184
185 /*
186 * There is a semaphore, try wait on it.
187 *
188 * We continue waiting after reaching DONE_HAVE_SEM if we
189 * already got the semaphore to avoid racing the first thread.
190 */
191 case RTONCESTATE_DONE_HAVE_SEM:
192 if (*phEvtM == NIL_RTSEMEVENTMULTI)
193 return VINF_SUCCESS;
194 /* fall thru */
195 case RTONCESTATE_BUSY_HAVE_SEM:
196 {
197 /*
198 * Grab the semaphore if we haven't got it yet.
199 * We must take care not to increment the counter if it
200 * is 0. This may happen if we're racing a state change.
201 */
202 if (*phEvtM == NIL_RTSEMEVENTMULTI)
203 {
204 int32_t cEventRefs = ASMAtomicUoReadS32(&pOnce->cEventRefs);
205 while ( cEventRefs > 0
206 && ASMAtomicUoReadS32(&pOnce->iState) == RTONCESTATE_BUSY_HAVE_SEM)
207 {
208 if (ASMAtomicCmpXchgExS32(&pOnce->cEventRefs, cEventRefs + 1, cEventRefs, &cEventRefs))
209 break;
210 ASMNopPause();
211 }
212 if (cEventRefs <= 0)
213 break;
214
215 ASMAtomicReadHandle(&pOnce->hEventMulti, phEvtM);
216 AssertReturn(*phEvtM != NIL_RTSEMEVENTMULTI, VERR_INTERNAL_ERROR_2);
217 }
218
219 /*
220 * We've got a sempahore, do the actual waiting.
221 */
222 do
223 RTSemEventMultiWaitNoResume(*phEvtM, RT_INDEFINITE_WAIT);
224 while (ASMAtomicReadS32(&pOnce->iState) == RTONCESTATE_BUSY_HAVE_SEM);
225 break;
226 }
227
228 case RTONCESTATE_DONE_CREATING_SEM:
229 case RTONCESTATE_DONE:
230 return VINF_SUCCESS;
231
232 default:
233 AssertMsgFailedReturn(("%d\n", iState), VERR_INTERNAL_ERROR_3);
234 }
235 }
236}
237
238
239RTDECL(int) RTOnceSlow(PRTONCE pOnce, PFNRTONCE pfnOnce, PFNRTONCECLEANUP pfnCleanUp, void *pvUser)
240{
241 /*
242 * Validate input (strict builds only).
243 */
244 AssertPtr(pOnce);
245 AssertPtr(pfnOnce);
246
247 /*
248 * Deal with the 'initialized' case first
249 */
250 int32_t iState = ASMAtomicUoReadS32(&pOnce->iState);
251 if (RT_LIKELY( iState == RTONCESTATE_DONE
252 || iState == RTONCESTATE_DONE_CREATING_SEM
253 || iState == RTONCESTATE_DONE_HAVE_SEM
254 ))
255 return ASMAtomicUoReadS32(&pOnce->rc);
256
257 AssertReturn( iState == RTONCESTATE_UNINITIALIZED
258 || iState == RTONCESTATE_BUSY_NO_SEM
259 || iState == RTONCESTATE_BUSY_SPIN
260 || iState == RTONCESTATE_BUSY_CREATING_SEM
261 || iState == RTONCESTATE_BUSY_HAVE_SEM
262 , VERR_INTERNAL_ERROR);
263
264#ifndef IN_RING3
265 AssertReturn(!pfnCleanUp, VERR_NOT_SUPPORTED);
266#else /* IN_RING3 */
267
268 /*
269 * Make sure our clean-up bits are working if needed later.
270 */
271 if (pfnCleanUp)
272 {
273 int rc = RTOnce(&g_OnceCleanUp, rtOnceInitCleanUp, NULL);
274 if (RT_FAILURE(rc))
275 return rc;
276 }
277#endif /* IN_RING3 */
278
279 /*
280 * Do we initialize it?
281 */
282 int32_t rcOnce;
283 if ( iState == RTONCESTATE_UNINITIALIZED
284 && ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_BUSY_NO_SEM, RTONCESTATE_UNINITIALIZED))
285 {
286 /*
287 * Yes, so do the execute once stuff.
288 */
289 rcOnce = pfnOnce(pvUser);
290 ASMAtomicWriteS32(&pOnce->rc, rcOnce);
291
292#ifdef IN_RING3
293 /*
294 * Register clean-up if requested and we were successful.
295 */
296 if (pfnCleanUp && RT_SUCCESS(rcOnce))
297 {
298 RTCritSectEnter(&g_CleanUpCritSect);
299 pOnce->pfnCleanUp = pfnCleanUp;
300 pOnce->pvUser = pvUser;
301 RTListAppend(&g_CleanUpList, &pOnce->CleanUpNode);
302 RTCritSectLeave(&g_CleanUpCritSect);
303 }
304#endif
305
306 /*
307 * If there is a sempahore to signal, we're in for some extra work here.
308 */
309 if ( !ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE, RTONCESTATE_BUSY_NO_SEM)
310 && !ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE, RTONCESTATE_BUSY_SPIN)
311 && !ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE_CREATING_SEM, RTONCESTATE_BUSY_CREATING_SEM)
312 )
313 {
314 /* Grab the sempahore by switching to 'DONE_HAVE_SEM' before reaching 'DONE'. */
315 AssertReturn(ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE_HAVE_SEM, RTONCESTATE_BUSY_HAVE_SEM),
316 VERR_INTERNAL_ERROR_2);
317
318 int32_t cRefs = ASMAtomicIncS32(&pOnce->cEventRefs);
319 Assert(cRefs > 1); NOREF(cRefs);
320
321 RTSEMEVENTMULTI hEvtM;
322 ASMAtomicReadHandle(&pOnce->hEventMulti, &hEvtM);
323 Assert(hEvtM != NIL_RTSEMEVENTMULTI);
324
325 ASMAtomicWriteS32(&pOnce->iState, RTONCESTATE_DONE);
326
327 /* Signal it and return. */
328 RTSemEventMultiSignal(hEvtM);
329 }
330 }
331 else
332 {
333 /*
334 * Wait for the first thread to complete. Delegate this to a helper
335 * function to simplify cleanup and keep things a bit shorter.
336 */
337 RTSEMEVENTMULTI hEvtM = NIL_RTSEMEVENTMULTI;
338 rcOnce = rtOnceOtherThread(pOnce, &hEvtM);
339 if (hEvtM != NIL_RTSEMEVENTMULTI)
340 {
341 if (ASMAtomicDecS32(&pOnce->cEventRefs) == 0)
342 {
343 bool fRc;
344 ASMAtomicCmpXchgHandle(&pOnce->hEventMulti, NIL_RTSEMEVENTMULTI, hEvtM, fRc); Assert(fRc);
345 fRc = ASMAtomicCmpXchgS32(&pOnce->iState, RTONCESTATE_DONE, RTONCESTATE_DONE_HAVE_SEM); Assert(fRc);
346 RTSemEventMultiDestroy(hEvtM);
347 }
348 }
349 if (RT_SUCCESS(rcOnce))
350 rcOnce = ASMAtomicUoReadS32(&pOnce->rc);
351 }
352
353 return rcOnce;
354}
355RT_EXPORT_SYMBOL(RTOnceSlow);
356
357
358RTDECL(void) RTOnceReset(PRTONCE pOnce)
359{
360 /* Cannot be done while busy! */
361 AssertPtr(pOnce);
362 Assert(pOnce->hEventMulti == NIL_RTSEMEVENTMULTI);
363 int32_t iState = ASMAtomicUoReadS32(&pOnce->iState);
364 AssertMsg( iState == RTONCESTATE_DONE
365 && iState == RTONCESTATE_UNINITIALIZED,
366 ("%d\n", iState));
367 NOREF(iState);
368
369#ifdef IN_RING3
370 /* Unregister clean-up. */
371 if (pOnce->pfnCleanUp)
372 {
373 RTCritSectEnter(&g_CleanUpCritSect);
374 RTListNodeRemove(&pOnce->CleanUpNode);
375 pOnce->pfnCleanUp = NULL;
376 pOnce->pvUser = NULL;
377 RTCritSectLeave(&g_CleanUpCritSect);
378 }
379#endif /* IN_RING3 */
380
381 /* Do the same as RTONCE_INITIALIZER does. */
382 ASMAtomicWriteS32(&pOnce->rc, VERR_INTERNAL_ERROR);
383 ASMAtomicWriteS32(&pOnce->iState, RTONCESTATE_UNINITIALIZED);
384}
385RT_EXPORT_SYMBOL(RTOnceReset);
386
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