VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/PDMAllCritSectRw.cpp@ 91247

Last change on this file since 91247 was 90940, checked in by vboxsync, 3 years ago

VMM/PDMAllCritSectRw: Removed blank line. bugref:6695

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 83.9 KB
Line 
1/* $Id: PDMAllCritSectRw.cpp 90940 2021-08-27 09:33:21Z vboxsync $ */
2/** @file
3 * IPRT - Read/Write Critical Section, Generic.
4 */
5
6/*
7 * Copyright (C) 2009-2020 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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_PDM_CRITSECTRW
23#include "PDMInternal.h"
24#include <VBox/vmm/pdmcritsectrw.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/vmm.h>
27#include <VBox/vmm/vmcc.h>
28#include <VBox/err.h>
29#include <VBox/vmm/hm.h>
30
31#include <VBox/log.h>
32#include <iprt/asm.h>
33#include <iprt/asm-amd64-x86.h>
34#include <iprt/assert.h>
35#ifdef IN_RING3
36# include <iprt/lockvalidator.h>
37#endif
38#if defined(IN_RING3) || defined(IN_RING0)
39# include <iprt/semaphore.h>
40# include <iprt/thread.h>
41#endif
42#ifdef IN_RING0
43# include <iprt/time.h>
44#endif
45#ifdef RT_ARCH_AMD64
46# include <iprt/x86.h>
47#endif
48
49
50/*********************************************************************************************************************************
51* Defined Constants And Macros *
52*********************************************************************************************************************************/
53#if 0 /* unused */
54/** The number loops to spin for shared access in ring-3. */
55#define PDMCRITSECTRW_SHRD_SPIN_COUNT_R3 20
56/** The number loops to spin for shared access in ring-0. */
57#define PDMCRITSECTRW_SHRD_SPIN_COUNT_R0 128
58/** The number loops to spin for shared access in the raw-mode context. */
59#define PDMCRITSECTRW_SHRD_SPIN_COUNT_RC 128
60
61/** The number loops to spin for exclusive access in ring-3. */
62#define PDMCRITSECTRW_EXCL_SPIN_COUNT_R3 20
63/** The number loops to spin for exclusive access in ring-0. */
64#define PDMCRITSECTRW_EXCL_SPIN_COUNT_R0 256
65/** The number loops to spin for exclusive access in the raw-mode context. */
66#define PDMCRITSECTRW_EXCL_SPIN_COUNT_RC 256
67#endif
68
69/** Max number of write or write/read recursions. */
70#define PDM_CRITSECTRW_MAX_RECURSIONS _1M
71
72/** Skips some of the overly paranoid atomic reads and updates.
73 * Makes some assumptions about cache coherence, though not brave enough not to
74 * always end with an atomic update. */
75#define PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
76
77/** For reading RTCRITSECTRWSTATE::s::u64State. */
78#ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
79# define PDMCRITSECTRW_READ_STATE(a_pu64State) ASMAtomicUoReadU64(a_pu64State)
80#else
81# define PDMCRITSECTRW_READ_STATE(a_pu64State) ASMAtomicReadU64(a_pu64State)
82#endif
83
84
85/* Undefine the automatic VBOX_STRICT API mappings. */
86#undef PDMCritSectRwEnterExcl
87#undef PDMCritSectRwTryEnterExcl
88#undef PDMCritSectRwEnterShared
89#undef PDMCritSectRwTryEnterShared
90
91
92/*********************************************************************************************************************************
93* Defined Constants And Macros *
94*********************************************************************************************************************************/
95#if defined(RTASM_HAVE_CMP_WRITE_U128) && defined(RT_ARCH_AMD64)
96static int32_t g_fCmpWriteSupported = -1;
97#endif
98
99
100/*********************************************************************************************************************************
101* Internal Functions *
102*********************************************************************************************************************************/
103static int pdmCritSectRwLeaveSharedWorker(PVMCC pVM, PPDMCRITSECTRW pThis, bool fNoVal);
104
105
106#ifdef RTASM_HAVE_CMP_WRITE_U128
107
108# ifdef RT_ARCH_AMD64
109/**
110 * Called once to initialize g_fCmpWriteSupported.
111 */
112DECL_NO_INLINE(static, bool) pdmCritSectRwIsCmpWriteU128SupportedSlow(void)
113{
114 bool const fCmpWriteSupported = RT_BOOL(ASMCpuId_ECX(1) & X86_CPUID_FEATURE_ECX_CX16);
115 ASMAtomicWriteS32(&g_fCmpWriteSupported, fCmpWriteSupported);
116 return fCmpWriteSupported;
117}
118# endif
119
120
121/**
122 * Indicates whether hardware actually supports 128-bit compare & write.
123 */
124DECL_FORCE_INLINE(bool) pdmCritSectRwIsCmpWriteU128Supported(void)
125{
126# ifdef RT_ARCH_AMD64
127 int32_t const fCmpWriteSupported = g_fCmpWriteSupported;
128 if (RT_LIKELY(fCmpWriteSupported >= 0))
129 return fCmpWriteSupported != 0;
130 return pdmCritSectRwIsCmpWriteU128SupportedSlow();
131# else
132 return true;
133# endif
134}
135
136#endif /* RTASM_HAVE_CMP_WRITE_U128 */
137
138/**
139 * Gets the ring-3 native thread handle of the calling thread.
140 *
141 * @returns native thread handle (ring-3).
142 * @param pVM The cross context VM structure.
143 * @param pThis The read/write critical section. This is only used in
144 * R0 and RC.
145 */
146DECL_FORCE_INLINE(RTNATIVETHREAD) pdmCritSectRwGetNativeSelf(PVMCC pVM, PCPDMCRITSECTRW pThis)
147{
148#ifdef IN_RING3
149 RT_NOREF(pVM, pThis);
150 RTNATIVETHREAD hNativeSelf = RTThreadNativeSelf();
151#else
152 AssertMsgReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, ("%RX32\n", pThis->s.Core.u32Magic),
153 NIL_RTNATIVETHREAD);
154 PVMCPUCC pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
155 RTNATIVETHREAD hNativeSelf = pVCpu ? pVCpu->hNativeThread : NIL_RTNATIVETHREAD;
156 Assert(hNativeSelf != NIL_RTNATIVETHREAD);
157#endif
158 return hNativeSelf;
159}
160
161
162DECL_NO_INLINE(static, int) pdmCritSectRwCorrupted(PPDMCRITSECTRW pThis, const char *pszMsg)
163{
164 ASMAtomicWriteU32(&pThis->s.Core.u32Magic, PDMCRITSECTRW_MAGIC_CORRUPT);
165 LogRel(("PDMCritSect: %s pCritSect=%p\n", pszMsg, pThis));
166 return VERR_PDM_CRITSECTRW_IPE;
167}
168
169
170
171#ifdef IN_RING3
172/**
173 * Changes the lock validator sub-class of the read/write critical section.
174 *
175 * It is recommended to try make sure that nobody is using this critical section
176 * while changing the value.
177 *
178 * @returns The old sub-class. RTLOCKVAL_SUB_CLASS_INVALID is returns if the
179 * lock validator isn't compiled in or either of the parameters are
180 * invalid.
181 * @param pThis Pointer to the read/write critical section.
182 * @param uSubClass The new sub-class value.
183 */
184VMMDECL(uint32_t) PDMR3CritSectRwSetSubClass(PPDMCRITSECTRW pThis, uint32_t uSubClass)
185{
186 AssertPtrReturn(pThis, RTLOCKVAL_SUB_CLASS_INVALID);
187 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, RTLOCKVAL_SUB_CLASS_INVALID);
188# if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
189 AssertReturn(!(pThis->s.Core.fFlags & RTCRITSECT_FLAGS_NOP), RTLOCKVAL_SUB_CLASS_INVALID);
190
191 RTLockValidatorRecSharedSetSubClass(pThis->s.Core.pValidatorRead, uSubClass);
192 return RTLockValidatorRecExclSetSubClass(pThis->s.Core.pValidatorWrite, uSubClass);
193# else
194 NOREF(uSubClass);
195 return RTLOCKVAL_SUB_CLASS_INVALID;
196# endif
197}
198#endif /* IN_RING3 */
199
200
201/**
202 * Worker for pdmCritSectRwEnterShared returning with read-ownership of the CS.
203 */
204DECL_FORCE_INLINE(int) pdmCritSectRwEnterSharedGotIt(PPDMCRITSECTRW pThis, PCRTLOCKVALSRCPOS pSrcPos,
205 bool fNoVal, RTTHREAD hThreadSelf)
206{
207#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
208 if (!fNoVal)
209 RTLockValidatorRecSharedAddOwner(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos);
210#else
211 RT_NOREF(pSrcPos, fNoVal, hThreadSelf);
212#endif
213
214 /* got it! */
215 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(Stat,EnterShared));
216 Assert((PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State) & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT));
217 return VINF_SUCCESS;
218}
219
220/**
221 * Worker for pdmCritSectRwEnterShared and pdmCritSectRwEnterSharedBailOut
222 * that decrement the wait count and maybe resets the semaphore.
223 */
224DECLINLINE(int) pdmCritSectRwEnterSharedGotItAfterWaiting(PVMCC pVM, PPDMCRITSECTRW pThis, uint64_t u64State,
225 PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal, RTTHREAD hThreadSelf)
226{
227 for (;;)
228 {
229 uint64_t const u64OldState = u64State;
230 uint64_t cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
231 AssertReturn(cWait > 0, pdmCritSectRwCorrupted(pThis, "Invalid waiting read count"));
232 AssertReturn((u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT > 0,
233 pdmCritSectRwCorrupted(pThis, "Invalid read count"));
234 cWait--;
235 u64State &= ~RTCSRW_WAIT_CNT_RD_MASK;
236 u64State |= cWait << RTCSRW_WAIT_CNT_RD_SHIFT;
237
238 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
239 {
240 if (cWait == 0)
241 {
242 if (ASMAtomicXchgBool(&pThis->s.Core.fNeedReset, false))
243 {
244 int rc = SUPSemEventMultiReset(pVM->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
245 AssertRCReturn(rc, rc);
246 }
247 }
248 return pdmCritSectRwEnterSharedGotIt(pThis, pSrcPos, fNoVal, hThreadSelf);
249 }
250
251 ASMNopPause();
252 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
253 ASMNopPause();
254
255 u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
256 }
257 /* not reached */
258}
259
260
261#if defined(IN_RING0) || (defined(IN_RING3) && defined(PDMCRITSECTRW_STRICT))
262/**
263 * Worker for pdmCritSectRwEnterSharedContended that decrements both read counts
264 * and returns @a rc.
265 *
266 * @note May return VINF_SUCCESS if we race the exclusive leave function and
267 * come out on the bottom.
268 *
269 * Ring-3 only calls in a case where it is _not_ acceptable to take the
270 * lock, so even if we get the lock we'll have to leave. In the ring-0
271 * contexts, we can safely return VINF_SUCCESS in case of a race.
272 */
273DECL_NO_INLINE(static, int) pdmCritSectRwEnterSharedBailOut(PVMCC pVM, PPDMCRITSECTRW pThis, int rc,
274 PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal, RTTHREAD hThreadSelf)
275{
276#ifdef IN_RING0
277 uint64_t const tsStart = RTTimeNanoTS();
278 uint64_t cNsElapsed = 0;
279#endif
280 for (;;)
281 {
282 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
283 uint64_t u64OldState = u64State;
284
285 uint64_t cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
286 AssertReturn(cWait > 0, pdmCritSectRwCorrupted(pThis, "Invalid waiting read count on bailout"));
287 cWait--;
288
289 uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
290 AssertReturn(c > 0, pdmCritSectRwCorrupted(pThis, "Invalid read count on bailout"));
291
292 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT))
293 {
294 c--;
295 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
296 u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
297 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
298 return rc;
299 }
300 else
301 {
302 /*
303 * The direction changed, so we can actually get the lock now.
304 *
305 * This means that we _have_ to wait on the semaphore to be signalled
306 * so we can properly reset it. Otherwise the stuff gets out of wack,
307 * because signalling and resetting will race one another. An
308 * exception would be if we're not the last reader waiting and don't
309 * need to worry about the resetting.
310 *
311 * An option would be to do the resetting in PDMCritSectRwEnterExcl,
312 * but that would still leave a racing PDMCritSectRwEnterShared
313 * spinning hard for a little bit, which isn't great...
314 */
315 if (cWait == 0)
316 {
317# ifdef IN_RING0
318 /* Do timeout processing first to avoid redoing the above. */
319 uint32_t cMsWait;
320 if (cNsElapsed <= RT_NS_10SEC)
321 cMsWait = 32;
322 else
323 {
324 u64State &= ~RTCSRW_WAIT_CNT_RD_MASK;
325 u64State |= cWait << RTCSRW_WAIT_CNT_RD_SHIFT;
326 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
327 {
328 LogFunc(("%p: giving up\n", pThis));
329 return rc;
330 }
331 cMsWait = 2;
332 }
333
334 int rcWait = SUPSemEventMultiWait(pVM->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead, cMsWait);
335 Log11Func(("%p: rc=%Rrc %'RU64 ns (hNativeWriter=%p u64State=%#RX64)\n", pThis, rcWait,
336 RTTimeNanoTS() - tsStart, pThis->s.Core.u.s.hNativeWriter, pThis->s.Core.u.s.u64State));
337# else
338 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_READ, false);
339 int rcWait = SUPSemEventMultiWaitNoResume(pVM->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead, RT_MS_5SEC);
340 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
341# endif
342 if (rcWait == VINF_SUCCESS)
343 {
344# ifdef IN_RING0
345 return pdmCritSectRwEnterSharedGotItAfterWaiting(pVM, pThis, u64State, pSrcPos, fNoVal, hThreadSelf);
346# else
347 /* ring-3: Cannot return VINF_SUCCESS. */
348 Assert(RT_FAILURE_NP(rc));
349 int rc2 = pdmCritSectRwEnterSharedGotItAfterWaiting(pVM, pThis, u64State, pSrcPos, fNoVal, hThreadSelf);
350 if (RT_SUCCESS(rc2))
351 rc2 = pdmCritSectRwLeaveSharedWorker(pVM, pThis, fNoVal);
352 return rc;
353# endif
354 }
355 AssertMsgReturn(rcWait == VERR_TIMEOUT || rcWait == VERR_INTERRUPTED,
356 ("%p: rcWait=%Rrc rc=%Rrc", pThis, rcWait, rc),
357 RT_FAILURE_NP(rcWait) ? rcWait : -rcWait);
358 }
359 else
360 {
361 u64State &= ~RTCSRW_WAIT_CNT_RD_MASK;
362 u64State |= cWait << RTCSRW_WAIT_CNT_RD_SHIFT;
363 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
364 return pdmCritSectRwEnterSharedGotIt(pThis, pSrcPos, fNoVal, hThreadSelf);
365 }
366
367# ifdef IN_RING0
368 /* Calculate the elapsed time here to avoid redoing state work. */
369 cNsElapsed = RTTimeNanoTS() - tsStart;
370# endif
371 }
372
373 ASMNopPause();
374 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
375 ASMNopPause();
376 }
377}
378#endif /* IN_RING0 || (IN_RING3 && PDMCRITSECTRW_STRICT) */
379
380
381/**
382 * Worker for pdmCritSectRwEnterShared that handles waiting for a contended CS.
383 * Caller has already added us to the read and read-wait counters.
384 */
385static int pdmCritSectRwEnterSharedContended(PVMCC pVM, PVMCPUCC pVCpu, PPDMCRITSECTRW pThis,
386 int rcBusy, PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal, RTTHREAD hThreadSelf)
387{
388 PSUPDRVSESSION const pSession = pVM->pSession;
389 SUPSEMEVENTMULTI const hEventMulti = (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead;
390# ifdef IN_RING0
391 uint64_t const tsStart = RTTimeNanoTS();
392 uint64_t const cNsMaxTotalDef = RT_NS_5MIN;
393 uint64_t cNsMaxTotal = cNsMaxTotalDef;
394 uint32_t cMsMaxOne = RT_MS_5SEC;
395 bool fNonInterruptible = false;
396# endif
397
398 for (uint32_t iLoop = 0; ; iLoop++)
399 {
400 /*
401 * Wait for the direction to switch.
402 */
403 int rc;
404# ifdef IN_RING3
405# if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
406 rc = RTLockValidatorRecSharedCheckBlocking(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos, true,
407 RT_INDEFINITE_WAIT, RTTHREADSTATE_RW_READ, false);
408 if (RT_FAILURE(rc))
409 return pdmCritSectRwEnterSharedBailOut(pVM, pThis, rc, pSrcPos, fNoVal, hThreadSelf);
410# else
411 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_READ, false);
412# endif
413# endif
414
415 for (;;)
416 {
417 /*
418 * We always wait with a timeout so we can re-check the structure sanity
419 * and not get stuck waiting on a corrupt or deleted section.
420 */
421# ifdef IN_RING3
422 rc = SUPSemEventMultiWaitNoResume(pSession, hEventMulti, RT_MS_5SEC);
423# else
424 rc = !fNonInterruptible
425 ? SUPSemEventMultiWaitNoResume(pSession, hEventMulti, cMsMaxOne)
426 : SUPSemEventMultiWait(pSession, hEventMulti, cMsMaxOne);
427 Log11Func(("%p: rc=%Rrc %'RU64 ns (cMsMaxOne=%RU64 hNativeWriter=%p u64State=%#RX64)\n", pThis, rc,
428 RTTimeNanoTS() - tsStart, cMsMaxOne, pThis->s.Core.u.s.hNativeWriter, pThis->s.Core.u.s.u64State));
429# endif
430 if (RT_LIKELY(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC))
431 { /* likely */ }
432 else
433 {
434# ifdef IN_RING3
435 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
436# endif
437 return VERR_SEM_DESTROYED;
438 }
439 if (RT_LIKELY(rc == VINF_SUCCESS))
440 break;
441
442 /*
443 * Timeout and interrupted waits needs careful handling in ring-0
444 * because we're cooperating with ring-3 on this critical section
445 * and thus need to make absolutely sure we won't get stuck here.
446 *
447 * The r0 interrupted case means something is pending (termination,
448 * signal, APC, debugger, whatever), so we must try our best to
449 * return to the caller and to ring-3 so it can be dealt with.
450 */
451 if (rc == VERR_TIMEOUT || rc == VERR_INTERRUPTED)
452 {
453# ifdef IN_RING0
454 uint64_t const cNsElapsed = RTTimeNanoTS() - tsStart;
455 int const rcTerm = RTThreadQueryTerminationStatus(NIL_RTTHREAD);
456 AssertMsg(rcTerm == VINF_SUCCESS || rcTerm == VERR_NOT_SUPPORTED || rcTerm == VINF_THREAD_IS_TERMINATING,
457 ("rcTerm=%Rrc\n", rcTerm));
458 if (rcTerm == VERR_NOT_SUPPORTED && cNsMaxTotal == cNsMaxTotalDef)
459 cNsMaxTotal = RT_NS_1MIN;
460
461 if (rc == VERR_TIMEOUT)
462 {
463 /* Try return get out of here with a non-VINF_SUCCESS status if
464 the thread is terminating or if the timeout has been exceeded. */
465 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwSharedVerrTimeout);
466 if ( rcTerm == VINF_THREAD_IS_TERMINATING
467 || cNsElapsed > cNsMaxTotal)
468 return pdmCritSectRwEnterSharedBailOut(pVM, pThis, rcBusy != VINF_SUCCESS ? rcBusy : rc,
469 pSrcPos, fNoVal, hThreadSelf);
470 }
471 else
472 {
473 /* For interrupt cases, we must return if we can. If rcBusy is VINF_SUCCESS,
474 we will try non-interruptible sleep for a while to help resolve the issue
475 w/o guru'ing. */
476 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwSharedVerrInterrupted);
477 if ( rcTerm != VINF_THREAD_IS_TERMINATING
478 && rcBusy == VINF_SUCCESS
479 && pVCpu != NULL
480 && cNsElapsed <= cNsMaxTotal)
481 {
482 if (!fNonInterruptible)
483 {
484 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwSharedNonInterruptibleWaits);
485 fNonInterruptible = true;
486 cMsMaxOne = 32;
487 uint64_t cNsLeft = cNsMaxTotal - cNsElapsed;
488 if (cNsLeft > RT_NS_10SEC)
489 cNsMaxTotal = cNsElapsed + RT_NS_10SEC;
490 }
491 }
492 else
493 return pdmCritSectRwEnterSharedBailOut(pVM, pThis, rcBusy != VINF_SUCCESS ? rcBusy : rc,
494 pSrcPos, fNoVal, hThreadSelf);
495 }
496# else /* IN_RING3 */
497 RT_NOREF(pVM, pVCpu, rcBusy);
498# endif /* IN_RING3 */
499 }
500 /*
501 * Any other return code is fatal.
502 */
503 else
504 {
505# ifdef IN_RING3
506 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
507# endif
508 AssertMsgFailed(("rc=%Rrc\n", rc));
509 return RT_FAILURE_NP(rc) ? rc : -rc;
510 }
511 }
512
513# ifdef IN_RING3
514 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_READ);
515# endif
516
517 /*
518 * Check the direction.
519 */
520 Assert(pThis->s.Core.fNeedReset);
521 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
522 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
523 {
524 /*
525 * Decrement the wait count and maybe reset the semaphore (if we're last).
526 */
527 return pdmCritSectRwEnterSharedGotItAfterWaiting(pVM, pThis, u64State, pSrcPos, fNoVal, hThreadSelf);
528 }
529
530 AssertMsg(iLoop < 1,
531 ("%p: %u u64State=%#RX64 hNativeWriter=%p\n", pThis, iLoop, u64State, pThis->s.Core.u.s.hNativeWriter));
532 RTThreadYield();
533 }
534
535 /* not reached */
536}
537
538
539/**
540 * Worker that enters a read/write critical section with shard access.
541 *
542 * @returns VBox status code.
543 * @param pVM The cross context VM structure.
544 * @param pThis Pointer to the read/write critical section.
545 * @param rcBusy The busy return code for ring-0 and ring-3.
546 * @param fTryOnly Only try enter it, don't wait.
547 * @param pSrcPos The source position. (Can be NULL.)
548 * @param fNoVal No validation records.
549 */
550static int pdmCritSectRwEnterShared(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy, bool fTryOnly,
551 PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal)
552{
553 /*
554 * Validate input.
555 */
556 AssertPtr(pThis);
557 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
558
559#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
560 RTTHREAD hThreadSelf = RTThreadSelfAutoAdopt();
561 if (!fTryOnly)
562 {
563 int rc9;
564 RTNATIVETHREAD hNativeWriter;
565 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
566 if (hNativeWriter != NIL_RTTHREAD && hNativeWriter == pdmCritSectRwGetNativeSelf(pVM, pThis))
567 rc9 = RTLockValidatorRecExclCheckOrder(pThis->s.Core.pValidatorWrite, hThreadSelf, pSrcPos, RT_INDEFINITE_WAIT);
568 else
569 rc9 = RTLockValidatorRecSharedCheckOrder(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos, RT_INDEFINITE_WAIT);
570 if (RT_FAILURE(rc9))
571 return rc9;
572 }
573#else
574 RTTHREAD hThreadSelf = NIL_RTTHREAD;
575#endif
576
577 /*
578 * Work the state.
579 */
580 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
581 uint64_t u64OldState = u64State;
582 for (;;)
583 {
584 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
585 {
586 /* It flows in the right direction, try follow it before it changes. */
587 uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
588 c++;
589 Assert(c < RTCSRW_CNT_MASK / 4);
590 AssertReturn(c < RTCSRW_CNT_MASK, VERR_PDM_CRITSECTRW_TOO_MANY_READERS);
591 u64State &= ~RTCSRW_CNT_RD_MASK;
592 u64State |= c << RTCSRW_CNT_RD_SHIFT;
593 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
594 return pdmCritSectRwEnterSharedGotIt(pThis, pSrcPos, fNoVal, hThreadSelf);
595 }
596 else if ((u64State & (RTCSRW_CNT_RD_MASK | RTCSRW_CNT_WR_MASK)) == 0)
597 {
598 /* Wrong direction, but we're alone here and can simply try switch the direction. */
599 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_CNT_WR_MASK | RTCSRW_DIR_MASK);
600 u64State |= (UINT64_C(1) << RTCSRW_CNT_RD_SHIFT) | (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT);
601 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
602 {
603 Assert(!pThis->s.Core.fNeedReset);
604 return pdmCritSectRwEnterSharedGotIt(pThis, pSrcPos, fNoVal, hThreadSelf);
605 }
606 }
607 else
608 {
609 /* Is the writer perhaps doing a read recursion? */
610 RTNATIVETHREAD hNativeWriter;
611 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
612 if (hNativeWriter != NIL_RTNATIVETHREAD)
613 {
614 RTNATIVETHREAD hNativeSelf = pdmCritSectRwGetNativeSelf(pVM, pThis);
615 if (hNativeSelf == hNativeWriter)
616 {
617#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
618 if (!fNoVal)
619 {
620 int rc9 = RTLockValidatorRecExclRecursionMixed(pThis->s.Core.pValidatorWrite, &pThis->s.Core.pValidatorRead->Core, pSrcPos);
621 if (RT_FAILURE(rc9))
622 return rc9;
623 }
624#endif
625 uint32_t const cReads = ASMAtomicIncU32(&pThis->s.Core.cWriterReads);
626 Assert(cReads < _16K);
627 AssertReturnStmt(cReads < PDM_CRITSECTRW_MAX_RECURSIONS, ASMAtomicDecU32(&pThis->s.Core.cWriterReads),
628 VERR_PDM_CRITSECTRW_TOO_MANY_RECURSIONS);
629 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(Stat,EnterShared));
630 return VINF_SUCCESS; /* don't break! */
631 }
632 }
633
634 /*
635 * If we're only trying, return already.
636 */
637 if (fTryOnly)
638 {
639 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterShared));
640 return VERR_SEM_BUSY;
641 }
642
643#if defined(IN_RING3) || defined(IN_RING0)
644 /*
645 * Add ourselves to the queue and wait for the direction to change.
646 */
647 uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
648 c++;
649 Assert(c < RTCSRW_CNT_MASK / 2);
650 AssertReturn(c < RTCSRW_CNT_MASK, VERR_PDM_CRITSECTRW_TOO_MANY_READERS);
651
652 uint64_t cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
653 cWait++;
654 Assert(cWait <= c);
655 Assert(cWait < RTCSRW_CNT_MASK / 2);
656 AssertReturn(cWait < RTCSRW_CNT_MASK, VERR_PDM_CRITSECTRW_TOO_MANY_READERS);
657
658 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
659 u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
660
661 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
662 {
663 /*
664 * In ring-3 it's straight forward, just optimize the RTThreadSelf() call.
665 */
666# if defined(IN_RING3) && defined(PDMCRITSECTRW_STRICT)
667 return pdmCritSectRwEnterSharedContended(pVM, NULL, pThis, rcBusy, pSrcPos, fNoVal, hThreadSelf);
668# elif defined(IN_RING3)
669 return pdmCritSectRwEnterSharedContended(pVM, NULL, pThis, rcBusy, pSrcPos, fNoVal, RTThreadSelf());
670# else /* IN_RING0 */
671 /*
672 * In ring-0 context we have to take the special VT-x/AMD-V HM context into
673 * account when waiting on contended locks.
674 */
675 PVMCPUCC pVCpu = VMMGetCpu(pVM);
676 if (pVCpu)
677 {
678 VMMR0EMTBLOCKCTX Ctx;
679 int rc = VMMR0EmtPrepareToBlock(pVCpu, rcBusy, __FUNCTION__, pThis, &Ctx);
680 if (rc == VINF_SUCCESS)
681 {
682 Assert(RTThreadPreemptIsEnabled(NIL_RTTHREAD));
683
684 rc = pdmCritSectRwEnterSharedContended(pVM, pVCpu, pThis, rcBusy, pSrcPos, fNoVal, hThreadSelf);
685
686 VMMR0EmtResumeAfterBlocking(pVCpu, &Ctx);
687 }
688 else
689 {
690 //STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLockBusy);
691 rc = pdmCritSectRwEnterSharedBailOut(pVM, pThis, rc, pSrcPos, fNoVal, hThreadSelf);
692 }
693 return rc;
694 }
695
696 /* Non-EMT. */
697 Assert(RTThreadPreemptIsEnabled(NIL_RTTHREAD));
698 return pdmCritSectRwEnterSharedContended(pVM, NULL, pThis, rcBusy, pSrcPos, fNoVal, hThreadSelf);
699# endif /* IN_RING0 */
700 }
701
702#else /* !IN_RING3 && !IN_RING0 */
703 /*
704 * We cannot call SUPSemEventMultiWaitNoResume in this context. Go
705 * back to ring-3 and do it there or return rcBusy.
706 */
707# error "Unused code."
708 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterShared));
709 if (rcBusy == VINF_SUCCESS)
710 {
711 PVMCPUCC pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
712 /** @todo Should actually do this in via VMMR0.cpp instead of going all the way
713 * back to ring-3. Goes for both kind of crit sects. */
714 return VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_PDM_CRIT_SECT_RW_ENTER_SHARED, MMHyperCCToR3(pVM, pThis));
715 }
716 return rcBusy;
717#endif /* !IN_RING3 && !IN_RING0 */
718 }
719
720 ASMNopPause();
721 if (RT_LIKELY(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC))
722 { /* likely */ }
723 else
724 return VERR_SEM_DESTROYED;
725 ASMNopPause();
726
727 u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
728 u64OldState = u64State;
729 }
730 /* not reached */
731}
732
733
734/**
735 * Enter a critical section with shared (read) access.
736 *
737 * @returns VBox status code.
738 * @retval VINF_SUCCESS on success.
739 * @retval rcBusy if in ring-0 or raw-mode context and it is busy.
740 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
741 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
742 * during the operation.
743 *
744 * @param pVM The cross context VM structure.
745 * @param pThis Pointer to the read/write critical section.
746 * @param rcBusy The status code to return when we're in RC or R0 and the
747 * section is busy. Pass VINF_SUCCESS to acquired the
748 * critical section thru a ring-3 call if necessary.
749 * @sa PDMCritSectRwEnterSharedDebug, PDMCritSectRwTryEnterShared,
750 * PDMCritSectRwTryEnterSharedDebug, PDMCritSectRwLeaveShared,
751 * RTCritSectRwEnterShared.
752 */
753VMMDECL(int) PDMCritSectRwEnterShared(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy)
754{
755#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
756 return pdmCritSectRwEnterShared(pVM, pThis, rcBusy, false /*fTryOnly*/, NULL, false /*fNoVal*/);
757#else
758 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
759 return pdmCritSectRwEnterShared(pVM, pThis, rcBusy, false /*fTryOnly*/, &SrcPos, false /*fNoVal*/);
760#endif
761}
762
763
764/**
765 * Enter a critical section with shared (read) access.
766 *
767 * @returns VBox status code.
768 * @retval VINF_SUCCESS on success.
769 * @retval rcBusy if in ring-0 or raw-mode context and it is busy.
770 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
771 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
772 * during the operation.
773 *
774 * @param pVM The cross context VM structure.
775 * @param pThis Pointer to the read/write critical section.
776 * @param rcBusy The status code to return when we're in RC or R0 and the
777 * section is busy. Pass VINF_SUCCESS to acquired the
778 * critical section thru a ring-3 call if necessary.
779 * @param uId Where we're entering the section.
780 * @param SRC_POS The source position.
781 * @sa PDMCritSectRwEnterShared, PDMCritSectRwTryEnterShared,
782 * PDMCritSectRwTryEnterSharedDebug, PDMCritSectRwLeaveShared,
783 * RTCritSectRwEnterSharedDebug.
784 */
785VMMDECL(int) PDMCritSectRwEnterSharedDebug(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy, RTHCUINTPTR uId, RT_SRC_POS_DECL)
786{
787 NOREF(uId); NOREF(pszFile); NOREF(iLine); NOREF(pszFunction);
788#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
789 return pdmCritSectRwEnterShared(pVM, pThis, rcBusy, false /*fTryOnly*/, NULL, false /*fNoVal*/);
790#else
791 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
792 return pdmCritSectRwEnterShared(pVM, pThis, rcBusy, false /*fTryOnly*/, &SrcPos, false /*fNoVal*/);
793#endif
794}
795
796
797/**
798 * Try enter a critical section with shared (read) access.
799 *
800 * @returns VBox status code.
801 * @retval VINF_SUCCESS on success.
802 * @retval VERR_SEM_BUSY if the critsect was owned.
803 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
804 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
805 * during the operation.
806 *
807 * @param pVM The cross context VM structure.
808 * @param pThis Pointer to the read/write critical section.
809 * @sa PDMCritSectRwTryEnterSharedDebug, PDMCritSectRwEnterShared,
810 * PDMCritSectRwEnterSharedDebug, PDMCritSectRwLeaveShared,
811 * RTCritSectRwTryEnterShared.
812 */
813VMMDECL(int) PDMCritSectRwTryEnterShared(PVMCC pVM, PPDMCRITSECTRW pThis)
814{
815#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
816 return pdmCritSectRwEnterShared(pVM, pThis, VERR_SEM_BUSY, true /*fTryOnly*/, NULL, false /*fNoVal*/);
817#else
818 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
819 return pdmCritSectRwEnterShared(pVM, pThis, VERR_SEM_BUSY, true /*fTryOnly*/, &SrcPos, false /*fNoVal*/);
820#endif
821}
822
823
824/**
825 * Try enter a critical section with shared (read) access.
826 *
827 * @returns VBox status code.
828 * @retval VINF_SUCCESS on success.
829 * @retval VERR_SEM_BUSY if the critsect was owned.
830 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
831 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
832 * during the operation.
833 *
834 * @param pVM The cross context VM structure.
835 * @param pThis Pointer to the read/write critical section.
836 * @param uId Where we're entering the section.
837 * @param SRC_POS The source position.
838 * @sa PDMCritSectRwTryEnterShared, PDMCritSectRwEnterShared,
839 * PDMCritSectRwEnterSharedDebug, PDMCritSectRwLeaveShared,
840 * RTCritSectRwTryEnterSharedDebug.
841 */
842VMMDECL(int) PDMCritSectRwTryEnterSharedDebug(PVMCC pVM, PPDMCRITSECTRW pThis, RTHCUINTPTR uId, RT_SRC_POS_DECL)
843{
844 NOREF(uId); NOREF(pszFile); NOREF(iLine); NOREF(pszFunction);
845#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
846 return pdmCritSectRwEnterShared(pVM, pThis, VERR_SEM_BUSY, true /*fTryOnly*/, NULL, false /*fNoVal*/);
847#else
848 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
849 return pdmCritSectRwEnterShared(pVM, pThis, VERR_SEM_BUSY, true /*fTryOnly*/, &SrcPos, false /*fNoVal*/);
850#endif
851}
852
853
854#ifdef IN_RING3
855/**
856 * Enters a PDM read/write critical section with shared (read) access.
857 *
858 * @returns VINF_SUCCESS if entered successfully.
859 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
860 * during the operation.
861 *
862 * @param pVM The cross context VM structure.
863 * @param pThis Pointer to the read/write critical section.
864 * @param fCallRing3 Whether this is a VMMRZCallRing3()request.
865 */
866VMMR3DECL(int) PDMR3CritSectRwEnterSharedEx(PVM pVM, PPDMCRITSECTRW pThis, bool fCallRing3)
867{
868 return pdmCritSectRwEnterShared(pVM, pThis, VERR_SEM_BUSY, false /*fTryAgain*/, NULL, fCallRing3);
869}
870#endif
871
872
873/**
874 * Leave a critical section held with shared access.
875 *
876 * @returns VBox status code.
877 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
878 * during the operation.
879 * @param pVM The cross context VM structure.
880 * @param pThis Pointer to the read/write critical section.
881 * @param fNoVal No validation records (i.e. queued release).
882 * @sa PDMCritSectRwEnterShared, PDMCritSectRwTryEnterShared,
883 * PDMCritSectRwEnterSharedDebug, PDMCritSectRwTryEnterSharedDebug,
884 * PDMCritSectRwLeaveExcl, RTCritSectRwLeaveShared.
885 */
886static int pdmCritSectRwLeaveSharedWorker(PVMCC pVM, PPDMCRITSECTRW pThis, bool fNoVal)
887{
888 /*
889 * Validate handle.
890 */
891 AssertPtr(pThis);
892 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
893
894#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
895 NOREF(fNoVal);
896#endif
897
898 /*
899 * Check the direction and take action accordingly.
900 */
901#ifdef IN_RING0
902 PVMCPUCC pVCpu = NULL;
903#endif
904 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
905 uint64_t u64OldState = u64State;
906 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
907 {
908#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
909 if (fNoVal)
910 Assert(!RTLockValidatorRecSharedIsOwner(pThis->s.Core.pValidatorRead, NIL_RTTHREAD));
911 else
912 {
913 int rc9 = RTLockValidatorRecSharedCheckAndRelease(pThis->s.Core.pValidatorRead, NIL_RTTHREAD);
914 if (RT_FAILURE(rc9))
915 return rc9;
916 }
917#endif
918 for (;;)
919 {
920 uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
921 AssertReturn(c > 0, VERR_NOT_OWNER);
922 c--;
923
924 if ( c > 0
925 || (u64State & RTCSRW_CNT_WR_MASK) == 0)
926 {
927 /* Don't change the direction. */
928 u64State &= ~RTCSRW_CNT_RD_MASK;
929 u64State |= c << RTCSRW_CNT_RD_SHIFT;
930 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
931 break;
932 }
933 else
934 {
935#if defined(IN_RING3) || defined(IN_RING0)
936# ifdef IN_RING0
937 Assert(RTSemEventIsSignalSafe() == RTSemEventMultiIsSignalSafe());
938 if (!pVCpu)
939 pVCpu = VMMGetCpu(pVM);
940 if ( pVCpu == NULL /* non-EMT access, if we implement it must be able to block */
941 || VMMRZCallRing3IsEnabled(pVCpu)
942 || RTSemEventIsSignalSafe()
943 || ( VMMR0ThreadCtxHookIsEnabled(pVCpu) /* Doesn't matter if Signal() blocks if we have hooks, ... */
944 && RTThreadPreemptIsEnabled(NIL_RTTHREAD) /* ... and preemption is still enabled, */
945 && ASMIntAreEnabled()) /* ... and interrupts hasn't yet been disabled. Special pre-GC HM env. */
946 )
947# endif
948 {
949 /* Reverse the direction and signal the writer threads. */
950 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_DIR_MASK);
951 u64State |= RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT;
952 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
953 {
954 int rc;
955# ifdef IN_RING0
956 STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveShared);
957 if (!RTSemEventIsSignalSafe() && pVCpu != NULL)
958 {
959 VMMR0EMTBLOCKCTX Ctx;
960 rc = VMMR0EmtPrepareToBlock(pVCpu, VINF_SUCCESS, __FUNCTION__, pThis, &Ctx);
961 VMM_ASSERT_RELEASE_MSG_RETURN(pVM, RT_SUCCESS(rc), ("rc=%Rrc\n", rc), rc);
962
963 rc = SUPSemEventSignal(pVM->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
964
965 VMMR0EmtResumeAfterBlocking(pVCpu, &Ctx);
966 }
967 else
968# endif
969 rc = SUPSemEventSignal(pVM->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
970 AssertRC(rc);
971 return rc;
972 }
973 }
974#endif /* IN_RING3 || IN_RING0 */
975#ifndef IN_RING3
976# ifdef IN_RING0
977 else
978# endif
979 {
980 /* Queue the exit request (ring-3). */
981# ifndef IN_RING0
982 PVMCPUCC pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
983# endif
984 uint32_t i = pVCpu->pdm.s.cQueuedCritSectRwShrdLeaves++;
985 LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3 c=%d (%#llx)\n", i, pThis, c, u64State));
986 VMM_ASSERT_RELEASE_MSG_RETURN(pVM, i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves),
987 ("i=%u\n", i), VERR_PDM_CRITSECTRW_IPE);
988 pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i] = pThis->s.pSelfR3;
989 VMM_ASSERT_RELEASE_MSG_RETURN(pVM,
990 RT_VALID_PTR(pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i])
991 && ((uintptr_t)pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i] & PAGE_OFFSET_MASK)
992 == ((uintptr_t)pThis & PAGE_OFFSET_MASK),
993 ("%p vs %p\n", pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i], pThis),
994 pdmCritSectRwCorrupted(pThis, "Invalid self pointer"));
995 VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
996 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
997 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
998 STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveShared);
999 break;
1000 }
1001#endif
1002 }
1003
1004 ASMNopPause();
1005 if (RT_LIKELY(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC))
1006 { }
1007 else
1008 return VERR_SEM_DESTROYED;
1009 ASMNopPause();
1010
1011 u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1012 u64OldState = u64State;
1013 }
1014 }
1015 else
1016 {
1017 /*
1018 * Write direction. Check that it's the owner calling and that it has reads to undo.
1019 */
1020 RTNATIVETHREAD hNativeSelf = pdmCritSectRwGetNativeSelf(pVM, pThis);
1021 AssertReturn(hNativeSelf != NIL_RTNATIVETHREAD, VERR_VM_THREAD_NOT_EMT);
1022
1023 RTNATIVETHREAD hNativeWriter;
1024 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
1025 AssertReturn(hNativeSelf == hNativeWriter, VERR_NOT_OWNER);
1026 AssertReturn(pThis->s.Core.cWriterReads > 0, VERR_NOT_OWNER);
1027#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1028 if (!fNoVal)
1029 {
1030 int rc = RTLockValidatorRecExclUnwindMixed(pThis->s.Core.pValidatorWrite, &pThis->s.Core.pValidatorRead->Core);
1031 if (RT_FAILURE(rc))
1032 return rc;
1033 }
1034#endif
1035 uint32_t cDepth = ASMAtomicDecU32(&pThis->s.Core.cWriterReads);
1036 AssertReturn(cDepth < PDM_CRITSECTRW_MAX_RECURSIONS, pdmCritSectRwCorrupted(pThis, "too many writer-read recursions"));
1037 }
1038
1039 return VINF_SUCCESS;
1040}
1041
1042
1043/**
1044 * Leave a critical section held with shared access.
1045 *
1046 * @returns VBox status code.
1047 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1048 * during the operation.
1049 * @param pVM The cross context VM structure.
1050 * @param pThis Pointer to the read/write critical section.
1051 * @sa PDMCritSectRwEnterShared, PDMCritSectRwTryEnterShared,
1052 * PDMCritSectRwEnterSharedDebug, PDMCritSectRwTryEnterSharedDebug,
1053 * PDMCritSectRwLeaveExcl, RTCritSectRwLeaveShared.
1054 */
1055VMMDECL(int) PDMCritSectRwLeaveShared(PVMCC pVM, PPDMCRITSECTRW pThis)
1056{
1057 return pdmCritSectRwLeaveSharedWorker(pVM, pThis, false /*fNoVal*/);
1058}
1059
1060
1061#if defined(IN_RING3) || defined(IN_RING0)
1062/**
1063 * PDMCritSectBothFF interface.
1064 *
1065 * @param pVM The cross context VM structure.
1066 * @param pThis Pointer to the read/write critical section.
1067 */
1068void pdmCritSectRwLeaveSharedQueued(PVMCC pVM, PPDMCRITSECTRW pThis)
1069{
1070 pdmCritSectRwLeaveSharedWorker(pVM, pThis, true /*fNoVal*/);
1071}
1072#endif
1073
1074
1075/**
1076 * Worker for pdmCritSectRwEnterExcl that bails out on wait failure.
1077 *
1078 * @returns @a rc unless corrupted.
1079 * @param pThis Pointer to the read/write critical section.
1080 * @param rc The status to return.
1081 */
1082DECL_NO_INLINE(static, int) pdmCritSectRwEnterExclBailOut(PPDMCRITSECTRW pThis, int rc)
1083{
1084 /*
1085 * Decrement the counts and return the error.
1086 */
1087 for (;;)
1088 {
1089 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1090 uint64_t const u64OldState = u64State;
1091 uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
1092 AssertReturn(c > 0, pdmCritSectRwCorrupted(pThis, "Invalid write count on bailout"));
1093 c--;
1094 u64State &= ~RTCSRW_CNT_WR_MASK;
1095 u64State |= c << RTCSRW_CNT_WR_SHIFT;
1096 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1097 return rc;
1098
1099 ASMNopPause();
1100 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
1101 ASMNopPause();
1102 }
1103}
1104
1105
1106/**
1107 * Worker for pdmCritSectRwEnterExcl that handles the red tape after we've
1108 * gotten exclusive ownership of the critical section.
1109 */
1110DECL_FORCE_INLINE(int) pdmCritSectRwEnterExclFirst(PPDMCRITSECTRW pThis, PCRTLOCKVALSRCPOS pSrcPos,
1111 bool fNoVal, RTTHREAD hThreadSelf)
1112{
1113 RT_NOREF(hThreadSelf, fNoVal, pSrcPos);
1114 Assert((PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State) & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT));
1115
1116#ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
1117 pThis->s.Core.cWriteRecursions = 1;
1118#else
1119 ASMAtomicWriteU32(&pThis->s.Core.cWriteRecursions, 1);
1120#endif
1121 Assert(pThis->s.Core.cWriterReads == 0);
1122
1123#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1124 if (!fNoVal)
1125 {
1126 if (hThreadSelf == NIL_RTTHREAD)
1127 hThreadSelf = RTThreadSelfAutoAdopt();
1128 RTLockValidatorRecExclSetOwner(pThis->s.Core.pValidatorWrite, hThreadSelf, pSrcPos, true);
1129 }
1130#endif
1131 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(Stat,EnterExcl));
1132 STAM_PROFILE_ADV_START(&pThis->s.StatWriteLocked, swl);
1133 return VINF_SUCCESS;
1134}
1135
1136
1137#if defined(IN_RING3) || defined(IN_RING0)
1138/**
1139 * Worker for pdmCritSectRwEnterExcl that handles waiting when the section is
1140 * contended.
1141 */
1142static int pdmR3R0CritSectRwEnterExclContended(PVMCC pVM, PVMCPUCC pVCpu, PPDMCRITSECTRW pThis, RTNATIVETHREAD hNativeSelf,
1143 PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal, int rcBusy, RTTHREAD hThreadSelf)
1144{
1145 RT_NOREF(hThreadSelf, rcBusy, pSrcPos, fNoVal, pVCpu);
1146
1147 PSUPDRVSESSION const pSession = pVM->pSession;
1148 SUPSEMEVENT const hEvent = (SUPSEMEVENT)pThis->s.Core.hEvtWrite;
1149# ifdef IN_RING0
1150 uint64_t const tsStart = RTTimeNanoTS();
1151 uint64_t const cNsMaxTotalDef = RT_NS_5MIN;
1152 uint64_t cNsMaxTotal = cNsMaxTotalDef;
1153 uint32_t cMsMaxOne = RT_MS_5SEC;
1154 bool fNonInterruptible = false;
1155# endif
1156
1157 for (uint32_t iLoop = 0; ; iLoop++)
1158 {
1159 /*
1160 * Wait for our turn.
1161 */
1162 int rc;
1163# ifdef IN_RING3
1164# ifdef PDMCRITSECTRW_STRICT
1165 rc = RTLockValidatorRecExclCheckBlocking(pThis->s.Core.pValidatorWrite, hThreadSelf, pSrcPos, true,
1166 RT_INDEFINITE_WAIT, RTTHREADSTATE_RW_WRITE, false);
1167 if (RT_SUCCESS(rc))
1168 { /* likely */ }
1169 else
1170 return pdmCritSectRwEnterExclBailOut(pThis, rc);
1171# else
1172 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_WRITE, false);
1173# endif
1174# endif
1175
1176 for (;;)
1177 {
1178 /*
1179 * We always wait with a timeout so we can re-check the structure sanity
1180 * and not get stuck waiting on a corrupt or deleted section.
1181 */
1182# ifdef IN_RING3
1183 rc = SUPSemEventWaitNoResume(pSession, hEvent, RT_MS_5SEC);
1184# else
1185 rc = !fNonInterruptible
1186 ? SUPSemEventWaitNoResume(pSession, hEvent, cMsMaxOne)
1187 : SUPSemEventWait(pSession, hEvent, cMsMaxOne);
1188 Log11Func(("%p: rc=%Rrc %'RU64 ns (cMsMaxOne=%RU64 hNativeWriter=%p)\n",
1189 pThis, rc, RTTimeNanoTS() - tsStart, cMsMaxOne, pThis->s.Core.u.s.hNativeWriter));
1190# endif
1191 if (RT_LIKELY(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC))
1192 { /* likely */ }
1193 else
1194 {
1195# ifdef IN_RING3
1196 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
1197# endif
1198 return VERR_SEM_DESTROYED;
1199 }
1200 if (RT_LIKELY(rc == VINF_SUCCESS))
1201 break;
1202
1203 /*
1204 * Timeout and interrupted waits needs careful handling in ring-0
1205 * because we're cooperating with ring-3 on this critical section
1206 * and thus need to make absolutely sure we won't get stuck here.
1207 *
1208 * The r0 interrupted case means something is pending (termination,
1209 * signal, APC, debugger, whatever), so we must try our best to
1210 * return to the caller and to ring-3 so it can be dealt with.
1211 */
1212 if (rc == VERR_TIMEOUT || rc == VERR_INTERRUPTED)
1213 {
1214# ifdef IN_RING0
1215 uint64_t const cNsElapsed = RTTimeNanoTS() - tsStart;
1216 int const rcTerm = RTThreadQueryTerminationStatus(NIL_RTTHREAD);
1217 AssertMsg(rcTerm == VINF_SUCCESS || rcTerm == VERR_NOT_SUPPORTED || rcTerm == VINF_THREAD_IS_TERMINATING,
1218 ("rcTerm=%Rrc\n", rcTerm));
1219 if (rcTerm == VERR_NOT_SUPPORTED && cNsMaxTotal == cNsMaxTotalDef)
1220 cNsMaxTotal = RT_NS_1MIN;
1221
1222 if (rc == VERR_TIMEOUT)
1223 {
1224 /* Try return get out of here with a non-VINF_SUCCESS status if
1225 the thread is terminating or if the timeout has been exceeded. */
1226 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwExclVerrTimeout);
1227 if ( rcTerm == VINF_THREAD_IS_TERMINATING
1228 || cNsElapsed > cNsMaxTotal)
1229 return pdmCritSectRwEnterExclBailOut(pThis, rcBusy != VINF_SUCCESS ? rcBusy : rc);
1230 }
1231 else
1232 {
1233 /* For interrupt cases, we must return if we can. If rcBusy is VINF_SUCCESS,
1234 we will try non-interruptible sleep for a while to help resolve the issue
1235 w/o guru'ing. */
1236 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwExclVerrInterrupted);
1237 if ( rcTerm != VINF_THREAD_IS_TERMINATING
1238 && rcBusy == VINF_SUCCESS
1239 && pVCpu != NULL
1240 && cNsElapsed <= cNsMaxTotal)
1241 {
1242 if (!fNonInterruptible)
1243 {
1244 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatCritSectRwExclNonInterruptibleWaits);
1245 fNonInterruptible = true;
1246 cMsMaxOne = 32;
1247 uint64_t cNsLeft = cNsMaxTotal - cNsElapsed;
1248 if (cNsLeft > RT_NS_10SEC)
1249 cNsMaxTotal = cNsElapsed + RT_NS_10SEC;
1250 }
1251 }
1252 else
1253 return pdmCritSectRwEnterExclBailOut(pThis, rcBusy != VINF_SUCCESS ? rcBusy : rc);
1254 }
1255# else /* IN_RING3 */
1256 RT_NOREF(pVM, pVCpu, rcBusy);
1257# endif /* IN_RING3 */
1258 }
1259 /*
1260 * Any other return code is fatal.
1261 */
1262 else
1263 {
1264# ifdef IN_RING3
1265 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
1266# endif
1267 AssertMsgFailed(("rc=%Rrc\n", rc));
1268 return RT_FAILURE_NP(rc) ? rc : -rc;
1269 }
1270 }
1271
1272# ifdef IN_RING3
1273 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
1274# endif
1275
1276 /*
1277 * Try take exclusive write ownership.
1278 */
1279 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1280 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT))
1281 {
1282 bool fDone;
1283 ASMAtomicCmpXchgHandle(&pThis->s.Core.u.s.hNativeWriter, hNativeSelf, NIL_RTNATIVETHREAD, fDone);
1284 if (fDone)
1285 return pdmCritSectRwEnterExclFirst(pThis, pSrcPos, fNoVal, hThreadSelf);
1286 }
1287 AssertMsg(iLoop < 1000, ("%u\n", iLoop)); /* may loop a few times here... */
1288 }
1289}
1290#endif /* IN_RING3 || IN_RING0 */
1291
1292
1293/**
1294 * Worker that enters a read/write critical section with exclusive access.
1295 *
1296 * @returns VBox status code.
1297 * @param pVM The cross context VM structure.
1298 * @param pThis Pointer to the read/write critical section.
1299 * @param rcBusy The busy return code for ring-0 and ring-3.
1300 * @param fTryOnly Only try enter it, don't wait.
1301 * @param pSrcPos The source position. (Can be NULL.)
1302 * @param fNoVal No validation records.
1303 */
1304static int pdmCritSectRwEnterExcl(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy, bool fTryOnly,
1305 PCRTLOCKVALSRCPOS pSrcPos, bool fNoVal)
1306{
1307 /*
1308 * Validate input.
1309 */
1310 AssertPtr(pThis);
1311 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
1312
1313 RTTHREAD hThreadSelf = NIL_RTTHREAD;
1314#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1315 if (!fTryOnly)
1316 {
1317 hThreadSelf = RTThreadSelfAutoAdopt();
1318 int rc9 = RTLockValidatorRecExclCheckOrder(pThis->s.Core.pValidatorWrite, hThreadSelf, pSrcPos, RT_INDEFINITE_WAIT);
1319 if (RT_FAILURE(rc9))
1320 return rc9;
1321 }
1322#endif
1323
1324 /*
1325 * Check if we're already the owner and just recursing.
1326 */
1327 RTNATIVETHREAD const hNativeSelf = pdmCritSectRwGetNativeSelf(pVM, pThis);
1328 AssertReturn(hNativeSelf != NIL_RTNATIVETHREAD, VERR_VM_THREAD_NOT_EMT);
1329 RTNATIVETHREAD hNativeWriter;
1330 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
1331 if (hNativeSelf == hNativeWriter)
1332 {
1333 Assert((PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State) & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT));
1334#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1335 if (!fNoVal)
1336 {
1337 int rc9 = RTLockValidatorRecExclRecursion(pThis->s.Core.pValidatorWrite, pSrcPos);
1338 if (RT_FAILURE(rc9))
1339 return rc9;
1340 }
1341#endif
1342 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(Stat,EnterExcl));
1343#ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
1344 uint32_t const cDepth = ++pThis->s.Core.cWriteRecursions;
1345#else
1346 uint32_t const cDepth = ASMAtomicIncU32(&pThis->s.Core.cWriteRecursions);
1347#endif
1348 AssertReturnStmt(cDepth > 1 && cDepth <= PDM_CRITSECTRW_MAX_RECURSIONS,
1349 ASMAtomicDecU32(&pThis->s.Core.cWriteRecursions),
1350 VERR_PDM_CRITSECTRW_TOO_MANY_RECURSIONS);
1351 return VINF_SUCCESS;
1352 }
1353
1354 /*
1355 * First we try grab an idle critical section using 128-bit atomics.
1356 */
1357 /** @todo This could be moved up before the recursion check. */
1358 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1359#ifdef RTASM_HAVE_CMP_WRITE_U128
1360 if ( (u64State & ~RTCSRW_DIR_MASK) == 0
1361 && pdmCritSectRwIsCmpWriteU128Supported())
1362 {
1363 RTCRITSECTRWSTATE OldState;
1364 OldState.s.u64State = u64State;
1365 OldState.s.hNativeWriter = NIL_RTNATIVETHREAD;
1366 AssertCompile(sizeof(OldState.s.hNativeWriter) == sizeof(OldState.u128.s.Lo));
1367
1368 RTCRITSECTRWSTATE NewState;
1369 NewState.s.u64State = (UINT64_C(1) << RTCSRW_CNT_WR_SHIFT) | (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT);
1370 NewState.s.hNativeWriter = hNativeSelf;
1371
1372 if (ASMAtomicCmpWriteU128U(&pThis->s.Core.u.u128, NewState.u128, OldState.u128))
1373 return pdmCritSectRwEnterExclFirst(pThis, pSrcPos, fNoVal, hThreadSelf);
1374
1375 u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1376 }
1377#endif
1378
1379 /*
1380 * Do it step by step. Update the state to reflect our desire.
1381 */
1382 uint64_t u64OldState = u64State;
1383
1384 for (;;)
1385 {
1386 if ( (u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT)
1387 || (u64State & (RTCSRW_CNT_RD_MASK | RTCSRW_CNT_WR_MASK)) != 0)
1388 {
1389 /* It flows in the right direction, try follow it before it changes. */
1390 uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
1391 AssertReturn(c < RTCSRW_CNT_MASK, VERR_PDM_CRITSECTRW_TOO_MANY_WRITERS);
1392 c++;
1393 Assert(c < RTCSRW_CNT_WR_MASK / 4);
1394 u64State &= ~RTCSRW_CNT_WR_MASK;
1395 u64State |= c << RTCSRW_CNT_WR_SHIFT;
1396 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1397 break;
1398 }
1399 else if ((u64State & (RTCSRW_CNT_RD_MASK | RTCSRW_CNT_WR_MASK)) == 0)
1400 {
1401 /* Wrong direction, but we're alone here and can simply try switch the direction. */
1402 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_CNT_WR_MASK | RTCSRW_DIR_MASK);
1403 u64State |= (UINT64_C(1) << RTCSRW_CNT_WR_SHIFT) | (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT);
1404 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1405 break;
1406 }
1407 else if (fTryOnly)
1408 {
1409 /* Wrong direction and we're not supposed to wait, just return. */
1410 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterExcl));
1411 return VERR_SEM_BUSY;
1412 }
1413 else
1414 {
1415 /* Add ourselves to the write count and break out to do the wait. */
1416 uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
1417 AssertReturn(c < RTCSRW_CNT_MASK, VERR_PDM_CRITSECTRW_TOO_MANY_WRITERS);
1418 c++;
1419 Assert(c < RTCSRW_CNT_WR_MASK / 4);
1420 u64State &= ~RTCSRW_CNT_WR_MASK;
1421 u64State |= c << RTCSRW_CNT_WR_SHIFT;
1422 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1423 break;
1424 }
1425
1426 ASMNopPause();
1427
1428 if (pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC)
1429 { /* likely */ }
1430 else
1431 return VERR_SEM_DESTROYED;
1432
1433 ASMNopPause();
1434 u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1435 u64OldState = u64State;
1436 }
1437
1438 /*
1439 * If we're in write mode now try grab the ownership. Play fair if there
1440 * are threads already waiting.
1441 */
1442 bool fDone = (u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT)
1443 && ( ((u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT) == 1
1444 || fTryOnly);
1445 if (fDone)
1446 {
1447 ASMAtomicCmpXchgHandle(&pThis->s.Core.u.s.hNativeWriter, hNativeSelf, NIL_RTNATIVETHREAD, fDone);
1448 if (fDone)
1449 return pdmCritSectRwEnterExclFirst(pThis, pSrcPos, fNoVal, hThreadSelf);
1450 }
1451
1452 /*
1453 * Okay, we have contention and will have to wait unless we're just trying.
1454 */
1455 if (fTryOnly)
1456 {
1457 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterExcl)); /** @todo different statistics for this */
1458 return pdmCritSectRwEnterExclBailOut(pThis, VERR_SEM_BUSY);
1459 }
1460
1461 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterExcl));
1462
1463 /*
1464 * Ring-3 is pretty straight forward.
1465 */
1466#if defined(IN_RING3) && defined(PDMCRITSECTRW_STRICT)
1467 return pdmR3R0CritSectRwEnterExclContended(pVM, NULL, pThis, hNativeSelf, pSrcPos, fNoVal, rcBusy, hThreadSelf);
1468#elif defined(IN_RING3)
1469 return pdmR3R0CritSectRwEnterExclContended(pVM, NULL, pThis, hNativeSelf, pSrcPos, fNoVal, rcBusy, RTThreadSelf());
1470
1471#elif defined(IN_RING0)
1472 /*
1473 * In ring-0 context we have to take the special VT-x/AMD-V HM context into
1474 * account when waiting on contended locks.
1475 */
1476 PVMCPUCC pVCpu = VMMGetCpu(pVM);
1477 if (pVCpu)
1478 {
1479 VMMR0EMTBLOCKCTX Ctx;
1480 int rc = VMMR0EmtPrepareToBlock(pVCpu, rcBusy, __FUNCTION__, pThis, &Ctx);
1481 if (rc == VINF_SUCCESS)
1482 {
1483 Assert(RTThreadPreemptIsEnabled(NIL_RTTHREAD));
1484
1485 rc = pdmR3R0CritSectRwEnterExclContended(pVM, pVCpu, pThis, hNativeSelf, pSrcPos, fNoVal, rcBusy, NIL_RTTHREAD);
1486
1487 VMMR0EmtResumeAfterBlocking(pVCpu, &Ctx);
1488 }
1489 else
1490 {
1491 //STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLockBusy);
1492 rc = pdmCritSectRwEnterExclBailOut(pThis, rc);
1493 }
1494 return rc;
1495 }
1496
1497 /* Non-EMT. */
1498 Assert(RTThreadPreemptIsEnabled(NIL_RTTHREAD));
1499 return pdmR3R0CritSectRwEnterExclContended(pVM, NULL, pThis, hNativeSelf, pSrcPos, fNoVal, rcBusy, NIL_RTTHREAD);
1500
1501#else
1502# error "Unused."
1503 /*
1504 * Raw-mode: Call host and take it there if rcBusy is VINF_SUCCESS.
1505 */
1506 rcBusy = pdmCritSectRwEnterExclBailOut(pThis, rcBusy);
1507 if (rcBusy == VINF_SUCCESS)
1508 {
1509 Assert(!fTryOnly);
1510 PVMCPUCC pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
1511 /** @todo Should actually do this in via VMMR0.cpp instead of going all the way
1512 * back to ring-3. Goes for both kind of crit sects. */
1513 return VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_PDM_CRIT_SECT_RW_ENTER_EXCL, MMHyperCCToR3(pVM, pThis));
1514 }
1515 return rcBusy;
1516#endif
1517}
1518
1519
1520/**
1521 * Try enter a critical section with exclusive (write) access.
1522 *
1523 * @returns VBox status code.
1524 * @retval VINF_SUCCESS on success.
1525 * @retval rcBusy if in ring-0 or raw-mode context and it is busy.
1526 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
1527 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1528 * during the operation.
1529 *
1530 * @param pVM The cross context VM structure.
1531 * @param pThis Pointer to the read/write critical section.
1532 * @param rcBusy The status code to return when we're in RC or R0 and the
1533 * section is busy. Pass VINF_SUCCESS to acquired the
1534 * critical section thru a ring-3 call if necessary.
1535 * @sa PDMCritSectRwEnterExclDebug, PDMCritSectRwTryEnterExcl,
1536 * PDMCritSectRwTryEnterExclDebug,
1537 * PDMCritSectEnterDebug, PDMCritSectEnter,
1538 * RTCritSectRwEnterExcl.
1539 */
1540VMMDECL(int) PDMCritSectRwEnterExcl(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy)
1541{
1542#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
1543 return pdmCritSectRwEnterExcl(pVM, pThis, rcBusy, false /*fTryAgain*/, NULL, false /*fNoVal*/);
1544#else
1545 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
1546 return pdmCritSectRwEnterExcl(pVM, pThis, rcBusy, false /*fTryAgain*/, &SrcPos, false /*fNoVal*/);
1547#endif
1548}
1549
1550
1551/**
1552 * Try enter a critical section with exclusive (write) access.
1553 *
1554 * @returns VBox status code.
1555 * @retval VINF_SUCCESS on success.
1556 * @retval rcBusy if in ring-0 or raw-mode context and it is busy.
1557 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
1558 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1559 * during the operation.
1560 *
1561 * @param pVM The cross context VM structure.
1562 * @param pThis Pointer to the read/write critical section.
1563 * @param rcBusy The status code to return when we're in RC or R0 and the
1564 * section is busy. Pass VINF_SUCCESS to acquired the
1565 * critical section thru a ring-3 call if necessary.
1566 * @param uId Where we're entering the section.
1567 * @param SRC_POS The source position.
1568 * @sa PDMCritSectRwEnterExcl, PDMCritSectRwTryEnterExcl,
1569 * PDMCritSectRwTryEnterExclDebug,
1570 * PDMCritSectEnterDebug, PDMCritSectEnter,
1571 * RTCritSectRwEnterExclDebug.
1572 */
1573VMMDECL(int) PDMCritSectRwEnterExclDebug(PVMCC pVM, PPDMCRITSECTRW pThis, int rcBusy, RTHCUINTPTR uId, RT_SRC_POS_DECL)
1574{
1575 NOREF(uId); NOREF(pszFile); NOREF(iLine); NOREF(pszFunction);
1576#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
1577 return pdmCritSectRwEnterExcl(pVM, pThis, rcBusy, false /*fTryAgain*/, NULL, false /*fNoVal*/);
1578#else
1579 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
1580 return pdmCritSectRwEnterExcl(pVM, pThis, rcBusy, false /*fTryAgain*/, &SrcPos, false /*fNoVal*/);
1581#endif
1582}
1583
1584
1585/**
1586 * Try enter a critical section with exclusive (write) access.
1587 *
1588 * @retval VINF_SUCCESS on success.
1589 * @retval VERR_SEM_BUSY if the critsect was owned.
1590 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
1591 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1592 * during the operation.
1593 *
1594 * @param pVM The cross context VM structure.
1595 * @param pThis Pointer to the read/write critical section.
1596 * @sa PDMCritSectRwEnterExcl, PDMCritSectRwTryEnterExclDebug,
1597 * PDMCritSectRwEnterExclDebug,
1598 * PDMCritSectTryEnter, PDMCritSectTryEnterDebug,
1599 * RTCritSectRwTryEnterExcl.
1600 */
1601VMMDECL(int) PDMCritSectRwTryEnterExcl(PVMCC pVM, PPDMCRITSECTRW pThis)
1602{
1603#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
1604 return pdmCritSectRwEnterExcl(pVM, pThis, VERR_SEM_BUSY, true /*fTryAgain*/, NULL, false /*fNoVal*/);
1605#else
1606 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
1607 return pdmCritSectRwEnterExcl(pVM, pThis, VERR_SEM_BUSY, true /*fTryAgain*/, &SrcPos, false /*fNoVal*/);
1608#endif
1609}
1610
1611
1612/**
1613 * Try enter a critical section with exclusive (write) access.
1614 *
1615 * @retval VINF_SUCCESS on success.
1616 * @retval VERR_SEM_BUSY if the critsect was owned.
1617 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
1618 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1619 * during the operation.
1620 *
1621 * @param pVM The cross context VM structure.
1622 * @param pThis Pointer to the read/write critical section.
1623 * @param uId Where we're entering the section.
1624 * @param SRC_POS The source position.
1625 * @sa PDMCritSectRwTryEnterExcl, PDMCritSectRwEnterExcl,
1626 * PDMCritSectRwEnterExclDebug,
1627 * PDMCritSectTryEnterDebug, PDMCritSectTryEnter,
1628 * RTCritSectRwTryEnterExclDebug.
1629 */
1630VMMDECL(int) PDMCritSectRwTryEnterExclDebug(PVMCC pVM, PPDMCRITSECTRW pThis, RTHCUINTPTR uId, RT_SRC_POS_DECL)
1631{
1632 NOREF(uId); NOREF(pszFile); NOREF(iLine); NOREF(pszFunction);
1633#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
1634 return pdmCritSectRwEnterExcl(pVM, pThis, VERR_SEM_BUSY, true /*fTryAgain*/, NULL, false /*fNoVal*/);
1635#else
1636 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
1637 return pdmCritSectRwEnterExcl(pVM, pThis, VERR_SEM_BUSY, true /*fTryAgain*/, &SrcPos, false /*fNoVal*/);
1638#endif
1639}
1640
1641
1642#ifdef IN_RING3
1643/**
1644 * Enters a PDM read/write critical section with exclusive (write) access.
1645 *
1646 * @returns VINF_SUCCESS if entered successfully.
1647 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1648 * during the operation.
1649 *
1650 * @param pVM The cross context VM structure.
1651 * @param pThis Pointer to the read/write critical section.
1652 * @param fCallRing3 Whether this is a VMMRZCallRing3()request.
1653 */
1654VMMR3DECL(int) PDMR3CritSectRwEnterExclEx(PVM pVM, PPDMCRITSECTRW pThis, bool fCallRing3)
1655{
1656 return pdmCritSectRwEnterExcl(pVM, pThis, VERR_SEM_BUSY, false /*fTryAgain*/, NULL, fCallRing3 /*fNoVal*/);
1657}
1658#endif /* IN_RING3 */
1659
1660
1661/**
1662 * Leave a critical section held exclusively.
1663 *
1664 * @returns VBox status code.
1665 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1666 * during the operation.
1667 * @param pVM The cross context VM structure.
1668 * @param pThis Pointer to the read/write critical section.
1669 * @param fNoVal No validation records (i.e. queued release).
1670 * @sa PDMCritSectRwLeaveShared, RTCritSectRwLeaveExcl.
1671 */
1672static int pdmCritSectRwLeaveExclWorker(PVMCC pVM, PPDMCRITSECTRW pThis, bool fNoVal)
1673{
1674 /*
1675 * Validate handle.
1676 */
1677 AssertPtr(pThis);
1678 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, VERR_SEM_DESTROYED);
1679
1680#if !defined(PDMCRITSECTRW_STRICT) || !defined(IN_RING3)
1681 NOREF(fNoVal);
1682#endif
1683
1684 /*
1685 * Check ownership.
1686 */
1687 RTNATIVETHREAD hNativeSelf = pdmCritSectRwGetNativeSelf(pVM, pThis);
1688 AssertReturn(hNativeSelf != NIL_RTNATIVETHREAD, VERR_VM_THREAD_NOT_EMT);
1689
1690 RTNATIVETHREAD hNativeWriter;
1691 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
1692 AssertReturn(hNativeSelf == hNativeWriter, VERR_NOT_OWNER);
1693
1694
1695 /*
1696 * Unwind one recursion. Not the last?
1697 */
1698 if (pThis->s.Core.cWriteRecursions != 1)
1699 {
1700#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1701 if (fNoVal)
1702 Assert(pThis->s.Core.pValidatorWrite->hThread == NIL_RTTHREAD);
1703 else
1704 {
1705 int rc9 = RTLockValidatorRecExclUnwind(pThis->s.Core.pValidatorWrite);
1706 if (RT_FAILURE(rc9))
1707 return rc9;
1708 }
1709#endif
1710#ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
1711 uint32_t const cDepth = --pThis->s.Core.cWriteRecursions;
1712#else
1713 uint32_t const cDepth = ASMAtomicDecU32(&pThis->s.Core.cWriteRecursions);
1714#endif
1715 AssertReturn(cDepth != 0 && cDepth < UINT32_MAX, pdmCritSectRwCorrupted(pThis, "Invalid write recursion value on leave"));
1716 return VINF_SUCCESS;
1717 }
1718
1719
1720 /*
1721 * Final recursion.
1722 */
1723 AssertReturn(pThis->s.Core.cWriterReads == 0, VERR_WRONG_ORDER); /* (must release all read recursions before the final write.) */
1724#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
1725 if (fNoVal)
1726 Assert(pThis->s.Core.pValidatorWrite->hThread == NIL_RTTHREAD);
1727 else
1728 {
1729 int rc9 = RTLockValidatorRecExclReleaseOwner(pThis->s.Core.pValidatorWrite, true);
1730 if (RT_FAILURE(rc9))
1731 return rc9;
1732 }
1733#endif
1734
1735
1736#ifdef RTASM_HAVE_CMP_WRITE_U128
1737 /*
1738 * See if we can get out w/o any signalling as this is a common case.
1739 */
1740 if (pdmCritSectRwIsCmpWriteU128Supported())
1741 {
1742 RTCRITSECTRWSTATE OldState;
1743 OldState.s.u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1744 if (OldState.s.u64State == ((UINT64_C(1) << RTCSRW_CNT_WR_SHIFT) | (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT)))
1745 {
1746 OldState.s.hNativeWriter = hNativeSelf;
1747 AssertCompile(sizeof(OldState.s.hNativeWriter) == sizeof(OldState.u128.s.Lo));
1748
1749 RTCRITSECTRWSTATE NewState;
1750 NewState.s.u64State = RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT;
1751 NewState.s.hNativeWriter = NIL_RTNATIVETHREAD;
1752
1753# ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
1754 pThis->s.Core.cWriteRecursions = 0;
1755# else
1756 ASMAtomicWriteU32(&pThis->s.Core.cWriteRecursions, 0);
1757# endif
1758 STAM_PROFILE_ADV_STOP(&pThis->s.StatWriteLocked, swl);
1759
1760 if (ASMAtomicCmpWriteU128U(&pThis->s.Core.u.u128, NewState.u128, OldState.u128))
1761 return VINF_SUCCESS;
1762
1763 /* bail out. */
1764 pThis->s.Core.cWriteRecursions = 1;
1765 }
1766 }
1767#endif /* RTASM_HAVE_CMP_WRITE_U128 */
1768
1769
1770#if defined(IN_RING3) || defined(IN_RING0)
1771 /*
1772 * Ring-3: Straight forward, just update the state and if necessary signal waiters.
1773 * Ring-0: Try leave for real, depends on host and context.
1774 */
1775# ifdef IN_RING0
1776 Assert(RTSemEventIsSignalSafe() == RTSemEventMultiIsSignalSafe());
1777 PVMCPUCC pVCpu = VMMGetCpu(pVM);
1778 if ( pVCpu == NULL /* non-EMT access, if we implement it must be able to block */
1779 || VMMRZCallRing3IsEnabled(pVCpu)
1780 || RTSemEventIsSignalSafe()
1781 || ( VMMR0ThreadCtxHookIsEnabled(pVCpu) /* Doesn't matter if Signal() blocks if we have hooks, ... */
1782 && RTThreadPreemptIsEnabled(NIL_RTTHREAD) /* ... and preemption is still enabled, */
1783 && ASMIntAreEnabled()) /* ... and interrupts hasn't yet been disabled. Special pre-GC HM env. */
1784 )
1785# endif
1786 {
1787# ifdef PDMCRITSECTRW_WITH_LESS_ATOMIC_STUFF
1788 pThis->s.Core.cWriteRecursions = 0;
1789# else
1790 ASMAtomicWriteU32(&pThis->s.Core.cWriteRecursions, 0);
1791# endif
1792 STAM_PROFILE_ADV_STOP(&pThis->s.StatWriteLocked, swl);
1793 ASMAtomicWriteHandle(&pThis->s.Core.u.s.hNativeWriter, NIL_RTNATIVETHREAD);
1794
1795 for (;;)
1796 {
1797 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1798 uint64_t u64OldState = u64State;
1799
1800 uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
1801 AssertReturn(c > 0, pdmCritSectRwCorrupted(pThis, "Invalid write count on leave"));
1802 c--;
1803
1804 if ( c > 0
1805 || (u64State & RTCSRW_CNT_RD_MASK) == 0)
1806 {
1807 /*
1808 * Don't change the direction, wake up the next writer if any.
1809 */
1810 u64State &= ~RTCSRW_CNT_WR_MASK;
1811 u64State |= c << RTCSRW_CNT_WR_SHIFT;
1812 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1813 {
1814 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,LeaveExcl));
1815 int rc;
1816 if (c == 0)
1817 rc = VINF_SUCCESS;
1818# ifdef IN_RING0
1819 else if (!RTSemEventIsSignalSafe() && pVCpu != NULL)
1820 {
1821 VMMR0EMTBLOCKCTX Ctx;
1822 rc = VMMR0EmtPrepareToBlock(pVCpu, VINF_SUCCESS, __FUNCTION__, pThis, &Ctx);
1823 VMM_ASSERT_RELEASE_MSG_RETURN(pVM, RT_SUCCESS(rc), ("rc=%Rrc\n", rc), rc);
1824
1825 rc = SUPSemEventSignal(pVM->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
1826
1827 VMMR0EmtResumeAfterBlocking(pVCpu, &Ctx);
1828 }
1829# endif
1830 else
1831 rc = SUPSemEventSignal(pVM->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
1832 AssertRC(rc);
1833 return rc;
1834 }
1835 }
1836 else
1837 {
1838 /*
1839 * Reverse the direction and signal the reader threads.
1840 */
1841 u64State &= ~(RTCSRW_CNT_WR_MASK | RTCSRW_DIR_MASK);
1842 u64State |= RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT;
1843 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u.s.u64State, u64State, u64OldState))
1844 {
1845 Assert(!pThis->s.Core.fNeedReset);
1846 ASMAtomicWriteBool(&pThis->s.Core.fNeedReset, true);
1847 STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,LeaveExcl));
1848
1849 int rc;
1850# ifdef IN_RING0
1851 if (!RTSemEventMultiIsSignalSafe() && pVCpu != NULL)
1852 {
1853 VMMR0EMTBLOCKCTX Ctx;
1854 rc = VMMR0EmtPrepareToBlock(pVCpu, VINF_SUCCESS, __FUNCTION__, pThis, &Ctx);
1855 VMM_ASSERT_RELEASE_MSG_RETURN(pVM, RT_SUCCESS(rc), ("rc=%Rrc\n", rc), rc);
1856
1857 rc = SUPSemEventMultiSignal(pVM->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
1858
1859 VMMR0EmtResumeAfterBlocking(pVCpu, &Ctx);
1860 }
1861 else
1862# endif
1863 rc = SUPSemEventMultiSignal(pVM->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
1864 AssertRC(rc);
1865 return rc;
1866 }
1867 }
1868
1869 ASMNopPause();
1870 if (pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC)
1871 { /*likely*/ }
1872 else
1873 return VERR_SEM_DESTROYED;
1874 ASMNopPause();
1875 }
1876 /* not reached! */
1877 }
1878#endif /* IN_RING3 || IN_RING0 */
1879
1880
1881#ifndef IN_RING3
1882 /*
1883 * Queue the requested exit for ring-3 execution.
1884 */
1885# ifndef IN_RING0
1886 PVMCPUCC pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
1887# endif
1888 uint32_t i = pVCpu->pdm.s.cQueuedCritSectRwExclLeaves++;
1889 LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3\n", i, pThis));
1890 VMM_ASSERT_RELEASE_MSG_RETURN(pVM, i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectRwExclLeaves),
1891 ("i=%u\n", i), VERR_PDM_CRITSECTRW_IPE);
1892 pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i] = pThis->s.pSelfR3;
1893 VMM_ASSERT_RELEASE_MSG_RETURN(pVM,
1894 RT_VALID_PTR(pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i])
1895 && ((uintptr_t)pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i] & PAGE_OFFSET_MASK)
1896 == ((uintptr_t)pThis & PAGE_OFFSET_MASK),
1897 ("%p vs %p\n", pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i], pThis),
1898 pdmCritSectRwCorrupted(pThis, "Invalid self pointer on queue (excl)"));
1899 VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
1900 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
1901 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
1902 STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveExcl);
1903 return VINF_SUCCESS;
1904#endif
1905}
1906
1907
1908/**
1909 * Leave a critical section held exclusively.
1910 *
1911 * @returns VBox status code.
1912 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
1913 * during the operation.
1914 * @param pVM The cross context VM structure.
1915 * @param pThis Pointer to the read/write critical section.
1916 * @sa PDMCritSectRwLeaveShared, RTCritSectRwLeaveExcl.
1917 */
1918VMMDECL(int) PDMCritSectRwLeaveExcl(PVMCC pVM, PPDMCRITSECTRW pThis)
1919{
1920 return pdmCritSectRwLeaveExclWorker(pVM, pThis, false /*fNoVal*/);
1921}
1922
1923
1924#if defined(IN_RING3) || defined(IN_RING0)
1925/**
1926 * PDMCritSectBothFF interface.
1927 *
1928 * @param pVM The cross context VM structure.
1929 * @param pThis Pointer to the read/write critical section.
1930 */
1931void pdmCritSectRwLeaveExclQueued(PVMCC pVM, PPDMCRITSECTRW pThis)
1932{
1933 pdmCritSectRwLeaveExclWorker(pVM, pThis, true /*fNoVal*/);
1934}
1935#endif
1936
1937
1938/**
1939 * Checks the caller is the exclusive (write) owner of the critical section.
1940 *
1941 * @retval true if owner.
1942 * @retval false if not owner.
1943 * @param pVM The cross context VM structure.
1944 * @param pThis Pointer to the read/write critical section.
1945 * @sa PDMCritSectRwIsReadOwner, PDMCritSectIsOwner,
1946 * RTCritSectRwIsWriteOwner.
1947 */
1948VMMDECL(bool) PDMCritSectRwIsWriteOwner(PVMCC pVM, PPDMCRITSECTRW pThis)
1949{
1950 /*
1951 * Validate handle.
1952 */
1953 AssertPtr(pThis);
1954 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, false);
1955
1956 /*
1957 * Check ownership.
1958 */
1959 RTNATIVETHREAD hNativeWriter;
1960 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hNativeWriter);
1961 if (hNativeWriter == NIL_RTNATIVETHREAD)
1962 return false;
1963 return hNativeWriter == pdmCritSectRwGetNativeSelf(pVM, pThis);
1964}
1965
1966
1967/**
1968 * Checks if the caller is one of the read owners of the critical section.
1969 *
1970 * @note !CAUTION! This API doesn't work reliably if lock validation isn't
1971 * enabled. Meaning, the answer is not trustworhty unless
1972 * RT_LOCK_STRICT or PDMCRITSECTRW_STRICT was defined at build time.
1973 * Also, make sure you do not use RTCRITSECTRW_FLAGS_NO_LOCK_VAL when
1974 * creating the semaphore. And finally, if you used a locking class,
1975 * don't disable deadlock detection by setting cMsMinDeadlock to
1976 * RT_INDEFINITE_WAIT.
1977 *
1978 * In short, only use this for assertions.
1979 *
1980 * @returns @c true if reader, @c false if not.
1981 * @param pVM The cross context VM structure.
1982 * @param pThis Pointer to the read/write critical section.
1983 * @param fWannaHear What you'd like to hear when lock validation is not
1984 * available. (For avoiding asserting all over the place.)
1985 * @sa PDMCritSectRwIsWriteOwner, RTCritSectRwIsReadOwner.
1986 */
1987VMMDECL(bool) PDMCritSectRwIsReadOwner(PVMCC pVM, PPDMCRITSECTRW pThis, bool fWannaHear)
1988{
1989 /*
1990 * Validate handle.
1991 */
1992 AssertPtr(pThis);
1993 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, false);
1994
1995 /*
1996 * Inspect the state.
1997 */
1998 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
1999 if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT))
2000 {
2001 /*
2002 * It's in write mode, so we can only be a reader if we're also the
2003 * current writer.
2004 */
2005 RTNATIVETHREAD hWriter;
2006 ASMAtomicUoReadHandle(&pThis->s.Core.u.s.hNativeWriter, &hWriter);
2007 if (hWriter == NIL_RTNATIVETHREAD)
2008 return false;
2009 return hWriter == pdmCritSectRwGetNativeSelf(pVM, pThis);
2010 }
2011
2012 /*
2013 * Read mode. If there are no current readers, then we cannot be a reader.
2014 */
2015 if (!(u64State & RTCSRW_CNT_RD_MASK))
2016 return false;
2017
2018#if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
2019 /*
2020 * Ask the lock validator.
2021 * Note! It doesn't know everything, let's deal with that if it becomes an issue...
2022 */
2023 NOREF(fWannaHear);
2024 return RTLockValidatorRecSharedIsOwner(pThis->s.Core.pValidatorRead, NIL_RTTHREAD);
2025#else
2026 /*
2027 * Ok, we don't know, just tell the caller what he want to hear.
2028 */
2029 return fWannaHear;
2030#endif
2031}
2032
2033
2034/**
2035 * Gets the write recursion count.
2036 *
2037 * @returns The write recursion count (0 if bad critsect).
2038 * @param pThis Pointer to the read/write critical section.
2039 * @sa PDMCritSectRwGetWriterReadRecursion, PDMCritSectRwGetReadCount,
2040 * RTCritSectRwGetWriteRecursion.
2041 */
2042VMMDECL(uint32_t) PDMCritSectRwGetWriteRecursion(PPDMCRITSECTRW pThis)
2043{
2044 /*
2045 * Validate handle.
2046 */
2047 AssertPtr(pThis);
2048 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, 0);
2049
2050 /*
2051 * Return the requested data.
2052 */
2053 return pThis->s.Core.cWriteRecursions;
2054}
2055
2056
2057/**
2058 * Gets the read recursion count of the current writer.
2059 *
2060 * @returns The read recursion count (0 if bad critsect).
2061 * @param pThis Pointer to the read/write critical section.
2062 * @sa PDMCritSectRwGetWriteRecursion, PDMCritSectRwGetReadCount,
2063 * RTCritSectRwGetWriterReadRecursion.
2064 */
2065VMMDECL(uint32_t) PDMCritSectRwGetWriterReadRecursion(PPDMCRITSECTRW pThis)
2066{
2067 /*
2068 * Validate handle.
2069 */
2070 AssertPtr(pThis);
2071 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, 0);
2072
2073 /*
2074 * Return the requested data.
2075 */
2076 return pThis->s.Core.cWriterReads;
2077}
2078
2079
2080/**
2081 * Gets the current number of reads.
2082 *
2083 * This includes all read recursions, so it might be higher than the number of
2084 * read owners. It does not include reads done by the current writer.
2085 *
2086 * @returns The read count (0 if bad critsect).
2087 * @param pThis Pointer to the read/write critical section.
2088 * @sa PDMCritSectRwGetWriteRecursion, PDMCritSectRwGetWriterReadRecursion,
2089 * RTCritSectRwGetReadCount.
2090 */
2091VMMDECL(uint32_t) PDMCritSectRwGetReadCount(PPDMCRITSECTRW pThis)
2092{
2093 /*
2094 * Validate input.
2095 */
2096 AssertPtr(pThis);
2097 AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, 0);
2098
2099 /*
2100 * Return the requested data.
2101 */
2102 uint64_t u64State = PDMCRITSECTRW_READ_STATE(&pThis->s.Core.u.s.u64State);
2103 if ((u64State & RTCSRW_DIR_MASK) != (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
2104 return 0;
2105 return (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
2106}
2107
2108
2109/**
2110 * Checks if the read/write critical section is initialized or not.
2111 *
2112 * @retval true if initialized.
2113 * @retval false if not initialized.
2114 * @param pThis Pointer to the read/write critical section.
2115 * @sa PDMCritSectIsInitialized, RTCritSectRwIsInitialized.
2116 */
2117VMMDECL(bool) PDMCritSectRwIsInitialized(PCPDMCRITSECTRW pThis)
2118{
2119 AssertPtr(pThis);
2120 return pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC;
2121}
2122
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