VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/PDMAllCritSect.cpp@ 57046

Last change on this file since 57046 was 56402, checked in by vboxsync, 10 years ago

DevATA,PDMCritSect: Changed the AsyncIOSem from RTSEMEVENT to SUPSEMEVENT so it can be signalled in ring-0 too.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 27.8 KB
Line 
1/* $Id: PDMAllCritSect.cpp 56402 2015-06-13 14:58:10Z vboxsync $ */
2/** @file
3 * PDM - Write-Only Critical Section, All Contexts.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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//_CRITSECT
23#include "PDMInternal.h"
24#include <VBox/vmm/pdmcritsect.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/vmm.h>
27#include <VBox/vmm/vm.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# include <iprt/semaphore.h>
38#endif
39#if defined(IN_RING3) || defined(IN_RING0)
40# include <iprt/thread.h>
41#endif
42
43
44/*******************************************************************************
45* Defined Constants And Macros *
46*******************************************************************************/
47/** The number loops to spin for in ring-3. */
48#define PDMCRITSECT_SPIN_COUNT_R3 20
49/** The number loops to spin for in ring-0. */
50#define PDMCRITSECT_SPIN_COUNT_R0 256
51/** The number loops to spin for in the raw-mode context. */
52#define PDMCRITSECT_SPIN_COUNT_RC 256
53
54
55/* Undefine the automatic VBOX_STRICT API mappings. */
56#undef PDMCritSectEnter
57#undef PDMCritSectTryEnter
58
59
60/**
61 * Gets the ring-3 native thread handle of the calling thread.
62 *
63 * @returns native thread handle (ring-3).
64 * @param pCritSect The critical section. This is used in R0 and RC.
65 */
66DECL_FORCE_INLINE(RTNATIVETHREAD) pdmCritSectGetNativeSelf(PCPDMCRITSECT pCritSect)
67{
68#ifdef IN_RING3
69 NOREF(pCritSect);
70 RTNATIVETHREAD hNativeSelf = RTThreadNativeSelf();
71#else
72 AssertMsgReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC, ("%RX32\n", pCritSect->s.Core.u32Magic),
73 NIL_RTNATIVETHREAD);
74 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
75 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
76 RTNATIVETHREAD hNativeSelf = pVCpu->hNativeThread; Assert(hNativeSelf != NIL_RTNATIVETHREAD);
77#endif
78 return hNativeSelf;
79}
80
81
82/**
83 * Tail code called when we've won the battle for the lock.
84 *
85 * @returns VINF_SUCCESS.
86 *
87 * @param pCritSect The critical section.
88 * @param hNativeSelf The native handle of this thread.
89 */
90DECL_FORCE_INLINE(int) pdmCritSectEnterFirst(PPDMCRITSECT pCritSect, RTNATIVETHREAD hNativeSelf, PCRTLOCKVALSRCPOS pSrcPos)
91{
92 AssertMsg(pCritSect->s.Core.NativeThreadOwner == NIL_RTNATIVETHREAD, ("NativeThreadOwner=%p\n", pCritSect->s.Core.NativeThreadOwner));
93 Assert(!(pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK));
94
95 ASMAtomicWriteS32(&pCritSect->s.Core.cNestings, 1);
96 Assert(pCritSect->s.Core.cNestings == 1);
97 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, hNativeSelf);
98
99# ifdef PDMCRITSECT_STRICT
100 RTLockValidatorRecExclSetOwner(pCritSect->s.Core.pValidatorRec, NIL_RTTHREAD, pSrcPos, true);
101# else
102 NOREF(pSrcPos);
103# endif
104
105 STAM_PROFILE_ADV_START(&pCritSect->s.StatLocked, l);
106 return VINF_SUCCESS;
107}
108
109
110#if defined(IN_RING3) || defined(IN_RING0)
111/**
112 * Deals with the contended case in ring-3 and ring-0.
113 *
114 * @retval VINF_SUCCESS on success.
115 * @retval VERR_SEM_DESTROYED if destroyed.
116 *
117 * @param pCritSect The critsect.
118 * @param hNativeSelf The native thread handle.
119 */
120static int pdmR3R0CritSectEnterContended(PPDMCRITSECT pCritSect, RTNATIVETHREAD hNativeSelf, PCRTLOCKVALSRCPOS pSrcPos)
121{
122 /*
123 * Start waiting.
124 */
125 if (ASMAtomicIncS32(&pCritSect->s.Core.cLockers) == 0)
126 return pdmCritSectEnterFirst(pCritSect, hNativeSelf, pSrcPos);
127# ifdef IN_RING3
128 STAM_COUNTER_INC(&pCritSect->s.StatContentionR3);
129# else
130 STAM_COUNTER_INC(&pCritSect->s.StatContentionRZLock);
131# endif
132
133 /*
134 * The wait loop.
135 */
136 PSUPDRVSESSION pSession = pCritSect->s.CTX_SUFF(pVM)->pSession;
137 SUPSEMEVENT hEvent = (SUPSEMEVENT)pCritSect->s.Core.EventSem;
138# ifdef IN_RING3
139# ifdef PDMCRITSECT_STRICT
140 RTTHREAD hThreadSelf = RTThreadSelfAutoAdopt();
141 int rc2 = RTLockValidatorRecExclCheckOrder(pCritSect->s.Core.pValidatorRec, hThreadSelf, pSrcPos, RT_INDEFINITE_WAIT);
142 if (RT_FAILURE(rc2))
143 return rc2;
144# else
145 RTTHREAD hThreadSelf = RTThreadSelf();
146# endif
147# endif
148 for (;;)
149 {
150 /*
151 * Do the wait.
152 *
153 * In ring-3 this gets cluttered by lock validation and thread state
154 * maintainence.
155 *
156 * In ring-0 we have to deal with the possibility that the thread has
157 * been signalled and the interruptible wait function returning
158 * immediately. In that case we do normal R0/RC rcBusy handling.
159 */
160# ifdef IN_RING3
161# ifdef PDMCRITSECT_STRICT
162 int rc9 = RTLockValidatorRecExclCheckBlocking(pCritSect->s.Core.pValidatorRec, hThreadSelf, pSrcPos,
163 !(pCritSect->s.Core.fFlags & RTCRITSECT_FLAGS_NO_NESTING),
164 RT_INDEFINITE_WAIT, RTTHREADSTATE_CRITSECT, true);
165 if (RT_FAILURE(rc9))
166 return rc9;
167# else
168 RTThreadBlocking(hThreadSelf, RTTHREADSTATE_CRITSECT, true);
169# endif
170 int rc = SUPSemEventWaitNoResume(pSession, hEvent, RT_INDEFINITE_WAIT);
171 RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_CRITSECT);
172# else /* IN_RING0 */
173 int rc = SUPSemEventWaitNoResume(pSession, hEvent, RT_INDEFINITE_WAIT);
174# endif /* IN_RING0 */
175
176 /*
177 * Deal with the return code and critsect destruction.
178 */
179 if (RT_UNLIKELY(pCritSect->s.Core.u32Magic != RTCRITSECT_MAGIC))
180 return VERR_SEM_DESTROYED;
181 if (rc == VINF_SUCCESS)
182 return pdmCritSectEnterFirst(pCritSect, hNativeSelf, pSrcPos);
183 AssertMsg(rc == VERR_INTERRUPTED, ("rc=%Rrc\n", rc));
184
185# ifdef IN_RING0
186 /* Something is pending (signal, APC, debugger, whatever), just go back
187 to ring-3 so the kernel can deal with it when leaving kernel context.
188
189 Note! We've incremented cLockers already and cannot safely decrement
190 it without creating a race with PDMCritSectLeave, resulting in
191 spurious wakeups. */
192 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
193 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
194 rc = VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_VM_R0_PREEMPT, NULL);
195 AssertRC(rc);
196# endif
197 }
198 /* won't get here */
199}
200#endif /* IN_RING3 || IN_RING0 */
201
202
203/**
204 * Common worker for the debug and normal APIs.
205 *
206 * @returns VINF_SUCCESS if entered successfully.
207 * @returns rcBusy when encountering a busy critical section in GC/R0.
208 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
209 * during the operation.
210 *
211 * @param pCritSect The PDM critical section to enter.
212 * @param rcBusy The status code to return when we're in GC or R0
213 * and the section is busy.
214 */
215DECL_FORCE_INLINE(int) pdmCritSectEnter(PPDMCRITSECT pCritSect, int rcBusy, PCRTLOCKVALSRCPOS pSrcPos)
216{
217 Assert(pCritSect->s.Core.cNestings < 8); /* useful to catch incorrect locking */
218 Assert(pCritSect->s.Core.cNestings >= 0);
219
220 /*
221 * If the critical section has already been destroyed, then inform the caller.
222 */
223 AssertMsgReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC,
224 ("%p %RX32\n", pCritSect, pCritSect->s.Core.u32Magic),
225 VERR_SEM_DESTROYED);
226
227 /*
228 * See if we're lucky.
229 */
230 /* NOP ... */
231 if (pCritSect->s.Core.fFlags & RTCRITSECT_FLAGS_NOP)
232 return VINF_SUCCESS;
233
234 RTNATIVETHREAD hNativeSelf = pdmCritSectGetNativeSelf(pCritSect);
235 /* ... not owned ... */
236 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
237 return pdmCritSectEnterFirst(pCritSect, hNativeSelf, pSrcPos);
238
239 /* ... or nested. */
240 if (pCritSect->s.Core.NativeThreadOwner == hNativeSelf)
241 {
242 ASMAtomicIncS32(&pCritSect->s.Core.cLockers);
243 ASMAtomicIncS32(&pCritSect->s.Core.cNestings);
244 Assert(pCritSect->s.Core.cNestings > 1);
245 return VINF_SUCCESS;
246 }
247
248 /*
249 * Spin for a bit without incrementing the counter.
250 */
251 /** @todo Move this to cfgm variables since it doesn't make sense to spin on UNI
252 * cpu systems. */
253 int32_t cSpinsLeft = CTX_SUFF(PDMCRITSECT_SPIN_COUNT_);
254 while (cSpinsLeft-- > 0)
255 {
256 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
257 return pdmCritSectEnterFirst(pCritSect, hNativeSelf, pSrcPos);
258 ASMNopPause();
259 /** @todo Should use monitor/mwait on e.g. &cLockers here, possibly with a
260 cli'ed pendingpreemption check up front using sti w/ instruction fusing
261 for avoiding races. Hmm ... This is assuming the other party is actually
262 executing code on another CPU ... which we could keep track of if we
263 wanted. */
264 }
265
266#ifdef IN_RING3
267 /*
268 * Take the slow path.
269 */
270 NOREF(rcBusy);
271 return pdmR3R0CritSectEnterContended(pCritSect, hNativeSelf, pSrcPos);
272
273#else
274# ifdef IN_RING0
275 /** @todo If preemption is disabled it means we're in VT-x/AMD-V context
276 * and would be better off switching out of that while waiting for
277 * the lock. Several of the locks jumps back to ring-3 just to
278 * get the lock, the ring-3 code will then call the kernel to do
279 * the lock wait and when the call return it will call ring-0
280 * again and resume via in setjmp style. Not very efficient. */
281# if 0
282 if (ASMIntAreEnabled()) /** @todo this can be handled as well by changing
283 * callers not prepared for longjmp/blocking to
284 * use PDMCritSectTryEnter. */
285 {
286 /*
287 * Leave HM context while waiting if necessary.
288 */
289 int rc;
290 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD))
291 {
292 STAM_REL_COUNTER_ADD(&pCritSect->s.StatContentionRZLock, 1000000);
293 rc = pdmR3R0CritSectEnterContended(pCritSect, hNativeSelf, pSrcPos);
294 }
295 else
296 {
297 STAM_REL_COUNTER_ADD(&pCritSect->s.StatContentionRZLock, 1000000000);
298 PVM pVM = pCritSect->s.CTX_SUFF(pVM);
299 PVMCPU pVCpu = VMMGetCpu(pVM);
300 HMR0Leave(pVM, pVCpu);
301 RTThreadPreemptRestore(NIL_RTTHREAD, XXX);
302
303 rc = pdmR3R0CritSectEnterContended(pCritSect, hNativeSelf, pSrcPos);
304
305 RTThreadPreemptDisable(NIL_RTTHREAD, XXX);
306 HMR0Enter(pVM, pVCpu);
307 }
308 return rc;
309 }
310# else
311 /*
312 * We preemption hasn't been disabled, we can block here in ring-0.
313 */
314 if ( RTThreadPreemptIsEnabled(NIL_RTTHREAD)
315 && ASMIntAreEnabled())
316 return pdmR3R0CritSectEnterContended(pCritSect, hNativeSelf, pSrcPos);
317# endif
318#endif /* IN_RING0 */
319
320 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZLock);
321
322 /*
323 * Call ring-3 to acquire the critical section?
324 */
325 if (rcBusy == VINF_SUCCESS)
326 {
327 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
328 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
329 return VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_PDM_CRIT_SECT_ENTER, MMHyperCCToR3(pVM, pCritSect));
330 }
331
332 /*
333 * Return busy.
334 */
335 LogFlow(("PDMCritSectEnter: locked => R3 (%Rrc)\n", rcBusy));
336 return rcBusy;
337#endif /* !IN_RING3 */
338}
339
340
341/**
342 * Enters a PDM critical section.
343 *
344 * @returns VINF_SUCCESS if entered successfully.
345 * @returns rcBusy when encountering a busy critical section in RC/R0.
346 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
347 * during the operation.
348 *
349 * @param pCritSect The PDM critical section to enter.
350 * @param rcBusy The status code to return when we're in RC or R0
351 * and the section is busy. Pass VINF_SUCCESS to
352 * acquired the critical section thru a ring-3
353 * call if necessary.
354 */
355VMMDECL(int) PDMCritSectEnter(PPDMCRITSECT pCritSect, int rcBusy)
356{
357#ifndef PDMCRITSECT_STRICT
358 return pdmCritSectEnter(pCritSect, rcBusy, NULL);
359#else
360 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
361 return pdmCritSectEnter(pCritSect, rcBusy, &SrcPos);
362#endif
363}
364
365
366/**
367 * Enters a PDM critical section, with location information for debugging.
368 *
369 * @returns VINF_SUCCESS if entered successfully.
370 * @returns rcBusy when encountering a busy critical section in RC/R0.
371 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
372 * during the operation.
373 *
374 * @param pCritSect The PDM critical section to enter.
375 * @param rcBusy The status code to return when we're in RC or R0
376 * and the section is busy. Pass VINF_SUCCESS to
377 * acquired the critical section thru a ring-3
378 * call if necessary.
379 * @param uId Some kind of locking location ID. Typically a
380 * return address up the stack. Optional (0).
381 * @param pszFile The file where the lock is being acquired from.
382 * Optional.
383 * @param iLine The line number in that file. Optional (0).
384 * @param pszFunction The function where the lock is being acquired
385 * from. Optional.
386 */
387VMMDECL(int) PDMCritSectEnterDebug(PPDMCRITSECT pCritSect, int rcBusy, RTHCUINTPTR uId, RT_SRC_POS_DECL)
388{
389#ifdef PDMCRITSECT_STRICT
390 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
391 return pdmCritSectEnter(pCritSect, rcBusy, &SrcPos);
392#else
393 NOREF(uId); RT_SRC_POS_NOREF();
394 return pdmCritSectEnter(pCritSect, rcBusy, NULL);
395#endif
396}
397
398
399/**
400 * Common worker for the debug and normal APIs.
401 *
402 * @retval VINF_SUCCESS on success.
403 * @retval VERR_SEM_BUSY if the critsect was owned.
404 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
405 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
406 * during the operation.
407 *
408 * @param pCritSect The critical section.
409 */
410static int pdmCritSectTryEnter(PPDMCRITSECT pCritSect, PCRTLOCKVALSRCPOS pSrcPos)
411{
412 /*
413 * If the critical section has already been destroyed, then inform the caller.
414 */
415 AssertMsgReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC,
416 ("%p %RX32\n", pCritSect, pCritSect->s.Core.u32Magic),
417 VERR_SEM_DESTROYED);
418
419 /*
420 * See if we're lucky.
421 */
422 /* NOP ... */
423 if (pCritSect->s.Core.fFlags & RTCRITSECT_FLAGS_NOP)
424 return VINF_SUCCESS;
425
426 RTNATIVETHREAD hNativeSelf = pdmCritSectGetNativeSelf(pCritSect);
427 /* ... not owned ... */
428 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
429 return pdmCritSectEnterFirst(pCritSect, hNativeSelf, pSrcPos);
430
431 /* ... or nested. */
432 if (pCritSect->s.Core.NativeThreadOwner == hNativeSelf)
433 {
434 ASMAtomicIncS32(&pCritSect->s.Core.cLockers);
435 ASMAtomicIncS32(&pCritSect->s.Core.cNestings);
436 Assert(pCritSect->s.Core.cNestings > 1);
437 return VINF_SUCCESS;
438 }
439
440 /* no spinning */
441
442 /*
443 * Return busy.
444 */
445#ifdef IN_RING3
446 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionR3);
447#else
448 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZLock);
449#endif
450 LogFlow(("PDMCritSectTryEnter: locked\n"));
451 return VERR_SEM_BUSY;
452}
453
454
455/**
456 * Try enter a critical section.
457 *
458 * @retval VINF_SUCCESS on success.
459 * @retval VERR_SEM_BUSY if the critsect was owned.
460 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
461 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
462 * during the operation.
463 *
464 * @param pCritSect The critical section.
465 */
466VMMDECL(int) PDMCritSectTryEnter(PPDMCRITSECT pCritSect)
467{
468#ifndef PDMCRITSECT_STRICT
469 return pdmCritSectTryEnter(pCritSect, NULL);
470#else
471 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_NORMAL_API();
472 return pdmCritSectTryEnter(pCritSect, &SrcPos);
473#endif
474}
475
476
477/**
478 * Try enter a critical section, with location information for debugging.
479 *
480 * @retval VINF_SUCCESS on success.
481 * @retval VERR_SEM_BUSY if the critsect was owned.
482 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
483 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
484 * during the operation.
485 *
486 * @param pCritSect The critical section.
487 * @param uId Some kind of locking location ID. Typically a
488 * return address up the stack. Optional (0).
489 * @param pszFile The file where the lock is being acquired from.
490 * Optional.
491 * @param iLine The line number in that file. Optional (0).
492 * @param pszFunction The function where the lock is being acquired
493 * from. Optional.
494 */
495VMMDECL(int) PDMCritSectTryEnterDebug(PPDMCRITSECT pCritSect, RTHCUINTPTR uId, RT_SRC_POS_DECL)
496{
497#ifdef PDMCRITSECT_STRICT
498 RTLOCKVALSRCPOS SrcPos = RTLOCKVALSRCPOS_INIT_DEBUG_API();
499 return pdmCritSectTryEnter(pCritSect, &SrcPos);
500#else
501 NOREF(uId); RT_SRC_POS_NOREF();
502 return pdmCritSectTryEnter(pCritSect, NULL);
503#endif
504}
505
506
507#ifdef IN_RING3
508/**
509 * Enters a PDM critical section.
510 *
511 * @returns VINF_SUCCESS if entered successfully.
512 * @returns rcBusy when encountering a busy critical section in GC/R0.
513 * @retval VERR_SEM_DESTROYED if the critical section is delete before or
514 * during the operation.
515 *
516 * @param pCritSect The PDM critical section to enter.
517 * @param fCallRing3 Whether this is a VMMRZCallRing3()request.
518 */
519VMMR3DECL(int) PDMR3CritSectEnterEx(PPDMCRITSECT pCritSect, bool fCallRing3)
520{
521 int rc = PDMCritSectEnter(pCritSect, VERR_IGNORED);
522 if ( rc == VINF_SUCCESS
523 && fCallRing3
524 && pCritSect->s.Core.pValidatorRec
525 && pCritSect->s.Core.pValidatorRec->hThread != NIL_RTTHREAD)
526 RTLockValidatorRecExclReleaseOwnerUnchecked(pCritSect->s.Core.pValidatorRec);
527 return rc;
528}
529#endif /* IN_RING3 */
530
531
532/**
533 * Leaves a critical section entered with PDMCritSectEnter().
534 *
535 * @returns Indication whether we really exited the critical section.
536 * @retval VINF_SUCCESS if we really exited.
537 * @retval VINF_SEM_NESTED if we only reduced the nesting count.
538 * @retval VERR_NOT_OWNER if you somehow ignore release assertions.
539 *
540 * @param pCritSect The PDM critical section to leave.
541 */
542VMMDECL(int) PDMCritSectLeave(PPDMCRITSECT pCritSect)
543{
544 AssertMsg(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC, ("%p %RX32\n", pCritSect, pCritSect->s.Core.u32Magic));
545 Assert(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC);
546
547 /* Check for NOP sections before asserting ownership. */
548 if (pCritSect->s.Core.fFlags & RTCRITSECT_FLAGS_NOP)
549 return VINF_SUCCESS;
550
551 /*
552 * Always check that the caller is the owner (screw performance).
553 */
554 RTNATIVETHREAD const hNativeSelf = pdmCritSectGetNativeSelf(pCritSect);
555 AssertReleaseMsgReturn(pCritSect->s.Core.NativeThreadOwner == hNativeSelf,
556 ("%p %s: %p != %p; cLockers=%d cNestings=%d\n", pCritSect, R3STRING(pCritSect->s.pszName),
557 pCritSect->s.Core.NativeThreadOwner, hNativeSelf,
558 pCritSect->s.Core.cLockers, pCritSect->s.Core.cNestings),
559 VERR_NOT_OWNER);
560 Assert(pCritSect->s.Core.cNestings >= 1);
561
562 /*
563 * Nested leave.
564 */
565 if (pCritSect->s.Core.cNestings > 1)
566 {
567 ASMAtomicDecS32(&pCritSect->s.Core.cNestings);
568 Assert(pCritSect->s.Core.cNestings >= 1);
569 ASMAtomicDecS32(&pCritSect->s.Core.cLockers);
570 Assert(pCritSect->s.Core.cLockers >= 0);
571 return VINF_SEM_NESTED;
572 }
573
574#ifdef IN_RING0
575# if 0 /** @todo Make SUPSemEventSignal interrupt safe (handle table++) and enable this for: defined(RT_OS_LINUX) || defined(RT_OS_OS2) */
576 if (1) /* SUPSemEventSignal is safe */
577# else
578 if (ASMIntAreEnabled())
579# endif
580#endif
581#if defined(IN_RING3) || defined(IN_RING0)
582 {
583 /*
584 * Leave for real.
585 */
586 /* update members. */
587 SUPSEMEVENT hEventToSignal = pCritSect->s.hEventToSignal;
588 pCritSect->s.hEventToSignal = NIL_SUPSEMEVENT;
589# ifdef IN_RING3
590# if defined(PDMCRITSECT_STRICT)
591 if (pCritSect->s.Core.pValidatorRec->hThread != NIL_RTTHREAD)
592 RTLockValidatorRecExclReleaseOwnerUnchecked(pCritSect->s.Core.pValidatorRec);
593# endif
594 Assert(!pCritSect->s.Core.pValidatorRec || pCritSect->s.Core.pValidatorRec->hThread == NIL_RTTHREAD);
595# endif
596 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
597 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, NIL_RTNATIVETHREAD);
598 ASMAtomicDecS32(&pCritSect->s.Core.cNestings);
599 Assert(pCritSect->s.Core.cNestings == 0);
600
601 /* stop and decrement lockers. */
602 STAM_PROFILE_ADV_STOP(&pCritSect->s.StatLocked, l);
603 ASMCompilerBarrier();
604 if (ASMAtomicDecS32(&pCritSect->s.Core.cLockers) >= 0)
605 {
606 /* Someone is waiting, wake up one of them. */
607 SUPSEMEVENT hEvent = (SUPSEMEVENT)pCritSect->s.Core.EventSem;
608 PSUPDRVSESSION pSession = pCritSect->s.CTX_SUFF(pVM)->pSession;
609 int rc = SUPSemEventSignal(pSession, hEvent);
610 AssertRC(rc);
611 }
612
613 /* Signal exit event. */
614 if (hEventToSignal != NIL_SUPSEMEVENT)
615 {
616 Log8(("Signalling %#p\n", hEventToSignal));
617 int rc = SUPSemEventSignal(pCritSect->s.CTX_SUFF(pVM)->pSession, hEventToSignal);
618 AssertRC(rc);
619 }
620
621# if defined(DEBUG_bird) && defined(IN_RING0)
622 VMMTrashVolatileXMMRegs();
623# endif
624 }
625#endif /* IN_RING3 || IN_RING0 */
626#ifdef IN_RING0
627 else
628#endif
629#if defined(IN_RING0) || defined(IN_RC)
630 {
631 /*
632 * Try leave it.
633 */
634 if (pCritSect->s.Core.cLockers == 0)
635 {
636 ASMAtomicWriteS32(&pCritSect->s.Core.cNestings, 0);
637 RTNATIVETHREAD hNativeThread = pCritSect->s.Core.NativeThreadOwner;
638 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
639 STAM_PROFILE_ADV_STOP(&pCritSect->s.StatLocked, l);
640
641 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, NIL_RTNATIVETHREAD);
642 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, -1, 0))
643 return VINF_SUCCESS;
644
645 /* darn, someone raced in on us. */
646 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, hNativeThread);
647 STAM_PROFILE_ADV_START(&pCritSect->s.StatLocked, l);
648 Assert(pCritSect->s.Core.cNestings == 0);
649 ASMAtomicWriteS32(&pCritSect->s.Core.cNestings, 1);
650 }
651 ASMAtomicOrU32(&pCritSect->s.Core.fFlags, PDMCRITSECT_FLAGS_PENDING_UNLOCK);
652
653 /*
654 * Queue the request.
655 */
656 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
657 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
658 uint32_t i = pVCpu->pdm.s.cQueuedCritSectLeaves++;
659 LogFlow(("PDMCritSectLeave: [%d]=%p => R3\n", i, pCritSect));
660 AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectLeaves));
661 pVCpu->pdm.s.apQueuedCritSectLeaves[i] = MMHyperCCToR3(pVM, pCritSect);
662 VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
663 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
664 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
665 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZUnlock);
666 }
667#endif /* IN_RING0 || IN_RC */
668
669 return VINF_SUCCESS;
670}
671
672
673#if defined(IN_RING0) || defined(IN_RING3)
674/**
675 * Schedule a event semaphore for signalling upon critsect exit.
676 *
677 * @returns VINF_SUCCESS on success.
678 * @returns VERR_TOO_MANY_SEMAPHORES if an event was already scheduled.
679 * @returns VERR_NOT_OWNER if we're not the critsect owner (ring-3 only).
680 * @returns VERR_SEM_DESTROYED if RTCritSectDelete was called while waiting.
681 *
682 * @param pCritSect The critical section.
683 * @param hEventToSignal The support driver event semaphore that should be
684 * signalled.
685 */
686VMMDECL(int) PDMHCCritSectScheduleExitEvent(PPDMCRITSECT pCritSect, SUPSEMEVENT hEventToSignal)
687{
688 AssertPtr(pCritSect);
689 Assert(!(pCritSect->s.Core.fFlags & RTCRITSECT_FLAGS_NOP));
690 Assert(hEventToSignal != NIL_SUPSEMEVENT);
691# ifdef IN_RING3
692 if (RT_UNLIKELY(!RTCritSectIsOwner(&pCritSect->s.Core)))
693 return VERR_NOT_OWNER;
694# endif
695 if (RT_LIKELY( pCritSect->s.hEventToSignal == NIL_RTSEMEVENT
696 || pCritSect->s.hEventToSignal == hEventToSignal))
697 {
698 pCritSect->s.hEventToSignal = hEventToSignal;
699 return VINF_SUCCESS;
700 }
701 return VERR_TOO_MANY_SEMAPHORES;
702}
703#endif /* IN_RING0 || IN_RING3 */
704
705
706/**
707 * Checks the caller is the owner of the critical section.
708 *
709 * @returns true if owner.
710 * @returns false if not owner.
711 * @param pCritSect The critical section.
712 */
713VMMDECL(bool) PDMCritSectIsOwner(PCPDMCRITSECT pCritSect)
714{
715#ifdef IN_RING3
716 return RTCritSectIsOwner(&pCritSect->s.Core);
717#else
718 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
719 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
720 if (pCritSect->s.Core.NativeThreadOwner != pVCpu->hNativeThread)
721 return false;
722 return (pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK) == 0
723 || pCritSect->s.Core.cNestings > 1;
724#endif
725}
726
727
728/**
729 * Checks the specified VCPU is the owner of the critical section.
730 *
731 * @returns true if owner.
732 * @returns false if not owner.
733 * @param pCritSect The critical section.
734 * @param pVCpu Pointer to the VMCPU.
735 */
736VMMDECL(bool) PDMCritSectIsOwnerEx(PCPDMCRITSECT pCritSect, PVMCPU pVCpu)
737{
738#ifdef IN_RING3
739 NOREF(pVCpu);
740 return RTCritSectIsOwner(&pCritSect->s.Core);
741#else
742 Assert(&pVCpu->CTX_SUFF(pVM)->aCpus[pVCpu->idCpu] == pVCpu);
743 if (pCritSect->s.Core.NativeThreadOwner != pVCpu->hNativeThread)
744 return false;
745 return (pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK) == 0
746 || pCritSect->s.Core.cNestings > 1;
747#endif
748}
749
750
751/**
752 * Checks if anyone is waiting on the critical section we own.
753 *
754 * @returns true if someone is waiting.
755 * @returns false if no one is waiting.
756 * @param pCritSect The critical section.
757 */
758VMMDECL(bool) PDMCritSectHasWaiters(PCPDMCRITSECT pCritSect)
759{
760 AssertReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC, false);
761 Assert(pCritSect->s.Core.NativeThreadOwner == pdmCritSectGetNativeSelf(pCritSect));
762 return pCritSect->s.Core.cLockers >= pCritSect->s.Core.cNestings;
763}
764
765
766/**
767 * Checks if a critical section is initialized or not.
768 *
769 * @returns true if initialized.
770 * @returns false if not initialized.
771 * @param pCritSect The critical section.
772 */
773VMMDECL(bool) PDMCritSectIsInitialized(PCPDMCRITSECT pCritSect)
774{
775 return RTCritSectIsInitialized(&pCritSect->s.Core);
776}
777
778
779/**
780 * Gets the recursion depth.
781 *
782 * @returns The recursion depth.
783 * @param pCritSect The critical section.
784 */
785VMMDECL(uint32_t) PDMCritSectGetRecursion(PCPDMCRITSECT pCritSect)
786{
787 return RTCritSectGetRecursion(&pCritSect->s.Core);
788}
789
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