VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/semeventmulti-r0drv-linux.c@ 9563

Last change on this file since 9563 was 8245, checked in by vboxsync, 17 years ago

rebranding: IPRT files again.

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