summaryrefslogtreecommitdiffstats
path: root/src/VBox/VMM/VMMAll/VMAll.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/VBox/VMM/VMMAll/VMAll.cpp')
-rw-r--r--src/VBox/VMM/VMMAll/VMAll.cpp444
1 files changed, 444 insertions, 0 deletions
diff --git a/src/VBox/VMM/VMMAll/VMAll.cpp b/src/VBox/VMM/VMMAll/VMAll.cpp
new file mode 100644
index 00000000..67909b83
--- /dev/null
+++ b/src/VBox/VMM/VMMAll/VMAll.cpp
@@ -0,0 +1,444 @@
+/* $Id: VMAll.cpp $ */
+/** @file
+ * VM - Virtual Machine All Contexts.
+ */
+
+/*
+ * Copyright (C) 2006-2023 Oracle and/or its affiliates.
+ *
+ * This file is part of VirtualBox base platform packages, as
+ * available from https://www.virtualbox.org.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation, in version 3 of the
+ * License.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, see <https://www.gnu.org/licenses>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-only
+ */
+
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#define LOG_GROUP LOG_GROUP_VM
+#include "VMInternal.h"
+#include <VBox/vmm/vmm.h>
+#include <VBox/vmm/mm.h>
+#include <VBox/vmm/vmcc.h>
+#include <VBox/err.h>
+#include <VBox/log.h>
+
+#include <iprt/assert.h>
+#include <iprt/string.h>
+#include <iprt/thread.h>
+
+
+#ifdef IN_RING3
+
+/**
+ * Sets the error message.
+ *
+ * @returns rc. Meaning you can do:
+ * @code
+ * return VM_SET_ERROR(pVM, VERR_OF_YOUR_CHOICE, "descriptive message");
+ * @endcode
+ * @param pVM The cross context VM structure.
+ * @param rc VBox status code.
+ * @param SRC_POS Use RT_SRC_POS.
+ * @param pszFormat Error message format string.
+ * @param ... Error message arguments.
+ * @thread Any
+ */
+VMMDECL(int) VMSetError(PVMCC pVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
+{
+ va_list args;
+ va_start(args, pszFormat);
+ int rc2 = VMSetErrorV(pVM, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc == rc2); NOREF(rc2);
+ va_end(args);
+ return rc;
+}
+
+
+/**
+ * Sets the error message.
+ *
+ * @returns rc. Meaning you can do:
+ * @code
+ * return VM_SET_ERROR(pVM, VERR_OF_YOUR_CHOICE, "descriptive message");
+ * @endcode
+ * @param pVM The cross context VM structure.
+ * @param rc VBox status code.
+ * @param SRC_POS Use RT_SRC_POS.
+ * @param pszFormat Error message format string.
+ * @param args Error message arguments.
+ * @thread Any
+ */
+VMMDECL(int) VMSetErrorV(PVMCC pVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list args)
+{
+# ifdef IN_RING3
+ /*
+ * Switch to EMT.
+ */
+ va_list va2;
+ va_copy(va2, args); /* Have to make a copy here or GCC will break. */
+ VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)vmR3SetErrorUV, 7, /* ASSUMES 3 source pos args! */
+ pVM->pUVM, rc, RT_SRC_POS_ARGS, pszFormat, &va2);
+ va_end(va2);
+
+# else
+ /*
+ * We're already on the EMT thread and can safely create a VMERROR chunk.
+ */
+ vmSetErrorCopy(pVM, rc, RT_SRC_POS_ARGS, pszFormat, args);
+ VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_VM_SET_ERROR, 0);
+# endif
+ return rc;
+}
+
+
+/**
+ * Copies the error to a VMERROR structure.
+ *
+ * This is mainly intended for Ring-0 and GC where the error must be copied to
+ * memory accessible from ring-3. But it's just possible that we might add
+ * APIs for retrieving the VMERROR copy later.
+ *
+ * @param pVM The cross context VM structure.
+ * @param rc VBox status code.
+ * @param SRC_POS Use RT_SRC_POS.
+ * @param pszFormat Error message format string.
+ * @param args Error message arguments.
+ * @thread EMT
+ */
+void vmSetErrorCopy(PVM pVM, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list args)
+{
+ NOREF(pVM); NOREF(rc); RT_SRC_POS_NOREF(); NOREF(pszFormat); NOREF(args);
+# if 0 /// @todo implement Ring-0 and GC VMSetError
+ /*
+ * Create the untranslated message copy.
+ */
+ /* free any old message. */
+ MMHyperFree(pVM, MMHyperR32Ctx(pVM, pVM->vm.s.pError));
+ pVM->vm.s.pError = NULL;
+
+ /* calc reasonable start size. */
+ size_t cchFile = pszFile ? strlen(pszFile) : 0;
+ size_t cchFunction = pszFunction ? strlen(pszFunction) : 0;
+ size_t cchFormat = strlen(pszFormat);
+ size_t cb = sizeof(VMERROR)
+ + cchFile + 1
+ + cchFunction + 1
+ + cchFormat + 32;
+
+ /* allocate it */
+ void *pv;
+ int rc2 = MMHyperAlloc(pVM, cb, 0, MM_TAG_VM, &pv);
+ if (RT_SUCCESS(rc2))
+ {
+ /* initialize it. */
+ PVMERROR pErr = (PVMERROR)pv;
+ pErr->cbAllocated = cb;
+ pErr->iLine = iLine;
+ pErr->off = sizeof(VMERROR);
+ pErr->offFile = pErr->offFunction = 0;
+
+ if (cchFile)
+ {
+ pErr->offFile = pErr->off;
+ memcpy((uint8_t *)pErr + pErr->off, pszFile, cchFile + 1);
+ pErr->off += cchFile + 1;
+ }
+
+ if (cchFunction)
+ {
+ pErr->offFunction = pErr->off;
+ memcpy((uint8_t *)pErr + pErr->off, pszFunction, cchFunction + 1);
+ pErr->off += cchFunction + 1;
+ }
+
+ pErr->offMessage = pErr->off;
+
+ /* format the message (pErr might be reallocated) */
+ VMSETERRORFMTARGS Args;
+ Args.pVM = pVM;
+ Args.pErr = pErr;
+
+ va_list va2;
+ va_copy(va2, args);
+ RTStrFormatV(vmSetErrorFmtOut, &pErr, NULL, NULL, &pszFormatTmp, args);
+ va_end(va2);
+
+ /* done. */
+ pVM->vm.s.pErrorR3 = MMHyper2HC(pVM, (uintptr_t)pArgs.pErr);
+ }
+# endif
+}
+
+#endif /* IN_RING3 */
+#ifdef IN_RING3
+
+/**
+ * Sets the runtime error message.
+ *
+ * As opposed VMSetError(), this method is intended to inform the VM user about
+ * errors and error-like conditions that happen at an arbitrary point during VM
+ * execution (like "host memory low" or "out of host disk space").
+ *
+ * @returns VBox status code. For some flags the status code <b>must</b> be
+ * propagated up the stack.
+ *
+ * @param pVM The cross context VM structure.
+ *
+ * @param fFlags Flags indicating which actions to take.
+ * See VMSETRTERR_FLAGS_* for details on each flag.
+ *
+ * @param pszErrorId Unique error identification string. This is used by
+ * the frontends and maybe other devices or drivers, so
+ * once an ID has been selected it's essentially
+ * unchangable. Employ camelcase when constructing the
+ * string, leave out spaces.
+ *
+ * The registered runtime error callbacks should string
+ * switch on this and handle the ones it knows
+ * specifically and the unknown ones generically.
+ *
+ * @param pszFormat Error message format string.
+ * @param ... Error message arguments.
+ *
+ * @thread Any
+ */
+VMMDECL(int) VMSetRuntimeError(PVMCC pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
+{
+ va_list va;
+ va_start(va, pszFormat);
+ int rc = VMSetRuntimeErrorV(pVM, fFlags, pszErrorId, pszFormat, va);
+ va_end(va);
+ return rc;
+}
+
+
+/**
+ * va_list version of VMSetRuntimeError.
+ *
+ * @returns VBox status code. For some flags the status code <b>must</b> be
+ * propagated up the stack.
+ *
+ * @param pVM The cross context VM structure.
+ * @param fFlags Flags indicating which actions to take. See
+ * VMSETRTERR_FLAGS_*.
+ * @param pszErrorId Error ID string.
+ * @param pszFormat Error message format string.
+ * @param va Error message arguments.
+ *
+ * @thread Any
+ */
+VMMDECL(int) VMSetRuntimeErrorV(PVMCC pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
+{
+ Log(("VMSetRuntimeErrorV: fFlags=%#x pszErrorId=%s\n", fFlags, pszErrorId));
+
+ /*
+ * Relaxed parameter validation.
+ */
+ AssertPtr(pVM);
+ AssertMsg(!(fFlags & ~(VMSETRTERR_FLAGS_NO_WAIT | VMSETRTERR_FLAGS_SUSPEND | VMSETRTERR_FLAGS_FATAL)), ("%#x\n", fFlags));
+ Assert(!(fFlags & VMSETRTERR_FLAGS_NO_WAIT) || !VM_IS_EMT(pVM));
+ Assert(!(fFlags & VMSETRTERR_FLAGS_SUSPEND) || !(fFlags & VMSETRTERR_FLAGS_FATAL));
+ AssertPtr(pszErrorId);
+ Assert(*pszErrorId);
+ Assert(RTStrEnd(pszErrorId, 128) != NULL);
+ AssertPtr(pszFormat);
+ Assert(RTStrEnd(pszFormat, 512) != NULL);
+
+# ifdef IN_RING3
+ /*
+ * Switch to EMT.
+ *
+ * If it's a no-wait request, we have to format the message into a buffer
+ * here since the variable arguments list will become invalid once we call
+ * va_end and return.
+ */
+ int rc;
+ if ( !(fFlags & VMSETRTERR_FLAGS_NO_WAIT)
+ || VM_IS_EMT(pVM))
+ {
+ fFlags &= ~VMSETRTERR_FLAGS_NO_WAIT;
+
+ va_list va2;
+ va_copy(va2, va); /* Have to make a copy here or GCC will break. */
+ rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY,
+ (PFNRT)vmR3SetRuntimeErrorV, 5, pVM, fFlags, pszErrorId, pszFormat, &va2);
+ va_end(va2);
+ }
+ else
+ {
+ char *pszMessage = MMR3HeapAPrintfV(pVM, MM_TAG_VM, pszFormat, va);
+ rc = VMR3ReqCallNoWait(pVM, VMCPUID_ANY,
+ (PFNRT)vmR3SetRuntimeError, 4, pVM, fFlags, pszErrorId, pszMessage);
+ if (RT_FAILURE(rc))
+ MMR3HeapFree(pszMessage);
+ }
+
+# else
+ /*
+ * We're already on the EMT and can safely create a VMRUNTIMEERROR chunk.
+ */
+ AssertReleaseMsgFailed(("Congratulations! You will have the pleasure of debugging the RC/R0 path.\n"));
+ vmSetRuntimeErrorCopy(pVM, fFlags, pszErrorId, pszFormat, va);
+
+ int rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_VM_SET_RUNTIME_ERROR, 0);
+# endif
+
+ Log(("VMSetRuntimeErrorV: returns %Rrc (pszErrorId=%s)\n", rc, pszErrorId));
+ return rc;
+}
+
+
+/**
+ * Copies the error to a VMRUNTIMEERROR structure.
+ *
+ * This is mainly intended for Ring-0 and RC where the error must be copied to
+ * memory accessible from ring-3. But it's just possible that we might add
+ * APIs for retrieving the VMRUNTIMEERROR copy later.
+ *
+ * @param pVM The cross context VM structure.
+ * @param fFlags The error flags.
+ * @param pszErrorId Error ID string.
+ * @param pszFormat Error message format string.
+ * @param va Error message arguments. This is of course spoiled
+ * by this call.
+ * @thread EMT
+ */
+void vmSetRuntimeErrorCopy(PVM pVM, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
+{
+ NOREF(pVM); NOREF(fFlags); NOREF(pszErrorId); NOREF(pszFormat); NOREF(va);
+# if 0 /// @todo implement Ring-0 and GC VMSetError
+ /*
+ * Create the untranslated message copy.
+ */
+ /* free any old message. */
+ MMHyperFree(pVM, MMHyperR32Ctx(pVM, pVM->vm.s.pRuntimeErrorR3));
+ pVM->vm.s.pRuntimeErrorR3 = NULL;
+
+ /* calc reasonable start size. */
+ size_t cchErrorID = pszErrorId ? strlen(pszErrorId) : 0;
+ size_t cchFormat = strlen(pszFormat);
+ size_t cb = sizeof(VMRUNTIMEERROR)
+ + cchErrorID + 1
+ + cchFormat + 32;
+
+ /* allocate it */
+ void *pv;
+ int rc2 = MMHyperAlloc(pVM, cb, 0, MM_TAG_VM, &pv);
+ if (RT_SUCCESS(rc2))
+ {
+ /* initialize it. */
+ PVMRUNTIMEERROR pErr = (PVMRUNTIMEERROR)pv;
+ pErr->cbAllocated = cb;
+ pErr->fFlags = fFlags;
+ pErr->off = sizeof(PVMRUNTIMEERROR);
+ pErr->offErrorID = 0;
+
+ if (cchErrorID)
+ {
+ pErr->offErrorID = pErr->off;
+ memcpy((uint8_t *)pErr + pErr->off, pszErrorId, cchErrorID + 1);
+ pErr->off += cchErrorID + 1;
+ }
+
+ pErr->offMessage = pErr->off;
+
+ /* format the message (pErr might be reallocated) */
+ VMSETRUNTIMEERRORFMTARGS Args;
+ Args.pVM = pVM;
+ Args.pErr = pErr;
+
+ va_list va2;
+ va_copy(va2, args);
+ RTStrFormatV(vmSetRuntimeErrorFmtOut, &pErr, NULL, NULL, &pszFormatTmp, args);
+ va_end(va2);
+
+ /* done. */
+ pVM->vm.s.pErrorRuntimeR3 = MMHyper2HC(pVM, (uintptr_t)pArgs.pErr);
+ }
+# endif
+}
+
+#endif /* IN_RING3 */
+
+/**
+ * Gets the name of VM state.
+ *
+ * @returns Pointer to a read-only string with the state name.
+ * @param enmState The state.
+ */
+VMMDECL(const char *) VMGetStateName(VMSTATE enmState)
+{
+ switch (enmState)
+ {
+#define MY_CASE(enm) case VMSTATE_##enm: return #enm;
+ MY_CASE(CREATING);
+ MY_CASE(CREATED);
+ MY_CASE(RUNNING);
+ MY_CASE(LOADING);
+ MY_CASE(LOAD_FAILURE);
+ MY_CASE(SAVING);
+ MY_CASE(SUSPENDED);
+ MY_CASE(RESETTING);
+ MY_CASE(GURU_MEDITATION);
+ MY_CASE(OFF);
+ MY_CASE(DESTROYING);
+ MY_CASE(TERMINATED);
+#undef MY_CASE
+ default:
+ return "Unknown";
+ }
+}
+
+
+/**
+ * Gets the total reset count.
+ *
+ * @returns Reset count. UINT32_MAX if @a pVM is invalid.
+ * @param pVM The VM handle.
+ */
+VMMDECL(uint32_t) VMGetResetCount(PVMCC pVM)
+{
+ VM_ASSERT_VALID_EXT_RETURN(pVM, UINT32_MAX);
+ return pVM->vm.s.cResets;
+}
+
+
+/**
+ * Gets the soft reset count.
+ *
+ * @returns Soft reset count. UINT32_MAX if @a pVM is invalid.
+ * @param pVM The VM handle.
+ */
+VMMDECL(uint32_t) VMGetSoftResetCount(PVMCC pVM)
+{
+ VM_ASSERT_VALID_EXT_RETURN(pVM, UINT32_MAX);
+ return pVM->vm.s.cSoftResets;
+}
+
+
+/**
+ * Gets the hard reset count.
+ *
+ * @returns Hard reset count. UINT32_MAX if @a pVM is invalid.
+ * @param pVM The VM handle.
+ */
+VMMDECL(uint32_t) VMGetHardResetCount(PVMCC pVM)
+{
+ VM_ASSERT_VALID_EXT_RETURN(pVM, UINT32_MAX);
+ return pVM->vm.s.cHardResets;
+}
+