VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/EMR3Dbg.cpp@ 80191

Last change on this file since 80191 was 80191, checked in by vboxsync, 5 years ago

VMM/r3: Refactored VMCPU enumeration in preparation that aCpus will be replaced with a pointer array. Removed two raw-mode offset members from the CPUM and CPUMCPU sub-structures. bugref:9217 bugref:9517

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.1 KB
Line 
1/* $Id: EMR3Dbg.cpp 80191 2019-08-08 00:36:57Z vboxsync $ */
2/** @file
3 * EM - Execution Monitor / Manager, Debugger Related Bits.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define VBOX_BUGREF_9217_PART_I
23#define LOG_GROUP LOG_GROUP_EM
24#include <VBox/vmm/em.h>
25#include <VBox/vmm/hm.h>
26#include <VBox/vmm/nem.h>
27#include <VBox/dbg.h>
28#include "EMInternal.h"
29#include <VBox/vmm/vm.h>
30#include <iprt/string.h>
31#include <iprt/ctype.h>
32
33
34/** @callback_method_impl{FNDBGCCMD,
35 * Implements the '.alliem' command. }
36 */
37static DECLCALLBACK(int) enmR3DbgCmdAllIem(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PUVM pUVM, PCDBGCVAR paArgs, unsigned cArgs)
38{
39 int rc;
40 bool f;
41
42 if (cArgs == 0)
43 {
44 rc = EMR3QueryExecutionPolicy(pUVM, EMEXECPOLICY_IEM_ALL, &f);
45 if (RT_FAILURE(rc))
46 return DBGCCmdHlpFailRc(pCmdHlp, pCmd, rc, "EMR3QueryExecutionPolicy(,EMEXECPOLICY_IEM_ALL,");
47 DBGCCmdHlpPrintf(pCmdHlp, f ? "alliem: enabled\n" : "alliem: disabled\n");
48 }
49 else
50 {
51 rc = DBGCCmdHlpVarToBool(pCmdHlp, &paArgs[0], &f);
52 if (RT_FAILURE(rc))
53 return DBGCCmdHlpFailRc(pCmdHlp, pCmd, rc, "DBGCCmdHlpVarToBool");
54 rc = EMR3SetExecutionPolicy(pUVM, EMEXECPOLICY_IEM_ALL, f);
55 if (RT_FAILURE(rc))
56 return DBGCCmdHlpFailRc(pCmdHlp, pCmd, rc, "EMR3SetExecutionPolicy(,EMEXECPOLICY_IEM_ALL,%RTbool)", f);
57 }
58 return VINF_SUCCESS;
59}
60
61
62/** Describes a optional boolean argument. */
63static DBGCVARDESC const g_BoolArg = { 0, 1, DBGCVAR_CAT_ANY, 0, "boolean", "Boolean value." };
64
65/** Commands. */
66static DBGCCMD const g_aCmds[] =
67{
68 {
69 "alliem", 0, 1, &g_BoolArg, 1, 0, enmR3DbgCmdAllIem, "[boolean]",
70 "Enables or disabled executing ALL code in IEM, if no arguments are given it displays the current status."
71 },
72};
73
74
75/**
76 * Translates EMEXITTYPE into a name.
77 *
78 * @returns Pointer to read-only name, NULL if unknown type.
79 * @param enmExitType The exit type to name.
80 */
81VMM_INT_DECL(const char *) EMR3GetExitTypeName(EMEXITTYPE enmExitType)
82{
83 switch (enmExitType)
84 {
85 case EMEXITTYPE_INVALID: return "invalid";
86 case EMEXITTYPE_IO_PORT_READ: return "I/O port read";
87 case EMEXITTYPE_IO_PORT_WRITE: return "I/O port write";
88 case EMEXITTYPE_IO_PORT_STR_READ: return "I/O port string read";
89 case EMEXITTYPE_IO_PORT_STR_WRITE: return "I/O port string write";
90 case EMEXITTYPE_MMIO: return "MMIO access";
91 case EMEXITTYPE_MMIO_READ: return "MMIO read";
92 case EMEXITTYPE_MMIO_WRITE: return "MMIO write";
93 case EMEXITTYPE_MSR_READ: return "MSR read";
94 case EMEXITTYPE_MSR_WRITE: return "MSR write";
95 case EMEXITTYPE_CPUID: return "CPUID";
96 case EMEXITTYPE_RDTSC: return "RDTSC";
97 case EMEXITTYPE_MOV_CRX: return "MOV CRx";
98 case EMEXITTYPE_MOV_DRX: return "MOV DRx";
99
100 /* Raw-mode only: */
101 case EMEXITTYPE_INVLPG: return "INVLPG";
102 case EMEXITTYPE_LLDT: return "LLDT";
103 case EMEXITTYPE_RDPMC: return "RDPMC";
104 case EMEXITTYPE_CLTS: return "CLTS";
105 case EMEXITTYPE_STI: return "STI";
106 case EMEXITTYPE_INT: return "INT";
107 case EMEXITTYPE_SYSCALL: return "SYSCALL";
108 case EMEXITTYPE_SYSENTER: return "SYSENTER";
109 case EMEXITTYPE_HLT: return "HLT";
110 }
111 return NULL;
112}
113
114
115/**
116 * Translates flags+type into an exit name.
117 *
118 * @returns Exit name.
119 * @param uFlagsAndType The exit to name.
120 * @param pszFallback Buffer for formatting a numeric fallback.
121 * @param cbFallback Size of fallback buffer.
122 */
123static const char *emR3HistoryGetExitName(uint32_t uFlagsAndType, char *pszFallback, size_t cbFallback)
124{
125 const char *pszExitName;
126 switch (uFlagsAndType & EMEXIT_F_KIND_MASK)
127 {
128 case EMEXIT_F_KIND_EM:
129 pszExitName = EMR3GetExitTypeName((EMEXITTYPE)(uFlagsAndType & EMEXIT_F_TYPE_MASK));
130 break;
131
132 case EMEXIT_F_KIND_VMX:
133 pszExitName = HMGetVmxExitName( uFlagsAndType & EMEXIT_F_TYPE_MASK);
134 break;
135
136 case EMEXIT_F_KIND_SVM:
137 pszExitName = HMGetSvmExitName( uFlagsAndType & EMEXIT_F_TYPE_MASK);
138 break;
139
140 case EMEXIT_F_KIND_NEM:
141 pszExitName = NEMR3GetExitName( uFlagsAndType & EMEXIT_F_TYPE_MASK);
142 break;
143
144 case EMEXIT_F_KIND_XCPT:
145 switch (uFlagsAndType & EMEXIT_F_TYPE_MASK)
146 {
147 case X86_XCPT_DE: return "Xcpt #DE";
148 case X86_XCPT_DB: return "Xcpt #DB";
149 case X86_XCPT_NMI: return "Xcpt #NMI";
150 case X86_XCPT_BP: return "Xcpt #BP";
151 case X86_XCPT_OF: return "Xcpt #OF";
152 case X86_XCPT_BR: return "Xcpt #BR";
153 case X86_XCPT_UD: return "Xcpt #UD";
154 case X86_XCPT_NM: return "Xcpt #NM";
155 case X86_XCPT_DF: return "Xcpt #DF";
156 case X86_XCPT_CO_SEG_OVERRUN: return "Xcpt #CO_SEG_OVERRUN";
157 case X86_XCPT_TS: return "Xcpt #TS";
158 case X86_XCPT_NP: return "Xcpt #NP";
159 case X86_XCPT_SS: return "Xcpt #SS";
160 case X86_XCPT_GP: return "Xcpt #GP";
161 case X86_XCPT_PF: return "Xcpt #PF";
162 case X86_XCPT_MF: return "Xcpt #MF";
163 case X86_XCPT_AC: return "Xcpt #AC";
164 case X86_XCPT_MC: return "Xcpt #MC";
165 case X86_XCPT_XF: return "Xcpt #XF";
166 case X86_XCPT_VE: return "Xcpt #VE";
167 case X86_XCPT_SX: return "Xcpt #SX";
168 default:
169 pszExitName = NULL;
170 break;
171 }
172 break;
173
174 default:
175 AssertFailed();
176 pszExitName = NULL;
177 break;
178 }
179 if (pszExitName)
180 return pszExitName;
181 RTStrPrintf(pszFallback, cbFallback, "%#06x", uFlagsAndType & (EMEXIT_F_KIND_MASK | EMEXIT_F_TYPE_MASK));
182 return pszFallback;
183}
184
185
186/**
187 * Displays the VM-exit history.
188 *
189 * @param pVM The cross context VM structure.
190 * @param pHlp The info helper functions.
191 * @param pszArgs Arguments, ignored.
192 */
193static DECLCALLBACK(void) emR3InfoExitHistory(PVM pVM, PCDBGFINFOHLP pHlp, const char *pszArgs)
194{
195 NOREF(pszArgs);
196
197 /*
198 * Figure out target cpu and parse arguments.
199 */
200 PVMCPU pVCpu = VMMGetCpu(pVM);
201 if (!pVCpu)
202 pVCpu = pVM->apCpusR3[0];
203 bool fReverse = true;
204 uint32_t cLeft = RT_ELEMENTS(pVCpu->em.s.aExitHistory);
205
206 while (pszArgs && *pszArgs)
207 {
208 pszArgs = RTStrStripL(pszArgs);
209 if (!*pszArgs)
210 break;
211 if (RT_C_IS_DIGIT(*pszArgs))
212 {
213 /* The number to dump. */
214 uint32_t uValue = cLeft;
215 RTStrToUInt32Ex(pszArgs, (char **)&pszArgs, 0, &uValue);
216 if (uValue > 0)
217 cLeft = RT_MIN(uValue, RT_ELEMENTS(pVCpu->em.s.aExitHistory));
218 }
219 else if (RTStrCmp(pszArgs, "reverse") == 0)
220 {
221 pszArgs += 7;
222 fReverse = true;
223 }
224 else if (RTStrCmp(pszArgs, "ascending") == 0)
225 {
226 pszArgs += 9;
227 fReverse = false;
228 }
229 else if (RTStrCmp(pszArgs, "asc") == 0)
230 {
231 pszArgs += 3;
232 fReverse = false;
233 }
234 else
235 {
236 const char *pszStart = pszArgs;
237 while (*pszArgs && !RT_C_IS_SPACE(*pszArgs))
238 pszArgs++;
239 pHlp->pfnPrintf(pHlp, "Unknown option: %.*s\n", pszArgs - pszStart, pszArgs);
240 }
241 }
242
243 /*
244 * Do the job.
245 */
246 uint64_t idx = pVCpu->em.s.iNextExit;
247 if (idx == 0)
248 pHlp->pfnPrintf(pHlp, "CPU[%u]: VM-exit history: empty\n", pVCpu->idCpu);
249 else
250 {
251 /*
252 * Print header.
253 */
254 pHlp->pfnPrintf(pHlp,
255 "CPU[%u]: VM-exit history:\n"
256 " Exit No.: TSC timestamp / delta RIP (Flat/*) Exit Name\n"
257 , pVCpu->idCpu);
258
259 /*
260 * Adjust bounds if ascending order.
261 */
262 if (!fReverse)
263 {
264 if (idx > cLeft)
265 idx -= cLeft;
266 else
267 {
268 cLeft = idx;
269 idx = 0;
270 }
271 }
272
273 /*
274 * Print the entries.
275 */
276 uint64_t uPrevTimestamp = 0;
277 do
278 {
279 if (fReverse)
280 idx -= 1;
281 PCEMEXITENTRY const pEntry = &pVCpu->em.s.aExitHistory[(uintptr_t)idx & 0xff];
282
283 /* Get the exit name. */
284 char szExitName[16];
285 const char *pszExitName = emR3HistoryGetExitName(pEntry->uFlagsAndType, szExitName, sizeof(szExitName));
286
287 /* Calc delta (negative if reverse order, positive ascending). */
288 int64_t offDelta = uPrevTimestamp != 0 && pEntry->uTimestamp != 0 ? pEntry->uTimestamp - uPrevTimestamp : 0;
289 uPrevTimestamp = pEntry->uTimestamp;
290
291 char szPC[32];
292 if (!(pEntry->uFlagsAndType & (EMEXIT_F_CS_EIP | EMEXIT_F_UNFLATTENED_PC)))
293 RTStrPrintf(szPC, sizeof(szPC), "%016RX64 ", pEntry->uFlatPC);
294 else if (pEntry->uFlagsAndType & EMEXIT_F_UNFLATTENED_PC)
295 RTStrPrintf(szPC, sizeof(szPC), "%016RX64*", pEntry->uFlatPC);
296 else
297 RTStrPrintf(szPC, sizeof(szPC), "%04x:%08RX32* ", (uint32_t)(pEntry->uFlatPC >> 32), (uint32_t)pEntry->uFlatPC);
298
299 /* Do the printing. */
300 if (pEntry->idxSlot == UINT32_MAX)
301 pHlp->pfnPrintf(pHlp, " %10RU64: %#018RX64/%+-9RI64 %s %#07x %s\n",
302 idx, pEntry->uTimestamp, offDelta, szPC, pEntry->uFlagsAndType, pszExitName);
303 else
304 {
305 /** @todo more on this later */
306 pHlp->pfnPrintf(pHlp, " %10RU64: %#018RX64/%+-9RI64 %s %#07x %s slot=%#x\n",
307 idx, pEntry->uTimestamp, offDelta, szPC, pEntry->uFlagsAndType, pszExitName, pEntry->idxSlot);
308 }
309
310 /* Advance if ascending. */
311 if (!fReverse)
312 idx += 1;
313 } while (--cLeft > 0 && idx > 0);
314 }
315}
316
317
318int emR3InitDbg(PVM pVM)
319{
320 /*
321 * Register info dumpers.
322 */
323 const char *pszExitsDesc = "Dumps the VM-exit history. Arguments: Number of entries; 'asc', 'ascending' or 'reverse'.";
324 int rc = DBGFR3InfoRegisterInternalEx(pVM, "exits", pszExitsDesc, emR3InfoExitHistory, DBGFINFO_FLAGS_ALL_EMTS);
325 AssertLogRelRCReturn(rc, rc);
326 rc = DBGFR3InfoRegisterInternalEx(pVM, "exithistory", pszExitsDesc, emR3InfoExitHistory, DBGFINFO_FLAGS_ALL_EMTS);
327 AssertLogRelRCReturn(rc, rc);
328
329#ifdef VBOX_WITH_DEBUGGER
330 /*
331 * Register debugger commands.
332 */
333 rc = DBGCRegisterCommands(&g_aCmds[0], RT_ELEMENTS(g_aCmds));
334 AssertLogRelRCReturn(rc, rc);
335#endif
336
337 return VINF_SUCCESS;
338}
339
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