VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/VMEmt.cpp@ 67073

Last change on this file since 67073 was 66156, checked in by vboxsync, 8 years ago

vmR3EmulationThreadWithId: Shadow variable warning fix.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 48.3 KB
Line 
1/* $Id: VMEmt.cpp 66156 2017-03-17 14:41:29Z vboxsync $ */
2/** @file
3 * VM - Virtual Machine, The Emulation Thread.
4 */
5
6/*
7 * Copyright (C) 2006-2016 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_VM
23#include <VBox/vmm/tm.h>
24#include <VBox/vmm/dbgf.h>
25#include <VBox/vmm/em.h>
26#include <VBox/vmm/pdmapi.h>
27#ifdef VBOX_WITH_REM
28# include <VBox/vmm/rem.h>
29#endif
30#include <VBox/vmm/tm.h>
31#include "VMInternal.h"
32#include <VBox/vmm/vm.h>
33#include <VBox/vmm/uvm.h>
34
35#include <VBox/err.h>
36#include <VBox/log.h>
37#include <iprt/assert.h>
38#include <iprt/asm.h>
39#include <iprt/asm-math.h>
40#include <iprt/semaphore.h>
41#include <iprt/string.h>
42#include <iprt/thread.h>
43#include <iprt/time.h>
44
45
46/*********************************************************************************************************************************
47* Internal Functions *
48*********************************************************************************************************************************/
49int vmR3EmulationThreadWithId(RTTHREAD hThreadSelf, PUVMCPU pUVCpu, VMCPUID idCpu);
50
51
52/**
53 * The emulation thread main function.
54 *
55 * @returns Thread exit code.
56 * @param hThreadSelf The handle to the executing thread.
57 * @param pvArgs Pointer to the user mode per-VCpu structure (UVMPCU).
58 */
59DECLCALLBACK(int) vmR3EmulationThread(RTTHREAD hThreadSelf, void *pvArgs)
60{
61 PUVMCPU pUVCpu = (PUVMCPU)pvArgs;
62 return vmR3EmulationThreadWithId(hThreadSelf, pUVCpu, pUVCpu->idCpu);
63}
64
65
66/**
67 * The emulation thread main function, with Virtual CPU ID for debugging.
68 *
69 * @returns Thread exit code.
70 * @param hThreadSelf The handle to the executing thread.
71 * @param pUVCpu Pointer to the user mode per-VCpu structure.
72 * @param idCpu The virtual CPU ID, for backtrace purposes.
73 */
74int vmR3EmulationThreadWithId(RTTHREAD hThreadSelf, PUVMCPU pUVCpu, VMCPUID idCpu)
75{
76 PUVM pUVM = pUVCpu->pUVM;
77 int rc;
78 RT_NOREF_PV(hThreadSelf);
79
80 AssertReleaseMsg(VALID_PTR(pUVM) && pUVM->u32Magic == UVM_MAGIC,
81 ("Invalid arguments to the emulation thread!\n"));
82
83 rc = RTTlsSet(pUVM->vm.s.idxTLS, pUVCpu);
84 AssertReleaseMsgRCReturn(rc, ("RTTlsSet %x failed with %Rrc\n", pUVM->vm.s.idxTLS, rc), rc);
85
86 if ( pUVM->pVmm2UserMethods
87 && pUVM->pVmm2UserMethods->pfnNotifyEmtInit)
88 pUVM->pVmm2UserMethods->pfnNotifyEmtInit(pUVM->pVmm2UserMethods, pUVM, pUVCpu);
89
90 /*
91 * The request loop.
92 */
93 rc = VINF_SUCCESS;
94 Log(("vmR3EmulationThread: Emulation thread starting the days work... Thread=%#x pUVM=%p\n", hThreadSelf, pUVM));
95 VMSTATE enmBefore = VMSTATE_CREATED; /* (only used for logging atm.) */
96 for (;;)
97 {
98 /*
99 * During early init there is no pVM and/or pVCpu, so make a special path
100 * for that to keep things clearly separate.
101 */
102 PVM pVM = pUVM->pVM;
103 PVMCPU pVCpu = pUVCpu->pVCpu;
104 if (!pVCpu || !pVM)
105 {
106 /*
107 * Check for termination first.
108 */
109 if (pUVM->vm.s.fTerminateEMT)
110 {
111 rc = VINF_EM_TERMINATE;
112 break;
113 }
114
115 /*
116 * Only the first VCPU may initialize the VM during early init
117 * and must therefore service all VMCPUID_ANY requests.
118 * See also VMR3Create
119 */
120 if ( (pUVM->vm.s.pNormalReqs || pUVM->vm.s.pPriorityReqs)
121 && pUVCpu->idCpu == 0)
122 {
123 /*
124 * Service execute in any EMT request.
125 */
126 rc = VMR3ReqProcessU(pUVM, VMCPUID_ANY, false /*fPriorityOnly*/);
127 Log(("vmR3EmulationThread: Req rc=%Rrc, VM state %s -> %s\n", rc, VMR3GetStateName(enmBefore), pUVM->pVM ? VMR3GetStateName(pUVM->pVM->enmVMState) : "CREATING"));
128 }
129 else if (pUVCpu->vm.s.pNormalReqs || pUVCpu->vm.s.pPriorityReqs)
130 {
131 /*
132 * Service execute in specific EMT request.
133 */
134 rc = VMR3ReqProcessU(pUVM, pUVCpu->idCpu, false /*fPriorityOnly*/);
135 Log(("vmR3EmulationThread: Req (cpu=%u) rc=%Rrc, VM state %s -> %s\n", pUVCpu->idCpu, rc, VMR3GetStateName(enmBefore), pUVM->pVM ? VMR3GetStateName(pUVM->pVM->enmVMState) : "CREATING"));
136 }
137 else
138 {
139 /*
140 * Nothing important is pending, so wait for something.
141 */
142 rc = VMR3WaitU(pUVCpu);
143 if (RT_FAILURE(rc))
144 {
145 AssertLogRelMsgFailed(("VMR3WaitU failed with %Rrc\n", rc));
146 break;
147 }
148 }
149 }
150 else
151 {
152 /*
153 * Pending requests which needs servicing?
154 *
155 * We check for state changes in addition to status codes when
156 * servicing requests. (Look after the ifs.)
157 */
158 enmBefore = pVM->enmVMState;
159 if (pUVM->vm.s.fTerminateEMT)
160 {
161 rc = VINF_EM_TERMINATE;
162 break;
163 }
164
165 if (VM_FF_IS_PENDING(pVM, VM_FF_EMT_RENDEZVOUS))
166 {
167 rc = VMMR3EmtRendezvousFF(pVM, &pVM->aCpus[idCpu]);
168 Log(("vmR3EmulationThread: Rendezvous rc=%Rrc, VM state %s -> %s\n", rc, VMR3GetStateName(enmBefore), VMR3GetStateName(pVM->enmVMState)));
169 }
170 else if (pUVM->vm.s.pNormalReqs || pUVM->vm.s.pPriorityReqs)
171 {
172 /*
173 * Service execute in any EMT request.
174 */
175 rc = VMR3ReqProcessU(pUVM, VMCPUID_ANY, false /*fPriorityOnly*/);
176 Log(("vmR3EmulationThread: Req rc=%Rrc, VM state %s -> %s\n", rc, VMR3GetStateName(enmBefore), VMR3GetStateName(pVM->enmVMState)));
177 }
178 else if (pUVCpu->vm.s.pNormalReqs || pUVCpu->vm.s.pPriorityReqs)
179 {
180 /*
181 * Service execute in specific EMT request.
182 */
183 rc = VMR3ReqProcessU(pUVM, pUVCpu->idCpu, false /*fPriorityOnly*/);
184 Log(("vmR3EmulationThread: Req (cpu=%u) rc=%Rrc, VM state %s -> %s\n", pUVCpu->idCpu, rc, VMR3GetStateName(enmBefore), VMR3GetStateName(pVM->enmVMState)));
185 }
186 else if ( VM_FF_IS_SET(pVM, VM_FF_DBGF)
187 || VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_DBGF))
188 {
189 /*
190 * Service the debugger request.
191 */
192 rc = DBGFR3VMMForcedAction(pVM, pVCpu);
193 Log(("vmR3EmulationThread: Dbg rc=%Rrc, VM state %s -> %s\n", rc, VMR3GetStateName(enmBefore), VMR3GetStateName(pVM->enmVMState)));
194 }
195 else if (VM_FF_TEST_AND_CLEAR(pVM, VM_FF_RESET))
196 {
197 /*
198 * Service a delayed reset request.
199 */
200 rc = VBOXSTRICTRC_VAL(VMR3ResetFF(pVM));
201 VM_FF_CLEAR(pVM, VM_FF_RESET);
202 Log(("vmR3EmulationThread: Reset rc=%Rrc, VM state %s -> %s\n", rc, VMR3GetStateName(enmBefore), VMR3GetStateName(pVM->enmVMState)));
203 }
204 else
205 {
206 /*
207 * Nothing important is pending, so wait for something.
208 */
209 rc = VMR3WaitU(pUVCpu);
210 if (RT_FAILURE(rc))
211 {
212 AssertLogRelMsgFailed(("VMR3WaitU failed with %Rrc\n", rc));
213 break;
214 }
215 }
216
217 /*
218 * Check for termination requests, these have extremely high priority.
219 */
220 if ( rc == VINF_EM_TERMINATE
221 || pUVM->vm.s.fTerminateEMT)
222 break;
223 }
224
225 /*
226 * Some requests (both VMR3Req* and the DBGF) can potentially resume
227 * or start the VM, in that case we'll get a change in VM status
228 * indicating that we're now running.
229 */
230 if (RT_SUCCESS(rc))
231 {
232 pVM = pUVM->pVM;
233 if (pVM)
234 {
235 pVCpu = &pVM->aCpus[idCpu];
236 if ( pVM->enmVMState == VMSTATE_RUNNING
237 && VMCPUSTATE_IS_STARTED(VMCPU_GET_STATE(pVCpu)))
238 {
239 rc = EMR3ExecuteVM(pVM, pVCpu);
240 Log(("vmR3EmulationThread: EMR3ExecuteVM() -> rc=%Rrc, enmVMState=%d\n", rc, pVM->enmVMState));
241 }
242 }
243 }
244
245 } /* forever */
246
247
248 /*
249 * Cleanup and exit.
250 */
251 Log(("vmR3EmulationThread: Terminating emulation thread! Thread=%#x pUVM=%p rc=%Rrc enmBefore=%d enmVMState=%d\n",
252 hThreadSelf, pUVM, rc, enmBefore, pUVM->pVM ? pUVM->pVM->enmVMState : VMSTATE_TERMINATED));
253 PVM pVM;
254 if ( idCpu == 0
255 && (pVM = pUVM->pVM) != NULL)
256 {
257 /* Wait for any other EMTs to terminate before we destroy the VM (see vmR3DestroyVM). */
258 for (VMCPUID iCpu = 1; iCpu < pUVM->cCpus; iCpu++)
259 {
260 RTTHREAD hThread;
261 ASMAtomicXchgHandle(&pUVM->aCpus[iCpu].vm.s.ThreadEMT, NIL_RTTHREAD, &hThread);
262 if (hThread != NIL_RTTHREAD)
263 {
264 int rc2 = RTThreadWait(hThread, 5 * RT_MS_1SEC, NULL);
265 AssertLogRelMsgRC(rc2, ("iCpu=%u rc=%Rrc\n", iCpu, rc2));
266 if (RT_FAILURE(rc2))
267 pUVM->aCpus[iCpu].vm.s.ThreadEMT = hThread;
268 }
269 }
270
271 /* Switch to the terminated state, clearing the VM pointer and finally destroy the VM. */
272 vmR3SetTerminated(pVM);
273
274 pUVM->pVM = NULL;
275
276 int rc2 = SUPR3CallVMMR0Ex(pVM->pVMR0, 0 /*idCpu*/, VMMR0_DO_GVMM_DESTROY_VM, 0, NULL);
277 AssertLogRelRC(rc2);
278 }
279
280 if ( pUVM->pVmm2UserMethods
281 && pUVM->pVmm2UserMethods->pfnNotifyEmtTerm)
282 pUVM->pVmm2UserMethods->pfnNotifyEmtTerm(pUVM->pVmm2UserMethods, pUVM, pUVCpu);
283
284 pUVCpu->vm.s.NativeThreadEMT = NIL_RTNATIVETHREAD;
285 Log(("vmR3EmulationThread: EMT is terminated.\n"));
286 return rc;
287}
288
289
290/**
291 * Gets the name of a halt method.
292 *
293 * @returns Pointer to a read only string.
294 * @param enmMethod The method.
295 */
296static const char *vmR3GetHaltMethodName(VMHALTMETHOD enmMethod)
297{
298 switch (enmMethod)
299 {
300 case VMHALTMETHOD_BOOTSTRAP: return "bootstrap";
301 case VMHALTMETHOD_DEFAULT: return "default";
302 case VMHALTMETHOD_OLD: return "old";
303 case VMHALTMETHOD_1: return "method1";
304 //case VMHALTMETHOD_2: return "method2";
305 case VMHALTMETHOD_GLOBAL_1: return "global1";
306 default: return "unknown";
307 }
308}
309
310
311/**
312 * Signal a fatal wait error.
313 *
314 * @returns Fatal error code to be propagated up the call stack.
315 * @param pUVCpu The user mode per CPU structure of the calling
316 * EMT.
317 * @param pszFmt The error format with a single %Rrc in it.
318 * @param rcFmt The status code to format.
319 */
320static int vmR3FatalWaitError(PUVMCPU pUVCpu, const char *pszFmt, int rcFmt)
321{
322 /** @todo This is wrong ... raise a fatal error / guru meditation
323 * instead. */
324 AssertLogRelMsgFailed((pszFmt, rcFmt));
325 ASMAtomicUoWriteBool(&pUVCpu->pUVM->vm.s.fTerminateEMT, true);
326 if (pUVCpu->pVM)
327 VM_FF_SET(pUVCpu->pVM, VM_FF_CHECK_VM_STATE);
328 return VERR_VM_FATAL_WAIT_ERROR;
329}
330
331
332/**
333 * The old halt loop.
334 */
335static DECLCALLBACK(int) vmR3HaltOldDoHalt(PUVMCPU pUVCpu, const uint32_t fMask, uint64_t /* u64Now*/)
336{
337 /*
338 * Halt loop.
339 */
340 PVM pVM = pUVCpu->pVM;
341 PVMCPU pVCpu = pUVCpu->pVCpu;
342
343 int rc = VINF_SUCCESS;
344 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
345 //unsigned cLoops = 0;
346 for (;;)
347 {
348 /*
349 * Work the timers and check if we can exit.
350 * The poll call gives us the ticks left to the next event in
351 * addition to perhaps set an FF.
352 */
353 uint64_t const u64StartTimers = RTTimeNanoTS();
354 TMR3TimerQueuesDo(pVM);
355 uint64_t const cNsElapsedTimers = RTTimeNanoTS() - u64StartTimers;
356 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltTimers, cNsElapsedTimers);
357 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
358 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
359 break;
360 uint64_t u64NanoTS;
361 TMTimerPollGIP(pVM, pVCpu, &u64NanoTS);
362 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
363 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
364 break;
365
366 /*
367 * Wait for a while. Someone will wake us up or interrupt the call if
368 * anything needs our attention.
369 */
370 if (u64NanoTS < 50000)
371 {
372 //RTLogPrintf("u64NanoTS=%RI64 cLoops=%d spin\n", u64NanoTS, cLoops++);
373 /* spin */;
374 }
375 else
376 {
377 VMMR3YieldStop(pVM);
378 //uint64_t u64Start = RTTimeNanoTS();
379 if (u64NanoTS < 870000) /* this is a bit speculative... works fine on linux. */
380 {
381 //RTLogPrintf("u64NanoTS=%RI64 cLoops=%d yield", u64NanoTS, cLoops++);
382 uint64_t const u64StartSchedYield = RTTimeNanoTS();
383 RTThreadYield(); /* this is the best we can do here */
384 uint64_t const cNsElapsedSchedYield = RTTimeNanoTS() - u64StartSchedYield;
385 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltYield, cNsElapsedSchedYield);
386 }
387 else if (u64NanoTS < 2000000)
388 {
389 //RTLogPrintf("u64NanoTS=%RI64 cLoops=%d sleep 1ms", u64NanoTS, cLoops++);
390 uint64_t const u64StartSchedHalt = RTTimeNanoTS();
391 rc = RTSemEventWait(pUVCpu->vm.s.EventSemWait, 1);
392 uint64_t const cNsElapsedSchedHalt = RTTimeNanoTS() - u64StartSchedHalt;
393 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlock, cNsElapsedSchedHalt);
394 }
395 else
396 {
397 //RTLogPrintf("u64NanoTS=%RI64 cLoops=%d sleep %dms", u64NanoTS, cLoops++, (uint32_t)RT_MIN((u64NanoTS - 500000) / 1000000, 15));
398 uint64_t const u64StartSchedHalt = RTTimeNanoTS();
399 rc = RTSemEventWait(pUVCpu->vm.s.EventSemWait, RT_MIN((u64NanoTS - 1000000) / 1000000, 15));
400 uint64_t const cNsElapsedSchedHalt = RTTimeNanoTS() - u64StartSchedHalt;
401 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlock, cNsElapsedSchedHalt);
402 }
403 //uint64_t u64Slept = RTTimeNanoTS() - u64Start;
404 //RTLogPrintf(" -> rc=%Rrc in %RU64 ns / %RI64 ns delta\n", rc, u64Slept, u64NanoTS - u64Slept);
405 }
406 if (rc == VERR_TIMEOUT)
407 rc = VINF_SUCCESS;
408 else if (RT_FAILURE(rc))
409 {
410 rc = vmR3FatalWaitError(pUVCpu, "RTSemEventWait->%Rrc\n", rc);
411 break;
412 }
413 }
414
415 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
416 return rc;
417}
418
419
420/**
421 * Initialize the configuration of halt method 1 & 2.
422 *
423 * @return VBox status code. Failure on invalid CFGM data.
424 * @param pUVM The user mode VM structure.
425 */
426static int vmR3HaltMethod12ReadConfigU(PUVM pUVM)
427{
428 /*
429 * The defaults.
430 */
431#if 1 /* DEBUGGING STUFF - REMOVE LATER */
432 pUVM->vm.s.Halt.Method12.u32LagBlockIntervalDivisorCfg = 4;
433 pUVM->vm.s.Halt.Method12.u32MinBlockIntervalCfg = 2*1000000;
434 pUVM->vm.s.Halt.Method12.u32MaxBlockIntervalCfg = 75*1000000;
435 pUVM->vm.s.Halt.Method12.u32StartSpinningCfg = 30*1000000;
436 pUVM->vm.s.Halt.Method12.u32StopSpinningCfg = 20*1000000;
437#else
438 pUVM->vm.s.Halt.Method12.u32LagBlockIntervalDivisorCfg = 4;
439 pUVM->vm.s.Halt.Method12.u32MinBlockIntervalCfg = 5*1000000;
440 pUVM->vm.s.Halt.Method12.u32MaxBlockIntervalCfg = 200*1000000;
441 pUVM->vm.s.Halt.Method12.u32StartSpinningCfg = 20*1000000;
442 pUVM->vm.s.Halt.Method12.u32StopSpinningCfg = 2*1000000;
443#endif
444
445 /*
446 * Query overrides.
447 *
448 * I don't have time to bother with niceties such as invalid value checks
449 * here right now. sorry.
450 */
451 PCFGMNODE pCfg = CFGMR3GetChild(CFGMR3GetRoot(pUVM->pVM), "/VMM/HaltedMethod1");
452 if (pCfg)
453 {
454 uint32_t u32;
455 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "LagBlockIntervalDivisor", &u32)))
456 pUVM->vm.s.Halt.Method12.u32LagBlockIntervalDivisorCfg = u32;
457 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "MinBlockInterval", &u32)))
458 pUVM->vm.s.Halt.Method12.u32MinBlockIntervalCfg = u32;
459 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "MaxBlockInterval", &u32)))
460 pUVM->vm.s.Halt.Method12.u32MaxBlockIntervalCfg = u32;
461 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "StartSpinning", &u32)))
462 pUVM->vm.s.Halt.Method12.u32StartSpinningCfg = u32;
463 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "StopSpinning", &u32)))
464 pUVM->vm.s.Halt.Method12.u32StopSpinningCfg = u32;
465 LogRel(("VMEmt: HaltedMethod1 config: %d/%d/%d/%d/%d\n",
466 pUVM->vm.s.Halt.Method12.u32LagBlockIntervalDivisorCfg,
467 pUVM->vm.s.Halt.Method12.u32MinBlockIntervalCfg,
468 pUVM->vm.s.Halt.Method12.u32MaxBlockIntervalCfg,
469 pUVM->vm.s.Halt.Method12.u32StartSpinningCfg,
470 pUVM->vm.s.Halt.Method12.u32StopSpinningCfg));
471 }
472
473 return VINF_SUCCESS;
474}
475
476
477/**
478 * Initialize halt method 1.
479 *
480 * @return VBox status code.
481 * @param pUVM Pointer to the user mode VM structure.
482 */
483static DECLCALLBACK(int) vmR3HaltMethod1Init(PUVM pUVM)
484{
485 return vmR3HaltMethod12ReadConfigU(pUVM);
486}
487
488
489/**
490 * Method 1 - Block whenever possible, and when lagging behind
491 * switch to spinning for 10-30ms with occasional blocking until
492 * the lag has been eliminated.
493 */
494static DECLCALLBACK(int) vmR3HaltMethod1Halt(PUVMCPU pUVCpu, const uint32_t fMask, uint64_t u64Now)
495{
496 PUVM pUVM = pUVCpu->pUVM;
497 PVMCPU pVCpu = pUVCpu->pVCpu;
498 PVM pVM = pUVCpu->pVM;
499
500 /*
501 * To simplify things, we decide up-front whether we should switch to spinning or
502 * not. This makes some ASSUMPTIONS about the cause of the spinning (PIT/RTC/PCNet)
503 * and that it will generate interrupts or other events that will cause us to exit
504 * the halt loop.
505 */
506 bool fBlockOnce = false;
507 bool fSpinning = false;
508 uint32_t u32CatchUpPct = TMVirtualSyncGetCatchUpPct(pVM);
509 if (u32CatchUpPct /* non-zero if catching up */)
510 {
511 if (pUVCpu->vm.s.Halt.Method12.u64StartSpinTS)
512 {
513 fSpinning = TMVirtualSyncGetLag(pVM) >= pUVM->vm.s.Halt.Method12.u32StopSpinningCfg;
514 if (fSpinning)
515 {
516 uint64_t u64Lag = TMVirtualSyncGetLag(pVM);
517 fBlockOnce = u64Now - pUVCpu->vm.s.Halt.Method12.u64LastBlockTS
518 > RT_MAX(pUVM->vm.s.Halt.Method12.u32MinBlockIntervalCfg,
519 RT_MIN(u64Lag / pUVM->vm.s.Halt.Method12.u32LagBlockIntervalDivisorCfg,
520 pUVM->vm.s.Halt.Method12.u32MaxBlockIntervalCfg));
521 }
522 else
523 {
524 //RTLogRelPrintf("Stopped spinning (%u ms)\n", (u64Now - pUVCpu->vm.s.Halt.Method12.u64StartSpinTS) / 1000000);
525 pUVCpu->vm.s.Halt.Method12.u64StartSpinTS = 0;
526 }
527 }
528 else
529 {
530 fSpinning = TMVirtualSyncGetLag(pVM) >= pUVM->vm.s.Halt.Method12.u32StartSpinningCfg;
531 if (fSpinning)
532 pUVCpu->vm.s.Halt.Method12.u64StartSpinTS = u64Now;
533 }
534 }
535 else if (pUVCpu->vm.s.Halt.Method12.u64StartSpinTS)
536 {
537 //RTLogRelPrintf("Stopped spinning (%u ms)\n", (u64Now - pUVCpu->vm.s.Halt.Method12.u64StartSpinTS) / 1000000);
538 pUVCpu->vm.s.Halt.Method12.u64StartSpinTS = 0;
539 }
540
541 /*
542 * Halt loop.
543 */
544 int rc = VINF_SUCCESS;
545 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
546 unsigned cLoops = 0;
547 for (;; cLoops++)
548 {
549 /*
550 * Work the timers and check if we can exit.
551 */
552 uint64_t const u64StartTimers = RTTimeNanoTS();
553 TMR3TimerQueuesDo(pVM);
554 uint64_t const cNsElapsedTimers = RTTimeNanoTS() - u64StartTimers;
555 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltTimers, cNsElapsedTimers);
556 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
557 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
558 break;
559
560 /*
561 * Estimate time left to the next event.
562 */
563 uint64_t u64NanoTS;
564 TMTimerPollGIP(pVM, pVCpu, &u64NanoTS);
565 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
566 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
567 break;
568
569 /*
570 * Block if we're not spinning and the interval isn't all that small.
571 */
572 if ( ( !fSpinning
573 || fBlockOnce)
574#if 1 /* DEBUGGING STUFF - REMOVE LATER */
575 && u64NanoTS >= 100000) /* 0.100 ms */
576#else
577 && u64NanoTS >= 250000) /* 0.250 ms */
578#endif
579 {
580 const uint64_t Start = pUVCpu->vm.s.Halt.Method12.u64LastBlockTS = RTTimeNanoTS();
581 VMMR3YieldStop(pVM);
582
583 uint32_t cMilliSecs = RT_MIN(u64NanoTS / 1000000, 15);
584 if (cMilliSecs <= pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLongAvg)
585 cMilliSecs = 1;
586 else
587 cMilliSecs -= pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLongAvg;
588
589 //RTLogRelPrintf("u64NanoTS=%RI64 cLoops=%3d sleep %02dms (%7RU64) ", u64NanoTS, cLoops, cMilliSecs, u64NanoTS);
590 uint64_t const u64StartSchedHalt = RTTimeNanoTS();
591 rc = RTSemEventWait(pUVCpu->vm.s.EventSemWait, cMilliSecs);
592 uint64_t const cNsElapsedSchedHalt = RTTimeNanoTS() - u64StartSchedHalt;
593 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlock, cNsElapsedSchedHalt);
594
595 if (rc == VERR_TIMEOUT)
596 rc = VINF_SUCCESS;
597 else if (RT_FAILURE(rc))
598 {
599 rc = vmR3FatalWaitError(pUVCpu, "RTSemEventWait->%Rrc\n", rc);
600 break;
601 }
602
603 /*
604 * Calc the statistics.
605 * Update averages every 16th time, and flush parts of the history every 64th time.
606 */
607 const uint64_t Elapsed = RTTimeNanoTS() - Start;
608 pUVCpu->vm.s.Halt.Method12.cNSBlocked += Elapsed;
609 if (Elapsed > u64NanoTS)
610 pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLong += Elapsed - u64NanoTS;
611 pUVCpu->vm.s.Halt.Method12.cBlocks++;
612 if (!(pUVCpu->vm.s.Halt.Method12.cBlocks & 0xf))
613 {
614 pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLongAvg = pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLong / pUVCpu->vm.s.Halt.Method12.cBlocks;
615 if (!(pUVCpu->vm.s.Halt.Method12.cBlocks & 0x3f))
616 {
617 pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLong = pUVCpu->vm.s.Halt.Method12.cNSBlockedTooLongAvg * 0x40;
618 pUVCpu->vm.s.Halt.Method12.cBlocks = 0x40;
619 }
620 }
621 //RTLogRelPrintf(" -> %7RU64 ns / %7RI64 ns delta%s\n", Elapsed, Elapsed - u64NanoTS, fBlockOnce ? " (block once)" : "");
622
623 /*
624 * Clear the block once flag if we actually blocked.
625 */
626 if ( fBlockOnce
627 && Elapsed > 100000 /* 0.1 ms */)
628 fBlockOnce = false;
629 }
630 }
631 //if (fSpinning) RTLogRelPrintf("spun for %RU64 ns %u loops; lag=%RU64 pct=%d\n", RTTimeNanoTS() - u64Now, cLoops, TMVirtualSyncGetLag(pVM), u32CatchUpPct);
632
633 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
634 return rc;
635}
636
637
638/**
639 * Initialize the global 1 halt method.
640 *
641 * @return VBox status code.
642 * @param pUVM Pointer to the user mode VM structure.
643 */
644static DECLCALLBACK(int) vmR3HaltGlobal1Init(PUVM pUVM)
645{
646 /*
647 * The defaults.
648 */
649 uint32_t cNsResolution = SUPSemEventMultiGetResolution(pUVM->vm.s.pSession);
650 if (cNsResolution > 5*RT_NS_100US)
651 pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg = 50000;
652 else if (cNsResolution > RT_NS_100US)
653 pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg = cNsResolution / 4;
654 else
655 pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg = 2000;
656
657 /*
658 * Query overrides.
659 *
660 * I don't have time to bother with niceties such as invalid value checks
661 * here right now. sorry.
662 */
663 PCFGMNODE pCfg = CFGMR3GetChild(CFGMR3GetRoot(pUVM->pVM), "/VMM/HaltedGlobal1");
664 if (pCfg)
665 {
666 uint32_t u32;
667 if (RT_SUCCESS(CFGMR3QueryU32(pCfg, "SpinBlockThreshold", &u32)))
668 pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg = u32;
669 }
670 LogRel(("VMEmt: HaltedGlobal1 config: cNsSpinBlockThresholdCfg=%u\n",
671 pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg));
672 return VINF_SUCCESS;
673}
674
675
676/**
677 * The global 1 halt method - Block in GMM (ring-0) and let it
678 * try take care of the global scheduling of EMT threads.
679 */
680static DECLCALLBACK(int) vmR3HaltGlobal1Halt(PUVMCPU pUVCpu, const uint32_t fMask, uint64_t u64Now)
681{
682 PUVM pUVM = pUVCpu->pUVM;
683 PVMCPU pVCpu = pUVCpu->pVCpu;
684 PVM pVM = pUVCpu->pVM;
685 Assert(VMMGetCpu(pVM) == pVCpu);
686 NOREF(u64Now);
687
688 /*
689 * Halt loop.
690 */
691 //uint64_t u64NowLog, u64Start;
692 //u64Start = u64NowLog = RTTimeNanoTS();
693 int rc = VINF_SUCCESS;
694 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
695 unsigned cLoops = 0;
696 for (;; cLoops++)
697 {
698 /*
699 * Work the timers and check if we can exit.
700 */
701 uint64_t const u64StartTimers = RTTimeNanoTS();
702 TMR3TimerQueuesDo(pVM);
703 uint64_t const cNsElapsedTimers = RTTimeNanoTS() - u64StartTimers;
704 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltTimers, cNsElapsedTimers);
705 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
706 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
707 break;
708
709 /*
710 * Estimate time left to the next event.
711 */
712 //u64NowLog = RTTimeNanoTS();
713 uint64_t u64Delta;
714 uint64_t u64GipTime = TMTimerPollGIP(pVM, pVCpu, &u64Delta);
715 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
716 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
717 break;
718
719 /*
720 * Block if we're not spinning and the interval isn't all that small.
721 */
722 if (u64Delta >= pUVM->vm.s.Halt.Global1.cNsSpinBlockThresholdCfg)
723 {
724 VMMR3YieldStop(pVM);
725 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
726 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
727 break;
728
729 //RTLogPrintf("loop=%-3d u64GipTime=%'llu / %'llu now=%'llu / %'llu\n", cLoops, u64GipTime, u64Delta, u64NowLog, u64GipTime - u64NowLog);
730 uint64_t const u64StartSchedHalt = RTTimeNanoTS();
731 rc = SUPR3CallVMMR0Ex(pVM->pVMR0, pVCpu->idCpu, VMMR0_DO_GVMM_SCHED_HALT, u64GipTime, NULL);
732 uint64_t const u64EndSchedHalt = RTTimeNanoTS();
733 uint64_t const cNsElapsedSchedHalt = u64EndSchedHalt - u64StartSchedHalt;
734 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlock, cNsElapsedSchedHalt);
735
736 if (rc == VERR_INTERRUPTED)
737 rc = VINF_SUCCESS;
738 else if (RT_FAILURE(rc))
739 {
740 rc = vmR3FatalWaitError(pUVCpu, "vmR3HaltGlobal1Halt: VMMR0_DO_GVMM_SCHED_HALT->%Rrc\n", rc);
741 break;
742 }
743 else
744 {
745 int64_t const cNsOverslept = u64EndSchedHalt - u64GipTime;
746 if (cNsOverslept > 50000)
747 STAM_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlockOverslept, cNsOverslept);
748 else if (cNsOverslept < -50000)
749 STAM_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlockInsomnia, cNsElapsedSchedHalt);
750 else
751 STAM_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltBlockOnTime, cNsElapsedSchedHalt);
752 }
753 }
754 /*
755 * When spinning call upon the GVMM and do some wakups once
756 * in a while, it's not like we're actually busy or anything.
757 */
758 else if (!(cLoops & 0x1fff))
759 {
760 uint64_t const u64StartSchedYield = RTTimeNanoTS();
761 rc = SUPR3CallVMMR0Ex(pVM->pVMR0, pVCpu->idCpu, VMMR0_DO_GVMM_SCHED_POLL, false /* don't yield */, NULL);
762 uint64_t const cNsElapsedSchedYield = RTTimeNanoTS() - u64StartSchedYield;
763 STAM_REL_PROFILE_ADD_PERIOD(&pUVCpu->vm.s.StatHaltYield, cNsElapsedSchedYield);
764 }
765 }
766 //RTLogPrintf("*** %u loops %'llu; lag=%RU64\n", cLoops, u64NowLog - u64Start, TMVirtualSyncGetLag(pVM));
767
768 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
769 return rc;
770}
771
772
773/**
774 * The global 1 halt method - VMR3Wait() worker.
775 *
776 * @returns VBox status code.
777 * @param pUVCpu Pointer to the user mode VMCPU structure.
778 */
779static DECLCALLBACK(int) vmR3HaltGlobal1Wait(PUVMCPU pUVCpu)
780{
781 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
782
783 PVM pVM = pUVCpu->pUVM->pVM;
784 PVMCPU pVCpu = VMMGetCpu(pVM);
785 Assert(pVCpu->idCpu == pUVCpu->idCpu);
786
787 int rc = VINF_SUCCESS;
788 for (;;)
789 {
790 /*
791 * Check Relevant FFs.
792 */
793 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_SUSPENDED_MASK)
794 || VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_EXTERNAL_SUSPENDED_MASK))
795 break;
796
797 /*
798 * Wait for a while. Someone will wake us up or interrupt the call if
799 * anything needs our attention.
800 */
801 rc = SUPR3CallVMMR0Ex(pVM->pVMR0, pVCpu->idCpu, VMMR0_DO_GVMM_SCHED_HALT, RTTimeNanoTS() + 1000000000 /* +1s */, NULL);
802 if (rc == VERR_INTERRUPTED)
803 rc = VINF_SUCCESS;
804 else if (RT_FAILURE(rc))
805 {
806 rc = vmR3FatalWaitError(pUVCpu, "vmR3HaltGlobal1Wait: VMMR0_DO_GVMM_SCHED_HALT->%Rrc\n", rc);
807 break;
808 }
809 }
810
811 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
812 return rc;
813}
814
815
816/**
817 * The global 1 halt method - VMR3NotifyFF() worker.
818 *
819 * @param pUVCpu Pointer to the user mode VMCPU structure.
820 * @param fFlags Notification flags, VMNOTIFYFF_FLAGS_*.
821 */
822static DECLCALLBACK(void) vmR3HaltGlobal1NotifyCpuFF(PUVMCPU pUVCpu, uint32_t fFlags)
823{
824 if (pUVCpu->vm.s.fWait)
825 {
826 int rc = SUPR3CallVMMR0Ex(pUVCpu->pVM->pVMR0, pUVCpu->idCpu, VMMR0_DO_GVMM_SCHED_WAKE_UP, 0, NULL);
827 AssertRC(rc);
828 }
829 else if ( ( (fFlags & VMNOTIFYFF_FLAGS_POKE)
830 || !(fFlags & VMNOTIFYFF_FLAGS_DONE_REM))
831 && pUVCpu->pVCpu)
832 {
833 VMCPUSTATE enmState = VMCPU_GET_STATE(pUVCpu->pVCpu);
834 if (enmState == VMCPUSTATE_STARTED_EXEC)
835 {
836 if (fFlags & VMNOTIFYFF_FLAGS_POKE)
837 {
838 int rc = SUPR3CallVMMR0Ex(pUVCpu->pVM->pVMR0, pUVCpu->idCpu, VMMR0_DO_GVMM_SCHED_POKE, 0, NULL);
839 AssertRC(rc);
840 }
841 }
842#ifdef VBOX_WITH_REM
843 else if (enmState == VMCPUSTATE_STARTED_EXEC_REM)
844 {
845 if (!(fFlags & VMNOTIFYFF_FLAGS_DONE_REM))
846 REMR3NotifyFF(pUVCpu->pVM);
847 }
848#endif
849 }
850}
851
852
853/**
854 * Bootstrap VMR3Wait() worker.
855 *
856 * @returns VBox status code.
857 * @param pUVCpu Pointer to the user mode VMCPU structure.
858 */
859static DECLCALLBACK(int) vmR3BootstrapWait(PUVMCPU pUVCpu)
860{
861 PUVM pUVM = pUVCpu->pUVM;
862
863 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
864
865 int rc = VINF_SUCCESS;
866 for (;;)
867 {
868 /*
869 * Check Relevant FFs.
870 */
871 if (pUVM->vm.s.pNormalReqs || pUVM->vm.s.pPriorityReqs) /* global requests pending? */
872 break;
873 if (pUVCpu->vm.s.pNormalReqs || pUVCpu->vm.s.pPriorityReqs) /* local requests pending? */
874 break;
875
876 if ( pUVCpu->pVM
877 && ( VM_FF_IS_PENDING(pUVCpu->pVM, VM_FF_EXTERNAL_SUSPENDED_MASK)
878 || VMCPU_FF_IS_PENDING(VMMGetCpu(pUVCpu->pVM), VMCPU_FF_EXTERNAL_SUSPENDED_MASK)
879 )
880 )
881 break;
882 if (pUVM->vm.s.fTerminateEMT)
883 break;
884
885 /*
886 * Wait for a while. Someone will wake us up or interrupt the call if
887 * anything needs our attention.
888 */
889 rc = RTSemEventWait(pUVCpu->vm.s.EventSemWait, 1000);
890 if (rc == VERR_TIMEOUT)
891 rc = VINF_SUCCESS;
892 else if (RT_FAILURE(rc))
893 {
894 rc = vmR3FatalWaitError(pUVCpu, "RTSemEventWait->%Rrc\n", rc);
895 break;
896 }
897 }
898
899 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
900 return rc;
901}
902
903
904/**
905 * Bootstrap VMR3NotifyFF() worker.
906 *
907 * @param pUVCpu Pointer to the user mode VMCPU structure.
908 * @param fFlags Notification flags, VMNOTIFYFF_FLAGS_*.
909 */
910static DECLCALLBACK(void) vmR3BootstrapNotifyCpuFF(PUVMCPU pUVCpu, uint32_t fFlags)
911{
912 if (pUVCpu->vm.s.fWait)
913 {
914 int rc = RTSemEventSignal(pUVCpu->vm.s.EventSemWait);
915 AssertRC(rc);
916 }
917 NOREF(fFlags);
918}
919
920
921/**
922 * Default VMR3Wait() worker.
923 *
924 * @returns VBox status code.
925 * @param pUVCpu Pointer to the user mode VMCPU structure.
926 */
927static DECLCALLBACK(int) vmR3DefaultWait(PUVMCPU pUVCpu)
928{
929 ASMAtomicWriteBool(&pUVCpu->vm.s.fWait, true);
930
931 PVM pVM = pUVCpu->pVM;
932 PVMCPU pVCpu = pUVCpu->pVCpu;
933 int rc = VINF_SUCCESS;
934 for (;;)
935 {
936 /*
937 * Check Relevant FFs.
938 */
939 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_SUSPENDED_MASK)
940 || VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_EXTERNAL_SUSPENDED_MASK))
941 break;
942
943 /*
944 * Wait for a while. Someone will wake us up or interrupt the call if
945 * anything needs our attention.
946 */
947 rc = RTSemEventWait(pUVCpu->vm.s.EventSemWait, 1000);
948 if (rc == VERR_TIMEOUT)
949 rc = VINF_SUCCESS;
950 else if (RT_FAILURE(rc))
951 {
952 rc = vmR3FatalWaitError(pUVCpu, "RTSemEventWait->%Rrc", rc);
953 break;
954 }
955 }
956
957 ASMAtomicUoWriteBool(&pUVCpu->vm.s.fWait, false);
958 return rc;
959}
960
961
962/**
963 * Default VMR3NotifyFF() worker.
964 *
965 * @param pUVCpu Pointer to the user mode VMCPU structure.
966 * @param fFlags Notification flags, VMNOTIFYFF_FLAGS_*.
967 */
968static DECLCALLBACK(void) vmR3DefaultNotifyCpuFF(PUVMCPU pUVCpu, uint32_t fFlags)
969{
970 if (pUVCpu->vm.s.fWait)
971 {
972 int rc = RTSemEventSignal(pUVCpu->vm.s.EventSemWait);
973 AssertRC(rc);
974 }
975#ifdef VBOX_WITH_REM
976 else if ( !(fFlags & VMNOTIFYFF_FLAGS_DONE_REM)
977 && pUVCpu->pVCpu
978 && pUVCpu->pVCpu->enmState == VMCPUSTATE_STARTED_EXEC_REM)
979 REMR3NotifyFF(pUVCpu->pVM);
980#else
981 RT_NOREF(fFlags);
982#endif
983}
984
985
986/**
987 * Array with halt method descriptors.
988 * VMINT::iHaltMethod contains an index into this array.
989 */
990static const struct VMHALTMETHODDESC
991{
992 /** The halt method id. */
993 VMHALTMETHOD enmHaltMethod;
994 /** The init function for loading config and initialize variables. */
995 DECLR3CALLBACKMEMBER(int, pfnInit,(PUVM pUVM));
996 /** The term function. */
997 DECLR3CALLBACKMEMBER(void, pfnTerm,(PUVM pUVM));
998 /** The VMR3WaitHaltedU function. */
999 DECLR3CALLBACKMEMBER(int, pfnHalt,(PUVMCPU pUVCpu, const uint32_t fMask, uint64_t u64Now));
1000 /** The VMR3WaitU function. */
1001 DECLR3CALLBACKMEMBER(int, pfnWait,(PUVMCPU pUVCpu));
1002 /** The VMR3NotifyCpuFFU function. */
1003 DECLR3CALLBACKMEMBER(void, pfnNotifyCpuFF,(PUVMCPU pUVCpu, uint32_t fFlags));
1004 /** The VMR3NotifyGlobalFFU function. */
1005 DECLR3CALLBACKMEMBER(void, pfnNotifyGlobalFF,(PUVM pUVM, uint32_t fFlags));
1006} g_aHaltMethods[] =
1007{
1008 { VMHALTMETHOD_BOOTSTRAP, NULL, NULL, NULL, vmR3BootstrapWait, vmR3BootstrapNotifyCpuFF, NULL },
1009 { VMHALTMETHOD_OLD, NULL, NULL, vmR3HaltOldDoHalt, vmR3DefaultWait, vmR3DefaultNotifyCpuFF, NULL },
1010 { VMHALTMETHOD_1, vmR3HaltMethod1Init, NULL, vmR3HaltMethod1Halt, vmR3DefaultWait, vmR3DefaultNotifyCpuFF, NULL },
1011 { VMHALTMETHOD_GLOBAL_1, vmR3HaltGlobal1Init, NULL, vmR3HaltGlobal1Halt, vmR3HaltGlobal1Wait, vmR3HaltGlobal1NotifyCpuFF, NULL },
1012};
1013
1014
1015/**
1016 * Notify the emulation thread (EMT) about pending Forced Action (FF).
1017 *
1018 * This function is called by thread other than EMT to make
1019 * sure EMT wakes up and promptly service an FF request.
1020 *
1021 * @param pUVM Pointer to the user mode VM structure.
1022 * @param fFlags Notification flags, VMNOTIFYFF_FLAGS_*.
1023 * @internal
1024 */
1025VMMR3_INT_DECL(void) VMR3NotifyGlobalFFU(PUVM pUVM, uint32_t fFlags)
1026{
1027 LogFlow(("VMR3NotifyGlobalFFU:\n"));
1028 uint32_t iHaldMethod = pUVM->vm.s.iHaltMethod;
1029
1030 if (g_aHaltMethods[iHaldMethod].pfnNotifyGlobalFF) /** @todo make mandatory. */
1031 g_aHaltMethods[iHaldMethod].pfnNotifyGlobalFF(pUVM, fFlags);
1032 else
1033 for (VMCPUID iCpu = 0; iCpu < pUVM->cCpus; iCpu++)
1034 g_aHaltMethods[iHaldMethod].pfnNotifyCpuFF(&pUVM->aCpus[iCpu], fFlags);
1035}
1036
1037
1038/**
1039 * Notify the emulation thread (EMT) about pending Forced Action (FF).
1040 *
1041 * This function is called by thread other than EMT to make
1042 * sure EMT wakes up and promptly service an FF request.
1043 *
1044 * @param pUVCpu Pointer to the user mode per CPU VM structure.
1045 * @param fFlags Notification flags, VMNOTIFYFF_FLAGS_*.
1046 * @internal
1047 */
1048VMMR3_INT_DECL(void) VMR3NotifyCpuFFU(PUVMCPU pUVCpu, uint32_t fFlags)
1049{
1050 PUVM pUVM = pUVCpu->pUVM;
1051
1052 LogFlow(("VMR3NotifyCpuFFU:\n"));
1053 g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnNotifyCpuFF(pUVCpu, fFlags);
1054}
1055
1056
1057/**
1058 * Halted VM Wait.
1059 * Any external event will unblock the thread.
1060 *
1061 * @returns VINF_SUCCESS unless a fatal error occurred. In the latter
1062 * case an appropriate status code is returned.
1063 * @param pVM The cross context VM structure.
1064 * @param pVCpu The cross context virtual CPU structure.
1065 * @param fIgnoreInterrupts If set the VM_FF_INTERRUPT flags is ignored.
1066 * @thread The emulation thread.
1067 * @remarks Made visible for implementing vmsvga sync register.
1068 * @internal
1069 */
1070VMMR3_INT_DECL(int) VMR3WaitHalted(PVM pVM, PVMCPU pVCpu, bool fIgnoreInterrupts)
1071{
1072 LogFlow(("VMR3WaitHalted: fIgnoreInterrupts=%d\n", fIgnoreInterrupts));
1073
1074 /*
1075 * Check Relevant FFs.
1076 */
1077 const uint32_t fMask = !fIgnoreInterrupts
1078 ? VMCPU_FF_EXTERNAL_HALTED_MASK
1079 : VMCPU_FF_EXTERNAL_HALTED_MASK & ~(VMCPU_FF_UPDATE_APIC | VMCPU_FF_INTERRUPT_APIC | VMCPU_FF_INTERRUPT_PIC);
1080 if ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_HALTED_MASK)
1081 || VMCPU_FF_IS_PENDING(pVCpu, fMask))
1082 {
1083 LogFlow(("VMR3WaitHalted: returns VINF_SUCCESS (FF %#x FFCPU %#x)\n", pVM->fGlobalForcedActions, pVCpu->fLocalForcedActions));
1084 return VINF_SUCCESS;
1085 }
1086
1087 /*
1088 * The yielder is suspended while we're halting, while TM might have clock(s) running
1089 * only at certain times and need to be notified..
1090 */
1091 if (pVCpu->idCpu == 0)
1092 VMMR3YieldSuspend(pVM);
1093 TMNotifyStartOfHalt(pVCpu);
1094
1095 /*
1096 * Record halt averages for the last second.
1097 */
1098 PUVMCPU pUVCpu = pVCpu->pUVCpu;
1099 uint64_t u64Now = RTTimeNanoTS();
1100 int64_t off = u64Now - pUVCpu->vm.s.u64HaltsStartTS;
1101 if (off > 1000000000)
1102 {
1103 if (off > _4G || !pUVCpu->vm.s.cHalts)
1104 {
1105 pUVCpu->vm.s.HaltInterval = 1000000000 /* 1 sec */;
1106 pUVCpu->vm.s.HaltFrequency = 1;
1107 }
1108 else
1109 {
1110 pUVCpu->vm.s.HaltInterval = (uint32_t)off / pUVCpu->vm.s.cHalts;
1111 pUVCpu->vm.s.HaltFrequency = ASMMultU64ByU32DivByU32(pUVCpu->vm.s.cHalts, 1000000000, (uint32_t)off);
1112 }
1113 pUVCpu->vm.s.u64HaltsStartTS = u64Now;
1114 pUVCpu->vm.s.cHalts = 0;
1115 }
1116 pUVCpu->vm.s.cHalts++;
1117
1118 /*
1119 * Do the halt.
1120 */
1121 VMCPU_ASSERT_STATE(pVCpu, VMCPUSTATE_STARTED);
1122 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED_HALTED);
1123 PUVM pUVM = pUVCpu->pUVM;
1124 int rc = g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnHalt(pUVCpu, fMask, u64Now);
1125 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
1126
1127 /*
1128 * Notify TM and resume the yielder
1129 */
1130 TMNotifyEndOfHalt(pVCpu);
1131 if (pVCpu->idCpu == 0)
1132 VMMR3YieldResume(pVM);
1133
1134 LogFlow(("VMR3WaitHalted: returns %Rrc (FF %#x)\n", rc, pVM->fGlobalForcedActions));
1135 return rc;
1136}
1137
1138
1139/**
1140 * Suspended VM Wait.
1141 * Only a handful of forced actions will cause the function to
1142 * return to the caller.
1143 *
1144 * @returns VINF_SUCCESS unless a fatal error occurred. In the latter
1145 * case an appropriate status code is returned.
1146 * @param pUVCpu Pointer to the user mode VMCPU structure.
1147 * @thread The emulation thread.
1148 * @internal
1149 */
1150VMMR3_INT_DECL(int) VMR3WaitU(PUVMCPU pUVCpu)
1151{
1152 LogFlow(("VMR3WaitU:\n"));
1153
1154 /*
1155 * Check Relevant FFs.
1156 */
1157 PVM pVM = pUVCpu->pVM;
1158 PVMCPU pVCpu = pUVCpu->pVCpu;
1159
1160 if ( pVM
1161 && ( VM_FF_IS_PENDING(pVM, VM_FF_EXTERNAL_SUSPENDED_MASK)
1162 || VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_EXTERNAL_SUSPENDED_MASK)
1163 )
1164 )
1165 {
1166 LogFlow(("VMR3Wait: returns VINF_SUCCESS (FF %#x)\n", pVM->fGlobalForcedActions));
1167 return VINF_SUCCESS;
1168 }
1169
1170 /*
1171 * Do waiting according to the halt method (so VMR3NotifyFF
1172 * doesn't have to special case anything).
1173 */
1174 PUVM pUVM = pUVCpu->pUVM;
1175 int rc = g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnWait(pUVCpu);
1176 LogFlow(("VMR3WaitU: returns %Rrc (FF %#x)\n", rc, pUVM->pVM ? pUVM->pVM->fGlobalForcedActions : 0));
1177 return rc;
1178}
1179
1180
1181/**
1182 * Interface that PDMR3Suspend, PDMR3PowerOff and PDMR3Reset uses when they wait
1183 * for the handling of asynchronous notifications to complete.
1184 *
1185 * @returns VINF_SUCCESS unless a fatal error occurred. In the latter
1186 * case an appropriate status code is returned.
1187 * @param pUVCpu Pointer to the user mode VMCPU structure.
1188 * @thread The emulation thread.
1189 */
1190VMMR3_INT_DECL(int) VMR3AsyncPdmNotificationWaitU(PUVMCPU pUVCpu)
1191{
1192 LogFlow(("VMR3AsyncPdmNotificationWaitU:\n"));
1193 return VMR3WaitU(pUVCpu);
1194}
1195
1196
1197/**
1198 * Interface that PDM the helper asynchronous notification completed methods
1199 * uses for EMT0 when it is waiting inside VMR3AsyncPdmNotificationWaitU().
1200 *
1201 * @param pUVM Pointer to the user mode VM structure.
1202 */
1203VMMR3_INT_DECL(void) VMR3AsyncPdmNotificationWakeupU(PUVM pUVM)
1204{
1205 LogFlow(("VMR3AsyncPdmNotificationWakeupU:\n"));
1206 VM_FF_SET(pUVM->pVM, VM_FF_REQUEST); /* this will have to do for now. */
1207 g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnNotifyCpuFF(&pUVM->aCpus[0], 0 /*fFlags*/);
1208}
1209
1210
1211/**
1212 * Rendezvous callback that will be called once.
1213 *
1214 * @returns VBox strict status code.
1215 * @param pVM The cross context VM structure.
1216 * @param pVCpu The cross context virtual CPU structure of the calling EMT.
1217 * @param pvUser The new g_aHaltMethods index.
1218 */
1219static DECLCALLBACK(VBOXSTRICTRC) vmR3SetHaltMethodCallback(PVM pVM, PVMCPU pVCpu, void *pvUser)
1220{
1221 PUVM pUVM = pVM->pUVM;
1222 uintptr_t i = (uintptr_t)pvUser;
1223 Assert(i < RT_ELEMENTS(g_aHaltMethods));
1224 NOREF(pVCpu);
1225
1226 /*
1227 * Terminate the old one.
1228 */
1229 if ( pUVM->vm.s.enmHaltMethod != VMHALTMETHOD_INVALID
1230 && g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnTerm)
1231 {
1232 g_aHaltMethods[pUVM->vm.s.iHaltMethod].pfnTerm(pUVM);
1233 pUVM->vm.s.enmHaltMethod = VMHALTMETHOD_INVALID;
1234 }
1235
1236 /* Assert that the failure fallback is where we expect. */
1237 Assert(g_aHaltMethods[0].enmHaltMethod == VMHALTMETHOD_BOOTSTRAP);
1238 Assert(!g_aHaltMethods[0].pfnTerm && !g_aHaltMethods[0].pfnInit);
1239
1240 /*
1241 * Init the new one.
1242 */
1243 int rc = VINF_SUCCESS;
1244 memset(&pUVM->vm.s.Halt, 0, sizeof(pUVM->vm.s.Halt));
1245 if (g_aHaltMethods[i].pfnInit)
1246 {
1247 rc = g_aHaltMethods[i].pfnInit(pUVM);
1248 if (RT_FAILURE(rc))
1249 {
1250 /* Fall back on the bootstrap method. This requires no
1251 init/term (see assertion above), and will always work. */
1252 AssertLogRelRC(rc);
1253 i = 0;
1254 }
1255 }
1256
1257 /*
1258 * Commit it.
1259 */
1260 pUVM->vm.s.enmHaltMethod = g_aHaltMethods[i].enmHaltMethod;
1261 ASMAtomicWriteU32(&pUVM->vm.s.iHaltMethod, i);
1262
1263 return rc;
1264}
1265
1266
1267/**
1268 * Changes the halt method.
1269 *
1270 * @returns VBox status code.
1271 * @param pUVM Pointer to the user mode VM structure.
1272 * @param enmHaltMethod The new halt method.
1273 * @thread EMT.
1274 */
1275int vmR3SetHaltMethodU(PUVM pUVM, VMHALTMETHOD enmHaltMethod)
1276{
1277 PVM pVM = pUVM->pVM; Assert(pVM);
1278 VM_ASSERT_EMT(pVM);
1279 AssertReturn(enmHaltMethod > VMHALTMETHOD_INVALID && enmHaltMethod < VMHALTMETHOD_END, VERR_INVALID_PARAMETER);
1280
1281 /*
1282 * Resolve default (can be overridden in the configuration).
1283 */
1284 if (enmHaltMethod == VMHALTMETHOD_DEFAULT)
1285 {
1286 uint32_t u32;
1287 int rc = CFGMR3QueryU32(CFGMR3GetChild(CFGMR3GetRoot(pVM), "VM"), "HaltMethod", &u32);
1288 if (RT_SUCCESS(rc))
1289 {
1290 enmHaltMethod = (VMHALTMETHOD)u32;
1291 if (enmHaltMethod <= VMHALTMETHOD_INVALID || enmHaltMethod >= VMHALTMETHOD_END)
1292 return VMSetError(pVM, VERR_INVALID_PARAMETER, RT_SRC_POS, N_("Invalid VM/HaltMethod value %d"), enmHaltMethod);
1293 }
1294 else if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_CHILD_NOT_FOUND)
1295 return VMSetError(pVM, rc, RT_SRC_POS, N_("Failed to Query VM/HaltMethod as uint32_t"));
1296 else
1297 enmHaltMethod = VMHALTMETHOD_GLOBAL_1;
1298 //enmHaltMethod = VMHALTMETHOD_1;
1299 //enmHaltMethod = VMHALTMETHOD_OLD;
1300 }
1301 LogRel(("VMEmt: Halt method %s (%d)\n", vmR3GetHaltMethodName(enmHaltMethod), enmHaltMethod));
1302
1303 /*
1304 * Find the descriptor.
1305 */
1306 unsigned i = 0;
1307 while ( i < RT_ELEMENTS(g_aHaltMethods)
1308 && g_aHaltMethods[i].enmHaltMethod != enmHaltMethod)
1309 i++;
1310 AssertReturn(i < RT_ELEMENTS(g_aHaltMethods), VERR_INVALID_PARAMETER);
1311
1312 /*
1313 * This needs to be done while the other EMTs are not sleeping or otherwise messing around.
1314 */
1315 return VMMR3EmtRendezvous(pVM, VMMEMTRENDEZVOUS_FLAGS_TYPE_ONCE, vmR3SetHaltMethodCallback, (void *)(uintptr_t)i);
1316}
1317
1318
1319/**
1320 * Special interface for implementing a HLT-like port on a device.
1321 *
1322 * This can be called directly from device code, provide the device is trusted
1323 * to access the VMM directly. Since we may not have an accurate register set
1324 * and the caller certainly shouldn't (device code does not access CPU
1325 * registers), this function will return when interrupts are pending regardless
1326 * of the actual EFLAGS.IF state.
1327 *
1328 * @returns VBox error status (never informational statuses).
1329 * @param pVM The cross context VM structure.
1330 * @param idCpu The id of the calling EMT.
1331 */
1332VMMR3DECL(int) VMR3WaitForDeviceReady(PVM pVM, VMCPUID idCpu)
1333{
1334 /*
1335 * Validate caller and resolve the CPU ID.
1336 */
1337 VM_ASSERT_VALID_EXT_RETURN(pVM, VERR_INVALID_VM_HANDLE);
1338 AssertReturn(idCpu < pVM->cCpus, VERR_INVALID_CPU_ID);
1339 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1340 VMCPU_ASSERT_EMT_RETURN(pVCpu, VERR_VM_THREAD_NOT_EMT);
1341
1342 /*
1343 * Tag along with the HLT mechanics for now.
1344 */
1345 int rc = VMR3WaitHalted(pVM, pVCpu, false /*fIgnoreInterrupts*/);
1346 if (RT_SUCCESS(rc))
1347 return VINF_SUCCESS;
1348 return rc;
1349}
1350
1351
1352/**
1353 * Wakes up a CPU that has called VMR3WaitForDeviceReady.
1354 *
1355 * @returns VBox error status (never informational statuses).
1356 * @param pVM The cross context VM structure.
1357 * @param idCpu The id of the calling EMT.
1358 */
1359VMMR3DECL(int) VMR3NotifyCpuDeviceReady(PVM pVM, VMCPUID idCpu)
1360{
1361 /*
1362 * Validate caller and resolve the CPU ID.
1363 */
1364 VM_ASSERT_VALID_EXT_RETURN(pVM, VERR_INVALID_VM_HANDLE);
1365 AssertReturn(idCpu < pVM->cCpus, VERR_INVALID_CPU_ID);
1366 PVMCPU pVCpu = &pVM->aCpus[idCpu];
1367
1368 /*
1369 * Pretend it was an FF that got set since we've got logic for that already.
1370 */
1371 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM);
1372 return VINF_SUCCESS;
1373}
1374
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