VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/semevent-r0drv-linux.c@ 7531

Last change on this file since 7531 was 7032, checked in by vboxsync, 17 years ago

Ring-0 semaphore debugging aid (linux).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 6.7 KB
Line 
1/* $Id: semevent-r0drv-linux.c 7032 2008-02-20 12:26:22Z vboxsync $ */
2/** @file
3 * innotek Portable Runtime - Single Release Event Semaphores, Ring-0 Driver, Linux.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek 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 (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 "the-linux-kernel.h"
32#include <iprt/semaphore.h>
33#include <iprt/alloc.h>
34#include <iprt/assert.h>
35#include <iprt/asm.h>
36#include <iprt/err.h>
37
38#include "internal/magics.h"
39
40
41/*******************************************************************************
42* Structures and Typedefs *
43*******************************************************************************/
44/**
45 * Linux event semaphore.
46 */
47typedef struct RTSEMEVENTINTERNAL
48{
49 /** Magic value (RTSEMEVENT_MAGIC). */
50 uint32_t volatile u32Magic;
51 /** The object status - !0 when signaled and 0 when reset. */
52 uint32_t volatile fState;
53 /** The wait queue. */
54 wait_queue_head_t Head;
55} RTSEMEVENTINTERNAL, *PRTSEMEVENTINTERNAL;
56
57
58
59RTDECL(int) RTSemEventCreate(PRTSEMEVENT pEventSem)
60{
61 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)RTMemAlloc(sizeof(*pEventInt));
62 if (pEventInt)
63 {
64 pEventInt->u32Magic = RTSEMEVENT_MAGIC;
65 init_waitqueue_head(&pEventInt->Head);
66 *pEventSem = pEventInt;
67 return VINF_SUCCESS;
68 }
69 return VERR_NO_MEMORY;
70}
71
72
73RTDECL(int) RTSemEventDestroy(RTSEMEVENT EventSem)
74{
75 /*
76 * Validate input.
77 */
78 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
79 if (!pEventInt)
80 return VERR_INVALID_PARAMETER;
81 if (pEventInt->u32Magic != RTSEMEVENT_MAGIC)
82 {
83 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt->u32Magic, pEventInt));
84 return VERR_INVALID_PARAMETER;
85 }
86
87 /*
88 * Invalidate it and signal the object just in case.
89 */
90 ASMAtomicIncU32(&pEventInt->u32Magic);
91 ASMAtomicXchgU32(&pEventInt->fState, 0);
92 Assert(!waitqueue_active(&pEventInt->Head));
93 wake_up_all(&pEventInt->Head);
94 RTMemFree(pEventInt);
95 return VINF_SUCCESS;
96}
97
98
99RTDECL(int) RTSemEventSignal(RTSEMEVENT EventSem)
100{
101 /*
102 * Validate input.
103 */
104 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
105 if (!pEventInt)
106 return VERR_INVALID_PARAMETER;
107 if ( !pEventInt
108 || pEventInt->u32Magic != RTSEMEVENT_MAGIC)
109 {
110 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt ? pEventInt->u32Magic : 0, pEventInt));
111 return VERR_INVALID_PARAMETER;
112 }
113
114 /*
115 * Signal the event object.
116 */
117 ASMAtomicXchgU32(&pEventInt->fState, 1);
118 wake_up(&pEventInt->Head);
119
120 return VINF_SUCCESS;
121}
122
123
124/**
125 * Worker for RTSemEvent and RTSemEventNoResume.
126 *
127 * @returns VBox status code.
128 * @param pEventInt The event semaphore.
129 * @param cMillies The number of milliseconds to wait.
130 * @param fInterruptible Whether it's an interruptible wait or not.
131 */
132static int rtSemEventWait(PRTSEMEVENTINTERNAL pEventInt, unsigned cMillies, bool fInterruptible)
133{
134 /*
135 * Ok wait for it.
136 */
137 DEFINE_WAIT(Wait);
138 int rc = VINF_SUCCESS;
139 long lTimeout = cMillies == RT_INDEFINITE_WAIT ? MAX_SCHEDULE_TIMEOUT : msecs_to_jiffies(cMillies);
140#ifdef IPRT_DEBUG_SEMS
141 snprintf(current->comm, TASK_COMM_LEN, "e%lx", IPRT_DEBUG_SEMS_ADDRESS(pEventInt));
142#endif
143 for (;;)
144 {
145 /* make everything thru schedule() atomic scheduling wise. */
146 prepare_to_wait(&pEventInt->Head, &Wait, fInterruptible ? TASK_INTERRUPTIBLE : TASK_UNINTERRUPTIBLE);
147
148 /* check the condition. */
149 if (ASMAtomicCmpXchgU32(&pEventInt->fState, 0, 1))
150 break;
151
152 /* check for pending signals. */
153 if (fInterruptible && signal_pending(current))
154 {
155 rc = VERR_INTERRUPTED;
156 break;
157 }
158
159 /* wait */
160 lTimeout = schedule_timeout(lTimeout);
161
162 /* Check if someone destroyed the semaphore while we were waiting. */
163 if (pEventInt->u32Magic != RTSEMEVENT_MAGIC)
164 {
165 rc = VERR_SEM_DESTROYED;
166 break;
167 }
168
169 /* check for timeout. */
170 if (!lTimeout)
171 {
172 rc = VERR_TIMEOUT;
173 break;
174 }
175 }
176
177 finish_wait(&pEventInt->Head, &Wait);
178#ifdef IPRT_DEBUG_SEMS
179 snprintf(current->comm, TASK_COMM_LEN, "e%lx:%d", IPRT_DEBUG_SEMS_ADDRESS(pEventInt), rc);
180#endif
181 return rc;
182}
183
184
185RTDECL(int) RTSemEventWait(RTSEMEVENT EventSem, unsigned cMillies)
186{
187 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
188 if (!pEventInt)
189 return VERR_INVALID_PARAMETER;
190 if ( !pEventInt
191 || pEventInt->u32Magic != RTSEMEVENT_MAGIC)
192 {
193 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt ? pEventInt->u32Magic : 0, pEventInt));
194 return VERR_INVALID_PARAMETER;
195 }
196
197 if (ASMAtomicCmpXchgU32(&pEventInt->fState, 0, 1))
198 return VINF_SUCCESS;
199 return rtSemEventWait(pEventInt, cMillies, false /* fInterruptible */);
200}
201
202
203RTDECL(int) RTSemEventWaitNoResume(RTSEMEVENT EventSem, unsigned cMillies)
204{
205 PRTSEMEVENTINTERNAL pEventInt = (PRTSEMEVENTINTERNAL)EventSem;
206 if (!pEventInt)
207 return VERR_INVALID_PARAMETER;
208 if ( !pEventInt
209 || pEventInt->u32Magic != RTSEMEVENT_MAGIC)
210 {
211 AssertMsgFailed(("pEventInt->u32Magic=%RX32 pEventInt=%p\n", pEventInt ? pEventInt->u32Magic : 0, pEventInt));
212 return VERR_INVALID_PARAMETER;
213 }
214
215 if (ASMAtomicCmpXchgU32(&pEventInt->fState, 0, 1))
216 return VINF_SUCCESS;
217 return rtSemEventWait(pEventInt, cMillies, true /* fInterruptible */);
218}
219
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