summaryrefslogtreecommitdiffstats
path: root/src/VBox/VMM/VMMR3/DBGFR3Trace.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/VBox/VMM/VMMR3/DBGFR3Trace.cpp')
-rw-r--r--src/VBox/VMM/VMMR3/DBGFR3Trace.cpp458
1 files changed, 458 insertions, 0 deletions
diff --git a/src/VBox/VMM/VMMR3/DBGFR3Trace.cpp b/src/VBox/VMM/VMMR3/DBGFR3Trace.cpp
new file mode 100644
index 00000000..ad6f8c89
--- /dev/null
+++ b/src/VBox/VMM/VMMR3/DBGFR3Trace.cpp
@@ -0,0 +1,458 @@
+/* $Id: DBGFR3Trace.cpp $ */
+/** @file
+ * DBGF - Debugger Facility, Tracing.
+ */
+
+/*
+ * Copyright (C) 2011-2022 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_DBGF
+#include <VBox/vmm/dbgftrace.h>
+#include <VBox/vmm/cfgm.h>
+#include <VBox/vmm/mm.h>
+#include <VBox/vmm/pdmapi.h>
+#include "DBGFInternal.h"
+#include <VBox/vmm/vm.h>
+#include "VMMTracing.h"
+
+#include <VBox/err.h>
+#include <VBox/log.h>
+#include <VBox/param.h>
+
+#include <iprt/assert.h>
+#include <iprt/ctype.h>
+#include <iprt/trace.h>
+
+
+/*********************************************************************************************************************************
+* Internal Functions *
+*********************************************************************************************************************************/
+static DECLCALLBACK(void) dbgfR3TraceInfo(PVM pVM, PCDBGFINFOHLP pHlp, const char *pszArgs);
+
+
+/*********************************************************************************************************************************
+* Global Variables *
+*********************************************************************************************************************************/
+/**
+ * VMM trace point group translation table.
+ */
+static const struct
+{
+ /** The group name. */
+ const char *pszName;
+ /** The name length. */
+ uint32_t cchName;
+ /** The mask. */
+ uint32_t fMask;
+} g_aVmmTpGroups[] =
+{
+ { RT_STR_TUPLE("em"), VMMTPGROUP_EM },
+ { RT_STR_TUPLE("hm"), VMMTPGROUP_HM },
+ { RT_STR_TUPLE("tm"), VMMTPGROUP_TM },
+};
+
+
+/**
+ * Initializes the tracing.
+ *
+ * @returns VBox status code
+ * @param pVM The cross context VM structure.
+ * @param cbEntry The trace entry size.
+ * @param cEntries The number of entries.
+ */
+static int dbgfR3TraceEnable(PVM pVM, uint32_t cbEntry, uint32_t cEntries)
+{
+ /*
+ * Don't enable it twice.
+ */
+ if (pVM->hTraceBufR3 != NIL_RTTRACEBUF)
+ return VERR_ALREADY_EXISTS;
+
+ /*
+ * Resolve default parameter values.
+ */
+ int rc;
+ if (!cbEntry)
+ {
+ rc = CFGMR3QueryU32Def(CFGMR3GetChild(CFGMR3GetRoot(pVM), "DBGF"), "TraceBufEntrySize", &cbEntry, 128);
+ AssertRCReturn(rc, rc);
+ }
+ if (!cEntries)
+ {
+ rc = CFGMR3QueryU32Def(CFGMR3GetChild(CFGMR3GetRoot(pVM), "DBGF"), "TraceBufEntries", &cEntries, 4096);
+ AssertRCReturn(rc, rc);
+ }
+
+ /*
+ * Figure the required size.
+ */
+ RTTRACEBUF hTraceBuf;
+ size_t cbBlock = 0;
+ rc = RTTraceBufCarve(&hTraceBuf, cEntries, cbEntry, 0 /*fFlags*/, NULL, &cbBlock);
+ if (rc != VERR_BUFFER_OVERFLOW)
+ {
+ AssertReturn(!RT_SUCCESS_NP(rc), VERR_IPE_UNEXPECTED_INFO_STATUS);
+ return rc;
+ }
+
+ /*
+ * Allocate a hyper heap block and carve a trace buffer out of it.
+ *
+ * Note! We ASSUME that the returned trace buffer handle has the same value
+ * as the heap block.
+ */
+ cbBlock = RT_ALIGN_Z(cbBlock, HOST_PAGE_SIZE);
+ RTR0PTR pvBlockR0 = NIL_RTR0PTR;
+ void *pvBlockR3 = NULL;
+ rc = SUPR3PageAllocEx(cbBlock >> HOST_PAGE_SHIFT, 0, &pvBlockR3, &pvBlockR0, NULL);
+ if (RT_FAILURE(rc))
+ return rc;
+
+ rc = RTTraceBufCarve(&hTraceBuf, cEntries, cbEntry, 0 /*fFlags*/, pvBlockR3, &cbBlock);
+ AssertRCReturn(rc, rc);
+ AssertReleaseReturn(hTraceBuf == (RTTRACEBUF)pvBlockR3, VERR_INTERNAL_ERROR_3);
+ AssertReleaseReturn((void *)hTraceBuf == pvBlockR3, VERR_INTERNAL_ERROR_3);
+
+ pVM->hTraceBufR3 = hTraceBuf;
+ pVM->hTraceBufR0 = pvBlockR0;
+ return VINF_SUCCESS;
+}
+
+
+/**
+ * Initializes the tracing.
+ *
+ * @returns VBox status code
+ * @param pVM The cross context VM structure.
+ */
+int dbgfR3TraceInit(PVM pVM)
+{
+ /*
+ * Initialize the trace buffer handles.
+ */
+ Assert(NIL_RTTRACEBUF == (RTTRACEBUF)NULL);
+ pVM->hTraceBufR3 = NIL_RTTRACEBUF;
+ pVM->hTraceBufR0 = NIL_RTR0PTR;
+
+ /*
+ * Check the config and enable tracing if requested.
+ */
+ PCFGMNODE pDbgfNode = CFGMR3GetChild(CFGMR3GetRoot(pVM), "DBGF");
+#if defined(DEBUG) || defined(RTTRACE_ENABLED)
+ bool const fDefault = false;
+ const char * const pszConfigDefault = "";
+#else
+ bool const fDefault = false;
+ const char * const pszConfigDefault = "";
+#endif
+ bool fTracingEnabled;
+ int rc = CFGMR3QueryBoolDef(pDbgfNode, "TracingEnabled", &fTracingEnabled, fDefault);
+ AssertRCReturn(rc, rc);
+ if (fTracingEnabled)
+ {
+ rc = dbgfR3TraceEnable(pVM, 0, 0);
+ if (RT_SUCCESS(rc))
+ {
+ if (pDbgfNode)
+ {
+ char *pszTracingConfig;
+ rc = CFGMR3QueryStringAllocDef(pDbgfNode, "TracingConfig", &pszTracingConfig, pszConfigDefault);
+ if (RT_SUCCESS(rc))
+ {
+ rc = DBGFR3TraceConfig(pVM, pszTracingConfig);
+ if (RT_FAILURE(rc))
+ rc = VMSetError(pVM, rc, RT_SRC_POS, "TracingConfig=\"%s\" -> %Rrc", pszTracingConfig, rc);
+ MMR3HeapFree(pszTracingConfig);
+ }
+ }
+ else
+ {
+ rc = DBGFR3TraceConfig(pVM, pszConfigDefault);
+ if (RT_FAILURE(rc))
+ rc = VMSetError(pVM, rc, RT_SRC_POS, "TracingConfig=\"%s\" (default) -> %Rrc", pszConfigDefault, rc);
+ }
+ }
+ }
+
+ /*
+ * Register a debug info item that will dump the trace buffer content.
+ */
+ if (RT_SUCCESS(rc))
+ rc = DBGFR3InfoRegisterInternal(pVM, "tracebuf", "Display the trace buffer content. No arguments.", dbgfR3TraceInfo);
+
+ return rc;
+}
+
+
+/**
+ * Terminates the tracing.
+ *
+ * @param pVM The cross context VM structure.
+ */
+void dbgfR3TraceTerm(PVM pVM)
+{
+ /* nothing to do */
+ NOREF(pVM);
+}
+
+
+/**
+ * Relocates the trace buffer handle in RC.
+ *
+ * @param pVM The cross context VM structure.
+ */
+void dbgfR3TraceRelocate(PVM pVM)
+{
+ RT_NOREF(pVM);
+}
+
+
+/**
+ * Change the traceing configuration of the VM.
+ *
+ * @returns VBox status code.
+ * @retval VINF_SUCCESS
+ * @retval VERR_NOT_FOUND if any of the trace point groups mentioned in the
+ * config string cannot be found. (Or if the string cannot be made
+ * sense of.) No change made.
+ * @retval VERR_INVALID_VM_HANDLE
+ * @retval VERR_INVALID_POINTER
+ *
+ * @param pVM The cross context VM structure.
+ * @param pszConfig The configuration change specification.
+ *
+ * Trace point group names, optionally prefixed by a '-' to
+ * indicate that the group is being disabled. A special
+ * group 'all' can be used to enable or disable all trace
+ * points.
+ *
+ * Drivers, devices and USB devices each have their own
+ * trace point group which can be accessed by prefixing
+ * their official PDM name by 'drv', 'dev' or 'usb'
+ * respectively.
+ */
+VMMDECL(int) DBGFR3TraceConfig(PVM pVM, const char *pszConfig)
+{
+ VM_ASSERT_VALID_EXT_RETURN(pVM, VERR_INVALID_VM_HANDLE);
+ AssertPtrReturn(pszConfig, VERR_INVALID_POINTER);
+ if (pVM->hTraceBufR3 == NIL_RTTRACEBUF)
+ return VERR_DBGF_NO_TRACE_BUFFER;
+
+ /*
+ * We do this in two passes, the first pass just validates the input string
+ * and the second applies the changes.
+ */
+ for (uint32_t uPass = 0; uPass < 1; uPass++)
+ {
+ char ch;
+ while ((ch = *pszConfig) != '\0')
+ {
+ if (RT_C_IS_SPACE(ch))
+ continue;
+
+ /*
+ * Operation prefix.
+ */
+ bool fNo = false;
+ do
+ {
+ if (ch == 'n' && pszConfig[1] == 'o')
+ {
+ fNo = !fNo;
+ pszConfig++;
+ }
+ else if (ch == '+')
+ fNo = false;
+ else if (ch == '-' || ch == '!' || ch == '~')
+ fNo = !fNo;
+ else
+ break;
+ } while ((ch = *++pszConfig) != '\0');
+ if (ch == '\0')
+ break;
+
+ /*
+ * Extract the name.
+ */
+ const char *pszName = pszConfig;
+ while ( ch != '\0'
+ && !RT_C_IS_SPACE(ch)
+ && !RT_C_IS_PUNCT(ch))
+ ch = *++pszConfig;
+ size_t const cchName = pszConfig - pszName;
+
+ /*
+ * 'all' - special group that enables or disables all trace points.
+ */
+ if (cchName == 3 && !strncmp(pszName, "all", 3))
+ {
+ if (uPass != 0)
+ {
+ uint32_t iCpu = pVM->cCpus;
+ if (!fNo)
+ while (iCpu-- > 0)
+ pVM->apCpusR3[iCpu]->fTraceGroups = UINT32_MAX;
+ else
+ while (iCpu-- > 0)
+ pVM->apCpusR3[iCpu]->fTraceGroups = 0;
+ PDMR3TracingConfig(pVM, NULL, 0, !fNo, uPass > 0);
+ }
+ }
+ else
+ {
+ /*
+ * A specific group, try the VMM first then PDM.
+ */
+ uint32_t i = RT_ELEMENTS(g_aVmmTpGroups);
+ while (i-- > 0)
+ if ( g_aVmmTpGroups[i].cchName == cchName
+ && !strncmp(g_aVmmTpGroups[i].pszName, pszName, cchName))
+ {
+ if (uPass != 0)
+ {
+ uint32_t iCpu = pVM->cCpus;
+ if (!fNo)
+ while (iCpu-- > 0)
+ pVM->apCpusR3[iCpu]->fTraceGroups |= g_aVmmTpGroups[i].fMask;
+ else
+ while (iCpu-- > 0)
+ pVM->apCpusR3[iCpu]->fTraceGroups &= ~g_aVmmTpGroups[i].fMask;
+ }
+ break;
+ }
+
+ if (i == UINT32_MAX)
+ {
+ int rc = PDMR3TracingConfig(pVM, pszName, cchName, !fNo, uPass > 0);
+ if (RT_FAILURE(rc))
+ return rc;
+ }
+ }
+ }
+ }
+
+ return VINF_SUCCESS;
+}
+
+
+/**
+ * Query the trace configuration specification string.
+ *
+ * @returns VBox status code.
+ * @retval VINF_SUCCESS
+ * @retval VERR_INVALID_VM_HANDLE
+ * @retval VERR_INVALID_POINTER
+ * @retval VERR_BUFFER_OVERFLOW if the buffer is too small. Buffer will be
+ * empty.
+
+ * @param pVM The cross context VM structure.
+ * @param pszConfig Pointer to the output buffer.
+ * @param cbConfig The size of the output buffer.
+ */
+VMMDECL(int) DBGFR3TraceQueryConfig(PVM pVM, char *pszConfig, size_t cbConfig)
+{
+ VM_ASSERT_VALID_EXT_RETURN(pVM, VERR_INVALID_VM_HANDLE);
+ AssertPtrReturn(pszConfig, VERR_INVALID_POINTER);
+ if (cbConfig < 1)
+ return VERR_BUFFER_OVERFLOW;
+ *pszConfig = '\0';
+
+ if (pVM->hTraceBufR3 == NIL_RTTRACEBUF)
+ return VERR_DBGF_NO_TRACE_BUFFER;
+
+ int rc = VINF_SUCCESS;
+ uint32_t const fTraceGroups = pVM->apCpusR3[0]->fTraceGroups;
+ if ( fTraceGroups == UINT32_MAX
+ && PDMR3TracingAreAll(pVM, true /*fEnabled*/))
+ rc = RTStrCopy(pszConfig, cbConfig, "all");
+ else if ( fTraceGroups == 0
+ && PDMR3TracingAreAll(pVM, false /*fEnabled*/))
+ rc = RTStrCopy(pszConfig, cbConfig, "-all");
+ else
+ {
+ char *pszDst = pszConfig;
+ size_t cbDst = cbConfig;
+ uint32_t i = RT_ELEMENTS(g_aVmmTpGroups);
+ while (i-- > 0)
+ if (g_aVmmTpGroups[i].fMask & fTraceGroups)
+ {
+ size_t cchThis = g_aVmmTpGroups[i].cchName + (pszDst != pszConfig);
+ if (cchThis >= cbDst)
+ {
+ rc = VERR_BUFFER_OVERFLOW;
+ break;
+ }
+ if (pszDst != pszConfig)
+ {
+ *pszDst = ' ';
+ memcpy(pszDst + 1, g_aVmmTpGroups[i].pszName, g_aVmmTpGroups[i].cchName + 1);
+ }
+ else
+ memcpy(pszDst, g_aVmmTpGroups[i].pszName, g_aVmmTpGroups[i].cchName + 1);
+ pszDst += cchThis;
+ cbDst -= cchThis;
+ }
+
+ if (RT_SUCCESS(rc))
+ rc = PDMR3TracingQueryConfig(pVM, pszDst, cbDst);
+ }
+
+ if (RT_FAILURE(rc))
+ *pszConfig = '\0';
+ return rc;
+}
+
+
+/**
+ * @callback_method_impl{FNRTTRACEBUFCALLBACK}
+ */
+static DECLCALLBACK(int)
+dbgfR3TraceInfoDumpEntry(RTTRACEBUF hTraceBuf, uint32_t iEntry, uint64_t NanoTS, RTCPUID idCpu, const char *pszMsg, void *pvUser)
+{
+ PCDBGFINFOHLP pHlp = (PCDBGFINFOHLP)pvUser;
+ pHlp->pfnPrintf(pHlp, "#%04u/%'llu/%02x: %s\n", iEntry, NanoTS, idCpu, pszMsg);
+ NOREF(hTraceBuf);
+ return VINF_SUCCESS;
+}
+
+
+/**
+ * @callback_method_impl{FNDBGFHANDLERINT, Info handler for displaying the trace buffer content.}
+ */
+static DECLCALLBACK(void) dbgfR3TraceInfo(PVM pVM, PCDBGFINFOHLP pHlp, const char *pszArgs)
+{
+ RTTRACEBUF hTraceBuf = pVM->hTraceBufR3;
+ if (hTraceBuf == NIL_RTTRACEBUF)
+ pHlp->pfnPrintf(pHlp, "Tracing is disabled\n");
+ else
+ {
+ pHlp->pfnPrintf(pHlp, "Trace buffer %p - %u entries of %u bytes\n",
+ hTraceBuf, RTTraceBufGetEntryCount(hTraceBuf), RTTraceBufGetEntrySize(hTraceBuf));
+ RTTraceBufEnumEntries(hTraceBuf, dbgfR3TraceInfoDumpEntry, (void *)pHlp);
+ }
+ NOREF(pszArgs);
+}
+