VirtualBox

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

Last change on this file since 21258 was 20998, checked in by vboxsync, 16 years ago

VMM: how to trigger the 64-bit windows issue.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 16.8 KB
Line 
1/* $Id: PDMAllCritSect.cpp 20998 2009-06-26 23:01:02Z vboxsync $ */
2/** @file
3 * PDM - Critical Sections, All Contexts.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_PDM//_CRITSECT
27#include "PDMInternal.h"
28#include <VBox/pdmcritsect.h>
29#include <VBox/mm.h>
30#include <VBox/vmm.h>
31#include <VBox/vm.h>
32#include <VBox/err.h>
33#include <VBox/hwaccm.h>
34
35#include <VBox/log.h>
36#include <iprt/asm.h>
37#include <iprt/assert.h>
38#ifdef IN_RING3
39# include <iprt/semaphore.h>
40#endif
41
42
43/*******************************************************************************
44* Defined Constants And Macros *
45*******************************************************************************/
46/** The number loops to spin for in ring-3. */
47#define PDMCRITSECT_SPIN_COUNT_R3 20
48/** The number loops to spin for in ring-0. */
49#define PDMCRITSECT_SPIN_COUNT_R0 256
50/** The number loops to spin for in the raw-mode context. */
51#define PDMCRITSECT_SPIN_COUNT_RC 256
52
53/** @def PDMCRITSECT_STRICT
54 * Enables/disables PDM critsect strictness like deadlock detection. */
55#if defined(VBOX_STRICT) || defined(DOXYGEN_RUNNING)
56# define PDMCRITSECT_STRICT
57#endif
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(PPDMCRITSECT 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 VERR_SEM_DESTROYED);
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 wont 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)
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 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, hNativeSelf);
97
98# if defined(PDMCRITSECT_STRICT) && defined(IN_RING3)
99 pCritSect->s.Core.Strict.pszEnterFile = NULL;
100 pCritSect->s.Core.Strict.u32EnterLine = 0;
101 pCritSect->s.Core.Strict.uEnterId = 0;
102 RTTHREAD hSelf = RTThreadSelf();
103 ASMAtomicWriteHandle(&pCritSect->s.Core.Strict.ThreadOwner, hSelf);
104 RTThreadWriteLockInc(hSelf);
105# endif
106
107 STAM_PROFILE_ADV_START(&pCritSect->s.StatLocked, l);
108 return VINF_SUCCESS;
109}
110
111
112#ifdef IN_RING3
113/**
114 * Deals with the contended case in ring-3.
115 *
116 * @returns VINF_SUCCESS or VERR_SEM_DESTROYED.
117 * @param pCritSect The critsect.
118 * @param hNativeSelf The native thread handle.
119 */
120static int pdmR3CritSectEnterContended(PPDMCRITSECT pCritSect, RTNATIVETHREAD hNativeSelf)
121{
122 /*
123 * Start waiting.
124 */
125 if (ASMAtomicIncS32(&pCritSect->s.Core.cLockers) == 0)
126 return pdmCritSectEnterFirst(pCritSect, hNativeSelf);
127 STAM_COUNTER_INC(&pCritSect->s.StatContentionR3);
128
129 /*
130 * The wait loop.
131 */
132 PSUPDRVSESSION pSession = pCritSect->s.CTX_SUFF(pVM)->pSession;
133 SUPSEMEVENT hEvent = (SUPSEMEVENT)pCritSect->s.Core.EventSem;
134# ifdef PDMCRITSECT_STRICT
135 RTTHREAD hSelf = RTThreadSelf();
136 if (hSelf == NIL_RTTHREAD)
137 RTThreadAdopt(RTTHREADTYPE_DEFAULT, 0, NULL, &hSelf);
138# endif
139 for (;;)
140 {
141# ifdef PDMCRITSECT_STRICT
142 RTThreadBlocking(hSelf, RTTHREADSTATE_CRITSECT, (uintptr_t)pCritSect, NULL, 0, 0);
143# endif
144 int rc = SUPSemEventWaitNoResume(pSession, hEvent, RT_INDEFINITE_WAIT);
145# ifdef PDMCRITSECT_STRICT
146 RTThreadUnblocked(hSelf, RTTHREADSTATE_CRITSECT);
147# endif
148 if (RT_UNLIKELY(pCritSect->s.Core.u32Magic != RTCRITSECT_MAGIC))
149 return VERR_SEM_DESTROYED;
150 if (rc == VINF_SUCCESS)
151 return pdmCritSectEnterFirst(pCritSect, hNativeSelf);
152 AssertMsg(rc == VERR_INTERRUPTED, ("rc=%Rrc\n", rc));
153 }
154 /* won't get here */
155}
156#endif /* IN_RING3 */
157
158
159/**
160 * Enters a PDM critical section.
161 *
162 * @returns VINF_SUCCESS if entered successfully.
163 * @returns rcBusy when encountering a busy critical section in GC/R0.
164 * @returns VERR_SEM_DESTROYED if the critical section is dead.
165 *
166 * @param pCritSect The PDM critical section to enter.
167 * @param rcBusy The status code to return when we're in GC or R0
168 * and the section is busy.
169 */
170VMMDECL(int) PDMCritSectEnter(PPDMCRITSECT pCritSect, int rcBusy)
171{
172 Assert(pCritSect->s.Core.cNestings < 8); /* useful to catch incorrect locking */
173
174 /*
175 * If the critical section has already been destroyed, then inform the caller.
176 */
177 AssertReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC, VERR_SEM_DESTROYED);
178
179 /*
180 * See if we're lucky.
181 */
182 RTNATIVETHREAD hNativeSelf = pdmCritSectGetNativeSelf(pCritSect);
183 /* Not owned ... */
184 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
185 return pdmCritSectEnterFirst(pCritSect, hNativeSelf);
186
187 /* ... or nested. */
188 if (pCritSect->s.Core.NativeThreadOwner == hNativeSelf)
189 {
190 ASMAtomicIncS32(&pCritSect->s.Core.cLockers);
191 ASMAtomicIncS32(&pCritSect->s.Core.cNestings);
192 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
193 return VINF_SUCCESS;
194 }
195
196 /*
197 * Spin for a bit without incrementing the counter.
198 */
199 /** @todo Move this to cfgm variables since it doesn't make sense to spin on UNI
200 * cpu systems. */
201 int32_t cSpinsLeft = CTX_SUFF(PDMCRITSECT_SPIN_COUNT_);
202 while (cSpinsLeft-- > 0)
203 {
204 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
205 return pdmCritSectEnterFirst(pCritSect, hNativeSelf);
206 /** @todo need pause/nop instruction here! */
207 /** @todo Should use monitor/mwait on e.g. &cLockers here, possibly with a
208 cli'ed pendingpreemption check up front using sti w/ instruction fusing
209 for avoiding races. Hmm ... This is assuming the other party is actually
210 executing code on another CPU... */
211 }
212
213#ifdef IN_RING3
214 /*
215 * Take the slow path.
216 */
217 return pdmR3CritSectEnterContended(pCritSect, hNativeSelf);
218#else
219 /*
220 * Return busy.
221 */
222 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZLock);
223 LogFlow(("PDMCritSectEnter: locked => R3 (%Rrc)\n", rcBusy));
224 return rcBusy;
225#endif
226}
227
228
229/**
230 * Try enter a critical section.
231 *
232 * @retval VINF_SUCCESS on success.
233 * @retval VERR_SEM_BUSY if the critsect was owned.
234 * @retval VERR_SEM_NESTED if nested enter on a no nesting section. (Asserted.)
235 * @retval VERR_SEM_DESTROYED if RTCritSectDelete was called while waiting.
236 *
237 * @param pCritSect The critical section.
238 */
239VMMDECL(int) PDMCritSectTryEnter(PPDMCRITSECT pCritSect)
240{
241 /*
242 * If the critical section has already been destroyed, then inform the caller.
243 */
244 AssertReturn(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC, VERR_SEM_DESTROYED);
245
246 /*
247 * See if we're lucky.
248 */
249 RTNATIVETHREAD hNativeSelf = pdmCritSectGetNativeSelf(pCritSect);
250 /* Not owned ... */
251 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, 0, -1))
252 return pdmCritSectEnterFirst(pCritSect, hNativeSelf);
253
254 /* ... or nested. */
255 if (pCritSect->s.Core.NativeThreadOwner == hNativeSelf)
256 {
257 ASMAtomicIncS32(&pCritSect->s.Core.cLockers);
258 ASMAtomicIncS32(&pCritSect->s.Core.cNestings);
259 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
260 return VINF_SUCCESS;
261 }
262
263 /* no spinning */
264
265 /*
266 * Return busy.
267 */
268#ifdef IN_RING3
269 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionR3);
270#else
271 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZLock);
272#endif
273 LogFlow(("PDMCritSectTryEnter: locked\n"));
274 return VERR_SEM_BUSY;
275}
276
277
278#ifdef IN_RING3
279/**
280 * Enters a PDM critical section.
281 *
282 * @returns VINF_SUCCESS if entered successfully.
283 * @returns rcBusy when encountering a busy critical section in GC/R0.
284 * @returns VERR_SEM_DESTROYED if the critical section is dead.
285 *
286 * @param pCritSect The PDM critical section to enter.
287 * @param fCallRing3 Whether this is a VMMRZCallRing3()request.
288 */
289VMMR3DECL(int) PDMR3CritSectEnterEx(PPDMCRITSECT pCritSect, bool fCallRing3)
290{
291 int rc = PDMCritSectEnter(pCritSect, VERR_INTERNAL_ERROR);
292 if ( rc == VINF_SUCCESS
293 && fCallRing3
294 && pCritSect->s.Core.Strict.ThreadOwner != NIL_RTTHREAD)
295 {
296 RTThreadWriteLockDec(pCritSect->s.Core.Strict.ThreadOwner);
297 ASMAtomicWriteHandle(&pCritSect->s.Core.Strict.ThreadOwner, NIL_RTTHREAD);
298 }
299 return rc;
300}
301#endif /* IN_RING3 */
302
303
304/**
305 * Leaves a critical section entered with PDMCritSectEnter().
306 *
307 * @param pCritSect The PDM critical section to leave.
308 */
309VMMDECL(void) PDMCritSectLeave(PPDMCRITSECT pCritSect)
310{
311 Assert(pCritSect->s.Core.u32Magic == RTCRITSECT_MAGIC);
312 Assert(pCritSect->s.Core.NativeThreadOwner == pdmCritSectGetNativeSelf(pCritSect));
313 Assert(pCritSect->s.Core.cNestings >= 1);
314
315 /*
316 * Nested leave.
317 */
318 if (pCritSect->s.Core.cNestings > 1)
319 {
320 ASMAtomicDecS32(&pCritSect->s.Core.cNestings);
321 ASMAtomicDecS32(&pCritSect->s.Core.cLockers);
322 return;
323 }
324
325#if defined(IN_RING3) || defined(IN_RING0)
326 /*
327 * Leave for real.
328 */
329 /* update members. */
330# ifdef IN_RING3
331 RTSEMEVENT hEventToSignal = pCritSect->s.EventToSignal;
332 pCritSect->s.EventToSignal = NIL_RTSEMEVENT;
333# if defined(PDMCRITSECT_STRICT)
334 if (pCritSect->s.Core.Strict.ThreadOwner != NIL_RTTHREAD)
335 RTThreadWriteLockDec(pCritSect->s.Core.Strict.ThreadOwner);
336 ASMAtomicWriteHandle(&pCritSect->s.Core.Strict.ThreadOwner, NIL_RTTHREAD);
337# endif
338# endif
339 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
340 Assert(pCritSect->s.Core.Strict.ThreadOwner == NIL_RTTHREAD);
341 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, NIL_RTNATIVETHREAD);
342 ASMAtomicDecS32(&pCritSect->s.Core.cNestings);
343
344 /* stop and decrement lockers. */
345 STAM_PROFILE_ADV_STOP(&pCritSect->s.StatLocked, l);
346 ASMCompilerBarrier();
347 if (ASMAtomicDecS32(&pCritSect->s.Core.cLockers) >= 0)
348 {
349 /* Someone is waiting, wake up one of them. */
350 SUPSEMEVENT hEvent = (SUPSEMEVENT)pCritSect->s.Core.EventSem;
351 PSUPDRVSESSION pSession = pCritSect->s.CTX_SUFF(pVM)->pSession;
352 int rc = SUPSemEventSignal(pSession, hEvent);
353 AssertRC(rc);
354 }
355
356# ifdef IN_RING3
357 /* Signal exit event. */
358 if (hEventToSignal != NIL_RTSEMEVENT)
359 {
360 LogBird(("Signalling %#x\n", hEventToSignal));
361 int rc = RTSemEventSignal(hEventToSignal);
362 AssertRC(rc);
363 }
364# endif
365
366# if defined(DEBUG_bird) && defined(IN_RING0)
367 VMMTrashVolatileXMMRegs();
368# endif
369
370#else /* IN_RC */
371 /*
372 * Try leave it.
373 */
374 if (pCritSect->s.Core.cLockers == 0)
375 {
376 ASMAtomicWriteS32(&pCritSect->s.Core.cNestings, 0);
377 RTNATIVETHREAD hNativeThread = pCritSect->s.Core.NativeThreadOwner;
378 ASMAtomicAndU32(&pCritSect->s.Core.fFlags, ~PDMCRITSECT_FLAGS_PENDING_UNLOCK);
379 STAM_PROFILE_ADV_STOP(&pCritSect->s.StatLocked, l);
380
381 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, NIL_RTNATIVETHREAD);
382 if (ASMAtomicCmpXchgS32(&pCritSect->s.Core.cLockers, -1, 0))
383 return;
384
385 /* darn, someone raced in on us. */
386 ASMAtomicWriteHandle(&pCritSect->s.Core.NativeThreadOwner, hNativeThread);
387 STAM_PROFILE_ADV_START(&pCritSect->s.StatLocked, l);
388 ASMAtomicWriteS32(&pCritSect->s.Core.cNestings, 1);
389 }
390 ASMAtomicOrU32(&pCritSect->s.Core.fFlags, PDMCRITSECT_FLAGS_PENDING_UNLOCK);
391
392 /*
393 * Queue the request.
394 */
395 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
396 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
397 uint32_t i = pVCpu->pdm.s.cQueuedCritSectLeaves++;
398 LogFlow(("PDMCritSectLeave: [%d]=%p => R3\n", i, pCritSect));
399 AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectsLeaves));
400 pVCpu->pdm.s.apQueuedCritSectsLeaves[i] = MMHyperCCToR3(pVM, pCritSect);
401 VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
402 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
403 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
404 STAM_REL_COUNTER_INC(&pCritSect->s.StatContentionRZUnlock);
405#endif /* IN_RC */
406}
407
408
409#if defined(IN_RING3) || defined(IN_RING0)
410/**
411 * Process the critical sections queued for ring-3 'leave'.
412 *
413 * @param pVCpu The VMCPU handle.
414 */
415VMMDECL(void) PDMCritSectFF(PVMCPU pVCpu)
416{
417 Assert(pVCpu->pdm.s.cQueuedCritSectLeaves > 0);
418
419 const RTUINT c = pVCpu->pdm.s.cQueuedCritSectLeaves;
420 for (RTUINT i = 0; i < c; i++)
421 {
422# ifdef IN_RING3
423 PPDMCRITSECT pCritSect = pVCpu->pdm.s.apQueuedCritSectsLeaves[i];
424# else
425 PPDMCRITSECT pCritSect = (PPDMCRITSECT)MMHyperR3ToCC(pVCpu->CTX_SUFF(pVM), pVCpu->pdm.s.apQueuedCritSectsLeaves[i]);
426# endif
427
428 PDMCritSectLeave(pCritSect);
429 LogFlow(("PDMR3CritSectFF: %p\n", pCritSect));
430 }
431
432 pVCpu->pdm.s.cQueuedCritSectLeaves = 0;
433 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_PDM_CRITSECT);
434}
435#endif /* IN_RING3 || IN_RING0 */
436
437
438/**
439 * Checks the caller is the owner of the critical section.
440 *
441 * @returns true if owner.
442 * @returns false if not owner.
443 * @param pCritSect The critical section.
444 */
445VMMDECL(bool) PDMCritSectIsOwner(PCPDMCRITSECT pCritSect)
446{
447#ifdef IN_RING3
448 return RTCritSectIsOwner(&pCritSect->s.Core);
449#else
450 PVM pVM = pCritSect->s.CTX_SUFF(pVM); AssertPtr(pVM);
451 PVMCPU pVCpu = VMMGetCpu(pVM); AssertPtr(pVCpu);
452 if (pCritSect->s.Core.NativeThreadOwner != pVCpu->hNativeThread)
453 return false;
454 return (pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK) == 0;
455#endif
456}
457
458
459/**
460 * Checks the specified VCPU is the owner of the critical section.
461 *
462 * @returns true if owner.
463 * @returns false if not owner.
464 * @param pCritSect The critical section.
465 * @param idCpu VCPU id
466 */
467VMMDECL(bool) PDMCritSectIsOwnerEx(PCPDMCRITSECT pCritSect, VMCPUID idCpu)
468{
469#ifdef IN_RING3
470 NOREF(idCpu);
471 return RTCritSectIsOwner(&pCritSect->s.Core);
472#else
473 PVM pVM = pCritSect->s.CTX_SUFF(pVM);
474 AssertPtr(pVM);
475 Assert(idCpu < pVM->cCPUs);
476 return pCritSect->s.Core.NativeThreadOwner == pVM->aCpus[idCpu].hNativeThread
477 && (pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK) == 0;
478#endif
479}
480
481
482/**
483 * Checks if somebody currently owns the critical section.
484 *
485 * @returns true if locked.
486 * @returns false if not locked.
487 *
488 * @param pCritSect The critical section.
489 *
490 * @remarks This doesn't prove that no deadlocks will occur later on; it's
491 * just a debugging tool
492 */
493VMMDECL(bool) PDMCritSectIsOwned(PCPDMCRITSECT pCritSect)
494{
495 return pCritSect->s.Core.NativeThreadOwner != NIL_RTNATIVETHREAD
496 && (pCritSect->s.Core.fFlags & PDMCRITSECT_FLAGS_PENDING_UNLOCK) == 0;
497}
498
499
500/**
501 * Checks if a critical section is initialized or not.
502 *
503 * @returns true if initialized.
504 * @returns false if not initialized.
505 * @param pCritSect The critical section.
506 */
507VMMDECL(bool) PDMCritSectIsInitialized(PCPDMCRITSECT pCritSect)
508{
509 return RTCritSectIsInitialized(&pCritSect->s.Core);
510}
511
512
513/**
514 * Gets the recursion depth.
515 *
516 * @returns The recursion depth.
517 * @param pCritSect The critical section.
518 */
519VMMDECL(uint32_t) PDMCritSectGetRecursion(PCPDMCRITSECT pCritSect)
520{
521 return RTCritSectGetRecursion(&pCritSect->s.Core);
522}
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