VirtualBox

source: vbox/trunk/src/VBox/VMM/VM.cpp@ 20833

Last change on this file since 20833 was 20564, checked in by vboxsync, 16 years ago

VM.cpp: Fixed crash in VMR3WaitU on creation error (for instance unavailable host NIC).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 109.0 KB
Line 
1/* $Id: VM.cpp 20564 2009-06-14 19:21:54Z vboxsync $ */
2/** @file
3 * VM - Virtual Machine
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/** @page pg_vm VM API
23 *
24 * This is the encapsulating bit. It provides the APIs that Main and VBoxBFE
25 * use to create a VMM instance for running a guest in. It also provides
26 * facilities for queuing request for execution in EMT (serialization purposes
27 * mostly) and for reporting error back to the VMM user (Main/VBoxBFE).
28 *
29 *
30 * @section sec_vm_design Design Critique / Things To Do
31 *
32 * In hindsight this component is a big design mistake, all this stuff really
33 * belongs in the VMM component. It just seemed like a kind of ok idea at a
34 * time when the VMM bit was a bit vague. 'VM' also happend to be the name of
35 * the per-VM instance structure (see vm.h), so it kind of made sense. However
36 * as it turned out, VMM(.cpp) is almost empty all it provides in ring-3 is some
37 * minor functionally and some "routing" services.
38 *
39 * Fixing this is just a matter of some more or less straight forward
40 * refactoring, the question is just when someone will get to it.
41 *
42 */
43
44/*******************************************************************************
45* Header Files *
46*******************************************************************************/
47#define LOG_GROUP LOG_GROUP_VM
48#include <VBox/cfgm.h>
49#include <VBox/vmm.h>
50#include <VBox/gvmm.h>
51#include <VBox/mm.h>
52#include <VBox/cpum.h>
53#include <VBox/selm.h>
54#include <VBox/trpm.h>
55#include <VBox/dbgf.h>
56#include <VBox/pgm.h>
57#include <VBox/pdmapi.h>
58#include <VBox/pdmcritsect.h>
59#include <VBox/em.h>
60#include <VBox/rem.h>
61#include <VBox/tm.h>
62#include <VBox/stam.h>
63#include <VBox/patm.h>
64#ifdef VBOX_WITH_VMI
65# include <VBox/parav.h>
66#endif
67#include <VBox/csam.h>
68#include <VBox/iom.h>
69#include <VBox/ssm.h>
70#include <VBox/hwaccm.h>
71#include "VMInternal.h"
72#include <VBox/vm.h>
73#include <VBox/uvm.h>
74
75#include <VBox/sup.h>
76#include <VBox/dbg.h>
77#include <VBox/err.h>
78#include <VBox/param.h>
79#include <VBox/log.h>
80#include <iprt/assert.h>
81#include <iprt/alloc.h>
82#include <iprt/asm.h>
83#include <iprt/env.h>
84#include <iprt/string.h>
85#include <iprt/time.h>
86#include <iprt/semaphore.h>
87#include <iprt/thread.h>
88
89
90/*******************************************************************************
91* Structures and Typedefs *
92*******************************************************************************/
93/**
94 * VM destruction callback registration record.
95 */
96typedef struct VMATDTOR
97{
98 /** Pointer to the next record in the list. */
99 struct VMATDTOR *pNext;
100 /** Pointer to the callback function. */
101 PFNVMATDTOR pfnAtDtor;
102 /** The user argument. */
103 void *pvUser;
104} VMATDTOR;
105/** Pointer to a VM destruction callback registration record. */
106typedef VMATDTOR *PVMATDTOR;
107
108
109/*******************************************************************************
110* Global Variables *
111*******************************************************************************/
112/** Pointer to the list of VMs. */
113static PUVM g_pUVMsHead = NULL;
114
115/** Pointer to the list of at VM destruction callbacks. */
116static PVMATDTOR g_pVMAtDtorHead = NULL;
117/** Lock the g_pVMAtDtorHead list. */
118#define VM_ATDTOR_LOCK() do { } while (0)
119/** Unlock the g_pVMAtDtorHead list. */
120#define VM_ATDTOR_UNLOCK() do { } while (0)
121
122
123/*******************************************************************************
124* Internal Functions *
125*******************************************************************************/
126static int vmR3CreateUVM(uint32_t cCPUs, PUVM *ppUVM);
127static int vmR3CreateU(PUVM pUVM, uint32_t cCPUs, PFNCFGMCONSTRUCTOR pfnCFGMConstructor, void *pvUserCFGM);
128static int vmR3InitRing3(PVM pVM, PUVM pUVM);
129static int vmR3InitVMCpu(PVM pVM);
130static int vmR3InitRing0(PVM pVM);
131static int vmR3InitGC(PVM pVM);
132static int vmR3InitDoCompleted(PVM pVM, VMINITCOMPLETED enmWhat);
133static DECLCALLBACK(int) vmR3PowerOn(PVM pVM);
134static DECLCALLBACK(int) vmR3Suspend(PVM pVM);
135static DECLCALLBACK(int) vmR3Resume(PVM pVM);
136static DECLCALLBACK(int) vmR3Save(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser);
137static DECLCALLBACK(int) vmR3Load(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser);
138static DECLCALLBACK(int) vmR3PowerOff(PVM pVM);
139static void vmR3DestroyUVM(PUVM pUVM, uint32_t cMilliesEMTWait);
140static void vmR3AtDtor(PVM pVM);
141static int vmR3AtResetU(PUVM pUVM);
142static DECLCALLBACK(int) vmR3Reset(PVM pVM);
143static DECLCALLBACK(int) vmR3AtStateRegisterU(PUVM pUVM, PFNVMATSTATE pfnAtState, void *pvUser);
144static DECLCALLBACK(int) vmR3AtStateDeregisterU(PUVM pUVM, PFNVMATSTATE pfnAtState, void *pvUser);
145static DECLCALLBACK(int) vmR3AtErrorRegisterU(PUVM pUVM, PFNVMATERROR pfnAtError, void *pvUser);
146static DECLCALLBACK(int) vmR3AtErrorDeregisterU(PUVM pUVM, PFNVMATERROR pfnAtError, void *pvUser);
147static int vmR3SetErrorU(PUVM pUVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...);
148static DECLCALLBACK(int) vmR3AtRuntimeErrorRegisterU(PUVM pUVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser);
149static DECLCALLBACK(int) vmR3AtRuntimeErrorDeregisterU(PUVM pUVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser);
150
151
152/**
153 * Do global VMM init.
154 *
155 * @returns VBox status code.
156 */
157VMMR3DECL(int) VMR3GlobalInit(void)
158{
159 /*
160 * Only once.
161 */
162 static bool volatile s_fDone = false;
163 if (s_fDone)
164 return VINF_SUCCESS;
165
166 /*
167 * We're done.
168 */
169 s_fDone = true;
170 return VINF_SUCCESS;
171}
172
173
174
175/**
176 * Creates a virtual machine by calling the supplied configuration constructor.
177 *
178 * On successful returned the VM is powered, i.e. VMR3PowerOn() should be
179 * called to start the execution.
180 *
181 * @returns 0 on success.
182 * @returns VBox error code on failure.
183 * @param cCPUs Number of virtual CPUs for the new VM.
184 * @param pfnVMAtError Pointer to callback function for setting VM
185 * errors. This was added as an implicit call to
186 * VMR3AtErrorRegister() since there is no way the
187 * caller can get to the VM handle early enough to
188 * do this on its own.
189 * This is called in the context of an EMT.
190 * @param pvUserVM The user argument passed to pfnVMAtError.
191 * @param pfnCFGMConstructor Pointer to callback function for constructing the VM configuration tree.
192 * This is called in the context of an EMT0.
193 * @param pvUserCFGM The user argument passed to pfnCFGMConstructor.
194 * @param ppVM Where to store the 'handle' of the created VM.
195 */
196VMMR3DECL(int) VMR3Create(uint32_t cCPUs, PFNVMATERROR pfnVMAtError, void *pvUserVM, PFNCFGMCONSTRUCTOR pfnCFGMConstructor, void *pvUserCFGM, PVM *ppVM)
197{
198 LogFlow(("VMR3Create: cCPUs=%RU32 pfnVMAtError=%p pvUserVM=%p pfnCFGMConstructor=%p pvUserCFGM=%p ppVM=%p\n", cCPUs, pfnVMAtError, pvUserVM, pfnCFGMConstructor, pvUserCFGM, ppVM));
199
200 /*
201 * Because of the current hackiness of the applications
202 * we'll have to initialize global stuff from here.
203 * Later the applications will take care of this in a proper way.
204 */
205 static bool fGlobalInitDone = false;
206 if (!fGlobalInitDone)
207 {
208 int rc = VMR3GlobalInit();
209 if (RT_FAILURE(rc))
210 return rc;
211 fGlobalInitDone = true;
212 }
213
214 /*
215 * Validate input.
216 */
217 AssertLogRelMsgReturn(cCPUs > 0 && cCPUs <= VMM_MAX_CPU_COUNT, ("%RU32\n", cCPUs), VERR_TOO_MANY_CPUS);
218
219 /*
220 * Create the UVM so we can register the at-error callback
221 * and consoliate a bit of cleanup code.
222 */
223 PUVM pUVM;
224 int rc = vmR3CreateUVM(cCPUs, &pUVM);
225 if (RT_FAILURE(rc))
226 return rc;
227 if (pfnVMAtError)
228 rc = VMR3AtErrorRegisterU(pUVM, pfnVMAtError, pvUserVM);
229 if (RT_SUCCESS(rc))
230 {
231 /*
232 * Initialize the support library creating the session for this VM.
233 */
234 rc = SUPR3Init(&pUVM->vm.s.pSession);
235 if (RT_SUCCESS(rc))
236 {
237 /*
238 * Call vmR3CreateU in the EMT thread and wait for it to finish.
239 *
240 * Note! VMCPUID_ANY is used here because VMR3ReqQueueU would have trouble
241 * submitting a request to a specific VCPU without a pVM. So, to make
242 * sure init is running on EMT(0), vmR3EmulationThreadWithId makes sure
243 * that only EMT(0) is servicing VMCPUID_ANY requests when pVM is NULL.
244 */
245 PVMREQ pReq;
246 rc = VMR3ReqCallU(pUVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, 0, (PFNRT)vmR3CreateU, 4,
247 pUVM, cCPUs, pfnCFGMConstructor, pvUserCFGM);
248 if (RT_SUCCESS(rc))
249 {
250 rc = pReq->iStatus;
251 VMR3ReqFree(pReq);
252 if (RT_SUCCESS(rc))
253 {
254 /*
255 * Success!
256 */
257 *ppVM = pUVM->pVM;
258 LogFlow(("VMR3Create: returns VINF_SUCCESS *ppVM=%p\n", *ppVM));
259 return VINF_SUCCESS;
260 }
261 }
262 else
263 AssertMsgFailed(("VMR3ReqCall failed rc=%Rrc\n", rc));
264
265 /*
266 * An error occurred during VM creation. Set the error message directly
267 * using the initial callback, as the callback list doesn't exist yet.
268 */
269 const char *pszError = NULL;
270 switch (rc)
271 {
272 case VERR_VMX_IN_VMX_ROOT_MODE:
273#ifdef RT_OS_LINUX
274 pszError = N_("VirtualBox can't operate in VMX root mode. "
275 "Please disable the KVM kernel extension, recompile your kernel and reboot");
276#else
277 pszError = N_("VirtualBox can't operate in VMX root mode. Please close all other virtualization programs.");
278#endif
279 break;
280
281 case VERR_VERSION_MISMATCH:
282 pszError = N_("VMMR0 driver version mismatch. Please terminate all VMs, make sure that "
283 "VBoxNetDHCP is not running and try again. If you still get this error, "
284 "re-install VirtualBox");
285 break;
286
287#ifdef RT_OS_LINUX
288 case VERR_SUPDRV_COMPONENT_NOT_FOUND:
289 pszError = N_("One of the kernel modules was not successfully loaded. Make sure "
290 "that no kernel modules from an older version of VirtualBox exist. "
291 "Then try to recompile and reload the kernel modules by executing "
292 "'/etc/init.d/vboxdrv setup' as root");
293 break;
294#endif
295
296 default:
297 pszError = N_("Unknown error creating VM");
298 break;
299 }
300 vmR3SetErrorU(pUVM, rc, RT_SRC_POS, pszError, rc);
301 }
302 else
303 {
304 /*
305 * An error occurred at support library initialization time (before the
306 * VM could be created). Set the error message directly using the
307 * initial callback, as the callback list doesn't exist yet.
308 */
309 const char *pszError;
310 switch (rc)
311 {
312 case VERR_VM_DRIVER_LOAD_ERROR:
313#ifdef RT_OS_LINUX
314 pszError = N_("VirtualBox kernel driver not loaded. The vboxdrv kernel module "
315 "was either not loaded or /dev/vboxdrv is not set up properly. "
316 "Re-setup the kernel module by executing "
317 "'/etc/init.d/vboxdrv setup' as root");
318#else
319 pszError = N_("VirtualBox kernel driver not loaded");
320#endif
321 break;
322 case VERR_VM_DRIVER_OPEN_ERROR:
323 pszError = N_("VirtualBox kernel driver cannot be opened");
324 break;
325 case VERR_VM_DRIVER_NOT_ACCESSIBLE:
326#ifdef VBOX_WITH_HARDENING
327 /* This should only happen if the executable wasn't hardened - bad code/build. */
328 pszError = N_("VirtualBox kernel driver not accessible, permission problem. "
329 "Re-install VirtualBox. If you are building it yourself, you "
330 "should make sure it installed correctly and that the setuid "
331 "bit is set on the executables calling VMR3Create.");
332#else
333 /* This should only happen when mixing builds or with the usual /dev/vboxdrv access issues. */
334# if defined(RT_OS_DARWIN)
335 pszError = N_("VirtualBox KEXT is not accessible, permission problem. "
336 "If you have built VirtualBox yourself, make sure that you do not "
337 "have the vboxdrv KEXT from a different build or installation loaded.");
338# elif defined(RT_OS_LINUX)
339 pszError = N_("VirtualBox kernel driver is not accessible, permission problem. "
340 "If you have built VirtualBox yourself, make sure that you do "
341 "not have the vboxdrv kernel module from a different build or "
342 "installation loaded. Also, make sure the vboxdrv udev rule gives "
343 "you the permission you need to access the device.");
344# elif defined(RT_OS_WINDOWS)
345 pszError = N_("VirtualBox kernel driver is not accessible, permission problem.");
346# else /* solaris, freebsd, ++. */
347 pszError = N_("VirtualBox kernel module is not accessible, permission problem. "
348 "If you have built VirtualBox yourself, make sure that you do "
349 "not have the vboxdrv kernel module from a different install loaded.");
350# endif
351#endif
352 break;
353 case VERR_INVALID_HANDLE: /** @todo track down and fix this error. */
354 case VERR_VM_DRIVER_NOT_INSTALLED:
355#ifdef RT_OS_LINUX
356 pszError = N_("VirtualBox kernel driver not installed. The vboxdrv kernel module "
357 "was either not loaded or /dev/vboxdrv was not created for some "
358 "reason. Re-setup the kernel module by executing "
359 "'/etc/init.d/vboxdrv setup' as root");
360#else
361 pszError = N_("VirtualBox kernel driver not installed");
362#endif
363 break;
364 case VERR_NO_MEMORY:
365 pszError = N_("VirtualBox support library out of memory");
366 break;
367 case VERR_VERSION_MISMATCH:
368 case VERR_VM_DRIVER_VERSION_MISMATCH:
369 pszError = N_("The VirtualBox support driver which is running is from a different "
370 "version of VirtualBox. You can correct this by stopping all "
371 "running instances of VirtualBox and reinstalling the software.");
372 break;
373 default:
374 pszError = N_("Unknown error initializing kernel driver");
375 AssertMsgFailed(("Add error message for rc=%d (%Rrc)\n", rc, rc));
376 }
377 vmR3SetErrorU(pUVM, rc, RT_SRC_POS, pszError, rc);
378 }
379 }
380
381 /* cleanup */
382 vmR3DestroyUVM(pUVM, 2000);
383 LogFlow(("VMR3Create: returns %Rrc\n", rc));
384 return rc;
385}
386
387
388/**
389 * Creates the UVM.
390 *
391 * This will not initialize the support library even if vmR3DestroyUVM
392 * will terminate that.
393 *
394 * @returns VBox status code.
395 * @param cCpus Number of virtual CPUs
396 * @param ppUVM Where to store the UVM pointer.
397 */
398static int vmR3CreateUVM(uint32_t cCpus, PUVM *ppUVM)
399{
400 uint32_t i;
401
402 /*
403 * Create and initialize the UVM.
404 */
405 PUVM pUVM = (PUVM)RTMemAllocZ(RT_OFFSETOF(UVM, aCpus[cCpus]));
406 AssertReturn(pUVM, VERR_NO_MEMORY);
407 pUVM->u32Magic = UVM_MAGIC;
408 pUVM->cCpus = cCpus;
409
410 AssertCompile(sizeof(pUVM->vm.s) <= sizeof(pUVM->vm.padding));
411
412 pUVM->vm.s.ppAtResetNext = &pUVM->vm.s.pAtReset;
413 pUVM->vm.s.ppAtStateNext = &pUVM->vm.s.pAtState;
414 pUVM->vm.s.ppAtErrorNext = &pUVM->vm.s.pAtError;
415 pUVM->vm.s.ppAtRuntimeErrorNext = &pUVM->vm.s.pAtRuntimeError;
416
417 pUVM->vm.s.enmHaltMethod = VMHALTMETHOD_BOOTSTRAP;
418
419 /* Initialize the VMCPU array in the UVM. */
420 for (i = 0; i < cCpus; i++)
421 {
422 pUVM->aCpus[i].pUVM = pUVM;
423 pUVM->aCpus[i].idCpu = i;
424 }
425
426 /* Allocate a TLS entry to store the VMINTUSERPERVMCPU pointer. */
427 int rc = RTTlsAllocEx(&pUVM->vm.s.idxTLS, NULL);
428 AssertRC(rc);
429 if (RT_SUCCESS(rc))
430 {
431 /* Allocate a halt method event semaphore for each VCPU. */
432 for (i = 0; i < cCpus; i++)
433 {
434 rc = RTSemEventCreate(&pUVM->aCpus[i].vm.s.EventSemWait);
435 if (RT_FAILURE(rc))
436 break;
437 }
438
439 if (RT_SUCCESS(rc))
440 {
441 /*
442 * Init fundamental (sub-)components - STAM, MMR3Heap and PDMLdr.
443 */
444 rc = STAMR3InitUVM(pUVM);
445 if (RT_SUCCESS(rc))
446 {
447 rc = MMR3InitUVM(pUVM);
448 if (RT_SUCCESS(rc))
449 {
450 rc = PDMR3InitUVM(pUVM);
451 if (RT_SUCCESS(rc))
452 {
453 /*
454 * Start the emulation threads for all VMCPUs.
455 */
456 for (i = 0; i < cCpus; i++)
457 {
458 rc = RTThreadCreateF(&pUVM->aCpus[i].vm.s.ThreadEMT, vmR3EmulationThread, &pUVM->aCpus[i], _1M,
459 RTTHREADTYPE_EMULATION, RTTHREADFLAGS_WAITABLE,
460 cCpus > 1 ? "EMT-%u" : "EMT", i);
461 if (RT_FAILURE(rc))
462 break;
463
464 pUVM->aCpus[i].vm.s.NativeThreadEMT = RTThreadGetNative(pUVM->aCpus[i].vm.s.ThreadEMT);
465 }
466
467 if (RT_SUCCESS(rc))
468 {
469 *ppUVM = pUVM;
470 return VINF_SUCCESS;
471 }
472
473 /* bail out. */
474 while (i-- > 0)
475 {
476 /** @todo rainy day: terminate the EMTs. */
477 }
478 PDMR3TermUVM(pUVM);
479 }
480 MMR3TermUVM(pUVM);
481 }
482 STAMR3TermUVM(pUVM);
483 }
484 for (i = 0; i < cCpus; i++)
485 {
486 RTSemEventDestroy(pUVM->aCpus[i].vm.s.EventSemWait);
487 pUVM->aCpus[i].vm.s.EventSemWait = NIL_RTSEMEVENT;
488 }
489 }
490 RTTlsFree(pUVM->vm.s.idxTLS);
491 }
492 RTMemFree(pUVM);
493 return rc;
494}
495
496
497/**
498 * Creates and initializes the VM.
499 *
500 * @thread EMT
501 */
502static int vmR3CreateU(PUVM pUVM, uint32_t cCpus, PFNCFGMCONSTRUCTOR pfnCFGMConstructor, void *pvUserCFGM)
503{
504 int rc = VINF_SUCCESS;
505
506 /*
507 * Load the VMMR0.r0 module so that we can call GVMMR0CreateVM.
508 */
509 rc = PDMR3LdrLoadVMMR0U(pUVM);
510 if (RT_FAILURE(rc))
511 {
512 /** @todo we need a cleaner solution for this (VERR_VMX_IN_VMX_ROOT_MODE).
513 * bird: what about moving the message down here? Main picks the first message, right? */
514 if (rc == VERR_VMX_IN_VMX_ROOT_MODE)
515 return rc; /* proper error message set later on */
516 return vmR3SetErrorU(pUVM, rc, RT_SRC_POS, N_("Failed to load VMMR0.r0"));
517 }
518
519 /*
520 * Request GVMM to create a new VM for us.
521 */
522 GVMMCREATEVMREQ CreateVMReq;
523 CreateVMReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
524 CreateVMReq.Hdr.cbReq = sizeof(CreateVMReq);
525 CreateVMReq.pSession = pUVM->vm.s.pSession;
526 CreateVMReq.pVMR0 = NIL_RTR0PTR;
527 CreateVMReq.pVMR3 = NULL;
528 CreateVMReq.cCpus = cCpus;
529 rc = SUPCallVMMR0Ex(NIL_RTR0PTR, NIL_VMCPUID, VMMR0_DO_GVMM_CREATE_VM, 0, &CreateVMReq.Hdr);
530 if (RT_SUCCESS(rc))
531 {
532 PVM pVM = pUVM->pVM = CreateVMReq.pVMR3;
533 AssertRelease(VALID_PTR(pVM));
534 AssertRelease(pVM->pVMR0 == CreateVMReq.pVMR0);
535 AssertRelease(pVM->pSession == pUVM->vm.s.pSession);
536 AssertRelease(pVM->cCPUs == cCpus);
537 AssertRelease(pVM->offVMCPU == RT_UOFFSETOF(VM, aCpus));
538
539 Log(("VMR3Create: Created pUVM=%p pVM=%p pVMR0=%p hSelf=%#x cCPUs=%RU32\n",
540 pUVM, pVM, pVM->pVMR0, pVM->hSelf, pVM->cCPUs));
541
542 /*
543 * Initialize the VM structure and our internal data (VMINT).
544 */
545 pVM->pUVM = pUVM;
546
547 for (uint32_t i = 0; i < pVM->cCPUs; i++)
548 {
549 pVM->aCpus[i].pUVCpu = &pUVM->aCpus[i];
550 pVM->aCpus[i].idCpu = i;
551 pVM->aCpus[i].hNativeThread = pUVM->aCpus[i].vm.s.NativeThreadEMT;
552 Assert(pVM->aCpus[i].hNativeThread != NIL_RTNATIVETHREAD);
553
554 pUVM->aCpus[i].pVCpu = &pVM->aCpus[i];
555 pUVM->aCpus[i].pVM = pVM;
556 }
557
558
559 /*
560 * Init the configuration.
561 */
562 rc = CFGMR3Init(pVM, pfnCFGMConstructor, pvUserCFGM);
563 if (RT_SUCCESS(rc))
564 {
565 rc = CFGMR3QueryBoolDef(CFGMR3GetRoot(pVM), "HwVirtExtForced", &pVM->fHwVirtExtForced, false);
566 if (RT_SUCCESS(rc) && pVM->fHwVirtExtForced)
567 pVM->fHWACCMEnabled = true;
568
569 /*
570 * If executing in fake suplib mode disable RR3 and RR0 in the config.
571 */
572 const char *psz = RTEnvGet("VBOX_SUPLIB_FAKE");
573 if (psz && !strcmp(psz, "fake"))
574 {
575 CFGMR3RemoveValue(CFGMR3GetRoot(pVM), "RawR3Enabled");
576 CFGMR3InsertInteger(CFGMR3GetRoot(pVM), "RawR3Enabled", 0);
577 CFGMR3RemoveValue(CFGMR3GetRoot(pVM), "RawR0Enabled");
578 CFGMR3InsertInteger(CFGMR3GetRoot(pVM), "RawR0Enabled", 0);
579 }
580
581 /*
582 * Make sure the CPU count in the config data matches.
583 */
584 if (RT_SUCCESS(rc))
585 {
586 uint32_t cCPUsCfg;
587 rc = CFGMR3QueryU32Def(CFGMR3GetRoot(pVM), "NumCPUs", &cCPUsCfg, 1);
588 AssertLogRelMsgRC(rc, ("Configuration error: Querying \"NumCPUs\" as integer failed, rc=%Rrc\n", rc));
589 if (RT_SUCCESS(rc) && cCPUsCfg != cCpus)
590 {
591 AssertLogRelMsgFailed(("Configuration error: \"NumCPUs\"=%RU32 and VMR3CreateVM::cCPUs=%RU32 does not match!\n",
592 cCPUsCfg, cCpus));
593 rc = VERR_INVALID_PARAMETER;
594 }
595 }
596 if (RT_SUCCESS(rc))
597 {
598 /*
599 * Init the ring-3 components and ring-3 per cpu data, finishing it off
600 * by a relocation round (intermediate context finalization will do this).
601 */
602 rc = vmR3InitRing3(pVM, pUVM);
603 if (RT_SUCCESS(rc))
604 {
605 rc = vmR3InitVMCpu(pVM);
606 if (RT_SUCCESS(rc))
607 rc = PGMR3FinalizeMappings(pVM);
608 if (RT_SUCCESS(rc))
609 {
610
611 LogFlow(("Ring-3 init succeeded\n"));
612
613 /*
614 * Init the Ring-0 components.
615 */
616 rc = vmR3InitRing0(pVM);
617 if (RT_SUCCESS(rc))
618 {
619 /* Relocate again, because some switcher fixups depends on R0 init results. */
620 VMR3Relocate(pVM, 0);
621
622#ifdef VBOX_WITH_DEBUGGER
623 /*
624 * Init the tcp debugger console if we're building
625 * with debugger support.
626 */
627 void *pvUser = NULL;
628 rc = DBGCTcpCreate(pVM, &pvUser);
629 if ( RT_SUCCESS(rc)
630 || rc == VERR_NET_ADDRESS_IN_USE)
631 {
632 pUVM->vm.s.pvDBGC = pvUser;
633#endif
634 /*
635 * Init the Guest Context components.
636 */
637 rc = vmR3InitGC(pVM);
638 if (RT_SUCCESS(rc))
639 {
640 /*
641 * Now we can safely set the VM halt method to default.
642 */
643 rc = vmR3SetHaltMethodU(pUVM, VMHALTMETHOD_DEFAULT);
644 if (RT_SUCCESS(rc))
645 {
646 /*
647 * Set the state and link into the global list.
648 */
649 vmR3SetState(pVM, VMSTATE_CREATED);
650 pUVM->pNext = g_pUVMsHead;
651 g_pUVMsHead = pUVM;
652 return VINF_SUCCESS;
653 }
654 }
655#ifdef VBOX_WITH_DEBUGGER
656 DBGCTcpTerminate(pVM, pUVM->vm.s.pvDBGC);
657 pUVM->vm.s.pvDBGC = NULL;
658 }
659#endif
660 //..
661 }
662 }
663 vmR3Destroy(pVM);
664 }
665 }
666 //..
667
668 /* Clean CFGM. */
669 int rc2 = CFGMR3Term(pVM);
670 AssertRC(rc2);
671 }
672
673 /*
674 * Drop all references to VM and the VMCPU structures, then
675 * tell GVMM to destroy the VM.
676 */
677 pUVM->pVM = NULL;
678 for (VMCPUID i = 0; i < pUVM->cCpus; i++)
679 {
680 pUVM->aCpus[i].pVM = NULL;
681 pUVM->aCpus[i].pVCpu = NULL;
682 }
683 Assert(pUVM->vm.s.enmHaltMethod == VMHALTMETHOD_BOOTSTRAP);
684
685 if (pUVM->cCpus > 1)
686 {
687 /* Poke the other EMTs since they may have stale pVM and pVCpu references
688 on the stack (see VMR3WaitU for instance) if they've been awakened after
689 VM creation. */
690 for (VMCPUID i = 1; i < pUVM->cCpus; i++)
691 VMR3NotifyCpuFFU(&pUVM->aCpus[i], 0);
692 RTThreadSleep(RT_MIN(100 + 25 *(pUVM->cCpus - 1), 500)); /* very sophisticated */
693 }
694
695 int rc2 = SUPCallVMMR0Ex(CreateVMReq.pVMR0, 0 /* VCPU 0 */, VMMR0_DO_GVMM_DESTROY_VM, 0, NULL);
696 AssertRC(rc2);
697 }
698 else
699 vmR3SetErrorU(pUVM, rc, RT_SRC_POS, N_("VM creation failed (GVMM)"));
700
701 LogFlow(("vmR3CreateU: returns %Rrc\n", rc));
702 return rc;
703}
704
705
706/**
707 * Register the calling EMT with GVM.
708 *
709 * @returns VBox status code.
710 * @param pVM The VM handle.
711 * @param idCpu The Virtual CPU ID.
712 */
713static DECLCALLBACK(int) vmR3RegisterEMT(PVM pVM, VMCPUID idCpu)
714{
715 Assert(VMMGetCpuId(pVM) == idCpu);
716 int rc = SUPCallVMMR0Ex(pVM->pVMR0, idCpu, VMMR0_DO_GVMM_REGISTER_VMCPU, 0, NULL);
717 if (RT_FAILURE(rc))
718 LogRel(("idCpu=%u rc=%Rrc\n", idCpu, rc));
719 return rc;
720}
721
722
723/**
724 * Initializes all R3 components of the VM
725 */
726static int vmR3InitRing3(PVM pVM, PUVM pUVM)
727{
728 int rc;
729
730 /*
731 * Register the other EMTs with GVM.
732 */
733 for (VMCPUID idCpu = 1; idCpu < pVM->cCPUs; idCpu++)
734 {
735 PVMREQ pReq;
736 rc = VMR3ReqCallU(pUVM, idCpu, &pReq, RT_INDEFINITE_WAIT, 0 /*fFlags*/,
737 (PFNRT)vmR3RegisterEMT, 2, pVM, idCpu);
738 if (RT_SUCCESS(rc))
739 rc = pReq->iStatus;
740 VMR3ReqFree(pReq);
741 if (RT_FAILURE(rc))
742 return rc;
743 }
744
745 /*
746 * Init all R3 components, the order here might be important.
747 */
748 rc = MMR3Init(pVM);
749 if (RT_SUCCESS(rc))
750 {
751 STAM_REG(pVM, &pVM->StatTotalInGC, STAMTYPE_PROFILE_ADV, "/PROF/VM/InGC", STAMUNIT_TICKS_PER_CALL, "Profiling the total time spent in GC.");
752 STAM_REG(pVM, &pVM->StatSwitcherToGC, STAMTYPE_PROFILE_ADV, "/PROF/VM/SwitchToGC", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
753 STAM_REG(pVM, &pVM->StatSwitcherToHC, STAMTYPE_PROFILE_ADV, "/PROF/VM/SwitchToHC", STAMUNIT_TICKS_PER_CALL, "Profiling switching to HC.");
754 STAM_REG(pVM, &pVM->StatSwitcherSaveRegs, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/SaveRegs", STAMUNIT_TICKS_PER_CALL,"Profiling switching to GC.");
755 STAM_REG(pVM, &pVM->StatSwitcherSysEnter, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/SysEnter", STAMUNIT_TICKS_PER_CALL,"Profiling switching to GC.");
756 STAM_REG(pVM, &pVM->StatSwitcherDebug, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/Debug", STAMUNIT_TICKS_PER_CALL,"Profiling switching to GC.");
757 STAM_REG(pVM, &pVM->StatSwitcherCR0, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/CR0", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
758 STAM_REG(pVM, &pVM->StatSwitcherCR4, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/CR4", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
759 STAM_REG(pVM, &pVM->StatSwitcherLgdt, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/Lgdt", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
760 STAM_REG(pVM, &pVM->StatSwitcherLidt, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/Lidt", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
761 STAM_REG(pVM, &pVM->StatSwitcherLldt, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/Lldt", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
762 STAM_REG(pVM, &pVM->StatSwitcherTSS, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/TSS", STAMUNIT_TICKS_PER_CALL, "Profiling switching to GC.");
763 STAM_REG(pVM, &pVM->StatSwitcherJmpCR3, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/JmpCR3", STAMUNIT_TICKS_PER_CALL,"Profiling switching to GC.");
764 STAM_REG(pVM, &pVM->StatSwitcherRstrRegs, STAMTYPE_PROFILE_ADV, "/VM/Switcher/ToGC/RstrRegs", STAMUNIT_TICKS_PER_CALL,"Profiling switching to GC.");
765
766 for (unsigned iCpu=0;iCpu<pVM->cCPUs;iCpu++)
767 {
768 rc = STAMR3RegisterF(pVM, &pUVM->aCpus[iCpu].vm.s.StatHaltYield, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling halted state yielding.", "/PROF/VM/CPU%d/Halt/Yield", iCpu);
769 AssertRC(rc);
770 rc = STAMR3RegisterF(pVM, &pUVM->aCpus[iCpu].vm.s.StatHaltBlock, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling halted state blocking.", "/PROF/VM/CPU%d/Halt/Block", iCpu);
771 AssertRC(rc);
772 rc = STAMR3RegisterF(pVM, &pUVM->aCpus[iCpu].vm.s.StatHaltTimers, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling halted state timer tasks.", "/PROF/VM/CPU%d/Halt/Timers", iCpu);
773 AssertRC(rc);
774 }
775
776 STAM_REG(pVM, &pUVM->vm.s.StatReqAllocNew, STAMTYPE_COUNTER, "/VM/Req/AllocNew", STAMUNIT_OCCURENCES, "Number of VMR3ReqAlloc returning a new packet.");
777 STAM_REG(pVM, &pUVM->vm.s.StatReqAllocRaces, STAMTYPE_COUNTER, "/VM/Req/AllocRaces", STAMUNIT_OCCURENCES, "Number of VMR3ReqAlloc causing races.");
778 STAM_REG(pVM, &pUVM->vm.s.StatReqAllocRecycled, STAMTYPE_COUNTER, "/VM/Req/AllocRecycled", STAMUNIT_OCCURENCES, "Number of VMR3ReqAlloc returning a recycled packet.");
779 STAM_REG(pVM, &pUVM->vm.s.StatReqFree, STAMTYPE_COUNTER, "/VM/Req/Free", STAMUNIT_OCCURENCES, "Number of VMR3ReqFree calls.");
780 STAM_REG(pVM, &pUVM->vm.s.StatReqFreeOverflow, STAMTYPE_COUNTER, "/VM/Req/FreeOverflow", STAMUNIT_OCCURENCES, "Number of times the request was actually freed.");
781
782 rc = CPUMR3Init(pVM);
783 if (RT_SUCCESS(rc))
784 {
785 rc = HWACCMR3Init(pVM);
786 if (RT_SUCCESS(rc))
787 {
788 rc = PGMR3Init(pVM);
789 if (RT_SUCCESS(rc))
790 {
791 rc = REMR3Init(pVM);
792 if (RT_SUCCESS(rc))
793 {
794 rc = MMR3InitPaging(pVM);
795 if (RT_SUCCESS(rc))
796 rc = TMR3Init(pVM);
797 if (RT_SUCCESS(rc))
798 {
799 rc = VMMR3Init(pVM);
800 if (RT_SUCCESS(rc))
801 {
802 rc = SELMR3Init(pVM);
803 if (RT_SUCCESS(rc))
804 {
805 rc = TRPMR3Init(pVM);
806 if (RT_SUCCESS(rc))
807 {
808 rc = CSAMR3Init(pVM);
809 if (RT_SUCCESS(rc))
810 {
811 rc = PATMR3Init(pVM);
812 if (RT_SUCCESS(rc))
813 {
814#ifdef VBOX_WITH_VMI
815 rc = PARAVR3Init(pVM);
816 if (RT_SUCCESS(rc))
817 {
818#endif
819 rc = IOMR3Init(pVM);
820 if (RT_SUCCESS(rc))
821 {
822 rc = EMR3Init(pVM);
823 if (RT_SUCCESS(rc))
824 {
825 rc = DBGFR3Init(pVM);
826 if (RT_SUCCESS(rc))
827 {
828 rc = PDMR3Init(pVM);
829 if (RT_SUCCESS(rc))
830 {
831 rc = PGMR3InitDynMap(pVM);
832 if (RT_SUCCESS(rc))
833 rc = MMR3HyperInitFinalize(pVM);
834 if (RT_SUCCESS(rc))
835 rc = PATMR3InitFinalize(pVM);
836 if (RT_SUCCESS(rc))
837 rc = PGMR3InitFinalize(pVM);
838 if (RT_SUCCESS(rc))
839 rc = SELMR3InitFinalize(pVM);
840 if (RT_SUCCESS(rc))
841 rc = TMR3InitFinalize(pVM);
842 if (RT_SUCCESS(rc))
843 rc = VMMR3InitFinalize(pVM);
844 if (RT_SUCCESS(rc))
845 rc = REMR3InitFinalize(pVM);
846 if (RT_SUCCESS(rc))
847 rc = vmR3InitDoCompleted(pVM, VMINITCOMPLETED_RING3);
848 if (RT_SUCCESS(rc))
849 {
850 LogFlow(("vmR3InitRing3: returns %Rrc\n", VINF_SUCCESS));
851 return VINF_SUCCESS;
852 }
853 int rc2 = PDMR3Term(pVM);
854 AssertRC(rc2);
855 }
856 int rc2 = DBGFR3Term(pVM);
857 AssertRC(rc2);
858 }
859 int rc2 = EMR3Term(pVM);
860 AssertRC(rc2);
861 }
862 int rc2 = IOMR3Term(pVM);
863 AssertRC(rc2);
864 }
865#ifdef VBOX_WITH_VMI
866 int rc2 = PARAVR3Term(pVM);
867 AssertRC(rc2);
868 }
869#endif
870 int rc2 = PATMR3Term(pVM);
871 AssertRC(rc2);
872 }
873 int rc2 = CSAMR3Term(pVM);
874 AssertRC(rc2);
875 }
876 int rc2 = TRPMR3Term(pVM);
877 AssertRC(rc2);
878 }
879 int rc2 = SELMR3Term(pVM);
880 AssertRC(rc2);
881 }
882 int rc2 = VMMR3Term(pVM);
883 AssertRC(rc2);
884 }
885 int rc2 = TMR3Term(pVM);
886 AssertRC(rc2);
887 }
888 int rc2 = REMR3Term(pVM);
889 AssertRC(rc2);
890 }
891 int rc2 = PGMR3Term(pVM);
892 AssertRC(rc2);
893 }
894 int rc2 = HWACCMR3Term(pVM);
895 AssertRC(rc2);
896 }
897 //int rc2 = CPUMR3Term(pVM);
898 //AssertRC(rc2);
899 }
900 /* MMR3Term is not called here because it'll kill the heap. */
901 }
902
903 LogFlow(("vmR3InitRing3: returns %Rrc\n", rc));
904 return rc;
905}
906
907
908/**
909 * Initializes all VM CPU components of the VM
910 */
911static int vmR3InitVMCpu(PVM pVM)
912{
913 int rc = VINF_SUCCESS;
914 int rc2;
915
916 rc = CPUMR3InitCPU(pVM);
917 if (RT_SUCCESS(rc))
918 {
919 rc = HWACCMR3InitCPU(pVM);
920 if (RT_SUCCESS(rc))
921 {
922 rc = PGMR3InitCPU(pVM);
923 if (RT_SUCCESS(rc))
924 {
925 rc = TMR3InitCPU(pVM);
926 if (RT_SUCCESS(rc))
927 {
928 rc = VMMR3InitCPU(pVM);
929 if (RT_SUCCESS(rc))
930 {
931 rc = EMR3InitCPU(pVM);
932 if (RT_SUCCESS(rc))
933 {
934 LogFlow(("vmR3InitVMCpu: returns %Rrc\n", VINF_SUCCESS));
935 return VINF_SUCCESS;
936 }
937
938 rc2 = VMMR3TermCPU(pVM);
939 AssertRC(rc2);
940 }
941 rc2 = TMR3TermCPU(pVM);
942 AssertRC(rc2);
943 }
944 rc2 = PGMR3TermCPU(pVM);
945 AssertRC(rc2);
946 }
947 rc2 = HWACCMR3TermCPU(pVM);
948 AssertRC(rc2);
949 }
950 rc2 = CPUMR3TermCPU(pVM);
951 AssertRC(rc2);
952 }
953 LogFlow(("vmR3InitVMCpu: returns %Rrc\n", rc));
954 return rc;
955}
956
957
958/**
959 * Initializes all R0 components of the VM
960 */
961static int vmR3InitRing0(PVM pVM)
962{
963 LogFlow(("vmR3InitRing0:\n"));
964
965 /*
966 * Check for FAKE suplib mode.
967 */
968 int rc = VINF_SUCCESS;
969 const char *psz = RTEnvGet("VBOX_SUPLIB_FAKE");
970 if (!psz || strcmp(psz, "fake"))
971 {
972 /*
973 * Call the VMMR0 component and let it do the init.
974 */
975 rc = VMMR3InitR0(pVM);
976 }
977 else
978 Log(("vmR3InitRing0: skipping because of VBOX_SUPLIB_FAKE=fake\n"));
979
980 /*
981 * Do notifications and return.
982 */
983 if (RT_SUCCESS(rc))
984 rc = vmR3InitDoCompleted(pVM, VMINITCOMPLETED_RING0);
985
986 /** todo: move this to the VMINITCOMPLETED_RING0 notification handler once implemented */
987 if (RT_SUCCESS(rc))
988 rc = HWACCMR3InitFinalizeR0(pVM);
989
990 LogFlow(("vmR3InitRing0: returns %Rrc\n", rc));
991 return rc;
992}
993
994
995/**
996 * Initializes all GC components of the VM
997 */
998static int vmR3InitGC(PVM pVM)
999{
1000 LogFlow(("vmR3InitGC:\n"));
1001
1002 /*
1003 * Check for FAKE suplib mode.
1004 */
1005 int rc = VINF_SUCCESS;
1006 const char *psz = RTEnvGet("VBOX_SUPLIB_FAKE");
1007 if (!psz || strcmp(psz, "fake"))
1008 {
1009 /*
1010 * Call the VMMR0 component and let it do the init.
1011 */
1012 rc = VMMR3InitRC(pVM);
1013 }
1014 else
1015 Log(("vmR3InitGC: skipping because of VBOX_SUPLIB_FAKE=fake\n"));
1016
1017 /*
1018 * Do notifications and return.
1019 */
1020 if (RT_SUCCESS(rc))
1021 rc = vmR3InitDoCompleted(pVM, VMINITCOMPLETED_GC);
1022 LogFlow(("vmR3InitGC: returns %Rrc\n", rc));
1023 return rc;
1024}
1025
1026
1027/**
1028 * Do init completed notifications.
1029 * This notifications can fail.
1030 *
1031 * @param pVM The VM handle.
1032 * @param enmWhat What's completed.
1033 */
1034static int vmR3InitDoCompleted(PVM pVM, VMINITCOMPLETED enmWhat)
1035{
1036 return VINF_SUCCESS;
1037}
1038
1039
1040/**
1041 * Calls the relocation functions for all VMM components so they can update
1042 * any GC pointers. When this function is called all the basic VM members
1043 * have been updated and the actual memory relocation have been done
1044 * by the PGM/MM.
1045 *
1046 * This is used both on init and on runtime relocations.
1047 *
1048 * @param pVM VM handle.
1049 * @param offDelta Relocation delta relative to old location.
1050 */
1051VMMR3DECL(void) VMR3Relocate(PVM pVM, RTGCINTPTR offDelta)
1052{
1053 LogFlow(("VMR3Relocate: offDelta=%RGv\n", offDelta));
1054
1055 /*
1056 * The order here is very important!
1057 */
1058 PGMR3Relocate(pVM, offDelta);
1059 PDMR3LdrRelocateU(pVM->pUVM, offDelta);
1060 PGMR3Relocate(pVM, 0); /* Repeat after PDM relocation. */
1061 CPUMR3Relocate(pVM);
1062 HWACCMR3Relocate(pVM);
1063 SELMR3Relocate(pVM);
1064 VMMR3Relocate(pVM, offDelta);
1065 SELMR3Relocate(pVM); /* !hack! fix stack! */
1066 TRPMR3Relocate(pVM, offDelta);
1067 PATMR3Relocate(pVM);
1068 CSAMR3Relocate(pVM, offDelta);
1069 IOMR3Relocate(pVM, offDelta);
1070 EMR3Relocate(pVM);
1071 TMR3Relocate(pVM, offDelta);
1072 DBGFR3Relocate(pVM, offDelta);
1073 PDMR3Relocate(pVM, offDelta);
1074}
1075
1076
1077/**
1078 * Power on the virtual machine.
1079 *
1080 * @returns 0 on success.
1081 * @returns VBox error code on failure.
1082 * @param pVM VM to power on.
1083 * @thread Any thread.
1084 * @vmstate Created
1085 * @vmstateto Running
1086 */
1087VMMR3DECL(int) VMR3PowerOn(PVM pVM)
1088{
1089 LogFlow(("VMR3PowerOn: pVM=%p\n", pVM));
1090
1091 /*
1092 * Validate input.
1093 */
1094 if (!pVM)
1095 {
1096 AssertMsgFailed(("Invalid VM pointer\n"));
1097 return VERR_INVALID_PARAMETER;
1098 }
1099
1100 /*
1101 * Request the operation in EMT (in order as VCPU 0 does all the work)
1102 */
1103 PVMREQ pReq;
1104 int rc = VMR3ReqCall(pVM, VMCPUID_ALL, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3PowerOn, 1, pVM);
1105 if (RT_SUCCESS(rc))
1106 {
1107 rc = pReq->iStatus;
1108 VMR3ReqFree(pReq);
1109 }
1110
1111 LogFlow(("VMR3PowerOn: returns %Rrc\n", rc));
1112 return rc;
1113}
1114
1115
1116/**
1117 * Power on the virtual machine.
1118 *
1119 * @returns 0 on success.
1120 * @returns VBox error code on failure.
1121 * @param pVM VM to power on.
1122 * @thread EMT
1123 */
1124static DECLCALLBACK(int) vmR3PowerOn(PVM pVM)
1125{
1126 LogFlow(("vmR3PowerOn: pVM=%p\n", pVM));
1127
1128 /*
1129 * EMT(0) does the actual power on work *before* the other EMTs
1130 * get here, they just need to set their state to STARTED so they
1131 * get out of the EMT loop and into EM.
1132 */
1133 PVMCPU pVCpu = VMMGetCpu(pVM);
1134 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
1135 if (pVCpu->idCpu != 0)
1136 return VINF_SUCCESS;
1137
1138 /*
1139 * Validate input.
1140 */
1141 if (pVM->enmVMState != VMSTATE_CREATED)
1142 {
1143 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1144 return VERR_VM_INVALID_VM_STATE;
1145 }
1146
1147 /*
1148 * Change the state, notify the components and resume the execution.
1149 */
1150 vmR3SetState(pVM, VMSTATE_RUNNING);
1151 PDMR3PowerOn(pVM);
1152
1153 return VINF_SUCCESS;
1154}
1155
1156
1157/**
1158 * Suspends a running VM.
1159 *
1160 * @returns 0 on success.
1161 * @returns VBox error code on failure.
1162 * @param pVM VM to suspend.
1163 * @thread Any thread.
1164 * @vmstate Running
1165 * @vmstateto Suspended
1166 */
1167VMMR3DECL(int) VMR3Suspend(PVM pVM)
1168{
1169 LogFlow(("VMR3Suspend: pVM=%p\n", pVM));
1170
1171 /*
1172 * Validate input.
1173 */
1174 if (!pVM)
1175 {
1176 AssertMsgFailed(("Invalid VM pointer\n"));
1177 return VERR_INVALID_PARAMETER;
1178 }
1179
1180 /*
1181 * Request the operation in EMT. (in reverse order as VCPU 0 does the actual work)
1182 */
1183 PVMREQ pReq = NULL;
1184 int rc = VMR3ReqCall(pVM, VMCPUID_ALL_REVERSE, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3Suspend, 1, pVM);
1185 if (RT_SUCCESS(rc))
1186 {
1187 rc = pReq->iStatus;
1188 VMR3ReqFree(pReq);
1189 }
1190 else
1191 Assert(pReq == NULL);
1192
1193 LogFlow(("VMR3Suspend: returns %Rrc\n", rc));
1194 return rc;
1195}
1196
1197
1198/**
1199 * Suspends a running VM and prevent state saving until the VM is resumed or stopped.
1200 *
1201 * @returns 0 on success.
1202 * @returns VBox error code on failure.
1203 * @param pVM VM to suspend.
1204 * @thread Any thread.
1205 * @vmstate Running
1206 * @vmstateto Suspended
1207 */
1208VMMR3DECL(int) VMR3SuspendNoSave(PVM pVM)
1209{
1210 pVM->vm.s.fPreventSaveState = true;
1211 return VMR3Suspend(pVM);
1212}
1213
1214
1215/**
1216 * Suspends a running VM.
1217 *
1218 * @returns 0 on success.
1219 * @returns VBox error code on failure.
1220 * @param pVM VM to suspend.
1221 * @thread EMT
1222 */
1223static DECLCALLBACK(int) vmR3Suspend(PVM pVM)
1224{
1225 LogFlow(("vmR3Suspend: pVM=%p\n", pVM));
1226
1227 /*
1228 * Validate input.
1229 */
1230 if (pVM->enmVMState != VMSTATE_RUNNING)
1231 {
1232 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1233 return VERR_VM_INVALID_VM_STATE;
1234 }
1235
1236 PVMCPU pVCpu = VMMGetCpu(pVM);
1237 /* Only VCPU 0 does the actual work (*after* all the other CPUs has been here). */
1238 if (pVCpu->idCpu != 0)
1239 return VINF_EM_SUSPEND;
1240
1241 /*
1242 * Change the state, notify the components and resume the execution.
1243 */
1244 vmR3SetState(pVM, VMSTATE_SUSPENDED);
1245 PDMR3Suspend(pVM);
1246
1247 return VINF_EM_SUSPEND;
1248}
1249
1250
1251/**
1252 * Resume VM execution.
1253 *
1254 * @returns 0 on success.
1255 * @returns VBox error code on failure.
1256 * @param pVM The VM to resume.
1257 * @thread Any thread.
1258 * @vmstate Suspended
1259 * @vmstateto Running
1260 */
1261VMMR3DECL(int) VMR3Resume(PVM pVM)
1262{
1263 LogFlow(("VMR3Resume: pVM=%p\n", pVM));
1264
1265 /*
1266 * Validate input.
1267 */
1268 if (!pVM)
1269 {
1270 AssertMsgFailed(("Invalid VM pointer\n"));
1271 return VERR_INVALID_PARAMETER;
1272 }
1273
1274 /*
1275 * Request the operation in EMT. (in VCPU order as VCPU 0 does the actual work)
1276 */
1277 PVMREQ pReq = NULL;
1278 int rc = VMR3ReqCall(pVM, VMCPUID_ALL, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3Resume, 1, pVM);
1279 if (RT_SUCCESS(rc))
1280 {
1281 rc = pReq->iStatus;
1282 VMR3ReqFree(pReq);
1283 }
1284 else
1285 Assert(pReq == NULL);
1286
1287 LogFlow(("VMR3Resume: returns %Rrc\n", rc));
1288 return rc;
1289}
1290
1291
1292/**
1293 * Resume VM execution.
1294 *
1295 * @returns 0 on success.
1296 * @returns VBox error code on failure.
1297 * @param pVM The VM to resume.
1298 * @thread EMT
1299 */
1300static DECLCALLBACK(int) vmR3Resume(PVM pVM)
1301{
1302 LogFlow(("vmR3Resume: pVM=%p\n", pVM));
1303
1304 PVMCPU pVCpu = VMMGetCpu(pVM);
1305 /* Only VCPU 0 does the actual work (*before* the others wake up). */
1306 if (pVCpu->idCpu != 0)
1307 return VINF_EM_RESUME;
1308
1309 /*
1310 * Validate input.
1311 */
1312 if (pVM->enmVMState != VMSTATE_SUSPENDED)
1313 {
1314 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1315 return VERR_VM_INVALID_VM_STATE;
1316 }
1317
1318 /*
1319 * Change the state, notify the components and resume the execution.
1320 */
1321 pVM->vm.s.fPreventSaveState = false;
1322 vmR3SetState(pVM, VMSTATE_RUNNING);
1323 PDMR3Resume(pVM);
1324
1325 return VINF_EM_RESUME;
1326}
1327
1328
1329/**
1330 * Save current VM state.
1331 *
1332 * To save and terminate the VM, the VM must be suspended before the call.
1333 *
1334 * @returns 0 on success.
1335 * @returns VBox error code on failure.
1336 * @param pVM VM which state should be saved.
1337 * @param pszFilename Name of the save state file.
1338 * @param pfnProgress Progress callback. Optional.
1339 * @param pvUser User argument for the progress callback.
1340 * @thread Any thread.
1341 * @vmstate Suspended
1342 * @vmstateto Unchanged state.
1343 */
1344VMMR3DECL(int) VMR3Save(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser)
1345{
1346 LogFlow(("VMR3Save: pVM=%p pszFilename=%p:{%s} pfnProgress=%p pvUser=%p\n", pVM, pszFilename, pszFilename, pfnProgress, pvUser));
1347
1348 /*
1349 * Validate input.
1350 */
1351 if (!pVM)
1352 {
1353 AssertMsgFailed(("Invalid VM pointer\n"));
1354 return VERR_INVALID_PARAMETER;
1355 }
1356 if (!pszFilename)
1357 {
1358 AssertMsgFailed(("Must specify a filename to save the state to, wise guy!\n"));
1359 return VERR_INVALID_PARAMETER;
1360 }
1361
1362 /*
1363 * Request the operation in EMT.
1364 */
1365 PVMREQ pReq;
1366 int rc = VMR3ReqCall(pVM, 0 /* VCPU 0 */, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3Save, 4, pVM, pszFilename, pfnProgress, pvUser);
1367 if (RT_SUCCESS(rc))
1368 {
1369 rc = pReq->iStatus;
1370 VMR3ReqFree(pReq);
1371 }
1372
1373 LogFlow(("VMR3Save: returns %Rrc\n", rc));
1374 return rc;
1375}
1376
1377
1378/**
1379 * Save current VM state.
1380 *
1381 * To save and terminate the VM, the VM must be suspended before the call.
1382 *
1383 * @returns 0 on success.
1384 * @returns VBox error code on failure.
1385 * @param pVM VM which state should be saved.
1386 * @param pszFilename Name of the save state file.
1387 * @param pfnProgress Progress callback. Optional.
1388 * @param pvUser User argument for the progress callback.
1389 * @thread EMT
1390 */
1391static DECLCALLBACK(int) vmR3Save(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser)
1392{
1393 LogFlow(("vmR3Save: pVM=%p pszFilename=%p:{%s} pfnProgress=%p pvUser=%p\n", pVM, pszFilename, pszFilename, pfnProgress, pvUser));
1394
1395 /*
1396 * Validate input.
1397 */
1398 if (pVM->enmVMState != VMSTATE_SUSPENDED)
1399 {
1400 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1401 return VERR_VM_INVALID_VM_STATE;
1402 }
1403
1404 /* If we are in an inconsistent state, then we don't allow state saving. */
1405 if (pVM->vm.s.fPreventSaveState)
1406 {
1407 LogRel(("VMM: vmR3Save: saving the VM state is not allowed at this moment\n"));
1408 return VERR_VM_SAVE_STATE_NOT_ALLOWED;
1409 }
1410
1411 /*
1412 * Change the state and perform the save.
1413 */
1414 vmR3SetState(pVM, VMSTATE_SAVING);
1415 int rc = SSMR3Save(pVM, pszFilename, SSMAFTER_CONTINUE, pfnProgress, pvUser);
1416 vmR3SetState(pVM, VMSTATE_SUSPENDED);
1417
1418 return rc;
1419}
1420
1421
1422/**
1423 * Loads a new VM state.
1424 *
1425 * To restore a saved state on VM startup, call this function and then
1426 * resume the VM instead of powering it on.
1427 *
1428 * @returns 0 on success.
1429 * @returns VBox error code on failure.
1430 * @param pVM VM which state should be saved.
1431 * @param pszFilename Name of the save state file.
1432 * @param pfnProgress Progress callback. Optional.
1433 * @param pvUser User argument for the progress callback.
1434 * @thread Any thread.
1435 * @vmstate Created, Suspended
1436 * @vmstateto Suspended
1437 */
1438VMMR3DECL(int) VMR3Load(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser)
1439{
1440 LogFlow(("VMR3Load: pVM=%p pszFilename=%p:{%s} pfnProgress=%p pvUser=%p\n", pVM, pszFilename, pszFilename, pfnProgress, pvUser));
1441
1442 /*
1443 * Validate input.
1444 */
1445 if (!pVM)
1446 {
1447 AssertMsgFailed(("Invalid VM pointer\n"));
1448 return VERR_INVALID_PARAMETER;
1449 }
1450 if (!pszFilename)
1451 {
1452 AssertMsgFailed(("Must specify a filename to load the state from, wise guy!\n"));
1453 return VERR_INVALID_PARAMETER;
1454 }
1455
1456 /*
1457 * Request the operation in EMT.
1458 */
1459 PVMREQ pReq;
1460 int rc = VMR3ReqCall(pVM, 0 /* VCPU 0 */, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3Load, 4, pVM, pszFilename, pfnProgress, pvUser);
1461 if (RT_SUCCESS(rc))
1462 {
1463 rc = pReq->iStatus;
1464 VMR3ReqFree(pReq);
1465 }
1466
1467 LogFlow(("VMR3Load: returns %Rrc\n", rc));
1468 return rc;
1469}
1470
1471
1472/**
1473 * Loads a new VM state.
1474 *
1475 * To restore a saved state on VM startup, call this function and then
1476 * resume the VM instead of powering it on.
1477 *
1478 * @returns 0 on success.
1479 * @returns VBox error code on failure.
1480 * @param pVM VM which state should be saved.
1481 * @param pszFilename Name of the save state file.
1482 * @param pfnProgress Progress callback. Optional.
1483 * @param pvUser User argument for the progress callback.
1484 * @thread EMT.
1485 */
1486static DECLCALLBACK(int) vmR3Load(PVM pVM, const char *pszFilename, PFNVMPROGRESS pfnProgress, void *pvUser)
1487{
1488 LogFlow(("vmR3Load: pVM=%p pszFilename=%p:{%s} pfnProgress=%p pvUser=%p\n", pVM, pszFilename, pszFilename, pfnProgress, pvUser));
1489
1490 /*
1491 * Validate input.
1492 */
1493 if ( pVM->enmVMState != VMSTATE_SUSPENDED
1494 && pVM->enmVMState != VMSTATE_CREATED)
1495 {
1496 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1497 return VMSetError(pVM, VERR_VM_INVALID_VM_STATE, RT_SRC_POS, N_("Invalid VM state (%s) for restoring state from '%s'"),
1498 VMR3GetStateName(pVM->enmVMState), pszFilename);
1499 }
1500
1501 /*
1502 * Change the state and perform the load.
1503 */
1504 vmR3SetState(pVM, VMSTATE_LOADING);
1505 int rc = SSMR3Load(pVM, pszFilename, SSMAFTER_RESUME, pfnProgress, pvUser);
1506 if (RT_SUCCESS(rc))
1507 {
1508 /* Not paranoia anymore; the saved guest might use different hypervisor selectors. We must call VMR3Relocate. */
1509 VMR3Relocate(pVM, 0);
1510 vmR3SetState(pVM, VMSTATE_SUSPENDED);
1511 }
1512 else
1513 {
1514 vmR3SetState(pVM, VMSTATE_LOAD_FAILURE);
1515 rc = VMSetError(pVM, rc, RT_SRC_POS, N_("Unable to restore the virtual machine's saved state from '%s'. It may be damaged or from an older version of VirtualBox. Please discard the saved state before starting the virtual machine"), pszFilename);
1516 }
1517
1518 return rc;
1519}
1520
1521
1522/**
1523 * Power Off the VM.
1524 *
1525 * @returns 0 on success.
1526 * @returns VBox error code on failure.
1527 * @param pVM VM which should be destroyed.
1528 * @thread Any thread.
1529 * @vmstate Suspended, Running, Guru Meditation, Load Failure
1530 * @vmstateto Off
1531 */
1532VMMR3DECL(int) VMR3PowerOff(PVM pVM)
1533{
1534 LogFlow(("VMR3PowerOff: pVM=%p\n", pVM));
1535
1536 /*
1537 * Validate input.
1538 */
1539 if (!pVM)
1540 {
1541 AssertMsgFailed(("Invalid VM pointer\n"));
1542 return VERR_INVALID_PARAMETER;
1543 }
1544
1545 /*
1546 * Request the operation in EMT. (in reverse order as VCPU 0 does the actual work)
1547 */
1548 PVMREQ pReq = NULL;
1549 int rc = VMR3ReqCall(pVM, VMCPUID_ALL_REVERSE, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3PowerOff, 1, pVM);
1550 if (RT_SUCCESS(rc))
1551 {
1552 rc = pReq->iStatus;
1553 VMR3ReqFree(pReq);
1554 }
1555 else
1556 Assert(pReq == NULL);
1557
1558 LogFlow(("VMR3PowerOff: returns %Rrc\n", rc));
1559 return rc;
1560}
1561
1562
1563/**
1564 * Power Off the VM.
1565 *
1566 * @returns 0 on success.
1567 * @returns VBox error code on failure.
1568 * @param pVM VM which should be destroyed.
1569 * @thread EMT.
1570 */
1571static DECLCALLBACK(int) vmR3PowerOff(PVM pVM)
1572{
1573 LogFlow(("vmR3PowerOff: pVM=%p\n", pVM));
1574
1575 /*
1576 * Validate input.
1577 */
1578 if ( pVM->enmVMState != VMSTATE_RUNNING
1579 && pVM->enmVMState != VMSTATE_SUSPENDED
1580 && pVM->enmVMState != VMSTATE_LOAD_FAILURE
1581 && pVM->enmVMState != VMSTATE_GURU_MEDITATION)
1582 {
1583 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
1584 return VERR_VM_INVALID_VM_STATE;
1585 }
1586
1587 /*
1588 * EMT(0) does the actual power off work here *after* all the other EMTs
1589 * have been thru and entered the STOPPED state.
1590 */
1591 PVMCPU pVCpu = VMMGetCpu(pVM);
1592 VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STOPPED);
1593 if (pVCpu->idCpu != 0)
1594 return VINF_EM_OFF;
1595
1596 /*
1597 * For debugging purposes, we will log a summary of the guest state at this point.
1598 */
1599 if (pVM->enmVMState != VMSTATE_GURU_MEDITATION)
1600 {
1601 /* @todo SMP support? */
1602 PVMCPU pVCpu = VMMGetCpu(pVM);
1603
1604 /** @todo make the state dumping at VMR3PowerOff optional. */
1605 RTLogRelPrintf("****************** Guest state at power off ******************\n");
1606 DBGFR3Info(pVM, "cpumguest", "verbose", DBGFR3InfoLogRelHlp());
1607 RTLogRelPrintf("***\n");
1608 DBGFR3Info(pVM, "mode", NULL, DBGFR3InfoLogRelHlp());
1609 RTLogRelPrintf("***\n");
1610 DBGFR3Info(pVM, "activetimers", NULL, DBGFR3InfoLogRelHlp());
1611 RTLogRelPrintf("***\n");
1612 DBGFR3Info(pVM, "gdt", NULL, DBGFR3InfoLogRelHlp());
1613 /** @todo dump guest call stack. */
1614#if 1 // temporary while debugging #1589
1615 RTLogRelPrintf("***\n");
1616 uint32_t esp = CPUMGetGuestESP(pVCpu);
1617 if ( CPUMGetGuestSS(pVCpu) == 0
1618 && esp < _64K)
1619 {
1620 uint8_t abBuf[PAGE_SIZE];
1621 RTLogRelPrintf("***\n"
1622 "ss:sp=0000:%04x ", esp);
1623 uint32_t Start = esp & ~(uint32_t)63;
1624 int rc = PGMPhysSimpleReadGCPhys(pVM, abBuf, Start, 0x100);
1625 if (RT_SUCCESS(rc))
1626 RTLogRelPrintf("0000:%04x TO 0000:%04x:\n"
1627 "%.*Rhxd\n",
1628 Start, Start + 0x100 - 1,
1629 0x100, abBuf);
1630 else
1631 RTLogRelPrintf("rc=%Rrc\n", rc);
1632
1633 /* grub ... */
1634 if (esp < 0x2000 && esp > 0x1fc0)
1635 {
1636 rc = PGMPhysSimpleReadGCPhys(pVM, abBuf, 0x8000, 0x800);
1637 if (RT_SUCCESS(rc))
1638 RTLogRelPrintf("0000:8000 TO 0000:87ff:\n"
1639 "%.*Rhxd\n",
1640 0x800, abBuf);
1641 }
1642 /* microsoft cdrom hang ... */
1643 if (true)
1644 {
1645 rc = PGMPhysSimpleReadGCPhys(pVM, abBuf, 0x8000, 0x200);
1646 if (RT_SUCCESS(rc))
1647 RTLogRelPrintf("2000:0000 TO 2000:01ff:\n"
1648 "%.*Rhxd\n",
1649 0x200, abBuf);
1650 }
1651 }
1652#endif
1653 RTLogRelPrintf("************** End of Guest state at power off ***************\n");
1654 }
1655
1656 /*
1657 * Change the state to OFF and notify the components.
1658 */
1659 vmR3SetState(pVM, VMSTATE_OFF);
1660 PDMR3PowerOff(pVM);
1661
1662 return VINF_EM_OFF;
1663}
1664
1665
1666/**
1667 * Destroys the VM.
1668 * The VM must be powered off (or never really powered on) to call this function.
1669 * The VM handle is destroyed and can no longer be used up successful return.
1670 *
1671 * @returns VBox status code.
1672 * @param pVM VM which should be destroyed.
1673 * @thread Any thread but the emulation thread.
1674 * @vmstate Off, Created
1675 * @vmstateto N/A
1676 */
1677VMMR3DECL(int) VMR3Destroy(PVM pVM)
1678{
1679 LogFlow(("VMR3Destroy: pVM=%p\n", pVM));
1680
1681 /*
1682 * Validate input.
1683 */
1684 if (!pVM)
1685 return VERR_INVALID_PARAMETER;
1686 AssertPtrReturn(pVM, VERR_INVALID_POINTER);
1687 AssertMsgReturn( pVM->enmVMState == VMSTATE_OFF
1688 || pVM->enmVMState == VMSTATE_CREATED,
1689 ("Invalid VM state %d\n", pVM->enmVMState),
1690 VERR_VM_INVALID_VM_STATE);
1691
1692 /*
1693 * Change VM state to destroying and unlink the VM.
1694 */
1695 vmR3SetState(pVM, VMSTATE_DESTROYING);
1696
1697 /** @todo lock this when we start having multiple machines in a process... */
1698 PUVM pUVM = pVM->pUVM; AssertPtr(pUVM);
1699 if (g_pUVMsHead == pUVM)
1700 g_pUVMsHead = pUVM->pNext;
1701 else
1702 {
1703 PUVM pPrev = g_pUVMsHead;
1704 while (pPrev && pPrev->pNext != pUVM)
1705 pPrev = pPrev->pNext;
1706 AssertMsgReturn(pPrev, ("pUVM=%p / pVM=%p is INVALID!\n", pUVM, pVM), VERR_INVALID_PARAMETER);
1707
1708 pPrev->pNext = pUVM->pNext;
1709 }
1710 pUVM->pNext = NULL;
1711
1712 /*
1713 * Notify registered at destruction listeners.
1714 * (That's the debugger console.)
1715 */
1716 vmR3AtDtor(pVM);
1717
1718 /*
1719 * If we are the EMT of VCPU 0, then we'll delay the cleanup till later.
1720 */
1721 if (VMMGetCpuId(pVM) == 0)
1722 {
1723 pUVM->vm.s.fEMTDoesTheCleanup = true;
1724 pUVM->vm.s.fTerminateEMT = true;
1725 VM_FF_SET(pVM, VM_FF_TERMINATE);
1726
1727 /* Inform all other VCPUs too. */
1728 for (VMCPUID idCpu = 1; idCpu < pVM->cCPUs; idCpu++)
1729 {
1730 /*
1731 * Request EMT to do the larger part of the destruction.
1732 */
1733 PVMREQ pReq = NULL;
1734 int rc = VMR3ReqCallU(pUVM, idCpu, &pReq, RT_INDEFINITE_WAIT, 0, (PFNRT)vmR3Destroy, 1, pVM);
1735 if (RT_SUCCESS(rc))
1736 rc = pReq->iStatus;
1737 AssertRC(rc);
1738 VMR3ReqFree(pReq);
1739 }
1740 }
1741 else
1742 {
1743 /*
1744 * Request EMT to do the larger part of the destruction. (in reverse order as VCPU 0 does the real cleanup)
1745 */
1746 PVMREQ pReq = NULL;
1747 int rc = VMR3ReqCallU(pUVM, VMCPUID_ALL_REVERSE, &pReq, RT_INDEFINITE_WAIT, 0, (PFNRT)vmR3Destroy, 1, pVM);
1748 if (RT_SUCCESS(rc))
1749 rc = pReq->iStatus;
1750 AssertRC(rc);
1751 VMR3ReqFree(pReq);
1752
1753 /*
1754 * Now do the final bit where the heap and VM structures are freed up.
1755 * This will also wait 30 secs for the emulation threads to terminate.
1756 */
1757 vmR3DestroyUVM(pUVM, 30000);
1758 }
1759
1760 LogFlow(("VMR3Destroy: returns VINF_SUCCESS\n"));
1761 return VINF_SUCCESS;
1762}
1763
1764
1765/**
1766 * Internal destruction worker.
1767 *
1768 * This will do nearly all of the job, including sacking the EMT.
1769 *
1770 * @returns VBox status.
1771 * @param pVM VM handle.
1772 */
1773DECLCALLBACK(int) vmR3Destroy(PVM pVM)
1774{
1775 PUVM pUVM = pVM->pUVM;
1776 PVMCPU pVCpu = VMMGetCpu(pVM);
1777
1778 NOREF(pUVM);
1779 LogFlow(("vmR3Destroy: pVM=%p pUVM=%p\n", pVM, pUVM));
1780 VM_ASSERT_EMT(pVM);
1781
1782 /* Only VCPU 0 does the full cleanup. */
1783 if (pVCpu->idCpu != 0)
1784 return VINF_EM_TERMINATE;
1785
1786 /*
1787 * Dump statistics to the log.
1788 */
1789#if defined(VBOX_WITH_STATISTICS) || defined(LOG_ENABLED)
1790 RTLogFlags(NULL, "nodisabled nobuffered");
1791#endif
1792#ifdef VBOX_WITH_STATISTICS
1793 STAMR3Dump(pVM, "*");
1794#else
1795 LogRel(("************************* Statistics *************************\n"));
1796 STAMR3DumpToReleaseLog(pVM, "*");
1797 LogRel(("********************* End of statistics **********************\n"));
1798#endif
1799
1800 /*
1801 * Destroy the VM components.
1802 */
1803 int rc = TMR3Term(pVM);
1804 AssertRC(rc);
1805#ifdef VBOX_WITH_DEBUGGER
1806 rc = DBGCTcpTerminate(pVM, pUVM->vm.s.pvDBGC);
1807 pVM->pUVM->vm.s.pvDBGC = NULL;
1808#endif
1809 AssertRC(rc);
1810 rc = DBGFR3Term(pVM);
1811 AssertRC(rc);
1812 rc = PDMR3Term(pVM);
1813 AssertRC(rc);
1814 rc = EMR3Term(pVM);
1815 AssertRC(rc);
1816 rc = IOMR3Term(pVM);
1817 AssertRC(rc);
1818 rc = CSAMR3Term(pVM);
1819 AssertRC(rc);
1820 rc = PATMR3Term(pVM);
1821 AssertRC(rc);
1822 rc = TRPMR3Term(pVM);
1823 AssertRC(rc);
1824 rc = SELMR3Term(pVM);
1825 AssertRC(rc);
1826 rc = REMR3Term(pVM);
1827 AssertRC(rc);
1828 rc = HWACCMR3Term(pVM);
1829 AssertRC(rc);
1830 rc = PGMR3Term(pVM);
1831 AssertRC(rc);
1832 rc = VMMR3Term(pVM); /* Terminates the ring-0 code! */
1833 AssertRC(rc);
1834 rc = CPUMR3Term(pVM);
1835 AssertRC(rc);
1836 rc = PDMR3CritSectTerm(pVM);
1837 AssertRC(rc);
1838 rc = MMR3Term(pVM);
1839 AssertRC(rc);
1840
1841 /*
1842 * We're done in this thread (EMT).
1843 */
1844 ASMAtomicUoWriteBool(&pVM->pUVM->vm.s.fTerminateEMT, true);
1845 ASMAtomicWriteU32(&pVM->fGlobalForcedActions, VM_FF_TERMINATE);
1846 LogFlow(("vmR3Destroy: returning %Rrc\n", VINF_EM_TERMINATE));
1847 return VINF_EM_TERMINATE;
1848}
1849
1850
1851/**
1852 * Destroys the shared VM structure, leaving only UVM behind.
1853 *
1854 * This is called by EMT right before terminating.
1855 *
1856 * @param pUVM The UVM handle.
1857 */
1858void vmR3DestroyFinalBitFromEMT(PUVM pUVM)
1859{
1860 if (pUVM->pVM)
1861 {
1862 PVMCPU pVCpu = VMMGetCpu(pUVM->pVM);
1863
1864 /* VCPU 0 does all the cleanup work. */
1865 if (pVCpu->idCpu != 0)
1866 return;
1867
1868 /*
1869 * Modify state and then terminate MM.
1870 * (MM must be delayed until this point so we don't destroy the callbacks and the request packet.)
1871 */
1872 vmR3SetState(pUVM->pVM, VMSTATE_TERMINATED);
1873
1874 /*
1875 * Tell GVMM to destroy the VM and free its resources.
1876 */
1877 int rc = SUPCallVMMR0Ex(pUVM->pVM->pVMR0, 0 /* VCPU 0 */, VMMR0_DO_GVMM_DESTROY_VM, 0, NULL);
1878 AssertRC(rc);
1879 pUVM->pVM = NULL;
1880 }
1881
1882 /*
1883 * Did an EMT call VMR3Destroy and end up having to do all the work?
1884 */
1885 if (pUVM->vm.s.fEMTDoesTheCleanup)
1886 vmR3DestroyUVM(pUVM, 30000);
1887}
1888
1889
1890/**
1891 * Destroys the UVM portion.
1892 *
1893 * This is called as the final step in the VM destruction or as the cleanup
1894 * in case of a creation failure. If EMT called VMR3Destroy, meaning
1895 * VMINTUSERPERVM::fEMTDoesTheCleanup is true, it will call this as
1896 * vmR3DestroyFinalBitFromEMT completes.
1897 *
1898 * @param pVM VM Handle.
1899 * @param cMilliesEMTWait The number of milliseconds to wait for the emulation
1900 * threads.
1901 */
1902static void vmR3DestroyUVM(PUVM pUVM, uint32_t cMilliesEMTWait)
1903{
1904 /*
1905 * Signal termination of each the emulation threads and
1906 * wait for them to complete.
1907 */
1908 /* Signal them. */
1909 ASMAtomicUoWriteBool(&pUVM->vm.s.fTerminateEMT, true);
1910 for (VMCPUID i = 0; i < pUVM->cCpus; i++)
1911 {
1912 ASMAtomicUoWriteBool(&pUVM->aCpus[i].vm.s.fTerminateEMT, true);
1913 if (pUVM->pVM)
1914 VM_FF_SET(pUVM->pVM, VM_FF_TERMINATE);
1915 VMR3NotifyGlobalFFU(pUVM, VMNOTIFYFF_FLAGS_DONE_REM);
1916 RTSemEventSignal(pUVM->aCpus[i].vm.s.EventSemWait);
1917 }
1918
1919 /* Wait for them. */
1920 uint64_t NanoTS = RTTimeNanoTS();
1921 RTTHREAD hSelf = RTThreadSelf();
1922 ASMAtomicUoWriteBool(&pUVM->vm.s.fTerminateEMT, true);
1923 for (VMCPUID i = 0; i < pUVM->cCpus; i++)
1924 {
1925 RTTHREAD hThread = pUVM->aCpus[i].vm.s.ThreadEMT;
1926 if ( hThread != NIL_RTTHREAD
1927 && hThread != hSelf)
1928 {
1929 uint64_t cMilliesElapsed = (RTTimeNanoTS() - NanoTS) / 1000000;
1930 int rc2 = RTThreadWait(hThread,
1931 cMilliesElapsed < cMilliesEMTWait
1932 ? RT_MAX(cMilliesEMTWait - cMilliesElapsed, 2000)
1933 : 2000,
1934 NULL);
1935 if (rc2 == VERR_TIMEOUT) /* avoid the assertion when debugging. */
1936 rc2 = RTThreadWait(hThread, 1000, NULL);
1937 AssertLogRelMsgRC(rc2, ("i=%u rc=%Rrc\n", i, rc2));
1938 if (RT_SUCCESS(rc2))
1939 pUVM->aCpus[0].vm.s.ThreadEMT = NIL_RTTHREAD;
1940 }
1941 }
1942
1943 /* Cleanup the semaphores. */
1944 for (VMCPUID i = 0; i < pUVM->cCpus; i++)
1945 {
1946 RTSemEventDestroy(pUVM->aCpus[i].vm.s.EventSemWait);
1947 pUVM->aCpus[i].vm.s.EventSemWait = NIL_RTSEMEVENT;
1948 }
1949
1950 /*
1951 * Free the event semaphores associated with the request packets.
1952 */
1953 unsigned cReqs = 0;
1954 for (unsigned i = 0; i < RT_ELEMENTS(pUVM->vm.s.apReqFree); i++)
1955 {
1956 PVMREQ pReq = pUVM->vm.s.apReqFree[i];
1957 pUVM->vm.s.apReqFree[i] = NULL;
1958 for (; pReq; pReq = pReq->pNext, cReqs++)
1959 {
1960 pReq->enmState = VMREQSTATE_INVALID;
1961 RTSemEventDestroy(pReq->EventSem);
1962 }
1963 }
1964 Assert(cReqs == pUVM->vm.s.cReqFree); NOREF(cReqs);
1965
1966 /*
1967 * Kill all queued requests. (There really shouldn't be any!)
1968 */
1969 for (unsigned i = 0; i < 10; i++)
1970 {
1971 PVMREQ pReqHead = (PVMREQ)ASMAtomicXchgPtr((void *volatile *)&pUVM->vm.s.pReqs, NULL);
1972 AssertMsg(!pReqHead, ("This isn't supposed to happen! VMR3Destroy caller has to serialize this.\n"));
1973 if (!pReqHead)
1974 break;
1975 for (PVMREQ pReq = pReqHead; pReq; pReq = pReq->pNext)
1976 {
1977 ASMAtomicUoWriteSize(&pReq->iStatus, VERR_INTERNAL_ERROR);
1978 ASMAtomicWriteSize(&pReq->enmState, VMREQSTATE_INVALID);
1979 RTSemEventSignal(pReq->EventSem);
1980 RTThreadSleep(2);
1981 RTSemEventDestroy(pReq->EventSem);
1982 }
1983 /* give them a chance to respond before we free the request memory. */
1984 RTThreadSleep(32);
1985 }
1986
1987 /*
1988 * Now all queued VCPU requests (again, there shouldn't be any).
1989 */
1990 for (VMCPUID i = 0; i < pUVM->cCpus; i++)
1991 {
1992 PUVMCPU pUVCpu = &pUVM->aCpus[i];
1993
1994 for (unsigned i = 0; i < 10; i++)
1995 {
1996 PVMREQ pReqHead = (PVMREQ)ASMAtomicXchgPtr((void *volatile *)&pUVCpu->vm.s.pReqs, NULL);
1997 AssertMsg(!pReqHead, ("This isn't supposed to happen! VMR3Destroy caller has to serialize this.\n"));
1998 if (!pReqHead)
1999 break;
2000 for (PVMREQ pReq = pReqHead; pReq; pReq = pReq->pNext)
2001 {
2002 ASMAtomicUoWriteSize(&pReq->iStatus, VERR_INTERNAL_ERROR);
2003 ASMAtomicWriteSize(&pReq->enmState, VMREQSTATE_INVALID);
2004 RTSemEventSignal(pReq->EventSem);
2005 RTThreadSleep(2);
2006 RTSemEventDestroy(pReq->EventSem);
2007 }
2008 /* give them a chance to respond before we free the request memory. */
2009 RTThreadSleep(32);
2010 }
2011 }
2012
2013 /*
2014 * Make sure the VMMR0.r0 module and whatever else is unloaded.
2015 */
2016 PDMR3TermUVM(pUVM);
2017
2018 /*
2019 * Terminate the support library if initialized.
2020 */
2021 if (pUVM->vm.s.pSession)
2022 {
2023 int rc = SUPTerm();
2024 AssertRC(rc);
2025 pUVM->vm.s.pSession = NIL_RTR0PTR;
2026 }
2027
2028 /*
2029 * Destroy the MM heap and free the UVM structure.
2030 */
2031 MMR3TermUVM(pUVM);
2032 STAMR3TermUVM(pUVM);
2033
2034 RTTlsFree(pUVM->vm.s.idxTLS);
2035
2036 ASMAtomicUoWriteU32(&pUVM->u32Magic, UINT32_MAX);
2037 RTMemFree(pUVM);
2038
2039 RTLogFlush(NULL);
2040}
2041
2042
2043/**
2044 * Enumerates the VMs in this process.
2045 *
2046 * @returns Pointer to the next VM.
2047 * @returns NULL when no more VMs.
2048 * @param pVMPrev The previous VM
2049 * Use NULL to start the enumeration.
2050 */
2051VMMR3DECL(PVM) VMR3EnumVMs(PVM pVMPrev)
2052{
2053 /*
2054 * This is quick and dirty. It has issues with VM being
2055 * destroyed during the enumeration.
2056 */
2057 PUVM pNext;
2058 if (pVMPrev)
2059 pNext = pVMPrev->pUVM->pNext;
2060 else
2061 pNext = g_pUVMsHead;
2062 return pNext ? pNext->pVM : NULL;
2063}
2064
2065
2066/**
2067 * Registers an at VM destruction callback.
2068 *
2069 * @returns VBox status code.
2070 * @param pfnAtDtor Pointer to callback.
2071 * @param pvUser User argument.
2072 */
2073VMMR3DECL(int) VMR3AtDtorRegister(PFNVMATDTOR pfnAtDtor, void *pvUser)
2074{
2075 /*
2076 * Check if already registered.
2077 */
2078 VM_ATDTOR_LOCK();
2079 PVMATDTOR pCur = g_pVMAtDtorHead;
2080 while (pCur)
2081 {
2082 if (pfnAtDtor == pCur->pfnAtDtor)
2083 {
2084 VM_ATDTOR_UNLOCK();
2085 AssertMsgFailed(("Already registered at destruction callback %p!\n", pfnAtDtor));
2086 return VERR_INVALID_PARAMETER;
2087 }
2088
2089 /* next */
2090 pCur = pCur->pNext;
2091 }
2092 VM_ATDTOR_UNLOCK();
2093
2094 /*
2095 * Allocate new entry.
2096 */
2097 PVMATDTOR pVMAtDtor = (PVMATDTOR)RTMemAlloc(sizeof(*pVMAtDtor));
2098 if (!pVMAtDtor)
2099 return VERR_NO_MEMORY;
2100
2101 VM_ATDTOR_LOCK();
2102 pVMAtDtor->pfnAtDtor = pfnAtDtor;
2103 pVMAtDtor->pvUser = pvUser;
2104 pVMAtDtor->pNext = g_pVMAtDtorHead;
2105 g_pVMAtDtorHead = pVMAtDtor;
2106 VM_ATDTOR_UNLOCK();
2107
2108 return VINF_SUCCESS;
2109}
2110
2111
2112/**
2113 * Deregisters an at VM destruction callback.
2114 *
2115 * @returns VBox status code.
2116 * @param pfnAtDtor Pointer to callback.
2117 */
2118VMMR3DECL(int) VMR3AtDtorDeregister(PFNVMATDTOR pfnAtDtor)
2119{
2120 /*
2121 * Find it, unlink it and free it.
2122 */
2123 VM_ATDTOR_LOCK();
2124 PVMATDTOR pPrev = NULL;
2125 PVMATDTOR pCur = g_pVMAtDtorHead;
2126 while (pCur)
2127 {
2128 if (pfnAtDtor == pCur->pfnAtDtor)
2129 {
2130 if (pPrev)
2131 pPrev->pNext = pCur->pNext;
2132 else
2133 g_pVMAtDtorHead = pCur->pNext;
2134 pCur->pNext = NULL;
2135 VM_ATDTOR_UNLOCK();
2136
2137 RTMemFree(pCur);
2138 return VINF_SUCCESS;
2139 }
2140
2141 /* next */
2142 pPrev = pCur;
2143 pCur = pCur->pNext;
2144 }
2145 VM_ATDTOR_UNLOCK();
2146
2147 return VERR_INVALID_PARAMETER;
2148}
2149
2150
2151/**
2152 * Walks the list of at VM destructor callbacks.
2153 * @param pVM The VM which is about to be destroyed.
2154 */
2155static void vmR3AtDtor(PVM pVM)
2156{
2157 /*
2158 * Find it, unlink it and free it.
2159 */
2160 VM_ATDTOR_LOCK();
2161 for (PVMATDTOR pCur = g_pVMAtDtorHead; pCur; pCur = pCur->pNext)
2162 pCur->pfnAtDtor(pVM, pCur->pvUser);
2163 VM_ATDTOR_UNLOCK();
2164}
2165
2166
2167/**
2168 * Reset the current VM.
2169 *
2170 * @returns VBox status code.
2171 * @param pVM VM to reset.
2172 */
2173VMMR3DECL(int) VMR3Reset(PVM pVM)
2174{
2175 int rc = VINF_SUCCESS;
2176
2177 /*
2178 * Check the state.
2179 */
2180 if (!pVM)
2181 return VERR_INVALID_PARAMETER;
2182 if ( pVM->enmVMState != VMSTATE_RUNNING
2183 && pVM->enmVMState != VMSTATE_SUSPENDED)
2184 {
2185 AssertMsgFailed(("Invalid VM state %d\n", pVM->enmVMState));
2186 return VERR_VM_INVALID_VM_STATE;
2187 }
2188
2189 /*
2190 * Queue reset request to the emulation thread
2191 * and wait for it to be processed. (in reverse order as VCPU 0 does the real cleanup)
2192 */
2193 PVMREQ pReq = NULL;
2194 rc = VMR3ReqCall(pVM, VMCPUID_ALL_REVERSE, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3Reset, 1, pVM);
2195 if (RT_SUCCESS(rc))
2196 rc = pReq->iStatus;
2197 AssertRC(rc);
2198 VMR3ReqFree(pReq);
2199
2200 return rc;
2201}
2202
2203
2204/**
2205 * Worker which checks integrity of some internal structures.
2206 * This is yet another attempt to track down that AVL tree crash.
2207 */
2208static void vmR3CheckIntegrity(PVM pVM)
2209{
2210#ifdef VBOX_STRICT
2211 int rc = PGMR3CheckIntegrity(pVM);
2212 AssertReleaseRC(rc);
2213#endif
2214}
2215
2216
2217/**
2218 * Reset request processor.
2219 *
2220 * This is called by the emulation threads as a response to the
2221 * reset request issued by VMR3Reset().
2222 *
2223 * @returns VBox status code.
2224 * @param pVM VM to reset.
2225 */
2226static DECLCALLBACK(int) vmR3Reset(PVM pVM)
2227{
2228 PVMCPU pVCpu = VMMGetCpu(pVM);
2229
2230 /*
2231 * EMT(0) does the full cleanup *after* all the other EMTs has been
2232 * thru here and been told to enter the EMSTATE_WAIT_SIPI state.
2233 */
2234 VMCPU_ASSERT_STATE(pVCpu, VMCPUSTATE_STARTED);
2235
2236 /* Clear all pending forced actions. */
2237 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_ALL_MASK & ~VMCPU_FF_REQUEST);
2238
2239 if (pVCpu->idCpu != 0)
2240 {
2241 CPUMR3ResetCpu(pVCpu);
2242 return VINF_EM_RESET;
2243 }
2244
2245 /*
2246 * As a safety precaution we temporarily change the state while resetting.
2247 * (If VMR3Reset was not called from EMT we might have change state... let's ignore that fact for now.)
2248 */
2249 VMSTATE enmVMState = pVM->enmVMState;
2250 Assert(enmVMState == VMSTATE_SUSPENDED || enmVMState == VMSTATE_RUNNING);
2251 vmR3SetState(pVM, VMSTATE_RESETTING);
2252 vmR3CheckIntegrity(pVM);
2253
2254
2255 /*
2256 * Reset the VM components.
2257 */
2258 PATMR3Reset(pVM);
2259 CSAMR3Reset(pVM);
2260 PGMR3Reset(pVM); /* We clear VM RAM in PGMR3Reset. It's vital PDMR3Reset is executed
2261 * _afterwards_. E.g. ACPI sets up RAM tables during init/reset. */
2262 MMR3Reset(pVM);
2263 PDMR3Reset(pVM);
2264 SELMR3Reset(pVM);
2265 TRPMR3Reset(pVM);
2266 vmR3AtResetU(pVM->pUVM);
2267 REMR3Reset(pVM);
2268 IOMR3Reset(pVM);
2269 CPUMR3Reset(pVM);
2270 TMR3Reset(pVM);
2271 EMR3Reset(pVM);
2272 HWACCMR3Reset(pVM); /* This must come *after* PATM, CSAM, CPUM, SELM and TRPM. */
2273
2274#ifdef LOG_ENABLED
2275 /*
2276 * Debug logging.
2277 */
2278 RTLogPrintf("\n\nThe VM was reset:\n");
2279 DBGFR3Info(pVM, "cpum", "verbose", NULL);
2280#endif
2281
2282 /*
2283 * Restore the state.
2284 */
2285 vmR3CheckIntegrity(pVM);
2286 Assert(pVM->enmVMState == VMSTATE_RESETTING);
2287 vmR3SetState(pVM, enmVMState);
2288
2289 return VINF_EM_RESET;
2290}
2291
2292
2293/**
2294 * Walks the list of at VM reset callbacks and calls them
2295 *
2296 * @returns VBox status code.
2297 * Any failure is fatal.
2298 * @param pUVM Pointe to the user mode VM structure.
2299 */
2300static int vmR3AtResetU(PUVM pUVM)
2301{
2302 /*
2303 * Walk the list and call them all.
2304 */
2305 int rc = VINF_SUCCESS;
2306 for (PVMATRESET pCur = pUVM->vm.s.pAtReset; pCur; pCur = pCur->pNext)
2307 {
2308 /* do the call */
2309 switch (pCur->enmType)
2310 {
2311 case VMATRESETTYPE_DEV:
2312 rc = pCur->u.Dev.pfnCallback(pCur->u.Dev.pDevIns, pCur->pvUser);
2313 break;
2314 case VMATRESETTYPE_INTERNAL:
2315 rc = pCur->u.Internal.pfnCallback(pUVM->pVM, pCur->pvUser);
2316 break;
2317 case VMATRESETTYPE_EXTERNAL:
2318 pCur->u.External.pfnCallback(pCur->pvUser);
2319 break;
2320 default:
2321 AssertMsgFailed(("Invalid at-reset type %d!\n", pCur->enmType));
2322 return VERR_INTERNAL_ERROR;
2323 }
2324
2325 if (RT_FAILURE(rc))
2326 {
2327 AssertMsgFailed(("At-reset handler %s failed with rc=%d\n", pCur->pszDesc, rc));
2328 return rc;
2329 }
2330 }
2331
2332 return VINF_SUCCESS;
2333}
2334
2335
2336/**
2337 * Internal registration function
2338 */
2339static int vmr3AtResetRegisterU(PUVM pUVM, void *pvUser, const char *pszDesc, PVMATRESET *ppNew)
2340{
2341 /*
2342 * Allocate restration structure.
2343 */
2344 PVMATRESET pNew = (PVMATRESET)MMR3HeapAllocU(pUVM, MM_TAG_VM, sizeof(*pNew));
2345 if (pNew)
2346 {
2347 /* fill data. */
2348 pNew->pszDesc = pszDesc;
2349 pNew->pvUser = pvUser;
2350
2351 /* insert */
2352 pNew->pNext = *pUVM->vm.s.ppAtResetNext;
2353 *pUVM->vm.s.ppAtResetNext = pNew;
2354 pUVM->vm.s.ppAtResetNext = &pNew->pNext;
2355
2356 *ppNew = pNew;
2357 return VINF_SUCCESS;
2358 }
2359 return VERR_NO_MEMORY;
2360}
2361
2362
2363/**
2364 * Registers an at VM reset callback.
2365 *
2366 * @returns VBox status code.
2367 * @param pVM The VM.
2368 * @param pDevInst Device instance.
2369 * @param pfnCallback Callback function.
2370 * @param pvUser User argument.
2371 * @param pszDesc Description (optional).
2372 */
2373VMMR3DECL(int) VMR3AtResetRegister(PVM pVM, PPDMDEVINS pDevInst, PFNVMATRESET pfnCallback, void *pvUser, const char *pszDesc)
2374{
2375 /*
2376 * Validate.
2377 */
2378 if (!pDevInst)
2379 {
2380 AssertMsgFailed(("pDevIns is NULL!\n"));
2381 return VERR_INVALID_PARAMETER;
2382 }
2383
2384 /*
2385 * Create the new entry.
2386 */
2387 PVMATRESET pNew;
2388 int rc = vmr3AtResetRegisterU(pVM->pUVM, pvUser, pszDesc, &pNew);
2389 if (RT_SUCCESS(rc))
2390 {
2391 /*
2392 * Fill in type data.
2393 */
2394 pNew->enmType = VMATRESETTYPE_DEV;
2395 pNew->u.Dev.pfnCallback = pfnCallback;
2396 pNew->u.Dev.pDevIns = pDevInst;
2397 }
2398
2399 return rc;
2400}
2401
2402
2403/**
2404 * Registers an at VM reset internal callback.
2405 *
2406 * @returns VBox status code.
2407 * @param pVM The VM.
2408 * @param pfnCallback Callback function.
2409 * @param pvUser User argument.
2410 * @param pszDesc Description (optional).
2411 */
2412VMMR3DECL(int) VMR3AtResetRegisterInternal(PVM pVM, PFNVMATRESETINT pfnCallback, void *pvUser, const char *pszDesc)
2413{
2414 /*
2415 * Validate.
2416 */
2417 if (!pfnCallback)
2418 {
2419 AssertMsgFailed(("pfnCallback is NULL!\n"));
2420 return VERR_INVALID_PARAMETER;
2421 }
2422
2423 /*
2424 * Create the new entry.
2425 */
2426 PVMATRESET pNew;
2427 int rc = vmr3AtResetRegisterU(pVM->pUVM, pvUser, pszDesc, &pNew);
2428 if (RT_SUCCESS(rc))
2429 {
2430 /*
2431 * Fill in type data.
2432 */
2433 pNew->enmType = VMATRESETTYPE_INTERNAL;
2434 pNew->u.Internal.pfnCallback = pfnCallback;
2435 }
2436
2437 return rc;
2438}
2439
2440
2441/**
2442 * Registers an at VM reset external callback.
2443 *
2444 * @returns VBox status code.
2445 * @param pVM The VM.
2446 * @param pfnCallback Callback function.
2447 * @param pvUser User argument.
2448 * @param pszDesc Description (optional).
2449 */
2450VMMR3DECL(int) VMR3AtResetRegisterExternal(PVM pVM, PFNVMATRESETEXT pfnCallback, void *pvUser, const char *pszDesc)
2451{
2452 /*
2453 * Validate.
2454 */
2455 if (!pfnCallback)
2456 {
2457 AssertMsgFailed(("pfnCallback is NULL!\n"));
2458 return VERR_INVALID_PARAMETER;
2459 }
2460
2461 /*
2462 * Create the new entry.
2463 */
2464 PVMATRESET pNew;
2465 int rc = vmr3AtResetRegisterU(pVM->pUVM, pvUser, pszDesc, &pNew);
2466 if (RT_SUCCESS(rc))
2467 {
2468 /*
2469 * Fill in type data.
2470 */
2471 pNew->enmType = VMATRESETTYPE_EXTERNAL;
2472 pNew->u.External.pfnCallback = pfnCallback;
2473 }
2474
2475 return rc;
2476}
2477
2478
2479/**
2480 * Unlinks and frees a callback.
2481 *
2482 * @returns Pointer to the next callback structure.
2483 * @param pUVM Pointer to the user mode VM structure.
2484 * @param pCur The one to free.
2485 * @param pPrev The one before pCur.
2486 */
2487static PVMATRESET vmr3AtResetFreeU(PUVM pUVM, PVMATRESET pCur, PVMATRESET pPrev)
2488{
2489 /*
2490 * Unlink it.
2491 */
2492 PVMATRESET pNext = pCur->pNext;
2493 if (pPrev)
2494 {
2495 pPrev->pNext = pNext;
2496 if (!pNext)
2497 pUVM->vm.s.ppAtResetNext = &pPrev->pNext;
2498 }
2499 else
2500 {
2501 pUVM->vm.s.pAtReset = pNext;
2502 if (!pNext)
2503 pUVM->vm.s.ppAtResetNext = &pUVM->vm.s.pAtReset;
2504 }
2505
2506 /*
2507 * Free it.
2508 */
2509 MMR3HeapFree(pCur);
2510
2511 return pNext;
2512}
2513
2514
2515/**
2516 * Deregisters an at VM reset callback.
2517 *
2518 * @returns VBox status code.
2519 * @param pVM The VM.
2520 * @param pDevIns Device instance.
2521 * @param pfnCallback Callback function.
2522 */
2523VMMR3DECL(int) VMR3AtResetDeregister(PVM pVM, PPDMDEVINS pDevIns, PFNVMATRESET pfnCallback)
2524{
2525 int rc = VERR_VM_ATRESET_NOT_FOUND;
2526 PVMATRESET pPrev = NULL;
2527 PVMATRESET pCur = pVM->pUVM->vm.s.pAtReset;
2528 while (pCur)
2529 {
2530 if ( pCur->enmType == VMATRESETTYPE_DEV
2531 && pCur->u.Dev.pDevIns == pDevIns
2532 && ( !pfnCallback
2533 || pCur->u.Dev.pfnCallback == pfnCallback))
2534 {
2535 pCur = vmr3AtResetFreeU(pVM->pUVM, pCur, pPrev);
2536 rc = VINF_SUCCESS;
2537 }
2538 else
2539 {
2540 pPrev = pCur;
2541 pCur = pCur->pNext;
2542 }
2543 }
2544
2545 AssertRC(rc);
2546 return rc;
2547}
2548
2549
2550/**
2551 * Deregisters an at VM reset internal callback.
2552 *
2553 * @returns VBox status code.
2554 * @param pVM The VM.
2555 * @param pfnCallback Callback function.
2556 */
2557VMMR3DECL(int) VMR3AtResetDeregisterInternal(PVM pVM, PFNVMATRESETINT pfnCallback)
2558{
2559 int rc = VERR_VM_ATRESET_NOT_FOUND;
2560 PVMATRESET pPrev = NULL;
2561 PVMATRESET pCur = pVM->pUVM->vm.s.pAtReset;
2562 while (pCur)
2563 {
2564 if ( pCur->enmType == VMATRESETTYPE_INTERNAL
2565 && pCur->u.Internal.pfnCallback == pfnCallback)
2566 {
2567 pCur = vmr3AtResetFreeU(pVM->pUVM, pCur, pPrev);
2568 rc = VINF_SUCCESS;
2569 }
2570 else
2571 {
2572 pPrev = pCur;
2573 pCur = pCur->pNext;
2574 }
2575 }
2576
2577 AssertRC(rc);
2578 return rc;
2579}
2580
2581
2582/**
2583 * Deregisters an at VM reset external callback.
2584 *
2585 * @returns VBox status code.
2586 * @param pVM The VM.
2587 * @param pfnCallback Callback function.
2588 */
2589VMMR3DECL(int) VMR3AtResetDeregisterExternal(PVM pVM, PFNVMATRESETEXT pfnCallback)
2590{
2591 int rc = VERR_VM_ATRESET_NOT_FOUND;
2592 PVMATRESET pPrev = NULL;
2593 PVMATRESET pCur = pVM->pUVM->vm.s.pAtReset;
2594 while (pCur)
2595 {
2596 if ( pCur->enmType == VMATRESETTYPE_INTERNAL
2597 && pCur->u.External.pfnCallback == pfnCallback)
2598 {
2599 pCur = vmr3AtResetFreeU(pVM->pUVM, pCur, pPrev);
2600 rc = VINF_SUCCESS;
2601 }
2602 else
2603 {
2604 pPrev = pCur;
2605 pCur = pCur->pNext;
2606 }
2607 }
2608
2609 AssertRC(rc);
2610 return rc;
2611}
2612
2613
2614/**
2615 * Gets the current VM state.
2616 *
2617 * @returns The current VM state.
2618 * @param pVM VM handle.
2619 * @thread Any
2620 */
2621VMMR3DECL(VMSTATE) VMR3GetState(PVM pVM)
2622{
2623 return pVM->enmVMState;
2624}
2625
2626
2627/**
2628 * Gets the state name string for a VM state.
2629 *
2630 * @returns Pointer to the state name. (readonly)
2631 * @param enmState The state.
2632 */
2633VMMR3DECL(const char *) VMR3GetStateName(VMSTATE enmState)
2634{
2635 switch (enmState)
2636 {
2637 case VMSTATE_CREATING: return "CREATING";
2638 case VMSTATE_CREATED: return "CREATED";
2639 case VMSTATE_RUNNING: return "RUNNING";
2640 case VMSTATE_LOADING: return "LOADING";
2641 case VMSTATE_LOAD_FAILURE: return "LOAD_FAILURE";
2642 case VMSTATE_SAVING: return "SAVING";
2643 case VMSTATE_SUSPENDED: return "SUSPENDED";
2644 case VMSTATE_RESETTING: return "RESETTING";
2645 case VMSTATE_GURU_MEDITATION: return "GURU_MEDITATION";
2646 case VMSTATE_OFF: return "OFF";
2647 case VMSTATE_DESTROYING: return "DESTROYING";
2648 case VMSTATE_TERMINATED: return "TERMINATED";
2649 default:
2650 AssertMsgFailed(("Unknown state %d\n", enmState));
2651 return "Unknown!\n";
2652 }
2653}
2654
2655
2656/**
2657 * Sets the current VM state.
2658 *
2659 * @returns The current VM state.
2660 * @param pVM VM handle.
2661 * @param enmStateNew The new state.
2662 */
2663void vmR3SetState(PVM pVM, VMSTATE enmStateNew)
2664{
2665 /*
2666 * Validate state machine transitions before doing the actual change.
2667 */
2668 VMSTATE enmStateOld = pVM->enmVMState;
2669 switch (enmStateOld)
2670 {
2671 case VMSTATE_OFF:
2672 Assert(enmStateNew != VMSTATE_GURU_MEDITATION);
2673 break;
2674
2675 default:
2676 /** @todo full validation. */
2677 break;
2678 }
2679
2680 pVM->enmVMState = enmStateNew;
2681 LogRel(("Changing the VM state from '%s' to '%s'.\n", VMR3GetStateName(enmStateOld), VMR3GetStateName(enmStateNew)));
2682
2683 /*
2684 * Call the at state change callbacks.
2685 */
2686 for (PVMATSTATE pCur = pVM->pUVM->vm.s.pAtState; pCur; pCur = pCur->pNext)
2687 {
2688 pCur->pfnAtState(pVM, enmStateNew, enmStateOld, pCur->pvUser);
2689 if ( pVM->enmVMState != enmStateNew
2690 && pVM->enmVMState == VMSTATE_DESTROYING)
2691 break;
2692 AssertMsg(pVM->enmVMState == enmStateNew,
2693 ("You are not allowed to change the state while in the change callback, except "
2694 "from destroying the VM. There are restrictions in the way the state changes "
2695 "are propagated up to the EM execution loop and it makes the program flow very "
2696 "difficult to follow.\n"));
2697 }
2698}
2699
2700
2701/**
2702 * Registers a VM state change callback.
2703 *
2704 * You are not allowed to call any function which changes the VM state from a
2705 * state callback, except VMR3Destroy().
2706 *
2707 * @returns VBox status code.
2708 * @param pVM VM handle.
2709 * @param pfnAtState Pointer to callback.
2710 * @param pvUser User argument.
2711 * @thread Any.
2712 */
2713VMMR3DECL(int) VMR3AtStateRegister(PVM pVM, PFNVMATSTATE pfnAtState, void *pvUser)
2714{
2715 LogFlow(("VMR3AtStateRegister: pfnAtState=%p pvUser=%p\n", pfnAtState, pvUser));
2716
2717 /*
2718 * Validate input.
2719 */
2720 if (!pfnAtState)
2721 {
2722 AssertMsgFailed(("callback is required\n"));
2723 return VERR_INVALID_PARAMETER;
2724 }
2725
2726 /*
2727 * Make sure we're in EMT (to avoid the logging).
2728 */
2729 PVMREQ pReq;
2730 int rc = VMR3ReqCall(pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3AtStateRegisterU, 3, pVM->pUVM, pfnAtState, pvUser);
2731 if (RT_FAILURE(rc))
2732 return rc;
2733 rc = pReq->iStatus;
2734 VMR3ReqFree(pReq);
2735
2736 LogFlow(("VMR3AtStateRegister: returns %Rrc\n", rc));
2737 return rc;
2738}
2739
2740
2741/**
2742 * Registers a VM state change callback.
2743 *
2744 * @returns VBox status code.
2745 * @param pUVM Pointer to the user mode VM structure.
2746 * @param pfnAtState Pointer to callback.
2747 * @param pvUser User argument.
2748 * @thread EMT
2749 */
2750static DECLCALLBACK(int) vmR3AtStateRegisterU(PUVM pUVM, PFNVMATSTATE pfnAtState, void *pvUser)
2751{
2752 /*
2753 * Allocate a new record.
2754 */
2755
2756 PVMATSTATE pNew = (PVMATSTATE)MMR3HeapAllocU(pUVM, MM_TAG_VM, sizeof(*pNew));
2757 if (!pNew)
2758 return VERR_NO_MEMORY;
2759
2760 /* fill */
2761 pNew->pfnAtState = pfnAtState;
2762 pNew->pvUser = pvUser;
2763
2764 /* insert */
2765 pNew->pNext = *pUVM->vm.s.ppAtStateNext;
2766 *pUVM->vm.s.ppAtStateNext = pNew;
2767 pUVM->vm.s.ppAtStateNext = &pNew->pNext;
2768
2769 return VINF_SUCCESS;
2770}
2771
2772
2773/**
2774 * Deregisters a VM state change callback.
2775 *
2776 * @returns VBox status code.
2777 * @param pVM VM handle.
2778 * @param pfnAtState Pointer to callback.
2779 * @param pvUser User argument.
2780 * @thread Any.
2781 */
2782VMMR3DECL(int) VMR3AtStateDeregister(PVM pVM, PFNVMATSTATE pfnAtState, void *pvUser)
2783{
2784 LogFlow(("VMR3AtStateDeregister: pfnAtState=%p pvUser=%p\n", pfnAtState, pvUser));
2785
2786 /*
2787 * Validate input.
2788 */
2789 if (!pfnAtState)
2790 {
2791 AssertMsgFailed(("callback is required\n"));
2792 return VERR_INVALID_PARAMETER;
2793 }
2794
2795 /*
2796 * Make sure we're in EMT (to avoid the logging).
2797 */
2798 PVMREQ pReq;
2799 int rc = VMR3ReqCall(pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3AtStateDeregisterU, 3, pVM->pUVM, pfnAtState, pvUser);
2800 if (RT_FAILURE(rc))
2801 return rc;
2802 rc = pReq->iStatus;
2803 VMR3ReqFree(pReq);
2804
2805 LogFlow(("VMR3AtStateDeregister: returns %Rrc\n", rc));
2806 return rc;
2807}
2808
2809
2810/**
2811 * Deregisters a VM state change callback.
2812 *
2813 * @returns VBox status code.
2814 * @param pUVM Pointer to the user mode VM structure.
2815 * @param pfnAtState Pointer to callback.
2816 * @param pvUser User argument.
2817 * @thread EMT
2818 */
2819static DECLCALLBACK(int) vmR3AtStateDeregisterU(PUVM pUVM, PFNVMATSTATE pfnAtState, void *pvUser)
2820{
2821 LogFlow(("vmR3AtStateDeregisterU: pfnAtState=%p pvUser=%p\n", pfnAtState, pvUser));
2822
2823 /*
2824 * Search the list for the entry.
2825 */
2826 PVMATSTATE pPrev = NULL;
2827 PVMATSTATE pCur = pUVM->vm.s.pAtState;
2828 while ( pCur
2829 && ( pCur->pfnAtState != pfnAtState
2830 || pCur->pvUser != pvUser))
2831 {
2832 pPrev = pCur;
2833 pCur = pCur->pNext;
2834 }
2835 if (!pCur)
2836 {
2837 AssertMsgFailed(("pfnAtState=%p was not found\n", pfnAtState));
2838 return VERR_FILE_NOT_FOUND;
2839 }
2840
2841 /*
2842 * Unlink it.
2843 */
2844 if (pPrev)
2845 {
2846 pPrev->pNext = pCur->pNext;
2847 if (!pCur->pNext)
2848 pUVM->vm.s.ppAtStateNext = &pPrev->pNext;
2849 }
2850 else
2851 {
2852 pUVM->vm.s.pAtState = pCur->pNext;
2853 if (!pCur->pNext)
2854 pUVM->vm.s.ppAtStateNext = &pUVM->vm.s.pAtState;
2855 }
2856
2857 /*
2858 * Free it.
2859 */
2860 pCur->pfnAtState = NULL;
2861 pCur->pNext = NULL;
2862 MMR3HeapFree(pCur);
2863
2864 return VINF_SUCCESS;
2865}
2866
2867
2868/**
2869 * Registers a VM error callback.
2870 *
2871 * @returns VBox status code.
2872 * @param pVM The VM handle.
2873 * @param pfnAtError Pointer to callback.
2874 * @param pvUser User argument.
2875 * @thread Any.
2876 */
2877VMMR3DECL(int) VMR3AtErrorRegister(PVM pVM, PFNVMATERROR pfnAtError, void *pvUser)
2878{
2879 return VMR3AtErrorRegisterU(pVM->pUVM, pfnAtError, pvUser);
2880}
2881
2882
2883/**
2884 * Registers a VM error callback.
2885 *
2886 * @returns VBox status code.
2887 * @param pUVM The VM handle.
2888 * @param pfnAtError Pointer to callback.
2889 * @param pvUser User argument.
2890 * @thread Any.
2891 */
2892VMMR3DECL(int) VMR3AtErrorRegisterU(PUVM pUVM, PFNVMATERROR pfnAtError, void *pvUser)
2893{
2894 LogFlow(("VMR3AtErrorRegister: pfnAtError=%p pvUser=%p\n", pfnAtError, pvUser));
2895 AssertPtrReturn(pfnAtError, VERR_INVALID_PARAMETER);
2896
2897 /*
2898 * Make sure we're in EMT (to avoid the logging).
2899 */
2900 PVMREQ pReq;
2901 int rc = VMR3ReqCallU(pUVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, 0, (PFNRT)vmR3AtErrorRegisterU, 3, pUVM, pfnAtError, pvUser);
2902 if (RT_FAILURE(rc))
2903 return rc;
2904 rc = pReq->iStatus;
2905 VMR3ReqFree(pReq);
2906
2907 LogFlow(("VMR3AtErrorRegister: returns %Rrc\n", rc));
2908 return rc;
2909}
2910
2911
2912/**
2913 * Registers a VM error callback.
2914 *
2915 * @returns VBox status code.
2916 * @param pUVM Pointer to the user mode VM structure.
2917 * @param pfnAtError Pointer to callback.
2918 * @param pvUser User argument.
2919 * @thread EMT
2920 */
2921static DECLCALLBACK(int) vmR3AtErrorRegisterU(PUVM pUVM, PFNVMATERROR pfnAtError, void *pvUser)
2922{
2923 /*
2924 * Allocate a new record.
2925 */
2926
2927 PVMATERROR pNew = (PVMATERROR)MMR3HeapAllocU(pUVM, MM_TAG_VM, sizeof(*pNew));
2928 if (!pNew)
2929 return VERR_NO_MEMORY;
2930
2931 /* fill */
2932 pNew->pfnAtError = pfnAtError;
2933 pNew->pvUser = pvUser;
2934
2935 /* insert */
2936 pNew->pNext = *pUVM->vm.s.ppAtErrorNext;
2937 *pUVM->vm.s.ppAtErrorNext = pNew;
2938 pUVM->vm.s.ppAtErrorNext = &pNew->pNext;
2939
2940 return VINF_SUCCESS;
2941}
2942
2943
2944/**
2945 * Deregisters a VM error callback.
2946 *
2947 * @returns VBox status code.
2948 * @param pVM The VM handle.
2949 * @param pfnAtError Pointer to callback.
2950 * @param pvUser User argument.
2951 * @thread Any.
2952 */
2953VMMR3DECL(int) VMR3AtErrorDeregister(PVM pVM, PFNVMATERROR pfnAtError, void *pvUser)
2954{
2955 LogFlow(("VMR3AtErrorDeregister: pfnAtError=%p pvUser=%p\n", pfnAtError, pvUser));
2956
2957 /*
2958 * Validate input.
2959 */
2960 if (!pfnAtError)
2961 {
2962 AssertMsgFailed(("callback is required\n"));
2963 return VERR_INVALID_PARAMETER;
2964 }
2965
2966 /*
2967 * Make sure we're in EMT (to avoid the logging).
2968 */
2969 PVMREQ pReq;
2970 int rc = VMR3ReqCall(pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3AtErrorDeregisterU, 3, pVM->pUVM, pfnAtError, pvUser);
2971 if (RT_FAILURE(rc))
2972 return rc;
2973 rc = pReq->iStatus;
2974 VMR3ReqFree(pReq);
2975
2976 LogFlow(("VMR3AtErrorDeregister: returns %Rrc\n", rc));
2977 return rc;
2978}
2979
2980
2981/**
2982 * Deregisters a VM error callback.
2983 *
2984 * @returns VBox status code.
2985 * @param pUVM Pointer to the user mode VM structure.
2986 * @param pfnAtError Pointer to callback.
2987 * @param pvUser User argument.
2988 * @thread EMT
2989 */
2990static DECLCALLBACK(int) vmR3AtErrorDeregisterU(PUVM pUVM, PFNVMATERROR pfnAtError, void *pvUser)
2991{
2992 LogFlow(("vmR3AtErrorDeregisterU: pfnAtError=%p pvUser=%p\n", pfnAtError, pvUser));
2993
2994 /*
2995 * Search the list for the entry.
2996 */
2997 PVMATERROR pPrev = NULL;
2998 PVMATERROR pCur = pUVM->vm.s.pAtError;
2999 while ( pCur
3000 && ( pCur->pfnAtError != pfnAtError
3001 || pCur->pvUser != pvUser))
3002 {
3003 pPrev = pCur;
3004 pCur = pCur->pNext;
3005 }
3006 if (!pCur)
3007 {
3008 AssertMsgFailed(("pfnAtError=%p was not found\n", pfnAtError));
3009 return VERR_FILE_NOT_FOUND;
3010 }
3011
3012 /*
3013 * Unlink it.
3014 */
3015 if (pPrev)
3016 {
3017 pPrev->pNext = pCur->pNext;
3018 if (!pCur->pNext)
3019 pUVM->vm.s.ppAtErrorNext = &pPrev->pNext;
3020 }
3021 else
3022 {
3023 pUVM->vm.s.pAtError = pCur->pNext;
3024 if (!pCur->pNext)
3025 pUVM->vm.s.ppAtErrorNext = &pUVM->vm.s.pAtError;
3026 }
3027
3028 /*
3029 * Free it.
3030 */
3031 pCur->pfnAtError = NULL;
3032 pCur->pNext = NULL;
3033 MMR3HeapFree(pCur);
3034
3035 return VINF_SUCCESS;
3036}
3037
3038
3039/**
3040 * Ellipsis to va_list wrapper for calling pfnAtError.
3041 */
3042static void vmR3SetErrorWorkerDoCall(PVM pVM, PVMATERROR pCur, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
3043{
3044 va_list va;
3045 va_start(va, pszFormat);
3046 pCur->pfnAtError(pVM, pCur->pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va);
3047 va_end(va);
3048}
3049
3050
3051/**
3052 * This is a worker function for GC and Ring-0 calls to VMSetError and VMSetErrorV.
3053 * The message is found in VMINT.
3054 *
3055 * @param pVM The VM handle.
3056 * @thread EMT.
3057 */
3058VMMR3DECL(void) VMR3SetErrorWorker(PVM pVM)
3059{
3060 VM_ASSERT_EMT(pVM);
3061 AssertReleaseMsgFailed(("And we have a winner! You get to implement Ring-0 and GC VMSetErrorV! Contrats!\n"));
3062
3063 /*
3064 * Unpack the error (if we managed to format one).
3065 */
3066 PVMERROR pErr = pVM->vm.s.pErrorR3;
3067 const char *pszFile = NULL;
3068 const char *pszFunction = NULL;
3069 uint32_t iLine = 0;
3070 const char *pszMessage;
3071 int32_t rc = VERR_MM_HYPER_NO_MEMORY;
3072 if (pErr)
3073 {
3074 AssertCompile(sizeof(const char) == sizeof(uint8_t));
3075 if (pErr->offFile)
3076 pszFile = (const char *)pErr + pErr->offFile;
3077 iLine = pErr->iLine;
3078 if (pErr->offFunction)
3079 pszFunction = (const char *)pErr + pErr->offFunction;
3080 if (pErr->offMessage)
3081 pszMessage = (const char *)pErr + pErr->offMessage;
3082 else
3083 pszMessage = "No message!";
3084 }
3085 else
3086 pszMessage = "No message! (Failed to allocate memory to put the error message in!)";
3087
3088 /*
3089 * Call the at error callbacks.
3090 */
3091 for (PVMATERROR pCur = pVM->pUVM->vm.s.pAtError; pCur; pCur = pCur->pNext)
3092 vmR3SetErrorWorkerDoCall(pVM, pCur, rc, RT_SRC_POS_ARGS, "%s", pszMessage);
3093}
3094
3095
3096/**
3097 * Creation time wrapper for vmR3SetErrorUV.
3098 *
3099 * @returns rc.
3100 * @param pUVM Pointer to the user mode VM structure.
3101 * @param rc The VBox status code.
3102 * @param RT_SRC_POS_DECL The source position of this error.
3103 * @param pszFormat Format string.
3104 * @param ... The arguments.
3105 * @thread Any thread.
3106 */
3107static int vmR3SetErrorU(PUVM pUVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
3108{
3109 va_list va;
3110 va_start(va, pszFormat);
3111 vmR3SetErrorUV(pUVM, rc, pszFile, iLine, pszFunction, pszFormat, &va);
3112 va_end(va);
3113 return rc;
3114}
3115
3116
3117/**
3118 * Worker which calls everyone listening to the VM error messages.
3119 *
3120 * @param pUVM Pointer to the user mode VM structure.
3121 * @param rc The VBox status code.
3122 * @param RT_SRC_POS_DECL The source position of this error.
3123 * @param pszFormat Format string.
3124 * @param pArgs Pointer to the format arguments.
3125 * @thread EMT
3126 */
3127DECLCALLBACK(void) vmR3SetErrorUV(PUVM pUVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list *pArgs)
3128{
3129#ifdef LOG_ENABLED
3130 /*
3131 * Log the error.
3132 */
3133 RTLogPrintf("VMSetError: %s(%d) %s\n", pszFile, iLine, pszFunction);
3134 va_list va3;
3135 va_copy(va3, *pArgs);
3136 RTLogPrintfV(pszFormat, va3);
3137 va_end(va3);
3138 RTLogPrintf("\n");
3139#endif
3140
3141 /*
3142 * Make a copy of the message.
3143 */
3144 if (pUVM->pVM)
3145 vmSetErrorCopy(pUVM->pVM, rc, RT_SRC_POS_ARGS, pszFormat, *pArgs);
3146
3147 /*
3148 * Call the at error callbacks.
3149 */
3150 for (PVMATERROR pCur = pUVM->vm.s.pAtError; pCur; pCur = pCur->pNext)
3151 {
3152 va_list va2;
3153 va_copy(va2, *pArgs);
3154 pCur->pfnAtError(pUVM->pVM, pCur->pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va2);
3155 va_end(va2);
3156 }
3157}
3158
3159
3160/**
3161 * Registers a VM runtime error callback.
3162 *
3163 * @returns VBox status code.
3164 * @param pVM The VM handle.
3165 * @param pfnAtRuntimeError Pointer to callback.
3166 * @param pvUser User argument.
3167 * @thread Any.
3168 */
3169VMMR3DECL(int) VMR3AtRuntimeErrorRegister(PVM pVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser)
3170{
3171 LogFlow(("VMR3AtRuntimeErrorRegister: pfnAtRuntimeError=%p pvUser=%p\n", pfnAtRuntimeError, pvUser));
3172
3173 /*
3174 * Validate input.
3175 */
3176 if (!pfnAtRuntimeError)
3177 {
3178 AssertMsgFailed(("callback is required\n"));
3179 return VERR_INVALID_PARAMETER;
3180 }
3181
3182 /*
3183 * Make sure we're in EMT (to avoid the logging).
3184 */
3185 PVMREQ pReq;
3186 int rc = VMR3ReqCall(pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3AtRuntimeErrorRegisterU, 3, pVM->pUVM, pfnAtRuntimeError, pvUser);
3187 if (RT_FAILURE(rc))
3188 return rc;
3189 rc = pReq->iStatus;
3190 VMR3ReqFree(pReq);
3191
3192 LogFlow(("VMR3AtRuntimeErrorRegister: returns %Rrc\n", rc));
3193 return rc;
3194}
3195
3196
3197/**
3198 * Registers a VM runtime error callback.
3199 *
3200 * @returns VBox status code.
3201 * @param pUVM Pointer to the user mode VM structure.
3202 * @param pfnAtRuntimeError Pointer to callback.
3203 * @param pvUser User argument.
3204 * @thread EMT
3205 */
3206static DECLCALLBACK(int) vmR3AtRuntimeErrorRegisterU(PUVM pUVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser)
3207{
3208 /*
3209 * Allocate a new record.
3210 */
3211
3212 PVMATRUNTIMEERROR pNew = (PVMATRUNTIMEERROR)MMR3HeapAllocU(pUVM, MM_TAG_VM, sizeof(*pNew));
3213 if (!pNew)
3214 return VERR_NO_MEMORY;
3215
3216 /* fill */
3217 pNew->pfnAtRuntimeError = pfnAtRuntimeError;
3218 pNew->pvUser = pvUser;
3219
3220 /* insert */
3221 pNew->pNext = *pUVM->vm.s.ppAtRuntimeErrorNext;
3222 *pUVM->vm.s.ppAtRuntimeErrorNext = pNew;
3223 pUVM->vm.s.ppAtRuntimeErrorNext = &pNew->pNext;
3224
3225 return VINF_SUCCESS;
3226}
3227
3228
3229/**
3230 * Deregisters a VM runtime error callback.
3231 *
3232 * @returns VBox status code.
3233 * @param pVM The VM handle.
3234 * @param pfnAtRuntimeError Pointer to callback.
3235 * @param pvUser User argument.
3236 * @thread Any.
3237 */
3238VMMR3DECL(int) VMR3AtRuntimeErrorDeregister(PVM pVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser)
3239{
3240 LogFlow(("VMR3AtRuntimeErrorDeregister: pfnAtRuntimeError=%p pvUser=%p\n", pfnAtRuntimeError, pvUser));
3241
3242 /*
3243 * Validate input.
3244 */
3245 if (!pfnAtRuntimeError)
3246 {
3247 AssertMsgFailed(("callback is required\n"));
3248 return VERR_INVALID_PARAMETER;
3249 }
3250
3251 /*
3252 * Make sure we're in EMT (to avoid the logging).
3253 */
3254 PVMREQ pReq;
3255 int rc = VMR3ReqCall(pVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)vmR3AtRuntimeErrorDeregisterU, 3, pVM->pUVM, pfnAtRuntimeError, pvUser);
3256 if (RT_FAILURE(rc))
3257 return rc;
3258 rc = pReq->iStatus;
3259 VMR3ReqFree(pReq);
3260
3261 LogFlow(("VMR3AtRuntimeErrorDeregister: returns %Rrc\n", rc));
3262 return rc;
3263}
3264
3265
3266/**
3267 * Deregisters a VM runtime error callback.
3268 *
3269 * @returns VBox status code.
3270 * @param pUVM Pointer to the user mode VM structure.
3271 * @param pfnAtRuntimeError Pointer to callback.
3272 * @param pvUser User argument.
3273 * @thread EMT
3274 */
3275static DECLCALLBACK(int) vmR3AtRuntimeErrorDeregisterU(PUVM pUVM, PFNVMATRUNTIMEERROR pfnAtRuntimeError, void *pvUser)
3276{
3277 LogFlow(("vmR3AtRuntimeErrorDeregisterU: pfnAtRuntimeError=%p pvUser=%p\n", pfnAtRuntimeError, pvUser));
3278
3279 /*
3280 * Search the list for the entry.
3281 */
3282 PVMATRUNTIMEERROR pPrev = NULL;
3283 PVMATRUNTIMEERROR pCur = pUVM->vm.s.pAtRuntimeError;
3284 while ( pCur
3285 && ( pCur->pfnAtRuntimeError != pfnAtRuntimeError
3286 || pCur->pvUser != pvUser))
3287 {
3288 pPrev = pCur;
3289 pCur = pCur->pNext;
3290 }
3291 if (!pCur)
3292 {
3293 AssertMsgFailed(("pfnAtRuntimeError=%p was not found\n", pfnAtRuntimeError));
3294 return VERR_FILE_NOT_FOUND;
3295 }
3296
3297 /*
3298 * Unlink it.
3299 */
3300 if (pPrev)
3301 {
3302 pPrev->pNext = pCur->pNext;
3303 if (!pCur->pNext)
3304 pUVM->vm.s.ppAtRuntimeErrorNext = &pPrev->pNext;
3305 }
3306 else
3307 {
3308 pUVM->vm.s.pAtRuntimeError = pCur->pNext;
3309 if (!pCur->pNext)
3310 pUVM->vm.s.ppAtRuntimeErrorNext = &pUVM->vm.s.pAtRuntimeError;
3311 }
3312
3313 /*
3314 * Free it.
3315 */
3316 pCur->pfnAtRuntimeError = NULL;
3317 pCur->pNext = NULL;
3318 MMR3HeapFree(pCur);
3319
3320 return VINF_SUCCESS;
3321}
3322
3323
3324/**
3325 * Worker for VMR3SetRuntimeErrorWorker and vmR3SetRuntimeErrorV.
3326 *
3327 * This does the common parts after the error has been saved / retrieved.
3328 *
3329 * @returns VBox status code with modifications, see VMSetRuntimeErrorV.
3330 *
3331 * @param pVM The VM handle.
3332 * @param fFlags The error flags.
3333 * @param pszErrorId Error ID string.
3334 * @param pszFormat Format string.
3335 * @param pVa Pointer to the format arguments.
3336 */
3337static int vmR3SetRuntimeErrorCommon(PVM pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list *pVa)
3338{
3339 LogRel(("VM: Raising runtime error '%s' (fFlags=%#x)\n", pszErrorId, fFlags));
3340
3341 /*
3342 * Take actions before the call.
3343 */
3344 int rc = VINF_SUCCESS;
3345 if (fFlags & VMSETRTERR_FLAGS_FATAL)
3346 /** @todo Add some special VM state for the FATAL variant that isn't resumable.
3347 * It's too risky for 2.2.0, do after branching. */
3348 rc = VMR3SuspendNoSave(pVM);
3349 else if (fFlags & VMSETRTERR_FLAGS_SUSPEND)
3350 rc = VMR3Suspend(pVM);
3351
3352 /*
3353 * Do the callback round.
3354 */
3355 for (PVMATRUNTIMEERROR pCur = pVM->pUVM->vm.s.pAtRuntimeError; pCur; pCur = pCur->pNext)
3356 {
3357 va_list va;
3358 va_copy(va, *pVa);
3359 pCur->pfnAtRuntimeError(pVM, pCur->pvUser, fFlags, pszErrorId, pszFormat, va);
3360 va_end(va);
3361 }
3362
3363 return rc;
3364}
3365
3366
3367/**
3368 * Ellipsis to va_list wrapper for calling vmR3SetRuntimeErrorCommon.
3369 */
3370static int vmR3SetRuntimeErrorCommonF(PVM pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
3371{
3372 va_list va;
3373 va_start(va, pszFormat);
3374 int rc = vmR3SetRuntimeErrorCommon(pVM, fFlags, pszErrorId, pszFormat, &va);
3375 va_end(va);
3376 return rc;
3377}
3378
3379
3380/**
3381 * This is a worker function for RC and Ring-0 calls to VMSetError and
3382 * VMSetErrorV.
3383 *
3384 * The message is found in VMINT.
3385 *
3386 * @returns VBox status code, see VMSetRuntimeError.
3387 * @param pVM The VM handle.
3388 * @thread EMT.
3389 */
3390VMMR3DECL(int) VMR3SetRuntimeErrorWorker(PVM pVM)
3391{
3392 VM_ASSERT_EMT(pVM);
3393 AssertReleaseMsgFailed(("And we have a winner! You get to implement Ring-0 and GC VMSetRuntimeErrorV! Congrats!\n"));
3394
3395 /*
3396 * Unpack the error (if we managed to format one).
3397 */
3398 const char *pszErrorId = "SetRuntimeError";
3399 const char *pszMessage = "No message!";
3400 uint32_t fFlags = VMSETRTERR_FLAGS_FATAL;
3401 PVMRUNTIMEERROR pErr = pVM->vm.s.pRuntimeErrorR3;
3402 if (pErr)
3403 {
3404 AssertCompile(sizeof(const char) == sizeof(uint8_t));
3405 if (pErr->offErrorId)
3406 pszErrorId = (const char *)pErr + pErr->offErrorId;
3407 if (pErr->offMessage)
3408 pszMessage = (const char *)pErr + pErr->offMessage;
3409 fFlags = pErr->fFlags;
3410 }
3411
3412 /*
3413 * Join cause with vmR3SetRuntimeErrorV.
3414 */
3415 return vmR3SetRuntimeErrorCommonF(pVM, fFlags, pszErrorId, "%s", pszMessage);
3416}
3417
3418
3419/**
3420 * Worker for VMSetRuntimeErrorV for doing the job on EMT in ring-3.
3421 *
3422 * @returns VBox status code with modifications, see VMSetRuntimeErrorV.
3423 *
3424 * @param pVM The VM handle.
3425 * @param fFlags The error flags.
3426 * @param pszErrorId Error ID string.
3427 * @param pszFormat Format string.
3428 * @param pVa Pointer to the format arguments.
3429 *
3430 * @thread EMT
3431 */
3432DECLCALLBACK(int) vmR3SetRuntimeErrorV(PVM pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list *pVa)
3433{
3434 /*
3435 * Make a copy of the message.
3436 */
3437 va_list va2;
3438 va_copy(va2, *pVa);
3439 vmSetRuntimeErrorCopy(pVM, fFlags, pszErrorId, pszFormat, va2);
3440 va_end(va2);
3441
3442 /*
3443 * Join paths with VMR3SetRuntimeErrorWorker.
3444 */
3445 return vmR3SetRuntimeErrorCommon(pVM, fFlags, pszErrorId, pszFormat, pVa);
3446}
3447
3448
3449/**
3450 * Gets the ID virtual of the virtual CPU assoicated with the calling thread.
3451 *
3452 * @returns The CPU ID. NIL_VMCPUID if the thread isn't an EMT.
3453 *
3454 * @param pVM The VM handle.
3455 */
3456VMMR3DECL(RTCPUID) VMR3GetVMCPUId(PVM pVM)
3457{
3458 PUVMCPU pUVCpu = (PUVMCPU)RTTlsGet(pVM->pUVM->vm.s.idxTLS);
3459 return pUVCpu
3460 ? pUVCpu->idCpu
3461 : NIL_VMCPUID;
3462}
3463
3464
3465/**
3466 * Returns the native handle of the current EMT VMCPU thread.
3467 *
3468 * @returns Handle if this is an EMT thread; NIL_RTNATIVETHREAD otherwise
3469 * @param pVM The VM handle.
3470 * @thread EMT
3471 */
3472VMMR3DECL(RTNATIVETHREAD) VMR3GetVMCPUNativeThread(PVM pVM)
3473{
3474 PUVMCPU pUVCpu = (PUVMCPU)RTTlsGet(pVM->pUVM->vm.s.idxTLS);
3475
3476 if (!pUVCpu)
3477 return NIL_RTNATIVETHREAD;
3478
3479 return pUVCpu->vm.s.NativeThreadEMT;
3480}
3481
3482
3483/**
3484 * Returns the native handle of the current EMT VMCPU thread.
3485 *
3486 * @returns Handle if this is an EMT thread; NIL_RTNATIVETHREAD otherwise
3487 * @param pVM The VM handle.
3488 * @thread EMT
3489 */
3490VMMR3DECL(RTNATIVETHREAD) VMR3GetVMCPUNativeThreadU(PUVM pUVM)
3491{
3492 PUVMCPU pUVCpu = (PUVMCPU)RTTlsGet(pUVM->vm.s.idxTLS);
3493
3494 if (!pUVCpu)
3495 return NIL_RTNATIVETHREAD;
3496
3497 return pUVCpu->vm.s.NativeThreadEMT;
3498}
3499
3500
3501/**
3502 * Returns the handle of the current EMT VMCPU thread.
3503 *
3504 * @returns Handle if this is an EMT thread; NIL_RTNATIVETHREAD otherwise
3505 * @param pVM The VM handle.
3506 * @thread EMT
3507 */
3508VMMR3DECL(RTTHREAD) VMR3GetVMCPUThread(PVM pVM)
3509{
3510 PUVMCPU pUVCpu = (PUVMCPU)RTTlsGet(pVM->pUVM->vm.s.idxTLS);
3511
3512 if (!pUVCpu)
3513 return NIL_RTTHREAD;
3514
3515 return pUVCpu->vm.s.ThreadEMT;
3516}
3517
3518
3519/**
3520 * Returns the handle of the current EMT VMCPU thread.
3521 *
3522 * @returns Handle if this is an EMT thread; NIL_RTNATIVETHREAD otherwise
3523 * @param pVM The VM handle.
3524 * @thread EMT
3525 */
3526VMMR3DECL(RTTHREAD) VMR3GetVMCPUThreadU(PUVM pUVM)
3527{
3528 PUVMCPU pUVCpu = (PUVMCPU)RTTlsGet(pUVM->vm.s.idxTLS);
3529
3530 if (!pUVCpu)
3531 return NIL_RTTHREAD;
3532
3533 return pUVCpu->vm.s.ThreadEMT;
3534}
3535
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