VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/semaphore-r0drv-linux.c@ 1507

Last change on this file since 1507 was 1, checked in by vboxsync, 55 years ago

import

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 12.6 KB
Line 
1/* $Id: semaphore-r0drv-linux.c 1 1970-01-01 00:00:00Z vboxsync $ */
2/** @file
3 * InnoTek Portable Runtime - Semaphores, Ring-0 Driver, Linux.
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#include "the-linux-kernel.h"
28#include <iprt/semaphore.h>
29#include <iprt/alloc.h>
30#include <iprt/assert.h>
31#include <iprt/asm.h>
32#include <iprt/err.h>
33
34
35/*******************************************************************************
36* Structures and Typedefs *
37*******************************************************************************/
38/**
39 * Linux event semaphore.
40 */
41typedef struct RTSEMEVENTINTERNAL
42{
43 /** Magic value (RTSEMEVENT_MAGIC). */
44 uint32_t volatile u32Magic;
45 /** The object status - !0 when signaled and 0 when reset. */
46 uint32_t volatile fState;
47 /** The wait queue. */
48 wait_queue_head_t Head;
49} RTSEMEVENTINTERNAL, *PRTSEMEVENTINTERNAL;
50
51/** Magic for the Linux event semaphore structure. (Neil Gaiman) */
52#define RTSEMEVENT_MAGIC 0x19601110
53
54
55/**
56 * Linux mutex semaphore.
57 */
58typedef struct RTSEMMUTEXINTERNAL
59{
60 /** Magic value (RTSEMMUTEX_MAGIC). */
61 uint32_t volatile u32Magic;
62 /** Number of recursive locks - 0 if not owned by anyone, > 0 if owned. */
63 uint32_t volatile cRecursion;
64 /** The wait queue. */
65 wait_queue_head_t Head;
66 /** The current owner. */
67 void * volatile pOwner;
68} RTSEMMUTEXINTERNAL, *PRTSEMMUTEXINTERNAL;
69
70/** Magic for the Linux mutex semaphore structure. (Douglas Adams) */
71#define RTSEMMUTEX_MAGIC 0x19520311
72
73
74/**
75 * Wrapper for the linux semaphore structure.
76 */
77typedef struct RTSEMFASTMUTEXINTERNAL
78{
79 /** Magic value (RTSEMFASTMUTEX_MAGIC). */
80 uint32_t u32Magic;
81 /** the linux semaphore. */
82 struct semaphore Semaphore;
83} RTSEMFASTMUTEXINTERNAL, *PRTSEMFASTMUTEXINTERNAL;
84
85/** Magic value for RTSEMFASTMUTEXINTERNAL::u32Magic (John Ronald Reuel Tolkien). */
86#define RTSEMFASTMUTEX_MAGIC 0x18920102
87
88
89
90
91RTDECL(int) RTSemEventCreate(PRTSEMEVENT pEventSem)
92{
93 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)RTMemAlloc(sizeof(*pEventInt));
94 if (pEventInt)
95 {
96 pEventInt->u32Magic = RTSEMEVENT_MAGIC;
97 init_waitqueue_head(&pEventInt->Head);
98 *pEventSem = pEventInt;
99 return VINF_SUCCESS;
100 }
101 return VERR_NO_MEMORY;
102}
103
104
105RTDECL(int) RTSemEventDestroy(RTSEMEVENT EventSem)
106{
107 /*
108 * Validate input.
109 */
110 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
111 if (!pEventInt)
112 return VERR_INVALID_PARAMETER;
113 if (pEventInt->u32Magic != RTSEMEVENT_MAGIC)
114 {
115 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt->u32Magic, pEventInt));
116 return VERR_INVALID_PARAMETER;
117 }
118
119 /*
120 * Invalidate it and signal the object just in case.
121 */
122 ASMAtomicIncU32(&pEventInt->u32Magic);
123 ASMAtomicXchgU32(&pEventInt->fState, 0);
124 Assert(!waitqueue_active(&pEventInt->Head));
125 wake_up_all(&pEventInt->Head);
126 RTMemFree(pEventInt);
127 return VINF_SUCCESS;
128}
129
130
131RTDECL(int) RTSemEventSignal(RTSEMEVENT EventSem)
132{
133 /*
134 * Validate input.
135 */
136 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
137 if (!pEventInt)
138 return VERR_INVALID_PARAMETER;
139 if ( !pEventInt
140 || pEventInt->u32Magic != RTSEMEVENT_MAGIC)
141 {
142 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt ? pEventInt->u32Magic : 0, pEventInt));
143 return VERR_INVALID_PARAMETER;
144 }
145
146 /*
147 * Signal the event object.
148 */
149 ASMAtomicXchgU32(&pEventInt->fState, 1);
150 wake_up(&pEventInt->Head);
151
152 return VINF_SUCCESS;
153}
154
155
156RTDECL(int) RTSemEventWait(RTSEMEVENT EventSem, unsigned cMillies)
157{
158 /*
159 * Validate input.
160 */
161 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
162 if (!pEventInt)
163 return VERR_INVALID_PARAMETER;
164 if ( !pEventInt
165 || pEventInt->u32Magic != RTSEMEVENT_MAGIC)
166 {
167 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt ? pEventInt->u32Magic : 0, pEventInt));
168 return VERR_INVALID_PARAMETER;
169 }
170
171 /*
172 * Try get it.
173 */
174 if (ASMAtomicCmpXchgU32(&pEventInt->fState, 0, 1))
175 return VINF_SUCCESS;
176 else
177 {
178 /*
179 * Ok wait for it.
180 */
181 DEFINE_WAIT(Wait);
182 int rc = VINF_SUCCESS;
183 long lTimeout = cMillies == RT_INDEFINITE_WAIT ? MAX_SCHEDULE_TIMEOUT : msecs_to_jiffies(cMillies);
184 for (;;)
185 {
186 /* make everything thru schedule() atomic scheduling wise. */
187 prepare_to_wait(&pEventInt->Head, &Wait, TASK_INTERRUPTIBLE);
188
189 /* check the condition. */
190 if (ASMAtomicCmpXchgU32(&pEventInt->fState, 0, 1))
191 break;
192
193 /* check for pending signals. */
194 if (signal_pending(current))
195 {
196 rc = VERR_INTERRUPTED; /** @todo VERR_INTERRUPTED isn't correct anylonger. please fix r0drv stuff! */
197 break;
198 }
199
200 /* wait */
201 lTimeout = schedule_timeout(lTimeout);
202
203 /* Check if someone destroyed the semaphore while we was waiting. */
204 if (pEventInt->u32Magic != RTSEMEVENT_MAGIC)
205 {
206 rc = VERR_SEM_DESTROYED;
207 break;
208 }
209
210 /* check for timeout. */
211 if (!lTimeout)
212 {
213 rc = VERR_TIMEOUT;
214 break;
215 }
216 }
217 finish_wait(&pEventInt->Head, &Wait);
218 return rc;
219 }
220}
221
222
223RTDECL(int) RTSemMutexCreate(PRTSEMMUTEX pMutexSem)
224{
225 PRTSEMMUTEXINTERNAL pMutexInt = (PRTSEMMUTEXINTERNAL)RTMemAlloc(sizeof(*pMutexInt));
226 if (pMutexInt)
227 {
228 pMutexInt->u32Magic = RTSEMMUTEX_MAGIC;
229 init_waitqueue_head(&pMutexInt->Head);
230 *pMutexSem = pMutexInt;
231AssertReleaseMsgFailed(("This mutex implementation is buggy, fix it!\n"));
232 return VINF_SUCCESS;
233 }
234 return VERR_NO_MEMORY;
235}
236
237
238RTDECL(int) RTSemMutexDestroy(RTSEMMUTEX MutexSem)
239{
240 /*
241 * Validate input.
242 */
243 PRTSEMMUTEXINTERNAL pMutexInt = (PRTSEMMUTEXINTERNAL)MutexSem;
244 if (!pMutexInt)
245 return VERR_INVALID_PARAMETER;
246 if (pMutexInt->u32Magic != RTSEMMUTEX_MAGIC)
247 {
248 AssertMsgFailed(("pMutexInt->u32Magic=%RX32 pMutexInt=%p\n", pMutexInt->u32Magic, pMutexInt));
249 return VERR_INVALID_PARAMETER;
250 }
251
252 /*
253 * Invalidate it and signal the object just in case.
254 */
255 ASMAtomicIncU32(&pMutexInt->u32Magic);
256 ASMAtomicXchgU32(&pMutexInt->cRecursion, 0);
257 Assert(!waitqueue_active(&pMutexInt->Head));
258 wake_up_all(&pMutexInt->Head);
259 RTMemFree(pMutexInt);
260 return VINF_SUCCESS;
261}
262
263
264RTDECL(int) RTSemMutexRequest(RTSEMMUTEX MutexSem, unsigned cMillies)
265{
266 /*
267 * Validate input.
268 */
269 PRTSEMMUTEXINTERNAL pMutexInt = (PRTSEMMUTEXINTERNAL)MutexSem;
270 if (!pMutexInt)
271 return VERR_INVALID_PARAMETER;
272 if ( !pMutexInt
273 || pMutexInt->u32Magic != RTSEMMUTEX_MAGIC)
274 {
275 AssertMsgFailed(("pMutexInt->u32Magic=%RX32 pMutexInt=%p\n", pMutexInt ? pMutexInt->u32Magic : 0, pMutexInt));
276 return VERR_INVALID_PARAMETER;
277 }
278
279 /*
280 * Check for recursive request.
281 */
282 if (pMutexInt->pOwner == current)
283 {
284 Assert(pMutexInt->cRecursion < 1000);
285 ASMAtomicIncU32(&pMutexInt->cRecursion);
286 return VINF_SUCCESS;
287 }
288
289 /*
290 * Try aquire it.
291 */
292 if (ASMAtomicCmpXchgU32(&pMutexInt->cRecursion, 1, 0))
293 {
294 ASMAtomicXchgPtr(&pMutexInt->pOwner, current);
295 return VINF_SUCCESS;
296 }
297 else
298 {
299 /*
300 * Ok wait for it.
301 */
302 DEFINE_WAIT(Wait);
303 int rc = VINF_SUCCESS;
304 long lTimeout = cMillies == RT_INDEFINITE_WAIT ? MAX_SCHEDULE_TIMEOUT : msecs_to_jiffies(cMillies);
305 for (;;)
306 {
307 /* make everything thru schedule() atomic scheduling wise. */
308 prepare_to_wait(&pMutexInt->Head, &Wait, TASK_INTERRUPTIBLE);
309
310 /* check the condition. */
311 if (ASMAtomicCmpXchgU32(&pMutexInt->cRecursion, 1, 0))
312 {
313 ASMAtomicXchgPtr(&pMutexInt->pOwner, current);
314 break;
315 }
316
317 /* check for pending signals. */
318 if (signal_pending(current))
319 {
320 rc = VERR_INTERRUPTED; /** @todo VERR_INTERRUPTED isn't correct anylonger. please fix r0drv stuff! */
321 break;
322 }
323
324 /* wait */
325 lTimeout = schedule_timeout(lTimeout);
326
327 /* Check if someone destroyed the semaphore while we was waiting. */
328 if (pMutexInt->u32Magic != RTSEMMUTEX_MAGIC)
329 {
330 rc = VERR_SEM_DESTROYED;
331 break;
332 }
333
334 /* check for timeout. */
335 if (!lTimeout)
336 {
337 rc = VERR_TIMEOUT;
338 break;
339 }
340 }
341 finish_wait(&pMutexInt->Head, &Wait);
342 return rc;
343 }
344 return VINF_SUCCESS;
345}
346
347
348RTDECL(int) RTSemMutexRelease(RTSEMMUTEX MutexSem)
349{
350 /*
351 * Validate input.
352 */
353 PRTSEMMUTEXINTERNAL pMutexInt = (PRTSEMMUTEXINTERNAL)MutexSem;
354 if (!pMutexInt)
355 return VERR_INVALID_PARAMETER;
356 if ( !pMutexInt
357 || pMutexInt->u32Magic != RTSEMMUTEX_MAGIC)
358 {
359 AssertMsgFailed(("pMutexInt->u32Magic=%RX32 pMutexInt=%p\n", pMutexInt ? pMutexInt->u32Magic : 0, pMutexInt));
360 return VERR_INVALID_PARAMETER;
361 }
362 if (pMutexInt->pOwner != current)
363 {
364 AssertMsgFailed(("Not owner, pOwner=%p current=%p\n", (void *)pMutexInt->pOwner, (void *)current));
365 return VERR_NOT_OWNER;
366 }
367
368 /*
369 * Release the mutex.
370 */
371 if (pMutexInt->cRecursion == 1)
372 {
373 ASMAtomicXchgPtr(&pMutexInt->pOwner, NULL);
374 ASMAtomicXchgU32(&pMutexInt->cRecursion, 0);
375 }
376 else
377 ASMAtomicDecU32(&pMutexInt->cRecursion);
378
379 return VINF_SUCCESS;
380}
381
382
383
384RTDECL(int) RTSemFastMutexCreate(PRTSEMFASTMUTEX pMutexSem)
385{
386 /*
387 * Allocate.
388 */
389 PRTSEMFASTMUTEXINTERNAL pFastInt;
390 pFastInt = (PRTSEMFASTMUTEXINTERNAL)RTMemAlloc(sizeof(*pFastInt));
391 if (!pFastInt)
392 return VERR_NO_MEMORY;
393
394 /*
395 * Initialize.
396 */
397 pFastInt->u32Magic = RTSEMFASTMUTEX_MAGIC;
398 sema_init(&pFastInt->Semaphore, 1);
399 *pMutexSem = pFastInt;
400 return VINF_SUCCESS;
401}
402
403
404RTDECL(int) RTSemFastMutexDestroy(RTSEMFASTMUTEX MutexSem)
405{
406 /*
407 * Validate.
408 */
409 PRTSEMFASTMUTEXINTERNAL pFastInt = (PRTSEMFASTMUTEXINTERNAL)MutexSem;
410 if (!pFastInt)
411 return VERR_INVALID_PARAMETER;
412 if (pFastInt->u32Magic != RTSEMFASTMUTEX_MAGIC)
413 {
414 AssertMsgFailed(("pFastInt->u32Magic=%RX32 pMutexInt=%p\n", pFastInt->u32Magic, pFastInt));
415 return VERR_INVALID_PARAMETER;
416 }
417
418 ASMAtomicIncU32(&pFastInt->u32Magic);
419 RTMemFree(pFastInt);
420 return VINF_SUCCESS;
421}
422
423
424RTDECL(int) RTSemFastMutexRequest(RTSEMFASTMUTEX MutexSem)
425{
426 /*
427 * Validate.
428 */
429 PRTSEMFASTMUTEXINTERNAL pFastInt = (PRTSEMFASTMUTEXINTERNAL)MutexSem;
430 if ( !pFastInt
431 || pFastInt->u32Magic != RTSEMFASTMUTEX_MAGIC)
432 {
433 AssertMsgFailed(("pFastInt->u32Magic=%RX32 pMutexInt=%p\n", pFastInt ? pFastInt->u32Magic : 0, pFastInt));
434 return VERR_INVALID_PARAMETER;
435 }
436
437 down(&pFastInt->Semaphore);
438 return VINF_SUCCESS;
439}
440
441
442RTDECL(int) RTSemFastMutexRelease(RTSEMFASTMUTEX MutexSem)
443{
444 /*
445 * Validate.
446 */
447 PRTSEMFASTMUTEXINTERNAL pFastInt = (PRTSEMFASTMUTEXINTERNAL)MutexSem;
448 if ( !pFastInt
449 || pFastInt->u32Magic != RTSEMFASTMUTEX_MAGIC)
450 {
451 AssertMsgFailed(("pFastInt->u32Magic=%RX32 pMutexInt=%p\n", pFastInt ? pFastInt->u32Magic : 0, pFastInt));
452 return VERR_INVALID_PARAMETER;
453 }
454
455 up(&pFastInt->Semaphore);
456 return VINF_SUCCESS;
457}
458
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