VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/netbsd/sleepqueue-r0drv-netbsd.h@ 72110

Last change on this file since 72110 was 69111, checked in by vboxsync, 7 years ago

(C) year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.9 KB
Line 
1/* $Id: sleepqueue-r0drv-netbsd.h 69111 2017-10-17 14:26:02Z vboxsync $ */
2/** @file
3 * IPRT - NetBSD Ring-0 Driver Helpers for Abstracting Sleep Queues,
4 */
5
6/*
7 * Copyright (C) 2006-2017 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#ifndef ___r0drv_netbsd_sleepqueue_r0drv_netbsd_h
29#define ___r0drv_netbsd_sleepqueue_r0drv_netbsd_h
30
31#include "the-netbsd-kernel.h"
32
33#include <iprt/asm-math.h>
34#include <iprt/err.h>
35#include <iprt/time.h>
36
37static syncobj_t vbox_syncobj = {
38 SOBJ_SLEEPQ_FIFO,
39 sleepq_unsleep,
40 sleepq_changepri,
41 sleepq_lendpri,
42 syncobj_noowner,
43};
44
45/**
46 * Kernel mode Linux wait state structure.
47 */
48typedef struct RTR0SEMBSDSLEEP
49{
50 /** The absolute timeout given as nano seconds since the start of the
51 * monotonic clock. */
52 uint64_t uNsAbsTimeout;
53 /** The timeout in ticks. Updated after waiting. */
54 int iTimeout;
55 /** Set if it's an indefinite wait. */
56 bool fIndefinite;
57 /** Set if we've already timed out.
58 * Set by rtR0SemBsdWaitDoIt and read by rtR0SemBsdWaitHasTimedOut. */
59 bool fTimedOut;
60 /** Flag whether the wait was interrupted. */
61 bool fInterrupted;
62 /** flag whether the wait is interruptible or not. */
63 bool fInterruptible;
64 /** Opaque wait channel id. */
65 wchan_t wchan;
66 sleepq_t *sq;
67 kmutex_t *sq_lock;
68} RTR0SEMBSDSLEEP;
69/** Pointer to a NetBSD wait state. */
70typedef RTR0SEMBSDSLEEP *PRTR0SEMBSDSLEEP;
71
72
73/**
74 * Updates the timeout of the NetBSD wait.
75 *
76 * @returns RTSEMWAIT_FLAGS_INDEFINITE if the timeout value is too big.
77 * 0 otherwise
78 * @param pWait The wait structure.
79 * @param uTimeout The relative timeout in nanoseconds.
80 */
81DECLINLINE(void) rtR0SemBsdWaitUpdateTimeout(PRTR0SEMBSDSLEEP pWait)
82{
83 /* Convert absolute timeout to ticks */
84 uint64_t now = RTTimeNanoTS();
85 if (now >= pWait->uNsAbsTimeout) {
86 pWait->iTimeout = 0;
87 } else {
88 uint64_t nanos = pWait->uNsAbsTimeout - now;
89 pWait->iTimeout = hz * nanos / 1000000000;
90 /* for 1ms wait, wait at least one tick ?? */
91 if ((pWait->iTimeout == 0) && (nanos >= 1000000)) {
92 pWait->iTimeout = 1;
93 }
94 }
95}
96
97/**
98 * Initializes a wait.
99 *
100 * The caller MUST check the wait condition BEFORE calling this function or the
101 * timeout logic will be flawed.
102 *
103 * @returns VINF_SUCCESS or VERR_TIMEOUT.
104 * @param pWait The wait structure.
105 * @param fFlags The wait flags.
106 * @param uTimeout The timeout.
107 * @param pvWaitChan The opaque wait channel.
108 */
109DECLINLINE(int) rtR0SemBsdWaitInit(PRTR0SEMBSDSLEEP pWait, uint32_t fFlags, uint64_t uTimeout,
110 void *pvWaitChan)
111{
112 if (fFlags & RTSEMWAIT_FLAGS_INDEFINITE) {
113 pWait->fIndefinite = true;
114 pWait->iTimeout = 0;
115 pWait->uNsAbsTimeout = 0;
116 } else {
117 if (fFlags & RTSEMWAIT_FLAGS_RELATIVE) {
118 if (fFlags & RTSEMWAIT_FLAGS_MILLISECS) {
119 pWait->uNsAbsTimeout = uTimeout * 1000000 + RTTimeSystemNanoTS();
120 } else {
121 pWait->uNsAbsTimeout = uTimeout + RTTimeSystemNanoTS();
122 }
123 } else {
124 if (fFlags & RTSEMWAIT_FLAGS_MILLISECS) {
125 pWait->uNsAbsTimeout = uTimeout * 1000000;
126 } else {
127 pWait->uNsAbsTimeout = uTimeout;
128 }
129 }
130 rtR0SemBsdWaitUpdateTimeout(pWait);
131 if (pWait->iTimeout == 0) {
132 return VERR_TIMEOUT;
133 }
134 }
135
136 pWait->fTimedOut = false;
137 /*
138 * Initialize the wait queue related bits.
139 */
140 pWait->fInterruptible = fFlags & RTSEMWAIT_FLAGS_INTERRUPTIBLE
141 ? true : false;
142 pWait->fInterrupted = false;
143 pWait->wchan = pvWaitChan;
144 pWait->sq = NULL;
145 pWait->sq_lock = NULL;
146
147 return VINF_SUCCESS;
148}
149
150/**
151 * Prepares the next wait.
152 *
153 * This must be called before rtR0SemBsdWaitDoIt, and the caller should check
154 * the exit conditions inbetween the two calls.
155 *
156 * @param pWait The wait structure.
157 */
158DECLINLINE(void) rtR0SemBsdWaitPrepare(PRTR0SEMBSDSLEEP pWait)
159{
160 pWait->sq = sleeptab_lookup(&sleeptab, pWait->wchan, &pWait->sq_lock);
161}
162
163/**
164 * Do the actual wait.
165 *
166 * @param pWait The wait structure.
167 */
168DECLINLINE(void) rtR0SemBsdWaitDoIt(PRTR0SEMBSDSLEEP pWait)
169{
170 sleepq_enter(pWait->sq, curlwp, pWait->sq_lock);
171 sleepq_enqueue(pWait->sq, pWait->wchan, "VBoxIS", &vbox_syncobj);
172
173 pWait->sq = NULL;
174 pWait->sq_lock = NULL;
175
176 int error = sleepq_block(pWait->iTimeout, pWait->fInterruptible);
177 if (error == EWOULDBLOCK) {
178 if (!pWait->fIndefinite) {
179 pWait->fTimedOut = true;
180 }
181 } else if (error == ERESTART) {
182 if (pWait->fInterruptible) {
183 pWait->fInterrupted = true;
184 } else if (!pWait->fIndefinite) {
185 rtR0SemBsdWaitUpdateTimeout(pWait);
186 if (pWait->iTimeout == 0) {
187 pWait->fTimedOut = true;
188 }
189 }
190 } else if (error == EINTR) {
191 if (pWait->fInterruptible) {
192 pWait->fInterrupted = true;
193 } else if (!pWait->fIndefinite) {
194 rtR0SemBsdWaitUpdateTimeout(pWait);
195 if (pWait->iTimeout == 0) {
196 pWait->fTimedOut = true;
197 }
198 }
199 } else if (error) {
200 AssertMsgFailed(("sleepq_block -> %d\n", error));
201 }
202}
203
204
205/**
206 * Checks if a NetBSD wait was interrupted.
207 *
208 * @returns true / false
209 * @param pWait The wait structure.
210 * @remarks This shall be called before the first rtR0SemLnxWaitDoIt().
211 */
212DECLINLINE(bool) rtR0SemBsdWaitWasInterrupted(PRTR0SEMBSDSLEEP pWait)
213{
214 return pWait->fInterrupted;
215}
216
217
218/**
219 * Checks if a NetBSD wait has timed out.
220 *
221 * @returns true / false
222 * @param pWait The wait structure.
223 */
224DECLINLINE(bool) rtR0SemBsdWaitHasTimedOut(PRTR0SEMBSDSLEEP pWait)
225{
226 return pWait->fTimedOut;
227}
228
229
230/**
231 * Deletes a NetBSD wait.
232 *
233 * @param pWait The wait structure.
234 */
235DECLINLINE(void) rtR0SemBsdWaitDelete(PRTR0SEMBSDSLEEP pWait)
236{
237 if (pWait->sq_lock != NULL) {
238 mutex_spin_exit(pWait->sq_lock);
239 pWait->sq = NULL;
240 pWait->sq_lock = NULL;
241 }
242}
243
244
245/**
246 * Signals the wait channel.
247 *
248 * @param pvWaitChan The opaque wait channel handle.
249 */
250DECLINLINE(void) rtR0SemBsdSignal(void *pvWaitChan)
251{
252 kmutex_t *mp;
253 sleepq_t *sq = sleeptab_lookup(&sleeptab, pvWaitChan, &mp);
254 sleepq_wake(sq, pvWaitChan, 1, mp);
255}
256
257/**
258 * Wakes up all waiters on the wait channel.
259 *
260 * @param pvWaitChan The opaque wait channel handle.
261 */
262DECLINLINE(void) rtR0SemBsdBroadcast(void *pvWaitChan)
263{
264 kmutex_t *mp;
265 sleepq_t *sq = sleeptab_lookup(&sleeptab, pvWaitChan, &mp);
266 sleepq_wake(sq, pvWaitChan, ~0u, mp);
267}
268
269/**
270 * Gets the max resolution of the timeout machinery.
271 *
272 * @returns Resolution specified in nanoseconds.
273 */
274DECLINLINE(uint32_t) rtR0SemBsdWaitGetResolution(void)
275{
276 return 1000000000 / hz; /* ns */
277}
278
279#endif /* ___r0drv_netbsd_sleepqueue_r0drv_netbsd_h */
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