VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/powernotification-r0drv.c@ 22150

Last change on this file since 22150 was 22052, checked in by vboxsync, 15 years ago

IPRT: RT_MORE_STRICT for r0rdv and r0drv/darwin.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.6 KB
Line 
1/* $Id: powernotification-r0drv.c 22052 2009-08-07 09:45:48Z vboxsync $ */
2/** @file
3 * IPRT - Power Management, Ring-0 Driver, Event Notifications.
4 */
5
6/*
7 * Copyright (C) 2008 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/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#include <iprt/power.h>
36#include "internal/iprt.h"
37
38#include <iprt/asm.h>
39#include <iprt/assert.h>
40#include <iprt/err.h>
41#include <iprt/mem.h>
42#include <iprt/spinlock.h>
43#include <iprt/string.h>
44#include <iprt/thread.h>
45#include "r0drv/mp-r0drv.h"
46#include "r0drv/power-r0drv.h"
47
48
49/*******************************************************************************
50* Structures and Typedefs *
51*******************************************************************************/
52/**
53 * Notification registration record tracking
54 * RTPowerRegisterNotification() calls.
55 */
56typedef struct RTPOWERNOTIFYREG
57{
58 /** Pointer to the next record. */
59 struct RTPOWERNOTIFYREG * volatile pNext;
60 /** The callback. */
61 PFNRTPOWERNOTIFICATION pfnCallback;
62 /** The user argument. */
63 void *pvUser;
64 /** Bit mask indicating whether we've done this callback or not. */
65 uint8_t bmDone[sizeof(void *)];
66} RTPOWERNOTIFYREG;
67/** Pointer to a registration record. */
68typedef RTPOWERNOTIFYREG *PRTPOWERNOTIFYREG;
69
70
71/*******************************************************************************
72* Global Variables *
73*******************************************************************************/
74/** The spinlock protecting the list. */
75static RTSPINLOCK volatile g_hRTPowerNotifySpinLock = NIL_RTSPINLOCK;
76/** List of callbacks, in registration order. */
77static PRTPOWERNOTIFYREG volatile g_pRTPowerCallbackHead = NULL;
78/** The current done bit. */
79static uint32_t volatile g_iRTPowerDoneBit;
80/** The list generation.
81 * This is increased whenever the list has been modified. The callback routine
82 * make use of this to avoid having restart at the list head after each callback. */
83static uint32_t volatile g_iRTPowerGeneration;
84/** The number of RTPowerNotification users.
85 * This is incremented on init and decremented on termination. */
86static uint32_t volatile g_cRTPowerUsers = 0;
87
88
89
90
91RTDECL(int) RTPowerSignalEvent(RTPOWEREVENT enmEvent)
92{
93 PRTPOWERNOTIFYREG pCur;
94 RTSPINLOCKTMP Tmp;
95 RTSPINLOCK hSpinlock;
96
97 /*
98 * This is a little bit tricky as we cannot be holding the spinlock
99 * while calling the callback. This means that the list might change
100 * while we're walking it, and that multiple events might be running
101 * concurrently (depending on the OS).
102 *
103 * So, the first measure is to employ a 32-bitmask for each
104 * record where we'll use a bit that rotates for each call to
105 * this function to indicate which records that has been
106 * processed. This will take care of both changes to the list
107 * and a reasonable amount of concurrent events.
108 *
109 * In order to avoid having to restart the list walks for every
110 * callback we make, we'll make use a list generation number that is
111 * incremented everytime the list is changed. So, if it remains
112 * unchanged over a callback we can safely continue the iteration.
113 */
114 uint32_t iDone = ASMAtomicIncU32(&g_iRTPowerDoneBit);
115 iDone %= RT_SIZEOFMEMB(RTPOWERNOTIFYREG, bmDone) * 8;
116
117 hSpinlock = g_hRTPowerNotifySpinLock;
118 if (hSpinlock == NIL_RTSPINLOCK)
119 return VERR_ACCESS_DENIED;
120 RTSpinlockAcquire(hSpinlock, &Tmp);
121
122 /* Clear the bit. */
123 for (pCur = g_pRTPowerCallbackHead; pCur; pCur = pCur->pNext)
124 ASMAtomicBitClear(&pCur->bmDone[0], iDone);
125
126 /* Iterate the records and perform the callbacks. */
127 do
128 {
129 uint32_t const iGeneration = ASMAtomicUoReadU32(&g_iRTPowerGeneration);
130
131 pCur = g_pRTPowerCallbackHead;
132 while (pCur)
133 {
134 if (!ASMAtomicBitTestAndSet(&pCur->bmDone[0], iDone))
135 {
136 PFNRTPOWERNOTIFICATION pfnCallback = pCur->pfnCallback;
137 void *pvUser = pCur->pvUser;
138 pCur = pCur->pNext;
139 RTSpinlockRelease(g_hRTPowerNotifySpinLock, &Tmp);
140
141 pfnCallback(enmEvent, pvUser);
142
143 /* carefully require the lock here, see RTR0MpNotificationTerm(). */
144 hSpinlock = g_hRTPowerNotifySpinLock;
145 if (hSpinlock == NIL_RTSPINLOCK)
146 return VERR_ACCESS_DENIED;
147 RTSpinlockAcquire(hSpinlock, &Tmp);
148 if (ASMAtomicUoReadU32(&g_iRTPowerGeneration) != iGeneration)
149 break;
150 }
151 else
152 pCur = pCur->pNext;
153 }
154 } while (pCur);
155
156 RTSpinlockRelease(hSpinlock, &Tmp);
157 return VINF_SUCCESS;
158}
159RT_EXPORT_SYMBOL(RTPowerSignalEvent);
160
161
162RTDECL(int) RTPowerNotificationRegister(PFNRTPOWERNOTIFICATION pfnCallback, void *pvUser)
163{
164 PRTPOWERNOTIFYREG pCur;
165 PRTPOWERNOTIFYREG pNew;
166 RTSPINLOCKTMP Tmp;
167
168 /*
169 * Validation.
170 */
171 AssertPtrReturn(pfnCallback, VERR_INVALID_POINTER);
172 AssertReturn(g_hRTPowerNotifySpinLock != NIL_RTSPINLOCK, VERR_WRONG_ORDER);
173 RT_ASSERT_PREEMPTIBLE();
174
175 RTSpinlockAcquire(g_hRTPowerNotifySpinLock, &Tmp);
176 for (pCur = g_pRTPowerCallbackHead; pCur; pCur = pCur->pNext)
177 if ( pCur->pvUser == pvUser
178 && pCur->pfnCallback == pfnCallback)
179 break;
180 RTSpinlockRelease(g_hRTPowerNotifySpinLock, &Tmp);
181 AssertMsgReturn(!pCur, ("pCur=%p pfnCallback=%p pvUser=%p\n", pCur, pfnCallback, pvUser), VERR_ALREADY_EXISTS);
182
183 /*
184 * Allocate a new record and attempt to insert it.
185 */
186 pNew = (PRTPOWERNOTIFYREG)RTMemAlloc(sizeof(*pNew));
187 if (!pNew)
188 return VERR_NO_MEMORY;
189
190 pNew->pNext = NULL;
191 pNew->pfnCallback = pfnCallback;
192 pNew->pvUser = pvUser;
193 memset(&pNew->bmDone[0], 0xff, sizeof(pNew->bmDone));
194
195 RTSpinlockAcquire(g_hRTPowerNotifySpinLock, &Tmp);
196
197 pCur = g_pRTPowerCallbackHead;
198 if (!pCur)
199 g_pRTPowerCallbackHead = pNew;
200 else
201 {
202 for (pCur = g_pRTPowerCallbackHead; ; pCur = pCur->pNext)
203 if ( pCur->pvUser == pvUser
204 && pCur->pfnCallback == pfnCallback)
205 break;
206 else if (!pCur->pNext)
207 {
208 pCur->pNext = pNew;
209 pCur = NULL;
210 break;
211 }
212 }
213
214 ASMAtomicIncU32(&g_iRTPowerGeneration);
215
216 RTSpinlockRelease(g_hRTPowerNotifySpinLock, &Tmp);
217
218 /* duplicate? */
219 if (pCur)
220 {
221 RTMemFree(pCur);
222 AssertMsgFailedReturn(("pCur=%p pfnCallback=%p pvUser=%p\n", pCur, pfnCallback, pvUser), VERR_ALREADY_EXISTS);
223 }
224
225 return VINF_SUCCESS;
226}
227RT_EXPORT_SYMBOL(RTPowerNotificationRegister);
228
229
230RTDECL(int) RTPowerNotificationDeregister(PFNRTPOWERNOTIFICATION pfnCallback, void *pvUser)
231{
232 PRTPOWERNOTIFYREG pPrev;
233 PRTPOWERNOTIFYREG pCur;
234 RTSPINLOCKTMP Tmp;
235
236 /*
237 * Validation.
238 */
239 AssertPtrReturn(pfnCallback, VERR_INVALID_POINTER);
240 AssertReturn(g_hRTPowerNotifySpinLock != NIL_RTSPINLOCK, VERR_WRONG_ORDER);
241 RT_ASSERT_INTS_ON();
242
243 /*
244 * Find and unlink the record from the list.
245 */
246 RTSpinlockAcquire(g_hRTPowerNotifySpinLock, &Tmp);
247 pPrev = NULL;
248 for (pCur = g_pRTPowerCallbackHead; pCur; pCur = pCur->pNext)
249 {
250 if ( pCur->pvUser == pvUser
251 && pCur->pfnCallback == pfnCallback)
252 break;
253 pPrev = pCur;
254 }
255 if (pCur)
256 {
257 if (pPrev)
258 pPrev->pNext = pCur->pNext;
259 else
260 g_pRTPowerCallbackHead = pCur->pNext;
261 ASMAtomicIncU32(&g_iRTPowerGeneration);
262 }
263 RTSpinlockRelease(g_hRTPowerNotifySpinLock, &Tmp);
264
265 if (!pCur)
266 return VERR_NOT_FOUND;
267
268 /*
269 * Invalidate and free the record.
270 */
271 pCur->pNext = NULL;
272 pCur->pfnCallback = NULL;
273 RTMemFree(pCur);
274
275 return VINF_SUCCESS;
276}
277RT_EXPORT_SYMBOL(RTPowerNotificationDeregister);
278
279
280int rtR0PowerNotificationInit(void)
281{
282 int rc = VINF_SUCCESS;
283
284 if (ASMAtomicIncU32(&g_cRTPowerUsers) == 1)
285 {
286 rc = RTSpinlockCreate((PRTSPINLOCK)&g_hRTPowerNotifySpinLock);
287 if (RT_SUCCESS(rc))
288 {
289 /** @todo OS specific init here */
290 return rc;
291#if 0
292 RTSpinlockDestroy(g_hRTPowerNotifySpinLock);
293 g_hRTPowerNotifySpinLock = NIL_RTSPINLOCK;
294#endif
295 }
296 ASMAtomicDecU32(&g_cRTPowerUsers);
297 }
298 return rc;
299}
300
301
302void rtR0PowerNotificationTerm(void)
303{
304 RTSPINLOCK hSpinlock = g_hRTPowerNotifySpinLock;
305 if (hSpinlock != NIL_RTSPINLOCK)
306 {
307 AssertMsg(g_cRTPowerUsers > 0, ("%d\n", g_cRTPowerUsers));
308 if (ASMAtomicDecU32(&g_cRTPowerUsers) == 0)
309 {
310 PRTPOWERNOTIFYREG pHead;
311 RTSPINLOCKTMP Tmp;
312
313 /** @todo OS specific term here */
314
315 /* pick up the list and the spinlock. */
316 RTSpinlockAcquire(hSpinlock, &Tmp);
317 ASMAtomicWriteSize(&g_hRTPowerNotifySpinLock, NIL_RTSPINLOCK);
318 pHead = g_pRTPowerCallbackHead;
319 g_pRTPowerCallbackHead = NULL;
320 ASMAtomicIncU32(&g_iRTPowerGeneration);
321 RTSpinlockRelease(hSpinlock, &Tmp);
322
323 /* free the list. */
324 while (pHead)
325 {
326 PRTPOWERNOTIFYREG pFree = pHead;
327 pHead = pHead->pNext;
328
329 pFree->pNext = NULL;
330 pFree->pfnCallback = NULL;
331 RTMemFree(pFree);
332 }
333
334 RTSpinlockDestroy(hSpinlock);
335 }
336 }
337}
338
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