VirtualBox

source: vbox/trunk/src/VBox/Debugger/DBGConsole.cpp@ 5320

Last change on this file since 5320 was 5079, checked in by vboxsync, 17 years ago

Fixed two copy & past bugs.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 342.9 KB
Line 
1/** $Id: DBGConsole.cpp 5079 2007-09-27 14:39:05Z vboxsync $ */
2/** @file
3 * DBGC - Debugger Console.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/** @page pg_dbgc DBGC - The Debug Console
20 *
21 * The debugger console is a first attempt to make some interactive
22 * debugging facilities for the VirtualBox backend (i.e. the VM). At a later
23 * stage we'll make a fancy gui around this, but for the present a telnet (or
24 * serial terminal) will have to suffice.
25 *
26 * The debugger is only built into the VM with debug builds or when
27 * VBOX_WITH_DEBUGGER is defined. There might be need for \#ifdef'ing on this
28 * define to enable special debugger hooks, but the general approach is to
29 * make generic interfaces. The individual components also can register
30 * external commands, and such code must be within \#ifdef.
31 *
32 *
33 * @section sec_dbgc_op Operation (intentions)
34 *
35 * The console will process commands in a manner similar to the OS/2 and
36 * windows kernel debuggers. This means ';' is a command separator and
37 * that when possible we'll use the same command names as these two uses.
38 *
39 *
40 * @subsection sec_dbg_op_numbers Numbers
41 *
42 * Numbers are hexadecimal unless specified with a prefix indicating
43 * elsewise. Prefixes:
44 * - '0x' - hexadecimal.
45 * - '0i' - decimal
46 * - '0t' - octal.
47 * - '0y' - binary.
48 *
49 *
50 * @subsection sec_dbg_op_address Addressing modes
51 *
52 * - Default is flat. For compatability '%' also means flat.
53 * - Segmented addresses are specified selector:offset.
54 * - Physical addresses are specified using '%%'.
55 * - The default target for the addressing is the guest context, the '#'
56 * will override this and set it to the host.
57 *
58 *
59 * @subsection sec_dbg_op_evalution Evaluation
60 *
61 * As time permits support will be implemented support for a subset of the C
62 * binary operators, starting with '+', '-', '*' and '/'. Support for variables
63 * are provided thru commands 'set' and 'unset' and the unary operator '$'. The
64 * unary '@' operator will indicate function calls. The debugger needs a set of
65 * memory read functions, but we might later extend this to allow registration of
66 * external functions too.
67 *
68 * A special command '?' will then be added which evalutates a given expression
69 * and prints it in all the different formats.
70 *
71 *
72 * @subsection sec_dbg_op_registers Registers
73 *
74 * Registers are addressed using their name. Some registers which have several fields
75 * (like gdtr) will have separate names indicating the different fields. The default
76 * register set is the guest one. To access the hypervisor register one have to
77 * prefix the register names with '.'.
78 *
79 *
80 * @subsection sec_dbg_op_commands Commands
81 *
82 * The commands are all lowercase, case sensitive, and starting with a letter. We will
83 * later add some special commands ('?' for evaulation) and perhaps command classes ('.', '!')
84 *
85 *
86 * @section sec_dbg_tasks Tasks
87 *
88 * To implement DBGT and instrument VMM for basic state inspection and log
89 * viewing, the follwing task must be executed:
90 *
91 * -# Basic threading layer in RT.
92 * -# Basic tcpip server abstration in RT.
93 * -# Write DBGC.
94 * -# Write DBCTCP.
95 * -# Integrate with VMM and the rest.
96 * -# Start writing DBGF (VMM).
97 */
98
99
100
101
102/*******************************************************************************
103* Header Files *
104*******************************************************************************/
105#define LOG_GROUP LOG_GROUP_DBGC
106#include <VBox/dbg.h>
107#include <VBox/dbgf.h>
108#include <VBox/vm.h>
109#include <VBox/vmm.h>
110#include <VBox/mm.h>
111#include <VBox/pgm.h>
112#include <VBox/selm.h>
113#include <VBox/dis.h>
114#include <VBox/param.h>
115#include <VBox/err.h>
116#include <VBox/log.h>
117
118#include <iprt/alloc.h>
119#include <iprt/alloca.h>
120#include <iprt/string.h>
121#include <iprt/assert.h>
122#include <iprt/ctype.h>
123
124#include <stdlib.h>
125#include <stdio.h>
126
127/* to err.h! */
128#define VERR_DBGC_QUIT (-11999)
129#define VERR_PARSE_FIRST (-11000)
130#define VERR_PARSE_TOO_FEW_ARGUMENTS (VERR_PARSE_FIRST - 0)
131#define VERR_PARSE_TOO_MANY_ARGUMENTS (VERR_PARSE_FIRST - 1)
132#define VERR_PARSE_ARGUMENT_OVERFLOW (VERR_PARSE_FIRST - 2)
133#define VERR_PARSE_ARGUMENT_TYPE_MISMATCH (VERR_PARSE_FIRST - 3)
134#define VERR_PARSE_NO_RANGE_ALLOWED (VERR_PARSE_FIRST - 4)
135#define VERR_PARSE_UNBALANCED_QUOTE (VERR_PARSE_FIRST - 5)
136#define VERR_PARSE_UNBALANCED_PARENTHESIS (VERR_PARSE_FIRST - 6)
137#define VERR_PARSE_EMPTY_ARGUMENT (VERR_PARSE_FIRST - 7)
138#define VERR_PARSE_UNEXPECTED_OPERATOR (VERR_PARSE_FIRST - 8)
139#define VERR_PARSE_INVALID_NUMBER (VERR_PARSE_FIRST - 9)
140#define VERR_PARSE_NUMBER_TOO_BIG (VERR_PARSE_FIRST - 10)
141#define VERR_PARSE_INVALID_OPERATION (VERR_PARSE_FIRST - 11)
142#define VERR_PARSE_FUNCTION_NOT_FOUND (VERR_PARSE_FIRST - 12)
143#define VERR_PARSE_NOT_A_FUNCTION (VERR_PARSE_FIRST - 13)
144#define VERR_PARSE_NO_MEMORY (VERR_PARSE_FIRST - 14)
145#define VERR_PARSE_INCORRECT_ARG_TYPE (VERR_PARSE_FIRST - 15)
146#define VERR_PARSE_VARIABLE_NOT_FOUND (VERR_PARSE_FIRST - 16)
147#define VERR_PARSE_CONVERSION_FAILED (VERR_PARSE_FIRST - 17)
148#define VERR_PARSE_NOT_IMPLEMENTED (VERR_PARSE_FIRST - 18)
149#define VERR_PARSE_BAD_RESULT_TYPE (VERR_PARSE_FIRST - 19)
150#define VERR_PARSE_WRITEONLY_SYMBOL (VERR_PARSE_FIRST - 20)
151#define VERR_PARSE_NO_ARGUMENT_MATCH (VERR_PARSE_FIRST - 21)
152#define VERR_PARSE_LAST (VERR_PARSE_FIRST - 30)
153
154#define VWRN_DBGC_CMD_PENDING 12000
155#define VWRN_DBGC_ALREADY_REGISTERED 12001
156#define VERR_DBGC_COMMANDS_NOT_REGISTERED (-12002)
157#define VERR_DBGC_BP_NOT_FOUND (-12003)
158#define VERR_DBGC_BP_EXISTS (-12004)
159#define VINF_DBGC_BP_NO_COMMAND 12005
160
161
162
163/*******************************************************************************
164* Defined Constants And Macros *
165*******************************************************************************/
166/** Makes a DBGC variable type pair.
167 * Typically used by binary operators. */
168#define BINARY_TYPE_PAIR(type1, type2) (type1 | (type2 << 16))
169
170
171/*******************************************************************************
172* Structures and Typedefs *
173*******************************************************************************/
174
175/**
176 * Debugger console per breakpoint data.
177 */
178typedef struct DBGCBP
179{
180 /** Pointer to the next breakpoint in the list. */
181 struct DBGCBP *pNext;
182 /** The breakpoint identifier. */
183 RTUINT iBp;
184 /** The size of the command. */
185 size_t cchCmd;
186 /** The command to execute when the breakpoint is hit. */
187 char szCmd[1];
188} DBGCBP;
189/** Pointer to a breakpoint. */
190typedef DBGCBP *PDBGCBP;
191
192
193/**
194 * Named variable.
195 *
196 * Always allocated from heap in one signle block.
197 */
198typedef struct DBGCNAMEDVAR
199{
200 /** The variable. */
201 DBGCVAR Var;
202 /** It's name. */
203 char szName[1];
204} DBGCNAMEDVAR;
205/** Pointer to named variable. */
206typedef DBGCNAMEDVAR *PDBGCNAMEDVAR;
207
208
209/**
210 * Debugger console status
211 */
212typedef enum DBGCSTATUS
213{
214 /** Normal status, .*/
215 DBGC_HALTED
216
217} DBGCSTATUS;
218
219
220/**
221 * Debugger console instance data.
222 */
223typedef struct DBGC
224{
225 /** Command helpers. */
226 DBGCCMDHLP CmdHlp;
227 /** Pointer to backend callback structure. */
228 PDBGCBACK pBack;
229 /** Log indicator. (If set we're writing the log to the console.) */
230 bool fLog;
231 /** Pointer to the current VM. */
232 PVM pVM;
233 /** Indicates whether or we're ready for input. */
234 bool fReady;
235
236 /** Indicates whether we're in guest (true) or hypervisor (false) register context. */
237 bool fRegCtxGuest;
238 /** Indicates whether the register are terse or sparse. */
239 bool fRegTerse;
240
241 /** Input buffer. */
242 char achInput[2048];
243 /** To ease debugging. */
244 unsigned uInputZero;
245 /** Write index in the input buffer. */
246 unsigned iWrite;
247 /** Read index in the input buffer. */
248 unsigned iRead;
249 /** The number of lines in the buffer. */
250 unsigned cInputLines;
251 /** Indicates that we have a buffer overflow condition.
252 * This means that input is ignored up to the next newline. */
253 bool fInputOverflow;
254
255 /** Scratch buffer position. */
256 char *pszScratch;
257 /** Scratch buffer. */
258 char achScratch[16384];
259 /** Argument array position. */
260 unsigned iArg;
261 /** Array of argument variables. */
262 DBGCVAR aArgs[100];
263
264 /** rc from last dbgcHlpPrintfV(). */
265 int rcOutput;
266
267 /** Number of variables in papVars. */
268 unsigned cVars;
269 /** Array of global variables.
270 * Global variables can be referenced using the $ operator and set
271 * and unset using command with those names. */
272 PDBGCNAMEDVAR *papVars;
273
274 /** Current dissassembler position. */
275 DBGCVAR DisasmPos;
276 /** Current source position. (flat GC) */
277 DBGCVAR SourcePos;
278 /** Current memory dump position. */
279 DBGCVAR DumpPos;
280 /** Size of the previous dump element. */
281 unsigned cbDumpElement;
282
283 /** The list of breakpoints. (singly linked) */
284 PDBGCBP pFirstBp;
285} DBGC;
286/** Pointer to debugger console instance data. */
287typedef DBGC *PDBGC;
288
289/** Converts a Command Helper pointer to a pointer to DBGC instance data. */
290#define DBGC_CMDHLP2DBGC(pCmdHlp) ( (PDBGC)((uintptr_t)(pCmdHlp) - RT_OFFSETOF(DBGC, CmdHlp)) )
291
292
293/**
294 * Chunk of external commands.
295 */
296typedef struct DBGCEXTCMDS
297{
298 /** Number of commands descriptors. */
299 unsigned cCmds;
300 /** Pointer to array of command descriptors. */
301 PCDBGCCMD paCmds;
302 /** Pointer to the next chunk. */
303 struct DBGCEXTCMDS *pNext;
304} DBGCEXTCMDS;
305/** Pointer to chunk of external commands. */
306typedef DBGCEXTCMDS *PDBGCEXTCMDS;
307
308
309
310/**
311 * Unary operator handler function.
312 *
313 * @returns 0 on success.
314 * @returns VBox evaluation / parsing error code on failure.
315 * The caller does the bitching.
316 * @param pDbgc Debugger console instance data.
317 * @param pArg The argument.
318 * @param pResult Where to store the result.
319 */
320typedef DECLCALLBACK(int) FNDBGCOPUNARY(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
321/** Pointer to a unary operator handler function. */
322typedef FNDBGCOPUNARY *PFNDBGCOPUNARY;
323
324
325/**
326 * Binary operator handler function.
327 *
328 * @returns 0 on success.
329 * @returns VBox evaluation / parsing error code on failure.
330 * The caller does the bitching.
331 * @param pDbgc Debugger console instance data.
332 * @param pArg1 The first argument.
333 * @param pArg2 The 2nd argument.
334 * @param pResult Where to store the result.
335 */
336typedef DECLCALLBACK(int) FNDBGCOPBINARY(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
337/** Pointer to a binary operator handler function. */
338typedef FNDBGCOPBINARY *PFNDBGCOPBINARY;
339
340
341/**
342 * Operator descriptor.
343 */
344typedef struct DBGCOP
345{
346 /** Operator mnemonic. */
347 char szName[4];
348 /** Length of name. */
349 const unsigned cchName;
350 /** Whether or not this is a binary operator.
351 * Unary operators are evaluated right-to-left while binary are left-to-right. */
352 bool fBinary;
353 /** Precedence level. */
354 unsigned iPrecedence;
355 /** Unary operator handler. */
356 PFNDBGCOPUNARY pfnHandlerUnary;
357 /** Binary operator handler. */
358 PFNDBGCOPBINARY pfnHandlerBinary;
359 /** Operator description. */
360 const char *pszDescription;
361} DBGCOP;
362/** Pointer to an operator descriptor. */
363typedef DBGCOP *PDBGCOP;
364/** Pointer to a const operator descriptor. */
365typedef const DBGCOP *PCDBGCOP;
366
367
368
369/** Pointer to symbol descriptor. */
370typedef struct DBGCSYM *PDBGCSYM;
371/** Pointer to const symbol descriptor. */
372typedef const struct DBGCSYM *PCDBGCSYM;
373
374/**
375 * Get builtin symbol.
376 *
377 * @returns 0 on success.
378 * @returns VBox evaluation / parsing error code on failure.
379 * The caller does the bitching.
380 * @param pSymDesc Pointer to the symbol descriptor.
381 * @param pCmdHlp Pointer to the command callback structure.
382 * @param enmType The result type.
383 * @param pResult Where to store the result.
384 */
385typedef DECLCALLBACK(int) FNDBGCSYMGET(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, DBGCVARTYPE enmType, PDBGCVAR pResult);
386/** Pointer to get function for a builtin symbol. */
387typedef FNDBGCSYMGET *PFNDBGCSYMGET;
388
389/**
390 * Set builtin symbol.
391 *
392 * @returns 0 on success.
393 * @returns VBox evaluation / parsing error code on failure.
394 * The caller does the bitching.
395 * @param pSymDesc Pointer to the symbol descriptor.
396 * @param pCmdHlp Pointer to the command callback structure.
397 * @param pValue The value to assign the symbol.
398 */
399typedef DECLCALLBACK(int) FNDBGCSYMSET(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, PCDBGCVAR pValue);
400/** Pointer to set function for a builtin symbol. */
401typedef FNDBGCSYMSET *PFNDBGCSYMSET;
402
403
404/**
405 * Symbol description (for builtin symbols).
406 */
407typedef struct DBGCSYM
408{
409 /** Symbol name. */
410 const char *pszName;
411 /** Get function. */
412 PFNDBGCSYMGET pfnGet;
413 /** Set function. (NULL if readonly) */
414 PFNDBGCSYMSET pfnSet;
415 /** User data. */
416 unsigned uUser;
417} DBGCSYM;
418
419
420/*******************************************************************************
421* Internal Functions *
422*******************************************************************************/
423static DECLCALLBACK(int) dbgcCmdBrkAccess(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
424static DECLCALLBACK(int) dbgcCmdBrkClear(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
425static DECLCALLBACK(int) dbgcCmdBrkDisable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
426static DECLCALLBACK(int) dbgcCmdBrkEnable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
427static DECLCALLBACK(int) dbgcCmdBrkList(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
428static DECLCALLBACK(int) dbgcCmdBrkSet(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
429static DECLCALLBACK(int) dbgcCmdBrkREM(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
430static DECLCALLBACK(int) dbgcCmdHelp(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
431static DECLCALLBACK(int) dbgcCmdQuit(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
432static DECLCALLBACK(int) dbgcCmdGo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
433static DECLCALLBACK(int) dbgcCmdStop(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
434static DECLCALLBACK(int) dbgcCmdDisasm(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
435static DECLCALLBACK(int) dbgcCmdSource(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
436static DECLCALLBACK(int) dbgcCmdReg(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
437static DECLCALLBACK(int) dbgcCmdRegGuest(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
438static DECLCALLBACK(int) dbgcCmdRegHyper(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
439static DECLCALLBACK(int) dbgcCmdRegTerse(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
440static DECLCALLBACK(int) dbgcCmdTrace(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
441static DECLCALLBACK(int) dbgcCmdStack(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
442static DECLCALLBACK(int) dbgcCmdDumpDT(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
443static DECLCALLBACK(int) dbgcCmdDumpIDT(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
444static DECLCALLBACK(int) dbgcCmdDumpMem(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
445static DECLCALLBACK(int) dbgcCmdDumpPageDir(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
446static DECLCALLBACK(int) dbgcCmdDumpPageDirBoth(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
447static DECLCALLBACK(int) dbgcCmdDumpPageTable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
448static DECLCALLBACK(int) dbgcCmdDumpPageTableBoth(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
449static DECLCALLBACK(int) dbgcCmdDumpTSS(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
450static DECLCALLBACK(int) dbgcCmdInfo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
451static DECLCALLBACK(int) dbgcCmdLog(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
452static DECLCALLBACK(int) dbgcCmdLogDest(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
453static DECLCALLBACK(int) dbgcCmdLogFlags(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
454static DECLCALLBACK(int) dbgcCmdMemoryInfo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
455static DECLCALLBACK(int) dbgcCmdFormat(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
456static DECLCALLBACK(int) dbgcCmdListNear(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
457static DECLCALLBACK(int) dbgcCmdLoadSyms(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
458static DECLCALLBACK(int) dbgcCmdSet(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
459static DECLCALLBACK(int) dbgcCmdUnset(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
460static DECLCALLBACK(int) dbgcCmdLoadVars(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
461static DECLCALLBACK(int) dbgcCmdShowVars(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
462static DECLCALLBACK(int) dbgcCmdHarakiri(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
463static DECLCALLBACK(int) dbgcCmdEcho(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
464static DECLCALLBACK(int) dbgcCmdRunScript(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult);
465
466static DECLCALLBACK(int) dbgcOpMinus(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
467static DECLCALLBACK(int) dbgcOpPluss(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
468static DECLCALLBACK(int) dbgcOpBooleanNot(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
469static DECLCALLBACK(int) dbgcOpBitwiseNot(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
470static DECLCALLBACK(int) dbgcOpAddrFlat(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
471static DECLCALLBACK(int) dbgcOpAddrPhys(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
472static DECLCALLBACK(int) dbgcOpAddrHost(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
473static DECLCALLBACK(int) dbgcOpAddrHostPhys(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
474static DECLCALLBACK(int) dbgcOpVar(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult);
475
476static DECLCALLBACK(int) dbgcOpAddrFar(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
477static DECLCALLBACK(int) dbgcOpMult(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
478static DECLCALLBACK(int) dbgcOpDiv(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
479static DECLCALLBACK(int) dbgcOpMod(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
480static DECLCALLBACK(int) dbgcOpAdd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
481static DECLCALLBACK(int) dbgcOpSub(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
482static DECLCALLBACK(int) dbgcOpBitwiseShiftLeft(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
483static DECLCALLBACK(int) dbgcOpBitwiseShiftRight(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
484static DECLCALLBACK(int) dbgcOpBitwiseAnd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
485static DECLCALLBACK(int) dbgcOpBitwiseXor(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
486static DECLCALLBACK(int) dbgcOpBitwiseOr(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
487static DECLCALLBACK(int) dbgcOpBooleanAnd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
488static DECLCALLBACK(int) dbgcOpBooleanOr(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
489static DECLCALLBACK(int) dbgcOpRangeLength(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
490static DECLCALLBACK(int) dbgcOpRangeLengthBytes(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
491static DECLCALLBACK(int) dbgcOpRangeTo(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult);
492
493static DECLCALLBACK(int) dbgcSymGetReg(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, DBGCVARTYPE enmType, PDBGCVAR pResult);
494static DECLCALLBACK(int) dbgcSymSetReg(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, PCDBGCVAR pValue);
495
496static void dbgcVarSetGCFlat(PDBGCVAR pVar, RTGCPTR GCFlat);
497static void dbgcVarSetGCFlatByteRange(PDBGCVAR pVar, RTGCPTR GCFlat, uint64_t cb);
498static void dbgcVarSetVar(PDBGCVAR pVar, PCDBGCVAR pVar2);
499static void dbgcVarSetNoRange(PDBGCVAR pVar);
500static void dbgcVarSetByteRange(PDBGCVAR pVar, uint64_t cb);
501static int dbgcVarToDbgfAddr(PDBGC pDbgc, PCDBGCVAR pVar, PDBGFADDRESS pAddress);
502
503static int dbgcBpAdd(PDBGC pDbgc, RTUINT iBp, const char *pszCmd);
504static int dbgcBpUpdate(PDBGC pDbgc, RTUINT iBp, const char *pszCmd);
505static int dbgcBpDelete(PDBGC pDbgc, RTUINT iBp);
506static PDBGCBP dbgcBpGet(PDBGC pDbgc, RTUINT iBp);
507static int dbgcBpExec(PDBGC pDbgc, RTUINT iBp);
508
509static PCDBGCSYM dbgcLookupRegisterSymbol(PDBGC pDbgc, const char *pszSymbol);
510static int dbgcSymbolGet(PDBGC pDbgc, const char *pszSymbol, DBGCVARTYPE enmType, PDBGCVAR pResult);
511static int dbgcEvalSub(PDBGC pDbgc, char *pszExpr, size_t cchExpr, PDBGCVAR pResult);
512static int dbgcProcessCommand(PDBGC pDbgc, char *pszCmd, size_t cchCmd);
513
514
515/*******************************************************************************
516* Global Variables *
517*******************************************************************************/
518/**
519 * Pointer to head of the list of external commands.
520 */
521static PDBGCEXTCMDS g_pExtCmdsHead; /** @todo rw protect g_pExtCmdsHead! */
522/** Locks the g_pExtCmdsHead list for reading. */
523#define DBGCEXTCMDS_LOCK_RD() do { } while (0)
524/** Locks the g_pExtCmdsHead list for writing. */
525#define DBGCEXTCMDS_LOCK_WR() do { } while (0)
526/** UnLocks the g_pExtCmdsHead list after reading. */
527#define DBGCEXTCMDS_UNLOCK_RD() do { } while (0)
528/** UnLocks the g_pExtCmdsHead list after writing. */
529#define DBGCEXTCMDS_UNLOCK_WR() do { } while (0)
530
531
532/** One argument of any kind. */
533static const DBGCVARDESC g_aArgAny[] =
534{
535 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
536 { 0, 1, DBGCVAR_CAT_ANY, 0, "var", "Any type of argument." },
537};
538
539/** Multiple string arguments (min 1). */
540static const DBGCVARDESC g_aArgMultiStr[] =
541{
542 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
543 { 1, ~0, DBGCVAR_CAT_STRING, 0, "strings", "One or more strings." },
544};
545
546/** Filename string. */
547static const DBGCVARDESC g_aArgFilename[] =
548{
549 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
550 { 1, 1, DBGCVAR_CAT_STRING, 0, "path", "Filename string." },
551};
552
553
554/** 'ba' arguments. */
555static const DBGCVARDESC g_aArgBrkAcc[] =
556{
557 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
558 { 1, 1, DBGCVAR_CAT_STRING, 0, "access", "The access type: x=execute, rw=read/write (alias r), w=write, i=not implemented." },
559 { 1, 1, DBGCVAR_CAT_NUMBER, 0, "size", "The access size: 1, 2, 4, or 8. 'x' access requires 1, and 8 requires amd64 long mode." },
560 { 1, 1, DBGCVAR_CAT_GC_POINTER, 0, "address", "The address." },
561 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "passes", "The number of passes before we trigger the breakpoint. (0 is default)" },
562 { 0, 1, DBGCVAR_CAT_NUMBER, DBGCVD_FLAGS_DEP_PREV, "max passes", "The number of passes after which we stop triggering the breakpoint. (~0 is default)" },
563 { 0, 1, DBGCVAR_CAT_STRING, 0, "cmds", "String of commands to be executed when the breakpoint is hit. Quote it!" },
564};
565
566
567/** 'bc', 'bd', 'be' arguments. */
568static const DBGCVARDESC g_aArgBrks[] =
569{
570 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
571 { 0, ~0, DBGCVAR_CAT_NUMBER, 0, "#bp", "Breakpoint number." },
572 { 0, 1, DBGCVAR_CAT_STRING, 0, "all", "All breakpoints." },
573};
574
575
576/** 'bp' arguments. */
577static const DBGCVARDESC g_aArgBrkSet[] =
578{
579 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
580 { 1, 1, DBGCVAR_CAT_GC_POINTER, 0, "address", "The address." },
581 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "passes", "The number of passes before we trigger the breakpoint. (0 is default)" },
582 { 0, 1, DBGCVAR_CAT_NUMBER, DBGCVD_FLAGS_DEP_PREV, "max passes", "The number of passes after which we stop triggering the breakpoint. (~0 is default)" },
583 { 0, 1, DBGCVAR_CAT_STRING, 0, "cmds", "String of commands to be executed when the breakpoint is hit. Quote it!" },
584};
585
586
587/** 'br' arguments. */
588static const DBGCVARDESC g_aArgBrkREM[] =
589{
590 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
591 { 1, 1, DBGCVAR_CAT_GC_POINTER, 0, "address", "The address." },
592 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "passes", "The number of passes before we trigger the breakpoint. (0 is default)" },
593 { 0, 1, DBGCVAR_CAT_NUMBER, DBGCVD_FLAGS_DEP_PREV, "max passes", "The number of passes after which we stop triggering the breakpoint. (~0 is default)" },
594 { 0, 1, DBGCVAR_CAT_STRING, 0, "cmds", "String of commands to be executed when the breakpoint is hit. Quote it!" },
595};
596
597
598/** 'd?' arguments. */
599static const DBGCVARDESC g_aArgDumpMem[] =
600{
601 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
602 { 0, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address where to start dumping memory." },
603};
604
605
606/** 'dg', 'dga', 'dl', 'dla' arguments. */
607static const DBGCVARDESC g_aArgDumpDT[] =
608{
609 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
610 { 0, ~0, DBGCVAR_CAT_NUMBER, 0, "sel", "Selector or selector range." },
611 { 0, ~0, DBGCVAR_CAT_POINTER, 0, "address", "Far address which selector should be dumped." },
612};
613
614
615/** 'di', 'dia' arguments. */
616static const DBGCVARDESC g_aArgDumpIDT[] =
617{
618 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
619 { 0, ~0, DBGCVAR_CAT_NUMBER, 0, "int", "The interrupt vector or interrupt vector range." },
620};
621
622
623/** 'dpd*' arguments. */
624static const DBGCVARDESC g_aArgDumpPD[] =
625{
626 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
627 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "index", "Index into the page directory." },
628 { 0, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address which page directory entry to start dumping from. Range is applied to the page directory." },
629};
630
631
632/** 'dpda' arguments. */
633static const DBGCVARDESC g_aArgDumpPDAddr[] =
634{
635 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
636 { 0, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address of the page directory entry to start dumping from." },
637};
638
639
640/** 'dpt?' arguments. */
641static const DBGCVARDESC g_aArgDumpPT[] =
642{
643 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
644 { 1, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address which page directory entry to start dumping from." },
645};
646
647
648/** 'dpta' arguments. */
649static const DBGCVARDESC g_aArgDumpPTAddr[] =
650{
651 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
652 { 1, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address of the page table entry to start dumping from." },
653};
654
655
656/** 'dt' arguments. */
657static const DBGCVARDESC g_aArgDumpTSS[] =
658{
659 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
660 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "tss", "TSS selector number." },
661 { 0, 1, DBGCVAR_CAT_POINTER, 0, "tss:ign|addr", "TSS address. If the selector is a TSS selector, the offset will be ignored." }
662};
663
664
665/** 'help' arguments. */
666static const DBGCVARDESC g_aArgHelp[] =
667{
668 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
669 { 0, ~0, DBGCVAR_CAT_STRING, 0, "cmd/op", "Zero or more command or operator names." },
670};
671
672
673/** 'info' arguments. */
674static const DBGCVARDESC g_aArgInfo[] =
675{
676 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
677 { 1, 1, DBGCVAR_CAT_STRING, 0, "info", "The name of the info to display." },
678 { 0, 1, DBGCVAR_CAT_STRING, 0, "args", "String arguments to the handler." },
679};
680
681
682
683/** 'ln' arguments. */
684static const DBGCVARDESC g_aArgListNear[] =
685{
686 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
687 { 0, ~0, DBGCVAR_CAT_POINTER, 0, "address", "Address of the symbol to look up." },
688 { 0, ~0, DBGCVAR_CAT_SYMBOL, 0, "symbol", "Symbol to lookup." },
689};
690
691/** 'ln' return. */
692static const DBGCVARDESC g_RetListNear =
693{
694 1, 1, DBGCVAR_CAT_POINTER, 0, "address", "The last resolved symbol/address with adjusted range."
695};
696
697
698/** loadsyms arguments. */
699static const DBGCVARDESC g_aArgLoadSyms[] =
700{
701 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
702 { 1, 1, DBGCVAR_CAT_STRING, 0, "path", "Filename string." },
703 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "delta", "Delta to add to the loaded symbols. (optional)" },
704 { 0, 1, DBGCVAR_CAT_STRING, 0, "module name", "Module name." },
705 { 0, 1, DBGCVAR_CAT_POINTER, DBGCVD_FLAGS_DEP_PREV, "module address", "Module address." },
706 { 0, 1, DBGCVAR_CAT_NUMBER, 0, "module size", "The module size. (optional)" },
707};
708
709
710/** log arguments. */
711static const DBGCVARDESC g_aArgLog[] =
712{
713 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
714 { 1, 1, DBGCVAR_CAT_STRING, 0, "groups", "Group modifier string (quote it!)." }
715};
716
717
718/** logdest arguments. */
719static const DBGCVARDESC g_aArgLogDest[] =
720{
721 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
722 { 1, 1, DBGCVAR_CAT_STRING, 0, "dests", "Destination modifier string (quote it!)." }
723};
724
725
726/** logflags arguments. */
727static const DBGCVARDESC g_aArgLogFlags[] =
728{
729 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
730 { 1, 1, DBGCVAR_CAT_STRING, 0, "flags", "Flag modifier string (quote it!)." }
731};
732
733
734/** 'm' argument. */
735static const DBGCVARDESC g_aArgMemoryInfo[] =
736{
737 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
738 { 1, 1, DBGCVAR_CAT_POINTER, 0, "address", "Pointer to obtain info about." },
739};
740
741
742/** 'r' arguments. */
743static const DBGCVARDESC g_aArgReg[] =
744{
745 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
746 { 0, 1, DBGCVAR_CAT_SYMBOL, 0, "register", "Register to show or set." },
747 { 0, 1, DBGCVAR_CAT_NUMBER_NO_RANGE, DBGCVD_FLAGS_DEP_PREV, "value", "New register value." },
748};
749
750
751/** 's' arguments. */
752static const DBGCVARDESC g_aArgSource[] =
753{
754 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
755 { 0, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address where to start looking for source lines." },
756};
757
758
759/** 'set' arguments */
760static const DBGCVARDESC g_aArgSet[] =
761{
762 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
763 { 1, 1, DBGCVAR_CAT_STRING, 0, "var", "Variable name." },
764 { 1, 1, DBGCVAR_CAT_ANY, 0, "value", "Value to assign to the variable." },
765};
766
767
768/** 'u' arguments. */
769static const DBGCVARDESC g_aArgDisasm[] =
770{
771 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
772 { 0, 1, DBGCVAR_CAT_POINTER, 0, "address", "Address where to start disassembling." },
773};
774
775
776
777
778
779/** Command descriptors. */
780static const DBGCCMD g_aCmds[] =
781{
782 /* pszCmd, cArgsMin, cArgsMax, paArgDescs, cArgDescs, pResultDesc, fFlags, pfnHandler pszSyntax, ....pszDescription */
783 { "ba", 3, 6, &g_aArgBrkAcc[0], ELEMENTS(g_aArgBrkAcc), NULL, 0, dbgcCmdBrkAccess, "<access> <size> <address> [passes [max passes]] [cmds]",
784 "Sets a data access breakpoint." },
785 { "bc", 1, ~0, &g_aArgBrks[0], ELEMENTS(g_aArgBrks), NULL, 0, dbgcCmdBrkClear, "all | <bp#> [bp# []]", "Enabled a set of breakpoints." },
786 { "bd", 1, ~0, &g_aArgBrks[0], ELEMENTS(g_aArgBrks), NULL, 0, dbgcCmdBrkDisable, "all | <bp#> [bp# []]", "Disables a set of breakpoints." },
787 { "be", 1, ~0, &g_aArgBrks[0], ELEMENTS(g_aArgBrks), NULL, 0, dbgcCmdBrkEnable, "all | <bp#> [bp# []]", "Enabled a set of breakpoints." },
788 { "bl", 0, 0, NULL, 0, NULL, 0, dbgcCmdBrkList, "", "Lists all the breakpoints." },
789 { "bp", 1, 4, &g_aArgBrkSet[0], ELEMENTS(g_aArgBrkSet), NULL, 0, dbgcCmdBrkSet, "<address> [passes [max passes]] [cmds]",
790 "Sets a breakpoint (int 3)." },
791 { "br", 1, 4, &g_aArgBrkREM[0], ELEMENTS(g_aArgBrkREM), NULL, 0, dbgcCmdBrkREM, "<address> [passes [max passes]] [cmds]",
792 "Sets a recompiler specific breakpoint." },
793 { "bye", 0, 0, NULL, 0, NULL, 0, dbgcCmdQuit, "", "Exits the debugger." },
794 { "d", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory using last element size." },
795 { "da", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory as ascii string." },
796 { "db", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory in bytes." },
797 { "dd", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory in double words." },
798 { "dg", 0, ~0, &g_aArgDumpDT[0], ELEMENTS(g_aArgDumpDT), NULL, 0, dbgcCmdDumpDT, "[sel [..]]", "Dump the global descriptor table (GDT)." },
799 { "dga", 0, ~0, &g_aArgDumpDT[0], ELEMENTS(g_aArgDumpDT), NULL, 0, dbgcCmdDumpDT, "[sel [..]]", "Dump the global descriptor table (GDT) including not-present entries." },
800 { "di", 0, ~0, &g_aArgDumpIDT[0], ELEMENTS(g_aArgDumpIDT), NULL, 0, dbgcCmdDumpIDT, "[int [..]]", "Dump the interrupt descriptor table (IDT)." },
801 { "dia", 0, ~0, &g_aArgDumpIDT[0], ELEMENTS(g_aArgDumpIDT), NULL, 0, dbgcCmdDumpIDT, "[int [..]]", "Dump the interrupt descriptor table (IDT) including not-present entries." },
802 { "dl", 0, ~0, &g_aArgDumpDT[0], ELEMENTS(g_aArgDumpDT), NULL, 0, dbgcCmdDumpDT, "[sel [..]]", "Dump the local descriptor table (LDT)." },
803 { "dla", 0, ~0, &g_aArgDumpDT[0], ELEMENTS(g_aArgDumpDT), NULL, 0, dbgcCmdDumpDT, "[sel [..]]", "Dump the local descriptor table (LDT) including not-present entries." },
804 { "dpd", 0, 1, &g_aArgDumpPD[0], ELEMENTS(g_aArgDumpPD), NULL, 0, dbgcCmdDumpPageDir, "[addr] [index]", "Dumps page directory entries of the default context." },
805 { "dpda", 0, 1, &g_aArgDumpPDAddr[0],ELEMENTS(g_aArgDumpPDAddr),NULL, 0, dbgcCmdDumpPageDir, "[addr]", "Dumps specified page directory." },
806 { "dpdb", 1, 1, &g_aArgDumpPD[0], ELEMENTS(g_aArgDumpPD), NULL, 0, dbgcCmdDumpPageDirBoth, "[addr] [index]", "Dumps page directory entries of the guest and the hypervisor. " },
807 { "dpdg", 0, 1, &g_aArgDumpPD[0], ELEMENTS(g_aArgDumpPD), NULL, 0, dbgcCmdDumpPageDir, "[addr] [index]", "Dumps page directory entries of the guest." },
808 { "dpdh", 0, 1, &g_aArgDumpPD[0], ELEMENTS(g_aArgDumpPD), NULL, 0, dbgcCmdDumpPageDir, "[addr] [index]", "Dumps page directory entries of the hypervisor. " },
809 { "dpt", 1, 1, &g_aArgDumpPT[0], ELEMENTS(g_aArgDumpPT), NULL, 0, dbgcCmdDumpPageTable,"<addr>", "Dumps page table entries of the default context." },
810 { "dpta", 1, 1, &g_aArgDumpPTAddr[0],ELEMENTS(g_aArgDumpPTAddr), NULL, 0, dbgcCmdDumpPageTable,"<addr>", "Dumps specified page table." },
811 { "dptb", 1, 1, &g_aArgDumpPT[0], ELEMENTS(g_aArgDumpPT), NULL, 0, dbgcCmdDumpPageTableBoth,"<addr>", "Dumps page table entries of the guest and the hypervisor." },
812 { "dptg", 1, 1, &g_aArgDumpPT[0], ELEMENTS(g_aArgDumpPT), NULL, 0, dbgcCmdDumpPageTable,"<addr>", "Dumps page table entries of the guest." },
813 { "dpth", 1, 1, &g_aArgDumpPT[0], ELEMENTS(g_aArgDumpPT), NULL, 0, dbgcCmdDumpPageTable,"<addr>", "Dumps page table entries of the hypervisor." },
814 { "dq", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory in quad words." },
815 { "dt", 0, 1, &g_aArgDumpTSS[0], ELEMENTS(g_aArgDumpTSS), NULL, 0, dbgcCmdDumpTSS, "[tss|tss:ign|addr]", "Dump the task state segment (TSS)." },
816 { "dw", 0, 1, &g_aArgDumpMem[0], ELEMENTS(g_aArgDumpMem), NULL, 0, dbgcCmdDumpMem, "[addr]", "Dump memory in words." },
817 { "echo", 1, ~0, &g_aArgMultiStr[0], ELEMENTS(g_aArgMultiStr), NULL, 0, dbgcCmdEcho, "<str1> [str2..[strN]]", "Displays the strings separated by one blank space and the last one followed by a newline." },
818 { "exit", 0, 0, NULL, 0, NULL, 0, dbgcCmdQuit, "", "Exits the debugger." },
819 { "format", 1, 1, &g_aArgAny[0], ELEMENTS(g_aArgAny), NULL, 0, dbgcCmdFormat, "", "Evaluates an expression and formats it." },
820 { "g", 0, 0, NULL, 0, NULL, 0, dbgcCmdGo, "", "Continue execution." },
821 { "harakiri", 0, 0, NULL, 0, NULL, 0, dbgcCmdHarakiri, "", "Kills debugger process." },
822 { "help", 0, ~0, &g_aArgHelp[0], ELEMENTS(g_aArgHelp), NULL, 0, dbgcCmdHelp, "[cmd/op [..]]", "Display help. For help about info items try 'info help'." },
823 { "info", 1, 2, &g_aArgInfo[0], ELEMENTS(g_aArgInfo), NULL, 0, dbgcCmdInfo, "<info> [args]", "Display info register in the DBGF. For a list of info items try 'info help'." },
824 { "k", 0, 0, NULL, 0, NULL, 0, dbgcCmdStack, "", "Callstack." },
825 { "kg", 0, 0, NULL, 0, NULL, 0, dbgcCmdStack, "", "Callstack - guest." },
826 { "kh", 0, 0, NULL, 0, NULL, 0, dbgcCmdStack, "", "Callstack - hypervisor." },
827 { "ln", 0, ~0, &g_aArgListNear[0], ELEMENTS(g_aArgListNear), &g_RetListNear, 0, dbgcCmdListNear, "[addr/sym [..]]", "List symbols near to the address. Default address is CS:EIP." },
828 { "loadsyms", 1, 5, &g_aArgLoadSyms[0], ELEMENTS(g_aArgLoadSyms), NULL, 0, dbgcCmdLoadSyms, "<filename> [delta] [module] [module address]", "Loads symbols from a text file. Optionally giving a delta and a module." },
829 { "loadvars", 1, 1, &g_aArgFilename[0], ELEMENTS(g_aArgFilename), NULL, 0, dbgcCmdLoadVars, "<filename>", "Load variables from file. One per line, same as the args to the set command." },
830 { "log", 1, 1, &g_aArgLog[0], ELEMENTS(g_aArgLog), NULL, 0, dbgcCmdLog, "<group string>", "Modifies the logging group settings (VBOX_LOG)" },
831 { "logdest", 1, 1, &g_aArgLogDest[0], ELEMENTS(g_aArgLogDest), NULL, 0, dbgcCmdLogDest, "<dest string>", "Modifies the logging destination (VBOX_LOG_DEST)." },
832 { "logflags", 1, 1, &g_aArgLogFlags[0], ELEMENTS(g_aArgLogFlags), NULL, 0, dbgcCmdLogFlags, "<flags string>", "Modifies the logging flags (VBOX_LOG_FLAGS)." },
833 { "m", 1, 1, &g_aArgMemoryInfo[0],ELEMENTS(g_aArgMemoryInfo),NULL, 0, dbgcCmdMemoryInfo, "<addr>", "Display information about that piece of memory." },
834 { "quit", 0, 0, NULL, 0, NULL, 0, dbgcCmdQuit, "", "Exits the debugger." },
835 { "r", 0, 2, &g_aArgReg[0], ELEMENTS(g_aArgReg), NULL, 0, dbgcCmdReg, "[reg [newval]]", "Show or set register(s) - active reg set." },
836 { "rg", 0, 2, &g_aArgReg[0], ELEMENTS(g_aArgReg), NULL, 0, dbgcCmdRegGuest, "[reg [newval]]", "Show or set register(s) - guest reg set." },
837 { "rh", 0, 2, &g_aArgReg[0], ELEMENTS(g_aArgReg), NULL, 0, dbgcCmdRegHyper, "[reg [newval]]", "Show or set register(s) - hypervisor reg set." },
838 { "rt", 0, 0, NULL, 0, NULL, 0, dbgcCmdRegTerse, "", "Toggles terse / verbose register info." },
839 { "runscript", 1, 1, &g_aArgFilename[0], ELEMENTS(g_aArgFilename), NULL, 0, dbgcCmdRunScript, "<filename>", "Runs the command listed in the script. Lines starting with '#' (after removing blanks) are comment. blank lines are ignored. Stops on failure." },
840 { "s", 0, 1, &g_aArgSource[0], ELEMENTS(g_aArgSource), NULL, 0, dbgcCmdSource, "[addr]", "Source." },
841 { "set", 2, 2, &g_aArgSet[0], ELEMENTS(g_aArgSet), NULL, 0, dbgcCmdSet, "<var> <value>", "Sets a global variable." },
842 { "showvars", 0, 0, NULL, 0, NULL, 0, dbgcCmdShowVars, "", "List all the defined variables." },
843 { "stop", 0, 0, NULL, 0, NULL, 0, dbgcCmdStop, "", "Stop execution." },
844 { "t", 0, 0, NULL, 0, NULL, 0, dbgcCmdTrace, "", "Instruction trace (step into)." },
845 { "u", 0, 1, &g_aArgDisasm[0], ELEMENTS(g_aArgDisasm), NULL, 0, dbgcCmdDisasm, "[addr]", "Disassemble." },
846 { "unset", 1, ~0, &g_aArgMultiStr[0], ELEMENTS(g_aArgMultiStr), NULL, 0, dbgcCmdUnset, "<var1> [var1..[varN]]", "Unsets (delete) one or more global variables." },
847};
848
849
850/** Operators. */
851static const DBGCOP g_aOps[] =
852{
853 /* szName is initialized as a 4 char array because of M$C elsewise optimizing it away in /Ox mode (the 'const char' vs 'char' problem). */
854 /* szName, cchName, fBinary, iPrecedence, pfnHandlerUnary, pfnHandlerBitwise */
855 { {'-'}, 1, false, 1, dbgcOpMinus, NULL, "Unary minus." },
856 { {'+'}, 1, false, 1, dbgcOpPluss, NULL, "Unary pluss." },
857 { {'!'}, 1, false, 1, dbgcOpBooleanNot, NULL, "Boolean not." },
858 { {'~'}, 1, false, 1, dbgcOpBitwiseNot, NULL, "Bitwise complement." },
859 { {':'}, 1, true, 2, NULL, dbgcOpAddrFar, "Far pointer." },
860 { {'%'}, 1, false, 3, dbgcOpAddrFlat, NULL, "Flat address." },
861 { {'%','%'}, 2, false, 3, dbgcOpAddrPhys, NULL, "Physical address." },
862 { {'#'}, 1, false, 3, dbgcOpAddrHost, NULL, "Flat host address." },
863 { {'#','%','%'}, 3, false, 3, dbgcOpAddrHostPhys, NULL, "Physical host address." },
864 { {'$'}, 1, false, 3, dbgcOpVar, NULL, "Reference a variable." },
865 { {'*'}, 1, true, 10, NULL, dbgcOpMult, "Multiplication." },
866 { {'/'}, 1, true, 11, NULL, dbgcOpDiv, "Division." },
867 { {'%'}, 1, true, 12, NULL, dbgcOpMod, "Modulus." },
868 { {'+'}, 1, true, 13, NULL, dbgcOpAdd, "Addition." },
869 { {'-'}, 1, true, 14, NULL, dbgcOpSub, "Subtraction." },
870 { {'<','<'}, 2, true, 15, NULL, dbgcOpBitwiseShiftLeft, "Bitwise left shift." },
871 { {'>','>'}, 2, true, 16, NULL, dbgcOpBitwiseShiftRight, "Bitwise right shift." },
872 { {'&'}, 1, true, 17, NULL, dbgcOpBitwiseAnd, "Bitwise and." },
873 { {'^'}, 1, true, 18, NULL, dbgcOpBitwiseXor, "Bitwise exclusiv or." },
874 { {'|'}, 1, true, 19, NULL, dbgcOpBitwiseOr, "Bitwise inclusive or." },
875 { {'&','&'}, 2, true, 20, NULL, dbgcOpBooleanAnd, "Boolean and." },
876 { {'|','|'}, 2, true, 21, NULL, dbgcOpBooleanOr, "Boolean or." },
877 { {'L'}, 1, true, 22, NULL, dbgcOpRangeLength, "Range elements." },
878 { {'L','B'}, 2, true, 23, NULL, dbgcOpRangeLengthBytes, "Range bytes." },
879 { {'T'}, 1, true, 24, NULL, dbgcOpRangeTo, "Range to." }
880};
881
882/** Bitmap where set bits indicates the characters the may start an operator name. */
883static uint32_t g_bmOperatorChars[256 / (4*8)];
884
885/** Register symbol uUser value.
886 * @{
887 */
888/** If set the register set is the hypervisor and not the guest one. */
889#define SYMREG_FLAGS_HYPER BIT(20)
890/** If set a far conversion of the value will use the high 16 bit for the selector.
891 * If clear the low 16 bit will be used. */
892#define SYMREG_FLAGS_HIGH_SEL BIT(21)
893/** The shift value to calc the size of a register symbol from the uUser value. */
894#define SYMREG_SIZE_SHIFT (24)
895/** Get the offset */
896#define SYMREG_OFFSET(uUser) (uUser & ((1 << 20) - 1))
897/** Get the size. */
898#define SYMREG_SIZE(uUser) ((uUser >> SYMREG_SIZE_SHIFT) & 0xff)
899/** 1 byte. */
900#define SYMREG_SIZE_1 ( 1 << SYMREG_SIZE_SHIFT)
901/** 2 byte. */
902#define SYMREG_SIZE_2 ( 2 << SYMREG_SIZE_SHIFT)
903/** 4 byte. */
904#define SYMREG_SIZE_4 ( 4 << SYMREG_SIZE_SHIFT)
905/** 6 byte. */
906#define SYMREG_SIZE_6 ( 6 << SYMREG_SIZE_SHIFT)
907/** 8 byte. */
908#define SYMREG_SIZE_8 ( 8 << SYMREG_SIZE_SHIFT)
909/** 12 byte. */
910#define SYMREG_SIZE_12 (12 << SYMREG_SIZE_SHIFT)
911/** 16 byte. */
912#define SYMREG_SIZE_16 (16 << SYMREG_SIZE_SHIFT)
913/** @} */
914
915/** Builtin Symbols.
916 * ASSUMES little endian register representation!
917 */
918static const DBGCSYM g_aSyms[] =
919{
920 { "eax", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_4 },
921 { "ax", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_2 },
922 { "al", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_1 },
923 { "ah", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) + 1 | SYMREG_SIZE_1 },
924
925 { "ebx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_4 },
926 { "bx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_2 },
927 { "bl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_1 },
928 { "bh", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) + 1 | SYMREG_SIZE_1 },
929
930 { "ecx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_4 },
931 { "cx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_2 },
932 { "cl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_1 },
933 { "ch", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) + 1 | SYMREG_SIZE_1 },
934
935 { "edx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_4 },
936 { "dx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_2 },
937 { "dl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_1 },
938 { "dh", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) + 1 | SYMREG_SIZE_1 },
939
940 { "edi", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edi) | SYMREG_SIZE_4 },
941 { "di", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edi) | SYMREG_SIZE_2 },
942
943 { "esi", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esi) | SYMREG_SIZE_4 },
944 { "si", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esi) | SYMREG_SIZE_2 },
945
946 { "ebp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebp) | SYMREG_SIZE_4 },
947 { "bp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebp) | SYMREG_SIZE_2 },
948
949 { "esp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esp) | SYMREG_SIZE_4 },
950 { "sp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esp) | SYMREG_SIZE_2 },
951
952 { "eip", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eip) | SYMREG_SIZE_4 },
953 { "ip", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eip) | SYMREG_SIZE_2 },
954
955 { "efl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_4 },
956 { "eflags", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_4 },
957 { "fl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_2 },
958 { "flags", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_2 },
959
960 { "cs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cs) | SYMREG_SIZE_2 },
961 { "ds", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ds) | SYMREG_SIZE_2 },
962 { "es", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, es) | SYMREG_SIZE_2 },
963 { "fs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, fs) | SYMREG_SIZE_2 },
964 { "gs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gs) | SYMREG_SIZE_2 },
965 { "ss", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ss) | SYMREG_SIZE_2 },
966
967 { "cr0", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr0) | SYMREG_SIZE_4 },
968 { "cr2", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr2) | SYMREG_SIZE_4 },
969 { "cr3", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr3) | SYMREG_SIZE_4 },
970 { "cr4", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr4) | SYMREG_SIZE_4 },
971
972 { "tr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, tr) | SYMREG_SIZE_2 },
973 { "ldtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ldtr) | SYMREG_SIZE_2 },
974
975 { "gdtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr) | SYMREG_SIZE_6 },
976 { "gdtr.limit", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr.cbGdt)| SYMREG_SIZE_2 },
977 { "gdtr.base", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr.pGdt)| SYMREG_SIZE_4 },
978
979 { "idtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr) | SYMREG_SIZE_6 },
980 { "idtr.limit", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr.cbIdt)| SYMREG_SIZE_2 },
981 { "idtr.base", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr.pIdt)| SYMREG_SIZE_4 },
982
983 /* hypervisor */
984
985 {".eax", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
986 {".ax", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
987 {".al", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
988 {".ah", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eax) + 1 | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
989
990 {".ebx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
991 {".bx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
992 {".bl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
993 {".bh", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebx) + 1 | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
994
995 {".ecx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
996 {".cx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
997 {".cl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
998 {".ch", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ecx) + 1 | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
999
1000 {".edx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1001 {".dx", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1002 {".dl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
1003 {".dh", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edx) + 1 | SYMREG_SIZE_1 | SYMREG_FLAGS_HYPER },
1004
1005 {".edi", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edi) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1006 {".di", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, edi) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1007
1008 {".esi", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esi) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1009 {".si", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esi) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1010
1011 {".ebp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebp) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1012 {".bp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ebp) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1013
1014 {".esp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esp) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1015 {".sp", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, esp) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1016
1017 {".eip", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eip) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1018 {".ip", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eip) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1019
1020 {".efl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1021 {".eflags", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1022 {".fl", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1023 {".flags", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, eflags) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1024
1025 {".cs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cs) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1026 {".ds", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ds) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1027 {".es", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, es) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1028 {".fs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, fs) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1029 {".gs", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gs) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1030 {".ss", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ss) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1031
1032 {".cr0", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr0) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1033 {".cr2", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr2) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1034 {".cr3", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr3) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1035 {".cr4", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, cr4) | SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1036
1037 {".tr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, tr) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1038 {".ldtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, ldtr) | SYMREG_SIZE_2 | SYMREG_FLAGS_HYPER },
1039
1040 {".gdtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr) | SYMREG_SIZE_6 | SYMREG_FLAGS_HYPER },
1041 {".gdtr.limit", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr.cbGdt)| SYMREG_SIZE_2| SYMREG_FLAGS_HYPER },
1042 {".gdtr.base", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, gdtr.pGdt)| SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1043
1044 {".idtr", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr) | SYMREG_SIZE_6 | SYMREG_FLAGS_HYPER },
1045 {".idtr.limit", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr.cbIdt)| SYMREG_SIZE_2| SYMREG_FLAGS_HYPER },
1046 {".idtr.base", dbgcSymGetReg, dbgcSymSetReg, offsetof(CPUMCTX, idtr.pIdt)| SYMREG_SIZE_4 | SYMREG_FLAGS_HYPER },
1047
1048};
1049
1050
1051/**
1052 * Prints full command help.
1053 */
1054static int dbgcPrintHelp(PDBGCCMDHLP pCmdHlp, PCDBGCCMD pCmd, bool fExternal)
1055{
1056 int rc;
1057
1058 /* the command */
1059 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1060 "%s%-*s %-30s %s",
1061 fExternal ? "." : "",
1062 fExternal ? 10 : 11,
1063 pCmd->pszCmd,
1064 pCmd->pszSyntax,
1065 pCmd->pszDescription);
1066 if (!pCmd->cArgsMin && pCmd->cArgsMin == pCmd->cArgsMax)
1067 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <no args>\n");
1068 else if (pCmd->cArgsMin == pCmd->cArgsMax)
1069 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <%u args>\n", pCmd->cArgsMin);
1070 else if (pCmd->cArgsMax == ~0U)
1071 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <%u+ args>\n", pCmd->cArgsMin);
1072 else
1073 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <%u to %u args>\n", pCmd->cArgsMin, pCmd->cArgsMax);
1074
1075 /* argument descriptions. */
1076 for (unsigned i = 0; i < pCmd->cArgDescs; i++)
1077 {
1078 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1079 " %-12s %s",
1080 pCmd->paArgDescs[i].pszName,
1081 pCmd->paArgDescs[i].pszDescription);
1082 if (!pCmd->paArgDescs[i].cTimesMin)
1083 {
1084 if (pCmd->paArgDescs[i].cTimesMax == ~0U)
1085 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <optional+>\n");
1086 else
1087 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <optional-%u>\n", pCmd->paArgDescs[i].cTimesMax);
1088 }
1089 else
1090 {
1091 if (pCmd->paArgDescs[i].cTimesMax == ~0U)
1092 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <%u+>\n", pCmd->paArgDescs[i].cTimesMin);
1093 else
1094 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " <%u-%u>\n", pCmd->paArgDescs[i].cTimesMin, pCmd->paArgDescs[i].cTimesMax);
1095 }
1096 }
1097 return rc;
1098}
1099
1100/**
1101 * The 'help' command.
1102 *
1103 * @returns VBox status.
1104 * @param pCmd Pointer to the command descriptor (as registered).
1105 * @param pCmdHlp Pointer to command helper functions.
1106 * @param pVM Pointer to the current VM (if any).
1107 * @param paArgs Pointer to (readonly) array of arguments.
1108 * @param cArgs Number of arguments in the array.
1109 */
1110static DECLCALLBACK(int) dbgcCmdHelp(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
1111{
1112 int rc = VINF_SUCCESS;
1113 unsigned i;
1114 if (!cArgs)
1115 {
1116 /*
1117 * All the stuff.
1118 */
1119 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1120 "VirtualBox Debugger\n"
1121 "-------------------\n"
1122 "\n"
1123 "Commands and Functions:\n");
1124 for (i = 0; i < ELEMENTS(g_aCmds); i++)
1125 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1126 "%-11s %-30s %s\n",
1127 g_aCmds[i].pszCmd,
1128 g_aCmds[i].pszSyntax,
1129 g_aCmds[i].pszDescription);
1130
1131 if (g_pExtCmdsHead)
1132 {
1133 DBGCEXTCMDS_LOCK_RD();
1134 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1135 "\n"
1136 "External Commands and Functions:\n");
1137 for (PDBGCEXTCMDS pExtCmd = g_pExtCmdsHead; pExtCmd; pExtCmd = pExtCmd->pNext)
1138 for (i = 0; i < pExtCmd->cCmds; i++)
1139 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1140 ".%-10s %-30s %s\n",
1141 pExtCmd->paCmds[i].pszCmd,
1142 pExtCmd->paCmds[i].pszSyntax,
1143 pExtCmd->paCmds[i].pszDescription);
1144 DBGCEXTCMDS_UNLOCK_RD();
1145 }
1146
1147 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1148 "\n"
1149 "Operators:\n");
1150 unsigned iPrecedence = 0;
1151 unsigned cLeft = ELEMENTS(g_aOps);
1152 while (cLeft > 0)
1153 {
1154 for (i = 0; i < ELEMENTS(g_aOps); i++)
1155 if (g_aOps[i].iPrecedence == iPrecedence)
1156 {
1157 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1158 "%-10s %s %s\n",
1159 g_aOps[i].szName,
1160 g_aOps[i].fBinary ? "Binary" : "Unary ",
1161 g_aOps[i].pszDescription);
1162 cLeft--;
1163 }
1164 iPrecedence++;
1165 }
1166 }
1167 else
1168 {
1169 /*
1170 * Search for the arguments (strings).
1171 */
1172 for (unsigned iArg = 0; iArg < cArgs; iArg++)
1173 {
1174 Assert(paArgs[iArg].enmType == DBGCVAR_TYPE_STRING);
1175
1176 /* lookup in the command list */
1177 bool fFound = false;
1178 for (i = 0; i < ELEMENTS(g_aCmds); i++)
1179 if (!strcmp(g_aCmds[i].pszCmd, paArgs[iArg].u.pszString))
1180 {
1181 rc = dbgcPrintHelp(pCmdHlp, &g_aCmds[i], false);
1182 fFound = true;
1183 break;
1184 }
1185
1186 /* external commands */
1187 if ( !fFound
1188 && g_pExtCmdsHead
1189 && paArgs[iArg].u.pszString[0] == '.')
1190 {
1191 DBGCEXTCMDS_LOCK_RD();
1192 for (PDBGCEXTCMDS pExtCmd = g_pExtCmdsHead; pExtCmd; pExtCmd = pExtCmd->pNext)
1193 for (i = 0; i < pExtCmd->cCmds; i++)
1194 if (!strcmp(pExtCmd->paCmds[i].pszCmd, paArgs[iArg].u.pszString + 1))
1195 {
1196 rc = dbgcPrintHelp(pCmdHlp, &g_aCmds[i], true);
1197 fFound = true;
1198 break;
1199 }
1200 DBGCEXTCMDS_UNLOCK_RD();
1201 }
1202
1203 /* operators */
1204 if (!fFound && strlen(paArgs[iArg].u.pszString) < sizeof(g_aOps[i].szName))
1205 {
1206 for (i = 0; i < ELEMENTS(g_aOps); i++)
1207 if (!strcmp(g_aOps[i].szName, paArgs[iArg].u.pszString))
1208 {
1209 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1210 "%-10s %s %s\n",
1211 g_aOps[i].szName,
1212 g_aOps[i].fBinary ? "Binary" : "Unary ",
1213 g_aOps[i].pszDescription);
1214 fFound = true;
1215 break;
1216 }
1217 }
1218
1219 /* found? */
1220 if (!fFound)
1221 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
1222 "error: '%s' was not found!\n",
1223 paArgs[iArg].u.pszString);
1224 } /* foreach argument */
1225 }
1226
1227 NOREF(pCmd);
1228 NOREF(pVM);
1229 NOREF(pResult);
1230 return rc;
1231}
1232
1233
1234/**
1235 * The 'quit', 'exit' and 'bye' commands.
1236 *
1237 * @returns VBox status.
1238 * @param pCmd Pointer to the command descriptor (as registered).
1239 * @param pCmdHlp Pointer to command helper functions.
1240 * @param pVM Pointer to the current VM (if any).
1241 * @param paArgs Pointer to (readonly) array of arguments.
1242 * @param cArgs Number of arguments in the array.
1243 */
1244static DECLCALLBACK(int) dbgcCmdQuit(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
1245{
1246 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Quitting console...\n");
1247 NOREF(pCmd);
1248 NOREF(pVM);
1249 NOREF(paArgs);
1250 NOREF(cArgs);
1251 NOREF(pResult);
1252 return VERR_DBGC_QUIT;
1253}
1254
1255
1256/**
1257 * The 'go' command.
1258 *
1259 * @returns VBox status.
1260 * @param pCmd Pointer to the command descriptor (as registered).
1261 * @param pCmdHlp Pointer to command helper functions.
1262 * @param pVM Pointer to the current VM (if any).
1263 * @param paArgs Pointer to (readonly) array of arguments.
1264 * @param cArgs Number of arguments in the array.
1265 */
1266static DECLCALLBACK(int) dbgcCmdGo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
1267{
1268 /*
1269 * Check if the VM is halted or not before trying to resume it.
1270 */
1271 if (!DBGFR3IsHalted(pVM))
1272 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "warning: The VM is already running...\n");
1273 else
1274 {
1275 int rc = DBGFR3Resume(pVM);
1276 if (VBOX_FAILURE(rc))
1277 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Executing DBGFR3Resume().");
1278 }
1279
1280 NOREF(pCmd);
1281 NOREF(paArgs);
1282 NOREF(cArgs);
1283 NOREF(pResult);
1284 return 0;
1285}
1286
1287/**
1288 * The 'stop' command.
1289 *
1290 * @returns VBox status.
1291 * @param pCmd Pointer to the command descriptor (as registered).
1292 * @param pCmdHlp Pointer to command helper functions.
1293 * @param pVM Pointer to the current VM (if any).
1294 * @param paArgs Pointer to (readonly) array of arguments.
1295 * @param cArgs Number of arguments in the array.
1296 */
1297static DECLCALLBACK(int) dbgcCmdStop(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
1298{
1299 /*
1300 * Check if the VM is halted or not before trying to halt it.
1301 */
1302 int rc;
1303 if (DBGFR3IsHalted(pVM))
1304 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "warning: The VM is already halted...\n");
1305 else
1306 {
1307 rc = DBGFR3Halt(pVM);
1308 if (VBOX_SUCCESS(rc))
1309 rc = VWRN_DBGC_CMD_PENDING;
1310 else
1311 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Executing DBGFR3Halt().");
1312 }
1313
1314 NOREF(pCmd); NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
1315 return rc;
1316}
1317
1318
1319/**
1320 * The 'ba' command.
1321 *
1322 * @returns VBox status.
1323 * @param pCmd Pointer to the command descriptor (as registered).
1324 * @param pCmdHlp Pointer to command helper functions.
1325 * @param pVM Pointer to the current VM (if any).
1326 * @param paArgs Pointer to (readonly) array of arguments.
1327 * @param cArgs Number of arguments in the array.
1328 */
1329static DECLCALLBACK(int) dbgcCmdBrkAccess(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1330{
1331 /*
1332 * Interpret access type.
1333 */
1334 if ( !strchr("xrwi", paArgs[0].u.pszString[0])
1335 || paArgs[0].u.pszString[1])
1336 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid access type '%s' for '%s'. Valid types are 'e', 'r', 'w' and 'i'.\n",
1337 paArgs[0].u.pszString, pCmd->pszCmd);
1338 uint8_t fType = 0;
1339 switch (paArgs[0].u.pszString[0])
1340 {
1341 case 'x': fType = X86_DR7_RW_EO; break;
1342 case 'r': fType = X86_DR7_RW_RW; break;
1343 case 'w': fType = X86_DR7_RW_WO; break;
1344 case 'i': fType = X86_DR7_RW_IO; break;
1345 }
1346
1347 /*
1348 * Validate size.
1349 */
1350 if (fType == X86_DR7_RW_EO && paArgs[1].u.u64Number != 1)
1351 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid access size %RX64 for '%s'. 'x' access type requires size 1!\n",
1352 paArgs[1].u.u64Number, pCmd->pszCmd);
1353 switch (paArgs[1].u.u64Number)
1354 {
1355 case 1:
1356 case 2:
1357 case 4:
1358 break;
1359 /*case 8: - later*/
1360 default:
1361 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid access size %RX64 for '%s'. 1, 2 or 4!\n",
1362 paArgs[1].u.u64Number, pCmd->pszCmd);
1363 }
1364 uint8_t cb = (uint8_t)paArgs[1].u.u64Number;
1365
1366 /*
1367 * Convert the pointer to a DBGF address.
1368 */
1369 DBGFADDRESS Address;
1370 int rc = pCmdHlp->pfnVarToDbgfAddr(pCmdHlp, &paArgs[2], &Address);
1371 if (VBOX_FAILURE(rc))
1372 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Couldn't convert '%DV' to a DBGF address, rc=%Vrc.\n", &paArgs[2], rc);
1373
1374 /*
1375 * Pick out the optional arguments.
1376 */
1377 uint64_t iHitTrigger = 0;
1378 uint64_t iHitDisable = ~0;
1379 const char *pszCmds = NULL;
1380 unsigned iArg = 3;
1381 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1382 {
1383 iHitTrigger = paArgs[iArg].u.u64Number;
1384 iArg++;
1385 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1386 {
1387 iHitDisable = paArgs[iArg].u.u64Number;
1388 iArg++;
1389 }
1390 }
1391 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_STRING)
1392 {
1393 pszCmds = paArgs[iArg].u.pszString;
1394 iArg++;
1395 }
1396
1397 /*
1398 * Try set the breakpoint.
1399 */
1400 RTUINT iBp;
1401 rc = DBGFR3BpSetReg(pVM, &Address, iHitTrigger, iHitDisable, fType, cb, &iBp);
1402 if (VBOX_SUCCESS(rc))
1403 {
1404 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1405 rc = dbgcBpAdd(pDbgc, iBp, pszCmds);
1406 if (VBOX_SUCCESS(rc))
1407 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Set access breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1408 if (rc == VERR_DBGC_BP_EXISTS)
1409 {
1410 rc = dbgcBpUpdate(pDbgc, iBp, pszCmds);
1411 if (VBOX_SUCCESS(rc))
1412 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Updated access breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1413 }
1414 int rc2 = DBGFR3BpClear(pDbgc->pVM, iBp);
1415 AssertRC(rc2);
1416 }
1417 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Failed to set access breakpoint at %VGv, rc=%Vrc.\n", Address.FlatPtr, rc);
1418}
1419
1420
1421/**
1422 * The 'bc' command.
1423 *
1424 * @returns VBox status.
1425 * @param pCmd Pointer to the command descriptor (as registered).
1426 * @param pCmdHlp Pointer to command helper functions.
1427 * @param pVM Pointer to the current VM (if any).
1428 * @param paArgs Pointer to (readonly) array of arguments.
1429 * @param cArgs Number of arguments in the array.
1430 */
1431static DECLCALLBACK(int) dbgcCmdBrkClear(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1432{
1433 /*
1434 * Enumerate the arguments.
1435 */
1436 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1437 int rc = VINF_SUCCESS;
1438 for (unsigned iArg = 0; iArg < cArgs && VBOX_SUCCESS(rc); iArg++)
1439 {
1440 if (paArgs[iArg].enmType != DBGCVAR_TYPE_STRING)
1441 {
1442 /* one */
1443 RTUINT iBp = (RTUINT)paArgs[iArg].u.u64Number;
1444 if (iBp != paArgs[iArg].u.u64Number)
1445 {
1446 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Breakpoint id %RX64 is too large!\n", paArgs[iArg].u.u64Number);
1447 break;
1448 }
1449 int rc2 = DBGFR3BpClear(pVM, iBp);
1450 if (VBOX_FAILURE(rc2))
1451 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc2, "DBGFR3BpClear failed for breakpoint %u!\n", iBp);
1452 if (VBOX_SUCCESS(rc2) || rc2 == VERR_DBGF_BP_NOT_FOUND)
1453 dbgcBpDelete(pDbgc, iBp);
1454 }
1455 else if (!strcmp(paArgs[iArg].u.pszString, "all"))
1456 {
1457 /* all */
1458 PDBGCBP pBp = pDbgc->pFirstBp;
1459 while (pBp)
1460 {
1461 RTUINT iBp = pBp->iBp;
1462 pBp = pBp->pNext;
1463
1464 int rc2 = DBGFR3BpClear(pVM, iBp);
1465 if (VBOX_FAILURE(rc2))
1466 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc2, "DBGFR3BpClear failed for breakpoint %u!\n", iBp);
1467 if (VBOX_SUCCESS(rc2) || rc2 == VERR_DBGF_BP_NOT_FOUND)
1468 dbgcBpDelete(pDbgc, iBp);
1469 }
1470 }
1471 else
1472 {
1473 /* invalid parameter */
1474 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid argument '%s' to '%s'!\n", paArgs[iArg].u.pszString, pCmd->pszCmd);
1475 break;
1476 }
1477 }
1478 return rc;
1479}
1480
1481
1482/**
1483 * The 'bd' command.
1484 *
1485 * @returns VBox status.
1486 * @param pCmd Pointer to the command descriptor (as registered).
1487 * @param pCmdHlp Pointer to command helper functions.
1488 * @param pVM Pointer to the current VM (if any).
1489 * @param paArgs Pointer to (readonly) array of arguments.
1490 * @param cArgs Number of arguments in the array.
1491 */
1492static DECLCALLBACK(int) dbgcCmdBrkDisable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1493{
1494 /*
1495 * Enumerate the arguments.
1496 */
1497 int rc = VINF_SUCCESS;
1498 for (unsigned iArg = 0; iArg < cArgs && VBOX_SUCCESS(rc); iArg++)
1499 {
1500 if (paArgs[iArg].enmType != DBGCVAR_TYPE_STRING)
1501 {
1502 /* one */
1503 RTUINT iBp = (RTUINT)paArgs[iArg].u.u64Number;
1504 if (iBp != paArgs[iArg].u.u64Number)
1505 {
1506 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Breakpoint id %RX64 is too large!\n", paArgs[iArg].u.u64Number);
1507 break;
1508 }
1509 rc = DBGFR3BpDisable(pVM, iBp);
1510 if (VBOX_FAILURE(rc))
1511 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3BpDisable failed for breakpoint %u!\n", iBp);
1512 }
1513 else if (!strcmp(paArgs[iArg].u.pszString, "all"))
1514 {
1515 /* all */
1516 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1517 for (PDBGCBP pBp = pDbgc->pFirstBp; pBp; pBp = pBp->pNext)
1518 {
1519 rc = DBGFR3BpDisable(pVM, pBp->iBp);
1520 if (VBOX_FAILURE(rc))
1521 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3BpDisable failed for breakpoint %u!\n", pBp->iBp);
1522 }
1523 }
1524 else
1525 {
1526 /* invalid parameter */
1527 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid argument '%s' to '%s'!\n", paArgs[iArg].u.pszString, pCmd->pszCmd);
1528 break;
1529 }
1530 }
1531 return rc;
1532}
1533
1534
1535/**
1536 * The 'be' command.
1537 *
1538 * @returns VBox status.
1539 * @param pCmd Pointer to the command descriptor (as registered).
1540 * @param pCmdHlp Pointer to command helper functions.
1541 * @param pVM Pointer to the current VM (if any).
1542 * @param paArgs Pointer to (readonly) array of arguments.
1543 * @param cArgs Number of arguments in the array.
1544 */
1545static DECLCALLBACK(int) dbgcCmdBrkEnable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1546{
1547 /*
1548 * Enumerate the arguments.
1549 */
1550 int rc = VINF_SUCCESS;
1551 for (unsigned iArg = 0; iArg < cArgs && VBOX_SUCCESS(rc); iArg++)
1552 {
1553 if (paArgs[iArg].enmType != DBGCVAR_TYPE_STRING)
1554 {
1555 /* one */
1556 RTUINT iBp = (RTUINT)paArgs[iArg].u.u64Number;
1557 if (iBp != paArgs[iArg].u.u64Number)
1558 {
1559 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Breakpoint id %RX64 is too large!\n", paArgs[iArg].u.u64Number);
1560 break;
1561 }
1562 rc = DBGFR3BpEnable(pVM, iBp);
1563 if (VBOX_FAILURE(rc))
1564 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3BpEnable failed for breakpoint %u!\n", iBp);
1565 }
1566 else if (!strcmp(paArgs[iArg].u.pszString, "all"))
1567 {
1568 /* all */
1569 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1570 for (PDBGCBP pBp = pDbgc->pFirstBp; pBp; pBp = pBp->pNext)
1571 {
1572 rc = DBGFR3BpEnable(pVM, pBp->iBp);
1573 if (VBOX_FAILURE(rc))
1574 rc = pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3BpEnable failed for breakpoint %u!\n", pBp->iBp);
1575 }
1576 }
1577 else
1578 {
1579 /* invalid parameter */
1580 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid argument '%s' to '%s'!\n", paArgs[iArg].u.pszString, pCmd->pszCmd);
1581 break;
1582 }
1583 }
1584 return rc;
1585}
1586
1587
1588/**
1589 * Breakpoint enumeration callback function.
1590 *
1591 * @returns VBox status code. Any failure will stop the enumeration.
1592 * @param pVM The VM handle.
1593 * @param pvUser The user argument.
1594 * @param pBp Pointer to the breakpoint information. (readonly)
1595 */
1596static DECLCALLBACK(int) dbgcEnumBreakpointsCallback(PVM pVM, void *pvUser, PCDBGFBP pBp)
1597{
1598 PDBGC pDbgc = (PDBGC)pvUser;
1599 PDBGCBP pDbgcBp = dbgcBpGet(pDbgc, pBp->iBp);
1600
1601 /*
1602 * BP type and size.
1603 */
1604 char chType;
1605 char cb = 1;
1606 switch (pBp->enmType)
1607 {
1608 case DBGFBPTYPE_INT3:
1609 chType = 'p';
1610 break;
1611 case DBGFBPTYPE_REG:
1612 switch (pBp->u.Reg.fType)
1613 {
1614 case X86_DR7_RW_EO: chType = 'x'; break;
1615 case X86_DR7_RW_WO: chType = 'w'; break;
1616 case X86_DR7_RW_IO: chType = 'i'; break;
1617 case X86_DR7_RW_RW: chType = 'r'; break;
1618 default: chType = '?'; break;
1619
1620 }
1621 cb = pBp->u.Reg.cb;
1622 break;
1623 case DBGFBPTYPE_REM:
1624 chType = 'r';
1625 break;
1626 default:
1627 chType = '?';
1628 break;
1629 }
1630
1631 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%2u %c %d %c %VGv %04RX64 (%04RX64 to ",
1632 pBp->iBp, pBp->fEnabled ? 'e' : 'd', cb, chType,
1633 pBp->GCPtr, pBp->cHits, pBp->iHitTrigger);
1634 if (pBp->iHitDisable == ~(uint64_t)0)
1635 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "~0) ");
1636 else
1637 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%04RX64)");
1638
1639 /*
1640 * Try resolve the address.
1641 */
1642 DBGFSYMBOL Sym;
1643 RTGCINTPTR off;
1644 int rc = DBGFR3SymbolByAddr(pVM, pBp->GCPtr, &off, &Sym);
1645 if (VBOX_SUCCESS(rc))
1646 {
1647 if (!off)
1648 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%s", Sym.szName);
1649 else if (off > 0)
1650 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%s+%VGv", Sym.szName, off);
1651 else
1652 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%s+%VGv", Sym.szName, -off);
1653 }
1654
1655 /*
1656 * The commands.
1657 */
1658 if (pDbgcBp)
1659 {
1660 if (pDbgcBp->cchCmd)
1661 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\n cmds: '%s'\n",
1662 pDbgcBp->szCmd);
1663 else
1664 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\n");
1665 }
1666 else
1667 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, " [unknown bp]\n");
1668
1669 return VINF_SUCCESS;
1670}
1671
1672
1673/**
1674 * The 'bl' command.
1675 *
1676 * @returns VBox status.
1677 * @param pCmd Pointer to the command descriptor (as registered).
1678 * @param pCmdHlp Pointer to command helper functions.
1679 * @param pVM Pointer to the current VM (if any).
1680 * @param paArgs Pointer to (readonly) array of arguments.
1681 * @param cArgs Number of arguments in the array.
1682 */
1683static DECLCALLBACK(int) dbgcCmdBrkList(PCDBGCCMD /*pCmd*/, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR /*paArgs*/, unsigned /*cArgs*/, PDBGCVAR /*pResult*/)
1684{
1685 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1686
1687 /*
1688 * Enumerate the breakpoints.
1689 */
1690 int rc = DBGFR3BpEnum(pVM, dbgcEnumBreakpointsCallback, pDbgc);
1691 if (VBOX_FAILURE(rc))
1692 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3BpEnum failed.\n");
1693 return rc;
1694}
1695
1696
1697/**
1698 * The 'bp' command.
1699 *
1700 * @returns VBox status.
1701 * @param pCmd Pointer to the command descriptor (as registered).
1702 * @param pCmdHlp Pointer to command helper functions.
1703 * @param pVM Pointer to the current VM (if any).
1704 * @param paArgs Pointer to (readonly) array of arguments.
1705 * @param cArgs Number of arguments in the array.
1706 */
1707static DECLCALLBACK(int) dbgcCmdBrkSet(PCDBGCCMD /*pCmd*/, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1708{
1709 /*
1710 * Convert the pointer to a DBGF address.
1711 */
1712 DBGFADDRESS Address;
1713 int rc = pCmdHlp->pfnVarToDbgfAddr(pCmdHlp, &paArgs[0], &Address);
1714 if (VBOX_FAILURE(rc))
1715 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Couldn't convert '%DV' to a DBGF address, rc=%Vrc.\n", &paArgs[0], rc);
1716
1717 /*
1718 * Pick out the optional arguments.
1719 */
1720 uint64_t iHitTrigger = 0;
1721 uint64_t iHitDisable = ~0;
1722 const char *pszCmds = NULL;
1723 unsigned iArg = 1;
1724 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1725 {
1726 iHitTrigger = paArgs[iArg].u.u64Number;
1727 iArg++;
1728 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1729 {
1730 iHitDisable = paArgs[iArg].u.u64Number;
1731 iArg++;
1732 }
1733 }
1734 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_STRING)
1735 {
1736 pszCmds = paArgs[iArg].u.pszString;
1737 iArg++;
1738 }
1739
1740 /*
1741 * Try set the breakpoint.
1742 */
1743 RTUINT iBp;
1744 rc = DBGFR3BpSet(pVM, &Address, iHitTrigger, iHitDisable, &iBp);
1745 if (VBOX_SUCCESS(rc))
1746 {
1747 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1748 rc = dbgcBpAdd(pDbgc, iBp, pszCmds);
1749 if (VBOX_SUCCESS(rc))
1750 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Set breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1751 if (rc == VERR_DBGC_BP_EXISTS)
1752 {
1753 rc = dbgcBpUpdate(pDbgc, iBp, pszCmds);
1754 if (VBOX_SUCCESS(rc))
1755 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Updated breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1756 }
1757 int rc2 = DBGFR3BpClear(pDbgc->pVM, iBp);
1758 AssertRC(rc2);
1759 }
1760 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Failed to set breakpoint at %VGv, rc=%Vrc.\n", Address.FlatPtr, rc);
1761}
1762
1763
1764/**
1765 * The 'br' command.
1766 *
1767 * @returns VBox status.
1768 * @param pCmd Pointer to the command descriptor (as registered).
1769 * @param pCmdHlp Pointer to command helper functions.
1770 * @param pVM Pointer to the current VM (if any).
1771 * @param paArgs Pointer to (readonly) array of arguments.
1772 * @param cArgs Number of arguments in the array.
1773 */
1774static DECLCALLBACK(int) dbgcCmdBrkREM(PCDBGCCMD /*pCmd*/, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR /*pResult*/)
1775{
1776 /*
1777 * Convert the pointer to a DBGF address.
1778 */
1779 DBGFADDRESS Address;
1780 int rc = pCmdHlp->pfnVarToDbgfAddr(pCmdHlp, &paArgs[0], &Address);
1781 if (VBOX_FAILURE(rc))
1782 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Couldn't convert '%DV' to a DBGF address, rc=%Vrc.\n", &paArgs[0], rc);
1783
1784 /*
1785 * Pick out the optional arguments.
1786 */
1787 uint64_t iHitTrigger = 0;
1788 uint64_t iHitDisable = ~0;
1789 const char *pszCmds = NULL;
1790 unsigned iArg = 1;
1791 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1792 {
1793 iHitTrigger = paArgs[iArg].u.u64Number;
1794 iArg++;
1795 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
1796 {
1797 iHitDisable = paArgs[iArg].u.u64Number;
1798 iArg++;
1799 }
1800 }
1801 if (iArg < cArgs && paArgs[iArg].enmType == DBGCVAR_TYPE_STRING)
1802 {
1803 pszCmds = paArgs[iArg].u.pszString;
1804 iArg++;
1805 }
1806
1807 /*
1808 * Try set the breakpoint.
1809 */
1810 RTUINT iBp;
1811 rc = DBGFR3BpSetREM(pVM, &Address, iHitTrigger, iHitDisable, &iBp);
1812 if (VBOX_SUCCESS(rc))
1813 {
1814 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1815 rc = dbgcBpAdd(pDbgc, iBp, pszCmds);
1816 if (VBOX_SUCCESS(rc))
1817 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Set REM breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1818 if (rc == VERR_DBGC_BP_EXISTS)
1819 {
1820 rc = dbgcBpUpdate(pDbgc, iBp, pszCmds);
1821 if (VBOX_SUCCESS(rc))
1822 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Updated REM breakpoint %u at %VGv\n", iBp, Address.FlatPtr);
1823 }
1824 int rc2 = DBGFR3BpClear(pDbgc->pVM, iBp);
1825 AssertRC(rc2);
1826 }
1827 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Failed to set REM breakpoint at %VGv, rc=%Vrc.\n", Address.FlatPtr, rc);
1828}
1829
1830
1831/**
1832 * The 'u' command.
1833 *
1834 * @returns VBox status.
1835 * @param pCmd Pointer to the command descriptor (as registered).
1836 * @param pCmdHlp Pointer to command helper functions.
1837 * @param pVM Pointer to the current VM (if any).
1838 * @param paArgs Pointer to (readonly) array of arguments.
1839 * @param cArgs Number of arguments in the array.
1840 */
1841static DECLCALLBACK(int) dbgcCmdDisasm(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
1842{
1843 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
1844
1845 /*
1846 * Validate input.
1847 */
1848 if ( cArgs > 1
1849 || (cArgs == 1 && !DBGCVAR_ISPOINTER(paArgs[0].enmType)))
1850 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: The parser doesn't do its job properly yet.. It might help to use the '%%' operator.\n");
1851 if (!pVM && !cArgs && !DBGCVAR_ISPOINTER(pDbgc->DisasmPos.enmType))
1852 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Don't know where to start disassembling...\n");
1853 if (!pVM && cArgs && DBGCVAR_ISGCPOINTER(paArgs[0].enmType))
1854 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: GC address but no VM.\n");
1855
1856 /*
1857 * Find address.
1858 */
1859 unsigned fFlags = DBGF_DISAS_FLAGS_NO_ADDRESS;
1860 if (!cArgs)
1861 {
1862 if (!DBGCVAR_ISPOINTER(pDbgc->DisasmPos.enmType))
1863 {
1864 pDbgc->DisasmPos.enmType = DBGCVAR_TYPE_GC_FAR;
1865 pDbgc->SourcePos.u.GCFar.off = pDbgc->fRegCtxGuest ? CPUMGetGuestEIP(pVM) : CPUMGetHyperEIP(pVM);
1866 pDbgc->SourcePos.u.GCFar.sel = pDbgc->fRegCtxGuest ? CPUMGetGuestCS(pVM) : CPUMGetHyperCS(pVM);
1867 if (pDbgc->fRegCtxGuest)
1868 fFlags |= DBGF_DISAS_FLAGS_CURRENT_GUEST;
1869 else
1870 fFlags |= DBGF_DISAS_FLAGS_CURRENT_HYPER;
1871 }
1872 pDbgc->DisasmPos.enmRangeType = DBGCVAR_RANGE_NONE;
1873 }
1874 else
1875 pDbgc->DisasmPos = paArgs[0];
1876
1877 /*
1878 * Range.
1879 */
1880 switch (pDbgc->DisasmPos.enmRangeType)
1881 {
1882 case DBGCVAR_RANGE_NONE:
1883 pDbgc->DisasmPos.enmRangeType = DBGCVAR_RANGE_ELEMENTS;
1884 pDbgc->DisasmPos.u64Range = 10;
1885 break;
1886
1887 case DBGCVAR_RANGE_ELEMENTS:
1888 if (pDbgc->DisasmPos.u64Range > 2048)
1889 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Too many lines requested. Max is 2048 lines.\n");
1890 break;
1891
1892 case DBGCVAR_RANGE_BYTES:
1893 if (pDbgc->DisasmPos.u64Range > 65536)
1894 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: The requested range is too big. Max is 64KB.\n");
1895 break;
1896
1897 default:
1898 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: Unknown range type %d.\n", pDbgc->DisasmPos.enmRangeType);
1899 }
1900
1901 /*
1902 * Convert physical and host addresses to guest addresses.
1903 */
1904 int rc;
1905 switch (pDbgc->DisasmPos.enmType)
1906 {
1907 case DBGCVAR_TYPE_GC_FLAT:
1908 case DBGCVAR_TYPE_GC_FAR:
1909 break;
1910 case DBGCVAR_TYPE_GC_PHYS:
1911 case DBGCVAR_TYPE_HC_FLAT:
1912 case DBGCVAR_TYPE_HC_PHYS:
1913 case DBGCVAR_TYPE_HC_FAR:
1914 {
1915 DBGCVAR VarTmp;
1916 rc = pCmdHlp->pfnEval(pCmdHlp, &VarTmp, "%%(%Dv)", &pDbgc->DisasmPos);
1917 if (VBOX_FAILURE(rc))
1918 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: failed to evaluate '%%(%Dv)' -> %Vrc .\n", pDbgc->DisasmPos, rc);
1919 pDbgc->DisasmPos = VarTmp;
1920 break;
1921 }
1922 default: AssertFailed(); break;
1923 }
1924
1925 /*
1926 * Print address.
1927 * todo: Change to list near.
1928 */
1929#if 0
1930 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV:\n", &pDbgc->DisasmPos);
1931 if (VBOX_FAILURE(rc))
1932 return rc;
1933#endif
1934
1935 /*
1936 * Do the disassembling.
1937 */
1938 unsigned cTries = 32;
1939 int iRangeLeft = (int)pDbgc->DisasmPos.u64Range;
1940 if (iRangeLeft == 0) /* klugde for 'r'. */
1941 iRangeLeft = -1;
1942 for (;;)
1943 {
1944 /*
1945 * Disassemble the instruction.
1946 */
1947 char szDis[256];
1948 uint32_t cbInstr = 1;
1949 if (pDbgc->DisasmPos.enmType == DBGCVAR_TYPE_GC_FLAT)
1950 rc = DBGFR3DisasInstrEx(pVM, DBGF_SEL_FLAT, pDbgc->DisasmPos.u.GCFlat, fFlags, &szDis[0], sizeof(szDis), &cbInstr);
1951 else
1952 rc = DBGFR3DisasInstrEx(pVM, pDbgc->DisasmPos.u.GCFar.sel, pDbgc->DisasmPos.u.GCFar.off, fFlags, &szDis[0], sizeof(szDis), &cbInstr);
1953 if (VBOX_SUCCESS(rc))
1954 {
1955 /* print it */
1956 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%-16DV %s\n", &pDbgc->DisasmPos, &szDis[0]);
1957 if (VBOX_FAILURE(rc))
1958 return rc;
1959 }
1960 else
1961 {
1962 /* bitch. */
1963 int rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Failed to disassemble instruction, skipping one byte.\n");
1964 if (VBOX_FAILURE(rc))
1965 return rc;
1966 if (cTries-- > 0)
1967 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Too many disassembly failures. Giving up.\n");
1968 cbInstr = 1;
1969 }
1970
1971 /* advance */
1972 if (iRangeLeft < 0) /* 'r' */
1973 break;
1974 if (pDbgc->DisasmPos.enmRangeType == DBGCVAR_RANGE_ELEMENTS)
1975 iRangeLeft--;
1976 else
1977 iRangeLeft -= cbInstr;
1978 rc = pCmdHlp->pfnEval(pCmdHlp, &pDbgc->DisasmPos, "(%Dv) + %x", &pDbgc->DisasmPos, cbInstr);
1979 if (VBOX_FAILURE(rc))
1980 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Expression: (%Dv) + %x\n", &pDbgc->DisasmPos, cbInstr);
1981 if (iRangeLeft <= 0)
1982 break;
1983 fFlags &= ~(DBGF_DISAS_FLAGS_CURRENT_GUEST | DBGF_DISAS_FLAGS_CURRENT_HYPER);
1984 }
1985
1986 NOREF(pCmd); NOREF(pResult);
1987 return 0;
1988}
1989
1990
1991/**
1992 * The 's' command.
1993 *
1994 * @returns VBox status.
1995 * @param pCmd Pointer to the command descriptor (as registered).
1996 * @param pCmdHlp Pointer to command helper functions.
1997 * @param pVM Pointer to the current VM (if any).
1998 * @param paArgs Pointer to (readonly) array of arguments.
1999 * @param cArgs Number of arguments in the array.
2000 */
2001static DECLCALLBACK(int) dbgcCmdSource(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2002{
2003 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2004
2005 /*
2006 * Validate input.
2007 */
2008 if ( cArgs > 1
2009 || (cArgs == 1 && !DBGCVAR_ISPOINTER(paArgs[0].enmType)))
2010 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: The parser doesn't do its job properly yet.. It might help to use the '%%' operator.\n");
2011 if (!pVM && !cArgs && !DBGCVAR_ISPOINTER(pDbgc->SourcePos.enmType))
2012 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Don't know where to start disassembling...\n");
2013 if (!pVM && cArgs && DBGCVAR_ISGCPOINTER(paArgs[0].enmType))
2014 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: GC address but no VM.\n");
2015
2016 /*
2017 * Find address.
2018 */
2019 if (!cArgs)
2020 {
2021 if (!DBGCVAR_ISPOINTER(pDbgc->SourcePos.enmType))
2022 {
2023 pDbgc->SourcePos.enmType = DBGCVAR_TYPE_GC_FAR;
2024 pDbgc->SourcePos.u.GCFar.off = pDbgc->fRegCtxGuest ? CPUMGetGuestEIP(pVM) : CPUMGetHyperEIP(pVM);
2025 pDbgc->SourcePos.u.GCFar.sel = pDbgc->fRegCtxGuest ? CPUMGetGuestCS(pVM) : CPUMGetHyperCS(pVM);
2026 }
2027 pDbgc->SourcePos.enmRangeType = DBGCVAR_RANGE_NONE;
2028 }
2029 else
2030 pDbgc->SourcePos = paArgs[0];
2031
2032 /*
2033 * Ensure the the source address is flat GC.
2034 */
2035 switch (pDbgc->SourcePos.enmType)
2036 {
2037 case DBGCVAR_TYPE_GC_FLAT:
2038 break;
2039 case DBGCVAR_TYPE_GC_PHYS:
2040 case DBGCVAR_TYPE_GC_FAR:
2041 case DBGCVAR_TYPE_HC_FLAT:
2042 case DBGCVAR_TYPE_HC_PHYS:
2043 case DBGCVAR_TYPE_HC_FAR:
2044 {
2045 int rc = pCmdHlp->pfnEval(pCmdHlp, &pDbgc->SourcePos, "%%(%Dv)", &pDbgc->SourcePos);
2046 if (VBOX_FAILURE(rc))
2047 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid address or address type. (rc=%d)\n", rc);
2048 break;
2049 }
2050 default: AssertFailed(); break;
2051 }
2052
2053 /*
2054 * Range.
2055 */
2056 switch (pDbgc->SourcePos.enmRangeType)
2057 {
2058 case DBGCVAR_RANGE_NONE:
2059 pDbgc->SourcePos.enmRangeType = DBGCVAR_RANGE_ELEMENTS;
2060 pDbgc->SourcePos.u64Range = 10;
2061 break;
2062
2063 case DBGCVAR_RANGE_ELEMENTS:
2064 if (pDbgc->SourcePos.u64Range > 2048)
2065 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Too many lines requested. Max is 2048 lines.\n");
2066 break;
2067
2068 case DBGCVAR_RANGE_BYTES:
2069 if (pDbgc->SourcePos.u64Range > 65536)
2070 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: The requested range is too big. Max is 64KB.\n");
2071 break;
2072
2073 default:
2074 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: Unknown range type %d.\n", pDbgc->SourcePos.enmRangeType);
2075 }
2076
2077 /*
2078 * Do the disassembling.
2079 */
2080 bool fFirst = 1;
2081 DBGFLINE LinePrev = { 0, 0, "" };
2082 int iRangeLeft = (int)pDbgc->SourcePos.u64Range;
2083 if (iRangeLeft == 0) /* klugde for 'r'. */
2084 iRangeLeft = -1;
2085 for (;;)
2086 {
2087 /*
2088 * Get line info.
2089 */
2090 DBGFLINE Line;
2091 RTGCINTPTR off;
2092 int rc = DBGFR3LineByAddr(pVM, pDbgc->SourcePos.u.GCFlat, &off, &Line);
2093 if (VBOX_FAILURE(rc))
2094 return VINF_SUCCESS;
2095
2096 unsigned cLines = 0;
2097 if (memcmp(&Line, &LinePrev, sizeof(Line)))
2098 {
2099 /*
2100 * Print filenamename
2101 */
2102 if (!fFirst && strcmp(Line.szFilename, LinePrev.szFilename))
2103 fFirst = true;
2104 if (fFirst)
2105 {
2106 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "[%s @ %d]\n", Line.szFilename, Line.uLineNo);
2107 if (VBOX_FAILURE(rc))
2108 return rc;
2109 }
2110
2111 /*
2112 * Try open the file and read the line.
2113 */
2114 FILE *phFile = fopen(Line.szFilename, "r");
2115 if (phFile)
2116 {
2117 /* Skip ahead to the desired line. */
2118 char szLine[4096];
2119 unsigned cBefore = fFirst ? RT_MIN(2, Line.uLineNo - 1) : Line.uLineNo - LinePrev.uLineNo - 1;
2120 if (cBefore > 7)
2121 cBefore = 0;
2122 unsigned cLeft = Line.uLineNo - cBefore;
2123 while (cLeft > 0)
2124 {
2125 szLine[0] = '\0';
2126 if (!fgets(szLine, sizeof(szLine), phFile))
2127 break;
2128 cLeft--;
2129 }
2130 if (!cLeft)
2131 {
2132 /* print the before lines */
2133 for (;;)
2134 {
2135 size_t cch = strlen(szLine);
2136 while (cch > 0 && (szLine[cch - 1] == '\r' || szLine[cch - 1] == '\n' || isspace(szLine[cch - 1])) )
2137 szLine[--cch] = '\0';
2138 if (cBefore-- <= 0)
2139 break;
2140
2141 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " %4d: %s\n", Line.uLineNo - cBefore - 1, szLine);
2142 szLine[0] = '\0';
2143 fgets(szLine, sizeof(szLine), phFile);
2144 cLines++;
2145 }
2146 /* print the actual line */
2147 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%08llx %4d: %s\n", Line.Address, Line.uLineNo, szLine);
2148 }
2149 fclose(phFile);
2150 if (VBOX_FAILURE(rc))
2151 return rc;
2152 fFirst = false;
2153 }
2154 else
2155 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Warning: couldn't open source file '%s'\n", Line.szFilename);
2156
2157 LinePrev = Line;
2158 }
2159
2160
2161 /*
2162 * Advance
2163 */
2164 if (iRangeLeft < 0) /* 'r' */
2165 break;
2166 if (pDbgc->SourcePos.enmRangeType == DBGCVAR_RANGE_ELEMENTS)
2167 iRangeLeft -= cLines;
2168 else
2169 iRangeLeft -= 1;
2170 rc = pCmdHlp->pfnEval(pCmdHlp, &pDbgc->SourcePos, "(%Dv) + %x", &pDbgc->SourcePos, 1);
2171 if (VBOX_FAILURE(rc))
2172 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Expression: (%Dv) + %x\n", &pDbgc->SourcePos, 1);
2173 if (iRangeLeft <= 0)
2174 break;
2175 }
2176
2177 NOREF(pCmd); NOREF(pResult);
2178 return 0;
2179}
2180
2181
2182/**
2183 * The 'r' command.
2184 *
2185 * @returns VBox status.
2186 * @param pCmd Pointer to the command descriptor (as registered).
2187 * @param pCmdHlp Pointer to command helper functions.
2188 * @param pVM Pointer to the current VM (if any).
2189 * @param paArgs Pointer to (readonly) array of arguments.
2190 * @param cArgs Number of arguments in the array.
2191 */
2192static DECLCALLBACK(int) dbgcCmdReg(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2193{
2194 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2195
2196 if (pDbgc->fRegCtxGuest)
2197 return dbgcCmdRegGuest(pCmd, pCmdHlp, pVM, paArgs, cArgs, pResult);
2198 else
2199 return dbgcCmdRegHyper(pCmd, pCmdHlp, pVM, paArgs, cArgs, pResult);
2200}
2201
2202
2203/**
2204 * Common worker for the dbgcCmdReg*() commands.
2205 *
2206 * @returns VBox status.
2207 * @param pCmd Pointer to the command descriptor (as registered).
2208 * @param pCmdHlp Pointer to command helper functions.
2209 * @param pVM Pointer to the current VM (if any).
2210 * @param paArgs Pointer to (readonly) array of arguments.
2211 * @param cArgs Number of arguments in the array.
2212 * @param pszPrefix The symbol prefix.
2213 */
2214static DECLCALLBACK(int) dbgcCmdRegCommon(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult, const char *pszPrefix)
2215{
2216 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2217
2218 /*
2219 * cArgs == 0: Show all
2220 */
2221 if (cArgs == 0)
2222 {
2223 /*
2224 * Get register context.
2225 */
2226 int rc;
2227 PCPUMCTX pCtx;
2228 PCCPUMCTXCORE pCtxCore;
2229 if (!*pszPrefix)
2230 {
2231 rc = CPUMQueryGuestCtxPtr(pVM, &pCtx);
2232 pCtxCore = CPUMGetGuestCtxCore(pVM);
2233 }
2234 else
2235 {
2236 rc = CPUMQueryHyperCtxPtr(pVM, &pCtx);
2237 pCtxCore = CPUMGetHyperCtxCore(pVM);
2238 }
2239 if (VBOX_FAILURE(rc))
2240 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Getting register context\n");
2241
2242 /*
2243 * Format the flags.
2244 */
2245 static struct
2246 {
2247 const char *pszSet; const char *pszClear; uint32_t fFlag;
2248 } aFlags[] =
2249 {
2250 { "vip",NULL, X86_EFL_VIP },
2251 { "vif",NULL, X86_EFL_VIF },
2252 { "ac", NULL, X86_EFL_AC },
2253 { "vm", NULL, X86_EFL_VM },
2254 { "rf", NULL, X86_EFL_RF },
2255 { "nt", NULL, X86_EFL_NT },
2256 { "ov", "nv", X86_EFL_OF },
2257 { "dn", "up", X86_EFL_DF },
2258 { "ei", "di", X86_EFL_IF },
2259 { "tf", NULL, X86_EFL_TF },
2260 { "nt", "pl", X86_EFL_SF },
2261 { "nz", "zr", X86_EFL_ZF },
2262 { "ac", "na", X86_EFL_AF },
2263 { "po", "pe", X86_EFL_PF },
2264 { "cy", "nc", X86_EFL_CF },
2265 };
2266 char szEFlags[80];
2267 char *psz = szEFlags;
2268 uint32_t efl = pCtxCore->eflags.u32;
2269 for (unsigned i = 0; i < ELEMENTS(aFlags); i++)
2270 {
2271 const char *pszAdd = aFlags[i].fFlag & efl ? aFlags[i].pszSet : aFlags[i].pszClear;
2272 if (pszAdd)
2273 {
2274 strcpy(psz, pszAdd);
2275 psz += strlen(pszAdd);
2276 *psz++ = ' ';
2277 }
2278 }
2279 psz[-1] = '\0';
2280
2281
2282 /*
2283 * Format the registers.
2284 */
2285 if (pDbgc->fRegTerse)
2286 {
2287 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
2288 "%seax=%08x %sebx=%08x %secx=%08x %sedx=%08x %sesi=%08x %sedi=%08x\n"
2289 "%seip=%08x %sesp=%08x %sebp=%08x %siopl=%d %*s\n"
2290 "%scs=%04x %sds=%04x %ses=%04x %sfs=%04x %sgs=%04x %sss=%04x %seflags=%08x\n",
2291 pszPrefix, pCtxCore->eax, pszPrefix, pCtxCore->ebx, pszPrefix, pCtxCore->ecx, pszPrefix, pCtxCore->edx, pszPrefix, pCtxCore->esi, pszPrefix, pCtxCore->edi,
2292 pszPrefix, pCtxCore->eip, pszPrefix, pCtxCore->esp, pszPrefix, pCtxCore->ebp, pszPrefix, X86_EFL_GET_IOPL(efl), *pszPrefix ? 34 : 31, szEFlags,
2293 pszPrefix, (RTSEL)pCtxCore->cs, pszPrefix, (RTSEL)pCtxCore->ds, pszPrefix, (RTSEL)pCtxCore->es,
2294 pszPrefix, (RTSEL)pCtxCore->fs, pszPrefix, (RTSEL)pCtxCore->gs, pszPrefix, (RTSEL)pCtxCore->ss, pszPrefix, efl);
2295 }
2296 else
2297 {
2298 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
2299 "%seax=%08x %sebx=%08x %secx=%08x %sedx=%08x %sesi=%08x %sedi=%08x\n"
2300 "%seip=%08x %sesp=%08x %sebp=%08x %siopl=%d %*s\n"
2301 "%scs={%04x base=%08x limit=%08x flags=%08x} %sdr0=%08x %sdr1=%08x\n"
2302 "%sds={%04x base=%08x limit=%08x flags=%08x} %sdr2=%08x %sdr3=%08x\n"
2303 "%ses={%04x base=%08x limit=%08x flags=%08x} %sdr4=%08x %sdr5=%08x\n"
2304 "%sfs={%04x base=%08x limit=%08x flags=%08x} %sdr6=%08x %sdr7=%08x\n"
2305 "%sgs={%04x base=%08x limit=%08x flags=%08x} %scr0=%08x %scr2=%08x\n"
2306 "%sss={%04x base=%08x limit=%08x flags=%08x} %scr3=%08x %scr4=%08x\n"
2307 "%sgdtr=%08x:%04x %sidtr=%08x:%04x %seflags=%08x\n"
2308 "%sldtr={%04x base=%08x limit=%08x flags=%08x}\n"
2309 "%str ={%04x base=%08x limit=%08x flags=%08x}\n"
2310 "%sSysEnter={cs=%04llx eip=%08llx esp=%08llx}\n"
2311 "%sFCW=%04x %sFSW=%04x %sFTW=%04x\n"
2312 ,
2313 pszPrefix, pCtxCore->eax, pszPrefix, pCtxCore->ebx, pszPrefix, pCtxCore->ecx, pszPrefix, pCtxCore->edx, pszPrefix, pCtxCore->esi, pszPrefix, pCtxCore->edi,
2314 pszPrefix, pCtxCore->eip, pszPrefix, pCtxCore->esp, pszPrefix, pCtxCore->ebp, pszPrefix, X86_EFL_GET_IOPL(efl), *pszPrefix ? 33 : 31, szEFlags,
2315 pszPrefix, (RTSEL)pCtxCore->cs, pCtx->csHid.u32Base, pCtx->csHid.u32Limit, pCtx->csHid.Attr.u, pszPrefix, pCtx->dr0, pszPrefix, pCtx->dr1,
2316 pszPrefix, (RTSEL)pCtxCore->ds, pCtx->dsHid.u32Base, pCtx->dsHid.u32Limit, pCtx->dsHid.Attr.u, pszPrefix, pCtx->dr2, pszPrefix, pCtx->dr3,
2317 pszPrefix, (RTSEL)pCtxCore->es, pCtx->esHid.u32Base, pCtx->esHid.u32Limit, pCtx->esHid.Attr.u, pszPrefix, pCtx->dr4, pszPrefix, pCtx->dr5,
2318 pszPrefix, (RTSEL)pCtxCore->fs, pCtx->fsHid.u32Base, pCtx->fsHid.u32Limit, pCtx->fsHid.Attr.u, pszPrefix, pCtx->dr6, pszPrefix, pCtx->dr7,
2319 pszPrefix, (RTSEL)pCtxCore->gs, pCtx->gsHid.u32Base, pCtx->gsHid.u32Limit, pCtx->gsHid.Attr.u, pszPrefix, pCtx->cr0, pszPrefix, pCtx->cr2,
2320 pszPrefix, (RTSEL)pCtxCore->ss, pCtx->ssHid.u32Base, pCtx->ssHid.u32Limit, pCtx->ssHid.Attr.u, pszPrefix, pCtx->cr3, pszPrefix, pCtx->cr4,
2321 pszPrefix, pCtx->gdtr.pGdt,pCtx->gdtr.cbGdt, pszPrefix, pCtx->idtr.pIdt, pCtx->idtr.cbIdt, pszPrefix, pCtxCore->eflags,
2322 pszPrefix, (RTSEL)pCtx->ldtr, pCtx->ldtrHid.u32Base, pCtx->ldtrHid.u32Limit, pCtx->ldtrHid.Attr.u,
2323 pszPrefix, (RTSEL)pCtx->tr, pCtx->trHid.u32Base, pCtx->trHid.u32Limit, pCtx->trHid.Attr.u,
2324 pszPrefix, pCtx->SysEnter.cs, pCtx->SysEnter.eip, pCtx->SysEnter.esp,
2325 pszPrefix, pCtx->fpu.FCW, pszPrefix, pCtx->fpu.FSW, pszPrefix, pCtx->fpu.FTW);
2326 }
2327
2328 /*
2329 * Disassemble one instruction at cs:eip.
2330 */
2331 return pCmdHlp->pfnExec(pCmdHlp, "u %04x:%08x L 0", pCtx->cs, pCtx->eip);
2332 }
2333
2334 /*
2335 * cArgs == 1: Show the register.
2336 * cArgs == 2: Modify the register.
2337 */
2338 if ( cArgs == 1
2339 || cArgs == 2)
2340 {
2341 /* locate the register symbol. */
2342 const char *pszReg = paArgs[0].u.pszString;
2343 if ( *pszPrefix
2344 && pszReg[0] != *pszPrefix)
2345 {
2346 /* prepend the prefix. */
2347 char *psz = (char *)alloca(strlen(pszReg) + 2);
2348 psz[0] = *pszPrefix;
2349 strcpy(psz + 1, paArgs[0].u.pszString);
2350 pszReg = psz;
2351 }
2352 PCDBGCSYM pSym = dbgcLookupRegisterSymbol(pDbgc, pszReg);
2353 if (!pSym)
2354 return pCmdHlp->pfnVBoxError(pCmdHlp, VERR_INVALID_PARAMETER /* VERR_DBGC_INVALID_REGISTER */, "Invalid register name '%s'.\n", pszReg);
2355
2356 /* show the register */
2357 if (cArgs == 1)
2358 {
2359 DBGCVAR Var;
2360 memset(&Var, 0, sizeof(Var));
2361 int rc = pSym->pfnGet(pSym, pCmdHlp, DBGCVAR_TYPE_NUMBER, &Var);
2362 if (VBOX_FAILURE(rc))
2363 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Failed getting value for register '%s'.\n", pszReg);
2364 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%s=%Dv\n", pszReg, &Var);
2365 }
2366
2367 /* change the register */
2368 int rc = pSym->pfnSet(pSym, pCmdHlp, &paArgs[1]);
2369 if (VBOX_FAILURE(rc))
2370 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Failed setting value for register '%s'.\n", pszReg);
2371 return VINF_SUCCESS;
2372 }
2373
2374
2375 NOREF(pCmd); NOREF(paArgs); NOREF(pResult);
2376 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Huh? cArgs=%d Expected 0, 1 or 2!\n", cArgs);
2377}
2378
2379
2380/**
2381 * The 'rg' command.
2382 *
2383 * @returns VBox status.
2384 * @param pCmd Pointer to the command descriptor (as registered).
2385 * @param pCmdHlp Pointer to command helper functions.
2386 * @param pVM Pointer to the current VM (if any).
2387 * @param paArgs Pointer to (readonly) array of arguments.
2388 * @param cArgs Number of arguments in the array.
2389 */
2390static DECLCALLBACK(int) dbgcCmdRegGuest(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2391{
2392 return dbgcCmdRegCommon(pCmd, pCmdHlp, pVM, paArgs, cArgs, pResult, "");
2393}
2394
2395
2396/**
2397 * The 'rh' command.
2398 *
2399 * @returns VBox status.
2400 * @param pCmd Pointer to the command descriptor (as registered).
2401 * @param pCmdHlp Pointer to command helper functions.
2402 * @param pVM Pointer to the current VM (if any).
2403 * @param paArgs Pointer to (readonly) array of arguments.
2404 * @param cArgs Number of arguments in the array.
2405 */
2406static DECLCALLBACK(int) dbgcCmdRegHyper(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2407{
2408 return dbgcCmdRegCommon(pCmd, pCmdHlp, pVM, paArgs, cArgs, pResult, ".");
2409}
2410
2411
2412/**
2413 * The 'rt' command.
2414 *
2415 * @returns VBox status.
2416 * @param pCmd Pointer to the command descriptor (as registered).
2417 * @param pCmdHlp Pointer to command helper functions.
2418 * @param pVM Pointer to the current VM (if any).
2419 * @param paArgs Pointer to (readonly) array of arguments.
2420 * @param cArgs Number of arguments in the array.
2421 */
2422static DECLCALLBACK(int) dbgcCmdRegTerse(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2423{
2424 NOREF(pCmd); NOREF(pVM); NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
2425
2426 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2427 pDbgc->fRegTerse = !pDbgc->fRegTerse;
2428 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, pDbgc->fRegTerse ? "info: Terse register info.\n" : "info: Verbose register info.\n");
2429}
2430
2431
2432/**
2433 * The 't' command.
2434 *
2435 * @returns VBox status.
2436 * @param pCmd Pointer to the command descriptor (as registered).
2437 * @param pCmdHlp Pointer to command helper functions.
2438 * @param pVM Pointer to the current VM (if any).
2439 * @param paArgs Pointer to (readonly) array of arguments.
2440 * @param cArgs Number of arguments in the array.
2441 */
2442static DECLCALLBACK(int) dbgcCmdTrace(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2443{
2444 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2445
2446 int rc = DBGFR3Step(pVM);
2447 if (VBOX_SUCCESS(rc))
2448 pDbgc->fReady = false;
2449 else
2450 rc = pDbgc->CmdHlp.pfnVBoxError(&pDbgc->CmdHlp, rc, "When trying to single step VM %p\n", pDbgc->pVM);
2451
2452 NOREF(pCmd); NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
2453 return rc;
2454}
2455
2456
2457/**
2458 * The 'k', 'kg' and 'kh' commands.
2459 *
2460 * @returns VBox status.
2461 * @param pCmd Pointer to the command descriptor (as registered).
2462 * @param pCmdHlp Pointer to command helper functions.
2463 * @param pVM Pointer to the current VM (if any).
2464 * @param paArgs Pointer to (readonly) array of arguments.
2465 * @param cArgs Number of arguments in the array.
2466 */
2467static DECLCALLBACK(int) dbgcCmdStack(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2468{
2469 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2470
2471 /*
2472 * Figure which context we're called for.
2473 */
2474 bool fGuest = pCmd->pszCmd[1] == 'g'
2475 || (!pCmd->pszCmd[1] && pDbgc->fRegCtxGuest);
2476
2477
2478 DBGFSTACKFRAME Frame;
2479 memset(&Frame, 0, sizeof(Frame));
2480 int rc;
2481 if (fGuest)
2482 rc = DBGFR3StackWalkBeginGuest(pVM, &Frame);
2483 else
2484 rc = DBGFR3StackWalkBeginHyper(pVM, &Frame);
2485 if (VBOX_FAILURE(rc))
2486 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Failed to begin stack walk, rc=%Vrc\n", rc);
2487
2488 /*
2489 * Print header.
2490 * 12345678 12345678 0023:87654321 12345678 87654321 12345678 87654321 symbol
2491 */
2492 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "EBP Ret EBP Ret CS:EIP Arg0 Arg1 Arg2 Arg3 CS:EIP / Symbol [line]\n");
2493 if (VBOX_FAILURE(rc))
2494 return rc;
2495 do
2496 {
2497 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%08RX32 %08RX32 %04RX32:%08RX32 %08RX32 %08RX32 %08RX32 %08RX32",
2498 (uint32_t)Frame.AddrFrame.off,
2499 (uint32_t)Frame.AddrReturnFrame.off,
2500 (uint32_t)Frame.AddrReturnPC.Sel,
2501 (uint32_t)Frame.AddrReturnPC.off,
2502 Frame.Args.au32[0],
2503 Frame.Args.au32[1],
2504 Frame.Args.au32[2],
2505 Frame.Args.au32[3]);
2506 if (VBOX_FAILURE(rc))
2507 return rc;
2508 if (!Frame.pSymPC)
2509 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " %RTsel:%08RGv", Frame.AddrPC.Sel, Frame.AddrPC.off);
2510 else
2511 {
2512 RTGCINTPTR offDisp = Frame.AddrPC.FlatPtr - Frame.pSymPC->Value; /** @todo this isn't 100% correct for segemnted stuff. */
2513 if (offDisp > 0)
2514 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " %s+%llx", Frame.pSymPC->szName, (int64_t)offDisp);
2515 else if (offDisp < 0)
2516 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " %s-%llx", Frame.pSymPC->szName, -(int64_t)offDisp);
2517 else
2518 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " %s", Frame.pSymPC->szName);
2519 }
2520 if (VBOX_SUCCESS(rc) && Frame.pLinePC)
2521 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " [%s @ 0i%d]", Frame.pLinePC->szFilename, Frame.pLinePC->uLineNo);
2522 if (VBOX_SUCCESS(rc))
2523 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
2524 if (VBOX_FAILURE(rc))
2525 return rc;
2526
2527 /* next */
2528 rc = DBGFR3StackWalkNext(pVM, &Frame);
2529 } while (VBOX_SUCCESS(rc));
2530
2531 NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
2532 return VINF_SUCCESS;
2533}
2534
2535
2536static int dbgcCmdDumpDTWorker64(PDBGCCMDHLP /*pCmdHlp*/, PCX86DESC64 /*pDesc*/, unsigned /*iEntry*/, bool /* fHyper */, bool * /*fDblEntry*/)
2537{
2538 /* GUEST64 */
2539 return VINF_SUCCESS;
2540}
2541
2542
2543/**
2544 * Wroker function that displays one descriptor entry (GDT, LDT, IDT).
2545 *
2546 * @returns pfnPrintf status code.
2547 * @param pCmdHlp The DBGC command helpers.
2548 * @param pDesc The descriptor to display.
2549 * @param iEntry The descriptor entry number.
2550 * @param fHyper Whether the selector belongs to the hypervisor or not.
2551 */
2552static int dbgcCmdDumpDTWorker32(PDBGCCMDHLP pCmdHlp, PCX86DESC pDesc, unsigned iEntry, bool fHyper)
2553{
2554 int rc;
2555
2556 const char *pszHyper = fHyper ? " HYPER" : "";
2557 const char *pszPresent = pDesc->Gen.u1Present ? "P " : "NP";
2558 if (pDesc->Gen.u1DescType)
2559 {
2560 static const char * const s_apszTypes[] =
2561 {
2562 "DataRO", /* 0 Read-Only */
2563 "DataRO", /* 1 Read-Only - Accessed */
2564 "DataRW", /* 2 Read/Write */
2565 "DataRW", /* 3 Read/Write - Accessed */
2566 "DownRO", /* 4 Expand-down, Read-Only */
2567 "DownRO", /* 5 Expand-down, Read-Only - Accessed */
2568 "DownRW", /* 6 Expand-down, Read/Write */
2569 "DownRO", /* 7 Expand-down, Read/Write - Accessed */
2570 "CodeEO", /* 8 Execute-Only */
2571 "CodeEO", /* 9 Execute-Only - Accessed */
2572 "CodeER", /* A Execute/Readable */
2573 "CodeER", /* B Execute/Readable - Accessed */
2574 "ConfE0", /* C Conforming, Execute-Only */
2575 "ConfE0", /* D Conforming, Execute-Only - Accessed */
2576 "ConfER", /* E Conforming, Execute/Readable */
2577 "ConfER" /* F Conforming, Execute/Readable - Accessed */
2578 };
2579 const char *pszAccessed = pDesc->Gen.u4Type & BIT(0) ? "A " : "NA";
2580 const char *pszGranularity = pDesc->Gen.u1Granularity ? "G" : " ";
2581 const char *pszBig = pDesc->Gen.u1DefBig ? "BIG" : " ";
2582 uint32_t u32Base = pDesc->Gen.u16BaseLow
2583 | ((uint32_t)pDesc->Gen.u8BaseHigh1 << 16)
2584 | ((uint32_t)pDesc->Gen.u8BaseHigh2 << 24);
2585 uint32_t cbLimit = pDesc->Gen.u16LimitLow | (pDesc->Gen.u4LimitHigh << 16);
2586 if (pDesc->Gen.u1Granularity)
2587 cbLimit <<= PAGE_SHIFT;
2588
2589 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s Bas=%08x Lim=%08x DPL=%d %s %s %s %s AVL=%d R=%d%s\n",
2590 iEntry, s_apszTypes[pDesc->Gen.u4Type], u32Base, cbLimit,
2591 pDesc->Gen.u2Dpl, pszPresent, pszAccessed, pszGranularity, pszBig,
2592 pDesc->Gen.u1Available, pDesc->Gen.u1Reserved, pszHyper);
2593 }
2594 else
2595 {
2596 static const char * const s_apszTypes[] =
2597 {
2598 "Ill-0 ", /* 0 0000 Reserved (Illegal) */
2599 "Tss16A", /* 1 0001 Available 16-bit TSS */
2600 "LDT ", /* 2 0010 LDT */
2601 "Tss16B", /* 3 0011 Busy 16-bit TSS */
2602 "Call16", /* 4 0100 16-bit Call Gate */
2603 "TaskG ", /* 5 0101 Task Gate */
2604 "Int16 ", /* 6 0110 16-bit Interrupt Gate */
2605 "Trap16", /* 7 0111 16-bit Trap Gate */
2606 "Ill-8 ", /* 8 1000 Reserved (Illegal) */
2607 "Tss32A", /* 9 1001 Available 32-bit TSS */
2608 "Ill-A ", /* A 1010 Reserved (Illegal) */
2609 "Tss32B", /* B 1011 Busy 32-bit TSS */
2610 "Call32", /* C 1100 32-bit Call Gate */
2611 "Ill-D ", /* D 1101 Reserved (Illegal) */
2612 "Int32 ", /* E 1110 32-bit Interrupt Gate */
2613 "Trap32" /* F 1111 32-bit Trap Gate */
2614 };
2615 switch (pDesc->Gen.u4Type)
2616 {
2617 /* raw */
2618 case X86_SEL_TYPE_SYS_UNDEFINED:
2619 case X86_SEL_TYPE_SYS_UNDEFINED2:
2620 case X86_SEL_TYPE_SYS_UNDEFINED4:
2621 case X86_SEL_TYPE_SYS_UNDEFINED3:
2622 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s %.8Rhxs DPL=%d %s%s\n",
2623 iEntry, s_apszTypes[pDesc->Gen.u4Type], pDesc,
2624 pDesc->Gen.u2Dpl, pszPresent, pszHyper);
2625 break;
2626
2627 case X86_SEL_TYPE_SYS_286_TSS_AVAIL:
2628 case X86_SEL_TYPE_SYS_386_TSS_AVAIL:
2629 case X86_SEL_TYPE_SYS_286_TSS_BUSY:
2630 case X86_SEL_TYPE_SYS_386_TSS_BUSY:
2631 case X86_SEL_TYPE_SYS_LDT:
2632 {
2633 const char *pszGranularity = pDesc->Gen.u1Granularity ? "G" : " ";
2634 const char *pszBusy = pDesc->Gen.u4Type & BIT(1) ? "B " : "NB";
2635 const char *pszBig = pDesc->Gen.u1DefBig ? "BIG" : " ";
2636 uint32_t u32Base = pDesc->Gen.u16BaseLow
2637 | ((uint32_t)pDesc->Gen.u8BaseHigh1 << 16)
2638 | ((uint32_t)pDesc->Gen.u8BaseHigh2 << 24);
2639 uint32_t cbLimit = pDesc->Gen.u16LimitLow | (pDesc->Gen.u4LimitHigh << 16);
2640 if (pDesc->Gen.u1Granularity)
2641 cbLimit <<= PAGE_SHIFT;
2642
2643 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s Bas=%08x Lim=%08x DPL=%d %s %s %s %s AVL=%d R=%d%s\n",
2644 iEntry, s_apszTypes[pDesc->Gen.u4Type], u32Base, cbLimit,
2645 pDesc->Gen.u2Dpl, pszPresent, pszBusy, pszGranularity, pszBig,
2646 pDesc->Gen.u1Available, pDesc->Gen.u1Reserved | (pDesc->Gen.u1DefBig << 1),
2647 pszHyper);
2648 break;
2649 }
2650
2651 case X86_SEL_TYPE_SYS_TASK_GATE:
2652 {
2653 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s TSS=%04x DPL=%d %s%s\n",
2654 iEntry, s_apszTypes[pDesc->Gen.u4Type], pDesc->au16[1],
2655 pDesc->Gen.u2Dpl, pszPresent, pszHyper);
2656 break;
2657 }
2658
2659 case X86_SEL_TYPE_SYS_286_CALL_GATE:
2660 case X86_SEL_TYPE_SYS_386_CALL_GATE:
2661 {
2662 unsigned cParams = pDesc->au8[0] & 0x1f;
2663 const char *pszCountOf = pDesc->Gen.u4Type & BIT(3) ? "DC" : "WC";
2664 RTSEL sel = pDesc->au16[1];
2665 uint32_t off = pDesc->au16[0] | ((uint32_t)pDesc->au16[3] << 16);
2666 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s Sel:Off=%04x:%08x DPL=%d %s %s=%d%s\n",
2667 iEntry, s_apszTypes[pDesc->Gen.u4Type], sel, off,
2668 pDesc->Gen.u2Dpl, pszPresent, pszCountOf, cParams, pszHyper);
2669 break;
2670 }
2671
2672 case X86_SEL_TYPE_SYS_286_INT_GATE:
2673 case X86_SEL_TYPE_SYS_386_INT_GATE:
2674 case X86_SEL_TYPE_SYS_286_TRAP_GATE:
2675 case X86_SEL_TYPE_SYS_386_TRAP_GATE:
2676 {
2677 RTSEL sel = pDesc->au16[1];
2678 uint32_t off = pDesc->au16[0] | ((uint32_t)pDesc->au16[3] << 16);
2679 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %s Sel:Off=%04x:%08x DPL=%d %s%s\n",
2680 iEntry, s_apszTypes[pDesc->Gen.u4Type], sel, off,
2681 pDesc->Gen.u2Dpl, pszPresent, pszHyper);
2682 break;
2683 }
2684
2685 /* impossible, just it's necessary to keep gcc happy. */
2686 default:
2687 return VINF_SUCCESS;
2688 }
2689 }
2690 return rc;
2691}
2692
2693
2694/**
2695 * The 'dg', 'dga', 'dl' and 'dla' commands.
2696 *
2697 * @returns VBox status.
2698 * @param pCmd Pointer to the command descriptor (as registered).
2699 * @param pCmdHlp Pointer to command helper functions.
2700 * @param pVM Pointer to the current VM (if any).
2701 * @param paArgs Pointer to (readonly) array of arguments.
2702 * @param cArgs Number of arguments in the array.
2703 */
2704static DECLCALLBACK(int) dbgcCmdDumpDT(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2705{
2706 /*
2707 * Validate input.
2708 */
2709 if (!pVM)
2710 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
2711
2712 /*
2713 * Get the CPU mode, check which command variation this is
2714 * and fix a default parameter if needed.
2715 */
2716 CPUMMODE enmMode = CPUMGetGuestMode(pVM);
2717 bool fGdt = pCmd->pszCmd[1] == 'g';
2718 bool fAll = pCmd->pszCmd[2] == 'a';
2719
2720 DBGCVAR Var;
2721 if (!cArgs)
2722 {
2723 cArgs = 1;
2724 paArgs = &Var;
2725 Var.enmType = DBGCVAR_TYPE_NUMBER;
2726 Var.u.u64Number = fGdt ? 0 : 4;
2727 Var.enmRangeType = DBGCVAR_RANGE_ELEMENTS;
2728 Var.u64Range = 1024;
2729 }
2730
2731 /*
2732 * Process the arguments.
2733 */
2734 for (unsigned i = 0; i < cArgs; i++)
2735 {
2736 /*
2737 * Retrive the selector value from the argument.
2738 * The parser may confuse pointers and numbers if more than one
2739 * argument is given, that that into account.
2740 */
2741 /* check that what've got makes sense as we don't trust the parser yet. */
2742 if ( paArgs[i].enmType != DBGCVAR_TYPE_NUMBER
2743 && !DBGCVAR_ISPOINTER(paArgs[i].enmType))
2744 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: arg #%u isn't of number or pointer type but %d.\n", i, paArgs[i].enmType);
2745 unsigned u64;
2746 unsigned cSels = 1;
2747 switch (paArgs[i].enmType)
2748 {
2749 case DBGCVAR_TYPE_NUMBER:
2750 u64 = paArgs[i].u.u64Number;
2751 if (paArgs[i].enmRangeType != DBGCVAR_RANGE_NONE)
2752 cSels = RT_MIN(paArgs[i].u64Range, 1024);
2753 break;
2754 case DBGCVAR_TYPE_GC_FAR: u64 = paArgs[i].u.GCFar.sel; break;
2755 case DBGCVAR_TYPE_GC_FLAT: u64 = paArgs[i].u.GCFlat; break;
2756 case DBGCVAR_TYPE_GC_PHYS: u64 = paArgs[i].u.GCPhys; break;
2757 case DBGCVAR_TYPE_HC_FAR: u64 = paArgs[i].u.HCFar.sel; break;
2758 case DBGCVAR_TYPE_HC_FLAT: u64 = (uintptr_t)paArgs[i].u.pvHCFlat; break;
2759 case DBGCVAR_TYPE_HC_PHYS: u64 = paArgs[i].u.HCPhys; break;
2760 default: u64 = _64K; break;
2761 }
2762 if (u64 < _64K)
2763 {
2764 unsigned Sel = (RTSEL)u64;
2765
2766 /*
2767 * Dump the specified range.
2768 */
2769 bool fSingle = cSels == 1;
2770 while ( cSels-- > 0
2771 && Sel < _64K)
2772 {
2773 SELMSELINFO SelInfo;
2774 int rc = SELMR3GetSelectorInfo(pVM, Sel, &SelInfo);
2775 if (RT_SUCCESS(rc))
2776 {
2777 if (SelInfo.fRealMode)
2778 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x RealM Bas=%04x Lim=%04x\n",
2779 Sel, (unsigned)SelInfo.GCPtrBase, (unsigned)SelInfo.cbLimit);
2780 else if (fAll || fSingle || SelInfo.Raw.Gen.u1Present)
2781 {
2782 if (enmMode == CPUMMODE_PROTECTED)
2783 rc = dbgcCmdDumpDTWorker32(pCmdHlp, (PX86DESC)&SelInfo.Raw, Sel, SelInfo.fHyper);
2784 else
2785 {
2786 bool fDblSkip = false;
2787 rc = dbgcCmdDumpDTWorker64(pCmdHlp, (PX86DESC64)&SelInfo.Raw, Sel, SelInfo.fHyper, &fDblSkip);
2788 if (fDblSkip)
2789 Sel += 4;
2790 }
2791 }
2792 }
2793 else
2794 {
2795 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %Vrc\n", Sel, rc);
2796 if (!fAll)
2797 return rc;
2798 }
2799 if (RT_FAILURE(rc))
2800 return rc;
2801
2802 /* next */
2803 Sel += 4;
2804 }
2805 }
2806 else
2807 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: %llx is out of bounds\n", u64);
2808 }
2809
2810 NOREF(pResult);
2811 return VINF_SUCCESS;
2812}
2813
2814
2815/**
2816 * The 'di' and 'dia' commands.
2817 *
2818 * @returns VBox status.
2819 * @param pCmd Pointer to the command descriptor (as registered).
2820 * @param pCmdHlp Pointer to command helper functions.
2821 * @param pVM Pointer to the current VM (if any).
2822 * @param paArgs Pointer to (readonly) array of arguments.
2823 * @param cArgs Number of arguments in the array.
2824 */
2825static DECLCALLBACK(int) dbgcCmdDumpIDT(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2826{
2827 /*
2828 * Validate input.
2829 */
2830 if (!pVM)
2831 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
2832
2833 /*
2834 * Establish some stuff like the current IDTR and CPU mode,
2835 * and fix a default parameter.
2836 */
2837 uint16_t cbLimit;
2838 RTGCUINTPTR GCPtrBase = CPUMGetGuestIDTR(pVM, &cbLimit);
2839 CPUMMODE enmMode = CPUMGetGuestMode(pVM);
2840 size_t cbEntry;
2841 switch (enmMode)
2842 {
2843 case CPUMMODE_REAL: cbEntry = sizeof(RTFAR16); break;
2844 case CPUMMODE_PROTECTED: cbEntry = sizeof(X86DESC); break;
2845 case CPUMMODE_LONG: cbEntry = sizeof(X86DESC64); break;
2846 default:
2847 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Invalid CPU mode %d.\n", enmMode);
2848 }
2849
2850 bool fAll = pCmd->pszCmd[2] == 'a';
2851 DBGCVAR Var;
2852 if (!cArgs)
2853 {
2854 cArgs = 1;
2855 paArgs = &Var;
2856 Var.enmType = DBGCVAR_TYPE_NUMBER;
2857 Var.u.u64Number = 0;
2858 Var.enmRangeType = DBGCVAR_RANGE_ELEMENTS;
2859 Var.u64Range = 256;
2860 }
2861
2862 /*
2863 * Process the arguments.
2864 */
2865 for (unsigned i = 0; i < cArgs; i++)
2866 {
2867 /* check that what've got makes sense as we don't trust the parser yet. */
2868 if (paArgs[i].enmType != DBGCVAR_TYPE_NUMBER)
2869 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: arg #%u isn't of number type but %d.\n", i, paArgs[i].enmType);
2870 if (paArgs[i].u.u64Number < 256)
2871 {
2872 RTGCUINTPTR iInt = paArgs[i].u.u64Number;
2873 unsigned cInts = paArgs[i].enmRangeType != DBGCVAR_RANGE_NONE
2874 ? paArgs[i].u64Range
2875 : 1;
2876 bool fSingle = cInts == 1;
2877 while ( cInts-- > 0
2878 && iInt < 256)
2879 {
2880 /*
2881 * Try read it.
2882 */
2883 union
2884 {
2885 RTFAR16 Real;
2886 X86DESC Prot;
2887 X86DESC64 Long;
2888 } u;
2889 if (iInt * cbEntry + (cbEntry - 1) > cbLimit)
2890 {
2891 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x not within the IDT\n", (unsigned)iInt);
2892 if (!fAll && !fSingle)
2893 return VINF_SUCCESS;
2894 }
2895 DBGCVAR AddrVar;
2896 AddrVar.enmType = DBGCVAR_TYPE_GC_FLAT;
2897 AddrVar.u.GCFlat = GCPtrBase + iInt * cbEntry;
2898 AddrVar.enmRangeType = DBGCVAR_RANGE_NONE;
2899 int rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &u, cbEntry, &AddrVar, NULL);
2900 if (VBOX_FAILURE(rc))
2901 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Reading IDT entry %#04x.\n", (unsigned)iInt);
2902
2903 /*
2904 * Display it.
2905 */
2906 switch (enmMode)
2907 {
2908 case CPUMMODE_REAL:
2909 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%04x %RTfp16\n", (unsigned)iInt, u.Real);
2910 /** @todo resolve 16:16 IDTE to a symbol */
2911 break;
2912 case CPUMMODE_PROTECTED:
2913 if (fAll || fSingle || u.Prot.Gen.u1Present)
2914 rc = dbgcCmdDumpDTWorker32(pCmdHlp, &u.Prot, iInt, false);
2915 break;
2916 case CPUMMODE_LONG:
2917 if (fAll || fSingle || u.Long.Gen.u1Present)
2918 rc = dbgcCmdDumpDTWorker64(pCmdHlp, &u.Long, iInt, false, NULL);
2919 break;
2920 default: break; /* to shut up gcc */
2921 }
2922 if (RT_FAILURE(rc))
2923 return rc;
2924
2925 /* next */
2926 iInt++;
2927 }
2928 }
2929 else
2930 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: %llx is out of bounds (max 256)\n", paArgs[i].u.u64Number);
2931 }
2932
2933 NOREF(pResult);
2934 return VINF_SUCCESS;
2935}
2936
2937
2938/**
2939 * The 'da', 'dq', 'dd', 'dw' and 'db' commands.
2940 *
2941 * @returns VBox status.
2942 * @param pCmd Pointer to the command descriptor (as registered).
2943 * @param pCmdHlp Pointer to command helper functions.
2944 * @param pVM Pointer to the current VM (if any).
2945 * @param paArgs Pointer to (readonly) array of arguments.
2946 * @param cArgs Number of arguments in the array.
2947 */
2948static DECLCALLBACK(int) dbgcCmdDumpMem(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
2949{
2950 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
2951
2952 /*
2953 * Validate input.
2954 */
2955 if ( cArgs > 1
2956 || (cArgs == 1 && !DBGCVAR_ISPOINTER(paArgs[0].enmType)))
2957 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: The parser doesn't do its job properly yet.. It might help to use the '%%' operator.\n");
2958 if (!pVM)
2959 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
2960
2961 /*
2962 * Figure out the element size.
2963 */
2964 size_t cbElement;
2965 bool fAscii = false;
2966 switch (pCmd->pszCmd[1])
2967 {
2968 default:
2969 case 'b': cbElement = 1; break;
2970 case 'w': cbElement = 2; break;
2971 case 'd': cbElement = 4; break;
2972 case 'q': cbElement = 8; break;
2973 case 'a':
2974 cbElement = 1;
2975 fAscii = true;
2976 break;
2977 case '\0':
2978 fAscii = !!(pDbgc->cbDumpElement & 0x80000000);
2979 cbElement = pDbgc->cbDumpElement & 0x7fffffff;
2980 if (!cbElement)
2981 cbElement = 1;
2982 break;
2983 }
2984
2985 /*
2986 * Find address.
2987 */
2988 if (!cArgs)
2989 pDbgc->DumpPos.enmRangeType = DBGCVAR_RANGE_NONE;
2990 else
2991 pDbgc->DumpPos = paArgs[0];
2992
2993 /*
2994 * Range.
2995 */
2996 switch (pDbgc->DumpPos.enmRangeType)
2997 {
2998 case DBGCVAR_RANGE_NONE:
2999 pDbgc->DumpPos.enmRangeType = DBGCVAR_RANGE_BYTES;
3000 pDbgc->DumpPos.u64Range = 0x60;
3001 break;
3002
3003 case DBGCVAR_RANGE_ELEMENTS:
3004 if (pDbgc->DumpPos.u64Range > 2048)
3005 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: Too many elements requested. Max is 2048 elements.\n");
3006 pDbgc->DumpPos.enmRangeType = DBGCVAR_RANGE_BYTES;
3007 pDbgc->DumpPos.u64Range = (cbElement ? cbElement : 1) * pDbgc->DumpPos.u64Range;
3008 break;
3009
3010 case DBGCVAR_RANGE_BYTES:
3011 if (pDbgc->DumpPos.u64Range > 65536)
3012 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: The requested range is too big. Max is 64KB.\n");
3013 break;
3014
3015 default:
3016 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: Unknown range type %d.\n", pDbgc->DumpPos.enmRangeType);
3017 }
3018
3019 /*
3020 * Do the dumping.
3021 */
3022 pDbgc->cbDumpElement = cbElement | (fAscii << 31);
3023 int cbLeft = (int)pDbgc->DumpPos.u64Range;
3024 uint8_t u8Prev = '\0';
3025 for (;;)
3026 {
3027 /*
3028 * Read memory.
3029 */
3030 char achBuffer[16];
3031 size_t cbReq = RT_MIN((int)sizeof(achBuffer), cbLeft);
3032 size_t cb = RT_MIN((int)sizeof(achBuffer), cbLeft);
3033 int rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &achBuffer, cbReq, &pDbgc->DumpPos, &cb);
3034 if (VBOX_FAILURE(rc))
3035 {
3036 if (u8Prev && u8Prev != '\n')
3037 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
3038 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Reading memory at %DV.\n", &pDbgc->DumpPos);
3039 }
3040
3041 /*
3042 * Display it.
3043 */
3044 memset(&achBuffer[cb], 0, sizeof(achBuffer) - cb);
3045 if (!fAscii)
3046 {
3047 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV:", &pDbgc->DumpPos);
3048 unsigned i;
3049 for (i = 0; i < cb; i += cbElement)
3050 {
3051 const char *pszSpace = " ";
3052 if (cbElement <= 2 && i == 8 && !fAscii)
3053 pszSpace = "-";
3054 switch (cbElement)
3055 {
3056 case 1: pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%s%02x", pszSpace, *(uint8_t *)&achBuffer[i]); break;
3057 case 2: pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%s%04x", pszSpace, *(uint16_t *)&achBuffer[i]); break;
3058 case 4: pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%s%08x", pszSpace, *(uint32_t *)&achBuffer[i]); break;
3059 case 8: pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%s%016llx", pszSpace, *(uint64_t *)&achBuffer[i]); break;
3060 }
3061 }
3062
3063 /* chars column */
3064 if (pDbgc->cbDumpElement == 1)
3065 {
3066 while (i++ < sizeof(achBuffer))
3067 pCmdHlp->pfnPrintf(pCmdHlp, NULL, " ");
3068 pCmdHlp->pfnPrintf(pCmdHlp, NULL, " ");
3069 for (i = 0; i < cb; i += cbElement)
3070 {
3071 uint8_t u8 = *(uint8_t *)&achBuffer[i];
3072 if (isprint(u8) && u8 < 127)
3073 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%c", u8);
3074 else
3075 pCmdHlp->pfnPrintf(pCmdHlp, NULL, ".");
3076 }
3077 }
3078 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
3079 }
3080 else
3081 {
3082 /*
3083 * We print up to the first zero and stop there.
3084 * Only printables + '\t' and '\n' are printed.
3085 */
3086 if (!u8Prev)
3087 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV:\n", &pDbgc->DumpPos);
3088 uint8_t u8 = '\0';
3089 unsigned i;
3090 for (i = 0; i < cb; i++)
3091 {
3092 u8Prev = u8;
3093 u8 = *(uint8_t *)&achBuffer[i];
3094 if ( u8 < 127
3095 && ( isprint(u8)
3096 || u8 == '\t'
3097 || u8 == '\n'))
3098 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%c", u8);
3099 else if (!u8)
3100 break;
3101 else
3102 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\\x%x", u8);
3103 }
3104 if (u8 == '\0')
3105 cbLeft = cb = i + 1;
3106 if (cbLeft - cb <= 0 && u8Prev != '\n')
3107 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
3108 }
3109
3110 /*
3111 * Advance
3112 */
3113 cbLeft -= cb;
3114 rc = pCmdHlp->pfnEval(pCmdHlp, &pDbgc->DumpPos, "(%Dv) + %x", &pDbgc->DumpPos, cb);
3115 if (VBOX_FAILURE(rc))
3116 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Expression: (%Dv) + %x\n", &pDbgc->DumpPos, cb);
3117 if (cbLeft <= 0)
3118 break;
3119 }
3120
3121 NOREF(pCmd); NOREF(pResult);
3122 return VINF_SUCCESS;
3123}
3124
3125
3126/**
3127 * Best guess at which paging mode currently applies to the guest
3128 * paging structures.
3129 *
3130 * This have to come up with a decent answer even when the guest
3131 * is in non-paged protected mode or real mode.
3132 *
3133 * @returns cr3.
3134 * @param pDbgc The DBGC instance.
3135 * @param pfPAE Where to store the page address extension indicator.
3136 * @param pfLME Where to store the long mode enabled indicator.
3137 * @param pfPSE Where to store the page size extension indicator.
3138 * @param pfPGE Where to store the page global enabled indicator.
3139 * @param pfNXE Where to store the no-execution enabled inidicator.
3140 */
3141static RTGCPHYS dbgcGetGuestPageMode(PDBGC pDbgc, bool *pfPAE, bool *pfLME, bool *pfPSE, bool *pfPGE, bool *pfNXE)
3142{
3143 RTGCUINTREG cr4 = CPUMGetGuestCR4(pDbgc->pVM);
3144 *pfPSE = !!(cr4 & X86_CR4_PSE);
3145 *pfPGE = !!(cr4 & X86_CR4_PGE);
3146 *pfPAE = !!(cr4 & X86_CR4_PAE);
3147 *pfLME = CPUMGetGuestMode(pDbgc->pVM) == CPUMMODE_LONG;
3148 *pfNXE = false; /* GUEST64 GUESTNX */
3149 return CPUMGetGuestCR3(pDbgc->pVM);
3150}
3151
3152
3153/**
3154 * Determin the shadow paging mode.
3155 *
3156 * @returns cr3.
3157 * @param pDbgc The DBGC instance.
3158 * @param pfPAE Where to store the page address extension indicator.
3159 * @param pfLME Where to store the long mode enabled indicator.
3160 * @param pfPSE Where to store the page size extension indicator.
3161 * @param pfPGE Where to store the page global enabled indicator.
3162 * @param pfNXE Where to store the no-execution enabled inidicator.
3163 */
3164static RTHCPHYS dbgcGetShadowPageMode(PDBGC pDbgc, bool *pfPAE, bool *pfLME, bool *pfPSE, bool *pfPGE, bool *pfNXE)
3165{
3166 *pfPSE = true;
3167 *pfPGE = false;
3168 switch (PGMGetShadowMode(pDbgc->pVM))
3169 {
3170 default:
3171 case PGMMODE_32_BIT:
3172 *pfPAE = *pfLME = *pfNXE = false;
3173 break;
3174 case PGMMODE_PAE:
3175 *pfLME = *pfNXE = false;
3176 *pfPAE = true;
3177 break;
3178 case PGMMODE_PAE_NX:
3179 *pfLME = false;
3180 *pfPAE = *pfNXE = true;
3181 break;
3182 case PGMMODE_AMD64:
3183 *pfNXE = false;
3184 *pfPAE = *pfLME = true;
3185 break;
3186 case PGMMODE_AMD64_NX:
3187 *pfPAE = *pfLME = *pfNXE = true;
3188 break;
3189 }
3190 return PGMGetHyperCR3(pDbgc->pVM);
3191}
3192
3193
3194/**
3195 * The 'dpd', 'dpda', 'dpdb', 'dpdg' and 'dpdh' commands.
3196 *
3197 * @returns VBox status.
3198 * @param pCmd Pointer to the command descriptor (as registered).
3199 * @param pCmdHlp Pointer to command helper functions.
3200 * @param pVM Pointer to the current VM (if any).
3201 * @param paArgs Pointer to (readonly) array of arguments.
3202 * @param cArgs Number of arguments in the array.
3203 */
3204static DECLCALLBACK(int) dbgcCmdDumpPageDir(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3205{
3206 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
3207
3208 /*
3209 * Validate input.
3210 */
3211 if ( cArgs > 1
3212 || (cArgs == 1 && pCmd->pszCmd[3] == 'a' && !DBGCVAR_ISPOINTER(paArgs[0].enmType))
3213 || (cArgs == 1 && pCmd->pszCmd[3] != 'a' && !(paArgs[0].enmType == DBGCVAR_TYPE_NUMBER || DBGCVAR_ISPOINTER(paArgs[0].enmType)))
3214 )
3215 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: The parser doesn't do its job properly yet.. It might help to use the '%%' operator.\n");
3216 if (!pVM)
3217 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
3218
3219 /*
3220 * Guest or shadow page directories? Get the paging parameters.
3221 */
3222 bool fGuest = pCmd->pszCmd[3] != 'h';
3223 if (!pCmd->pszCmd[3] || pCmd->pszCmd[3] == 'a')
3224 fGuest = paArgs[0].enmType == DBGCVAR_TYPE_NUMBER
3225 ? pDbgc->fRegCtxGuest
3226 : DBGCVAR_ISGCPOINTER(paArgs[0].enmType);
3227
3228 bool fPAE, fLME, fPSE, fPGE, fNXE;
3229 uint64_t cr3 = fGuest
3230 ? dbgcGetGuestPageMode(pDbgc, &fPAE, &fLME, &fPSE, &fPGE, &fNXE)
3231 : dbgcGetShadowPageMode(pDbgc, &fPAE, &fLME, &fPSE, &fPGE, &fNXE);
3232 const unsigned cbEntry = fPAE ? sizeof(X86PTEPAE) : sizeof(X86PTE);
3233
3234 /*
3235 * Setup default arugment if none was specified.
3236 * Fix address / index confusion.
3237 */
3238 DBGCVAR VarDefault;
3239 if (!cArgs)
3240 {
3241 if (pCmd->pszCmd[3] == 'a')
3242 {
3243 if (fLME || fPAE)
3244 return DBGCCmdHlpPrintf(pCmdHlp, "Default argument for 'dpda' hasn't been fully implemented yet. Try with an address or use one of the other commands.\n");
3245 if (fGuest)
3246 DBGCVAR_INIT_GC_PHYS(&VarDefault, cr3);
3247 else
3248 DBGCVAR_INIT_HC_PHYS(&VarDefault, cr3);
3249 }
3250 else
3251 DBGCVAR_INIT_GC_FLAT(&VarDefault, 0);
3252 paArgs = &VarDefault;
3253 cArgs = 1;
3254 }
3255 else if (paArgs[0].enmType == DBGCVAR_TYPE_NUMBER)
3256 {
3257 Assert(pCmd->pszCmd[3] != 'a');
3258 VarDefault = paArgs[0];
3259 if (VarDefault.u.u64Number <= 1024)
3260 {
3261 if (fPAE)
3262 return DBGCCmdHlpPrintf(pCmdHlp, "PDE indexing is only implemented for 32-bit paging.\n");
3263 if (VarDefault.u.u64Number >= PAGE_SIZE / cbEntry)
3264 return DBGCCmdHlpPrintf(pCmdHlp, "PDE index is out of range [0..%d].\n", PAGE_SIZE / cbEntry - 1);
3265 VarDefault.u.u64Number <<= X86_PD_SHIFT;
3266 }
3267 VarDefault.enmType = DBGCVAR_TYPE_GC_FLAT;
3268 paArgs = &VarDefault;
3269 }
3270
3271 /*
3272 * Locate the PDE to start displaying at.
3273 *
3274 * The 'dpda' command takes the address of a PDE, while the others are guest
3275 * virtual address which PDEs should be displayed. So, 'dpda' is rather simple
3276 * while the others require us to do all the tedious walking thru the paging
3277 * hierarchy to find the intended PDE.
3278 */
3279 unsigned iEntry = ~0U; /* The page directory index. ~0U for 'dpta'. */
3280 DBGCVAR VarGCPtr; /* The GC address corresponding to the current PDE (iEntry != ~0U). */
3281 DBGCVAR VarPDEAddr; /* The address of the current PDE. */
3282 unsigned cEntries; /* The number of entries to display. */
3283 unsigned cEntriesMax; /* The max number of entries to display. */
3284 int rc;
3285 if (pCmd->pszCmd[3] == 'a')
3286 {
3287 VarPDEAddr = paArgs[0];
3288 switch (VarPDEAddr.enmRangeType)
3289 {
3290 case DBGCVAR_RANGE_BYTES: cEntries = VarPDEAddr.u64Range / cbEntry; break;
3291 case DBGCVAR_RANGE_ELEMENTS: cEntries = VarPDEAddr.u64Range; break;
3292 default: cEntries = 10; break;
3293 }
3294 cEntriesMax = PAGE_SIZE / cbEntry;
3295 }
3296 else
3297 {
3298 /*
3299 * Determin the range.
3300 */
3301 switch (paArgs[0].enmRangeType)
3302 {
3303 case DBGCVAR_RANGE_BYTES: cEntries = paArgs[0].u64Range / PAGE_SIZE; break;
3304 case DBGCVAR_RANGE_ELEMENTS: cEntries = paArgs[0].u64Range; break;
3305 default: cEntries = 10; break;
3306 }
3307
3308 /*
3309 * Normalize the input address, it must be a flat GC address.
3310 */
3311 rc = pCmdHlp->pfnEval(pCmdHlp, &VarGCPtr, "%%(%Dv)", &paArgs[0]);
3312 if (VBOX_FAILURE(rc))
3313 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "%%(%Dv)", &paArgs[0]);
3314 if (VarGCPtr.enmType == DBGCVAR_TYPE_HC_FLAT)
3315 {
3316 VarGCPtr.u.GCFlat = (uintptr_t)VarGCPtr.u.pvHCFlat;
3317 VarGCPtr.enmType = DBGCVAR_TYPE_GC_FLAT;
3318 }
3319 if (fPAE)
3320 VarGCPtr.u.GCFlat &= ~(((RTGCPTR)1 << X86_PD_PAE_SHIFT) - 1);
3321 else
3322 VarGCPtr.u.GCFlat &= ~(((RTGCPTR)1 << X86_PD_SHIFT) - 1);
3323
3324 /*
3325 * Do the paging walk until we get to the page directory.
3326 */
3327 DBGCVAR VarCur;
3328 if (fGuest)
3329 DBGCVAR_INIT_GC_PHYS(&VarCur, cr3);
3330 else
3331 DBGCVAR_INIT_HC_PHYS(&VarCur, cr3);
3332 if (fLME)
3333 {
3334 /* Page Map Level 4 Lookup. */
3335 /* Check if it's a valid address first? */
3336 VarCur.u.u64Number &= X86_PTE_PAE_PG_MASK;
3337 VarCur.u.u64Number += (((uint64_t)VarGCPtr.u.GCFlat >> X86_PML4_SHIFT) & X86_PML4_MASK) * sizeof(X86PML4E);
3338 X86PML4E Pml4e;
3339 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pml4e, sizeof(Pml4e), &VarCur, NULL);
3340 if (VBOX_FAILURE(rc))
3341 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PML4E memory at %DV.\n", &VarCur);
3342 if (!Pml4e.n.u1Present)
3343 return DBGCCmdHlpPrintf(pCmdHlp, "Page directory pointer table is not present for %Dv.\n", &VarGCPtr);
3344
3345 VarCur.u.u64Number = Pml4e.u & X86_PML4E_PG_MASK;
3346 Assert(fPAE);
3347 }
3348 if (fPAE)
3349 {
3350 /* Page directory pointer table. */
3351 X86PDPE Pdpe;
3352 VarCur.u.u64Number += ((VarGCPtr.u.GCFlat >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK) * sizeof(Pdpe);
3353 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pdpe, sizeof(Pdpe), &VarCur, NULL);
3354 if (VBOX_FAILURE(rc))
3355 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PDPE memory at %DV.\n", &VarCur);
3356 if (!Pdpe.n.u1Present)
3357 return DBGCCmdHlpPrintf(pCmdHlp, "Page directory is not present for %Dv.\n", &VarGCPtr);
3358
3359 iEntry = (VarGCPtr.u.GCFlat >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK;
3360 VarPDEAddr = VarCur;
3361 VarPDEAddr.u.u64Number = Pdpe.u & X86_PDPE_PG_MASK;
3362 VarPDEAddr.u.u64Number += iEntry * sizeof(X86PDEPAE);
3363 }
3364 else
3365 {
3366 /* 32-bit legacy - CR3 == page directory. */
3367 iEntry = (VarGCPtr.u.GCFlat >> X86_PD_SHIFT) & X86_PD_MASK;
3368 VarPDEAddr = VarCur;
3369 VarPDEAddr.u.u64Number += iEntry * sizeof(X86PDE);
3370 }
3371 cEntriesMax = (PAGE_SIZE - iEntry) / cbEntry;
3372 iEntry /= cbEntry;
3373 }
3374
3375 /* adjust cEntries */
3376 cEntries = RT_MAX(1, cEntries);
3377 cEntries = RT_MIN(cEntries, cEntriesMax);
3378
3379 /*
3380 * The display loop.
3381 */
3382 DBGCCmdHlpPrintf(pCmdHlp, iEntry != ~0U ? "%DV (index %#x):\n" : "%DV:\n",
3383 &VarPDEAddr, iEntry);
3384 do
3385 {
3386 /*
3387 * Read.
3388 */
3389 X86PDEPAE Pde;
3390 Pde.u = 0;
3391 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pde, cbEntry, &VarPDEAddr, NULL);
3392 if (VBOX_FAILURE(rc))
3393 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Reading PDE memory at %DV.\n", &VarPDEAddr);
3394
3395 /*
3396 * Display.
3397 */
3398 if (iEntry != ~0U)
3399 {
3400 DBGCCmdHlpPrintf(pCmdHlp, "%03x %DV: ", iEntry, &VarGCPtr);
3401 iEntry++;
3402 }
3403 if (fPSE && Pde.b.u1Size)
3404 DBGCCmdHlpPrintf(pCmdHlp,
3405 fPAE
3406 ? "%016llx big phys=%016llx %s %s %s %s %s avl=%02x %s %s %s %s %s"
3407 : "%08llx big phys=%08llx %s %s %s %s %s avl=%02x %s %s %s %s %s",
3408 Pde.u,
3409 Pde.u & X86_PDE_PAE_PG_MASK,
3410 Pde.b.u1Present ? "p " : "np",
3411 Pde.b.u1Write ? "w" : "r",
3412 Pde.b.u1User ? "u" : "s",
3413 Pde.b.u1Accessed ? "a " : "na",
3414 Pde.b.u1Dirty ? "d " : "nd",
3415 Pde.b.u3Available,
3416 Pde.b.u1Global ? (fPGE ? "g" : "G") : " ",
3417 Pde.b.u1WriteThru ? "pwt" : " ",
3418 Pde.b.u1CacheDisable ? "pcd" : " ",
3419 Pde.b.u1PAT ? "pat" : "",
3420 Pde.b.u1NoExecute ? (fNXE ? "nx" : "NX") : " ");
3421 else
3422 DBGCCmdHlpPrintf(pCmdHlp,
3423 fPAE
3424 ? "%016llx 4kb phys=%016llx %s %s %s %s %s avl=%02x %s %s %s %s"
3425 : "%08llx 4kb phys=%08llx %s %s %s %s %s avl=%02x %s %s %s %s",
3426 Pde.u,
3427 Pde.u & X86_PDE_PAE_PG_MASK,
3428 Pde.n.u1Present ? "p " : "np",
3429 Pde.n.u1Write ? "w" : "r",
3430 Pde.n.u1User ? "u" : "s",
3431 Pde.n.u1Accessed ? "a " : "na",
3432 Pde.u & BIT(6) ? "6 " : " ",
3433 Pde.n.u3Available,
3434 Pde.u & BIT(8) ? "8" : " ",
3435 Pde.n.u1WriteThru ? "pwt" : " ",
3436 Pde.n.u1CacheDisable ? "pcd" : " ",
3437 Pde.u & BIT(7) ? "7" : "",
3438 Pde.n.u1NoExecute ? (fNXE ? "nx" : "NX") : " ");
3439 if (Pde.u & UINT64_C(0x7fff000000000000))
3440 DBGCCmdHlpPrintf(pCmdHlp, " weird=%RX64", (Pde.u & UINT64_C(0x7fff000000000000)));
3441 rc = DBGCCmdHlpPrintf(pCmdHlp, "\n");
3442 if (VBOX_FAILURE(rc))
3443 return rc;
3444
3445 /*
3446 * Advance.
3447 */
3448 VarPDEAddr.u.u64Number += cbEntry;
3449 if (iEntry != ~0U)
3450 VarGCPtr.u.GCFlat += 1 << (fPAE ? X86_PD_PAE_SHIFT : X86_PD_SHIFT);
3451 } while (cEntries-- > 0);
3452
3453 NOREF(pResult);
3454 return VINF_SUCCESS;
3455}
3456
3457
3458/**
3459 * The 'dpdb' command.
3460 *
3461 * @returns VBox status.
3462 * @param pCmd Pointer to the command descriptor (as registered).
3463 * @param pCmdHlp Pointer to command helper functions.
3464 * @param pVM Pointer to the current VM (if any).
3465 * @param paArgs Pointer to (readonly) array of arguments.
3466 * @param cArgs Number of arguments in the array.
3467 */
3468static DECLCALLBACK(int) dbgcCmdDumpPageDirBoth(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3469{
3470 if (!pVM)
3471 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
3472 int rc1 = pCmdHlp->pfnExec(pCmdHlp, "dpdg %DV", &paArgs[0]);
3473 int rc2 = pCmdHlp->pfnExec(pCmdHlp, "dpdh %DV", &paArgs[0]);
3474 if (VBOX_FAILURE(rc1))
3475 return rc1;
3476 NOREF(pCmd); NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
3477 return rc2;
3478}
3479
3480
3481/**
3482 * The 'dpg*' commands.
3483 *
3484 * @returns VBox status.
3485 * @param pCmd Pointer to the command descriptor (as registered).
3486 * @param pCmdHlp Pointer to command helper functions.
3487 * @param pVM Pointer to the current VM (if any).
3488 * @param paArgs Pointer to (readonly) array of arguments.
3489 * @param cArgs Number of arguments in the array.
3490 */
3491static DECLCALLBACK(int) dbgcCmdDumpPageTable(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3492{
3493 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
3494
3495 /*
3496 * Validate input.
3497 */
3498 if ( cArgs != 1
3499 || (pCmd->pszCmd[3] == 'a' && !DBGCVAR_ISPOINTER(paArgs[0].enmType))
3500 || (pCmd->pszCmd[3] != 'a' && !(paArgs[0].enmType == DBGCVAR_TYPE_NUMBER || DBGCVAR_ISPOINTER(paArgs[0].enmType)))
3501 )
3502 return DBGCCmdHlpPrintf(pCmdHlp, "internal error: The parser doesn't do its job properly yet.. It might help to use the '%%' operator.\n");
3503 if (!pVM)
3504 return DBGCCmdHlpPrintf(pCmdHlp, "error: No VM.\n");
3505
3506 /*
3507 * Guest or shadow page tables? Get the paging parameters.
3508 */
3509 bool fGuest = pCmd->pszCmd[3] != 'h';
3510 if (!pCmd->pszCmd[3] || pCmd->pszCmd[3] == 'a')
3511 fGuest = paArgs[0].enmType == DBGCVAR_TYPE_NUMBER
3512 ? pDbgc->fRegCtxGuest
3513 : DBGCVAR_ISGCPOINTER(paArgs[0].enmType);
3514
3515 bool fPAE, fLME, fPSE, fPGE, fNXE;
3516 uint64_t cr3 = fGuest
3517 ? dbgcGetGuestPageMode(pDbgc, &fPAE, &fLME, &fPSE, &fPGE, &fNXE)
3518 : dbgcGetShadowPageMode(pDbgc, &fPAE, &fLME, &fPSE, &fPGE, &fNXE);
3519 const unsigned cbEntry = fPAE ? sizeof(X86PTEPAE) : sizeof(X86PTE);
3520
3521 /*
3522 * Locate the PTE to start displaying at.
3523 *
3524 * The 'dpta' command takes the address of a PTE, while the others are guest
3525 * virtual address which PTEs should be displayed. So, 'pdta' is rather simple
3526 * while the others require us to do all the tedious walking thru the paging
3527 * hierarchy to find the intended PTE.
3528 */
3529 unsigned iEntry = ~0U; /* The page table index. ~0U for 'dpta'. */
3530 DBGCVAR VarGCPtr; /* The GC address corresponding to the current PTE (iEntry != ~0U). */
3531 DBGCVAR VarPTEAddr; /* The address of the current PTE. */
3532 unsigned cEntries; /* The number of entries to display. */
3533 unsigned cEntriesMax; /* The max number of entries to display. */
3534 int rc;
3535 if (pCmd->pszCmd[3] == 'a')
3536 {
3537 VarPTEAddr = paArgs[0];
3538 switch (VarPTEAddr.enmRangeType)
3539 {
3540 case DBGCVAR_RANGE_BYTES: cEntries = VarPTEAddr.u64Range / cbEntry; break;
3541 case DBGCVAR_RANGE_ELEMENTS: cEntries = VarPTEAddr.u64Range; break;
3542 default: cEntries = 10; break;
3543 }
3544 cEntriesMax = PAGE_SIZE / cbEntry;
3545 }
3546 else
3547 {
3548 /*
3549 * Determin the range.
3550 */
3551 switch (paArgs[0].enmRangeType)
3552 {
3553 case DBGCVAR_RANGE_BYTES: cEntries = paArgs[0].u64Range / PAGE_SIZE; break;
3554 case DBGCVAR_RANGE_ELEMENTS: cEntries = paArgs[0].u64Range; break;
3555 default: cEntries = 10; break;
3556 }
3557
3558 /*
3559 * Normalize the input address, it must be a flat GC address.
3560 */
3561 rc = pCmdHlp->pfnEval(pCmdHlp, &VarGCPtr, "%%(%Dv)", &paArgs[0]);
3562 if (VBOX_FAILURE(rc))
3563 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "%%(%Dv)", &paArgs[0]);
3564 if (VarGCPtr.enmType == DBGCVAR_TYPE_HC_FLAT)
3565 {
3566 VarGCPtr.u.GCFlat = (uintptr_t)VarGCPtr.u.pvHCFlat;
3567 VarGCPtr.enmType = DBGCVAR_TYPE_GC_FLAT;
3568 }
3569 VarGCPtr.u.GCFlat &= ~(RTGCPTR)PAGE_OFFSET_MASK;
3570
3571 /*
3572 * Do the paging walk until we get to the page table.
3573 */
3574 DBGCVAR VarCur;
3575 if (fGuest)
3576 DBGCVAR_INIT_GC_PHYS(&VarCur, cr3);
3577 else
3578 DBGCVAR_INIT_HC_PHYS(&VarCur, cr3);
3579 if (fLME)
3580 {
3581 /* Page Map Level 4 Lookup. */
3582 /* Check if it's a valid address first? */
3583 VarCur.u.u64Number &= X86_PTE_PAE_PG_MASK;
3584 VarCur.u.u64Number += (((uint64_t)VarGCPtr.u.GCFlat >> X86_PML4_SHIFT) & X86_PML4_MASK) * sizeof(X86PML4E);
3585 X86PML4E Pml4e;
3586 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pml4e, sizeof(Pml4e), &VarCur, NULL);
3587 if (VBOX_FAILURE(rc))
3588 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PML4E memory at %DV.\n", &VarCur);
3589 if (!Pml4e.n.u1Present)
3590 return DBGCCmdHlpPrintf(pCmdHlp, "Page directory pointer table is not present for %Dv.\n", &VarGCPtr);
3591
3592 VarCur.u.u64Number = Pml4e.u & X86_PML4E_PG_MASK;
3593 Assert(fPAE);
3594 }
3595 if (fPAE)
3596 {
3597 /* Page directory pointer table. */
3598 X86PDPE Pdpe;
3599 VarCur.u.u64Number += ((VarGCPtr.u.GCFlat >> X86_PDPTR_SHIFT) & X86_PDPTR_MASK) * sizeof(Pdpe);
3600 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pdpe, sizeof(Pdpe), &VarCur, NULL);
3601 if (VBOX_FAILURE(rc))
3602 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PDPE memory at %DV.\n", &VarCur);
3603 if (!Pdpe.n.u1Present)
3604 return DBGCCmdHlpPrintf(pCmdHlp, "Page directory is not present for %Dv.\n", &VarGCPtr);
3605
3606 VarCur.u.u64Number = Pdpe.u & X86_PDPE_PG_MASK;
3607
3608 /* Page directory (PAE). */
3609 X86PDEPAE Pde;
3610 VarCur.u.u64Number += ((VarGCPtr.u.GCFlat >> X86_PD_PAE_SHIFT) & X86_PD_PAE_MASK) * sizeof(Pde);
3611 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pde, sizeof(Pde), &VarCur, NULL);
3612 if (VBOX_FAILURE(rc))
3613 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PDE memory at %DV.\n", &VarCur);
3614 if (!Pde.n.u1Present)
3615 return DBGCCmdHlpPrintf(pCmdHlp, "Page table is not present for %Dv.\n", &VarGCPtr);
3616 if (fPSE && Pde.n.u1Size)
3617 return pCmdHlp->pfnExec(pCmdHlp, "dpd%s %Dv L3", &pCmd->pszCmd[3], &VarGCPtr);
3618
3619 iEntry = (VarGCPtr.u.GCFlat >> X86_PT_PAE_SHIFT) & X86_PT_PAE_MASK;
3620 VarPTEAddr = VarCur;
3621 VarPTEAddr.u.u64Number = Pde.u & X86_PDE_PAE_PG_MASK;
3622 VarPTEAddr.u.u64Number += iEntry * sizeof(X86PTEPAE);
3623 }
3624 else
3625 {
3626 /* Page directory (legacy). */
3627 X86PDE Pde;
3628 VarCur.u.u64Number += ((VarGCPtr.u.GCFlat >> X86_PD_SHIFT) & X86_PD_MASK) * sizeof(Pde);
3629 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pde, sizeof(Pde), &VarCur, NULL);
3630 if (VBOX_FAILURE(rc))
3631 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PDE memory at %DV.\n", &VarCur);
3632 if (!Pde.n.u1Present)
3633 return DBGCCmdHlpPrintf(pCmdHlp, "Page table is not present for %Dv.\n", &VarGCPtr);
3634 if (fPSE && Pde.n.u1Size)
3635 return pCmdHlp->pfnExec(pCmdHlp, "dpd%s %Dv L3", &pCmd->pszCmd[3], &VarGCPtr);
3636
3637 iEntry = (VarGCPtr.u.GCFlat >> X86_PT_SHIFT) & X86_PT_MASK;
3638 VarPTEAddr = VarCur;
3639 VarPTEAddr.u.u64Number = Pde.u & X86_PDE_PG_MASK;
3640 VarPTEAddr.u.u64Number += iEntry * sizeof(X86PTE);
3641 }
3642 cEntriesMax = (PAGE_SIZE - iEntry) / cbEntry;
3643 iEntry /= cbEntry;
3644 }
3645
3646 /* adjust cEntries */
3647 cEntries = RT_MAX(1, cEntries);
3648 cEntries = RT_MIN(cEntries, cEntriesMax);
3649
3650 /*
3651 * The display loop.
3652 */
3653 DBGCCmdHlpPrintf(pCmdHlp, iEntry != ~0U ? "%DV (base %DV / index %#x):\n" : "%DV:\n",
3654 &VarPTEAddr, &VarGCPtr, iEntry);
3655 do
3656 {
3657 /*
3658 * Read.
3659 */
3660 X86PTEPAE Pte;
3661 Pte.u = 0;
3662 rc = pCmdHlp->pfnMemRead(pCmdHlp, pVM, &Pte, cbEntry, &VarPTEAddr, NULL);
3663 if (VBOX_FAILURE(rc))
3664 return DBGCCmdHlpVBoxError(pCmdHlp, rc, "Reading PTE memory at %DV.\n", &VarPTEAddr);
3665
3666 /*
3667 * Display.
3668 */
3669 if (iEntry != ~0U)
3670 {
3671 DBGCCmdHlpPrintf(pCmdHlp, "%03x %DV: ", iEntry, &VarGCPtr);
3672 iEntry++;
3673 }
3674 DBGCCmdHlpPrintf(pCmdHlp,
3675 fPAE
3676 ? "%016llx 4kb phys=%016llx %s %s %s %s %s avl=%02x %s %s %s %s %s"
3677 : "%08llx 4kb phys=%08llx %s %s %s %s %s avl=%02x %s %s %s %s %s",
3678 Pte.u,
3679 Pte.u & X86_PTE_PAE_PG_MASK,
3680 Pte.n.u1Present ? "p " : "np",
3681 Pte.n.u1Write ? "w" : "r",
3682 Pte.n.u1User ? "u" : "s",
3683 Pte.n.u1Accessed ? "a " : "na",
3684 Pte.n.u1Dirty ? "d " : "nd",
3685 Pte.n.u3Available,
3686 Pte.n.u1Global ? (fPGE ? "g" : "G") : " ",
3687 Pte.n.u1WriteThru ? "pwt" : " ",
3688 Pte.n.u1CacheDisable ? "pcd" : " ",
3689 Pte.n.u1PAT ? "pat" : " ",
3690 Pte.n.u1NoExecute ? (fNXE ? "nx" : "NX") : " "
3691 );
3692 if (Pte.u & UINT64_C(0x7fff000000000000))
3693 DBGCCmdHlpPrintf(pCmdHlp, " weird=%RX64", (Pte.u & UINT64_C(0x7fff000000000000)));
3694 rc = DBGCCmdHlpPrintf(pCmdHlp, "\n");
3695 if (VBOX_FAILURE(rc))
3696 return rc;
3697
3698 /*
3699 * Advance.
3700 */
3701 VarPTEAddr.u.u64Number += cbEntry;
3702 if (iEntry != ~0U)
3703 VarGCPtr.u.GCFlat += PAGE_SIZE;
3704 } while (cEntries-- > 0);
3705
3706 NOREF(pResult);
3707 return VINF_SUCCESS;
3708}
3709
3710
3711/**
3712 * The 'dptb' command.
3713 *
3714 * @returns VBox status.
3715 * @param pCmd Pointer to the command descriptor (as registered).
3716 * @param pCmdHlp Pointer to command helper functions.
3717 * @param pVM Pointer to the current VM (if any).
3718 * @param paArgs Pointer to (readonly) array of arguments.
3719 * @param cArgs Number of arguments in the array.
3720 */
3721static DECLCALLBACK(int) dbgcCmdDumpPageTableBoth(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3722{
3723 if (!pVM)
3724 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
3725 int rc1 = pCmdHlp->pfnExec(pCmdHlp, "dptg %DV", &paArgs[0]);
3726 int rc2 = pCmdHlp->pfnExec(pCmdHlp, "dpth %DV", &paArgs[0]);
3727 if (VBOX_FAILURE(rc1))
3728 return rc1;
3729 NOREF(pCmd); NOREF(cArgs); NOREF(pResult);
3730 return rc2;
3731}
3732
3733
3734/**
3735 * The 'dt' command.
3736 *
3737 * @returns VBox status.
3738 * @param pCmd Pointer to the command descriptor (as registered).
3739 * @param pCmdHlp Pointer to command helper functions.
3740 * @param pVM Pointer to the current VM (if any).
3741 * @param paArgs Pointer to (readonly) array of arguments.
3742 * @param cArgs Number of arguments in the array.
3743 */
3744static DECLCALLBACK(int) dbgcCmdDumpTSS(PCDBGCCMD /*pCmd*/, PDBGCCMDHLP pCmdHlp, PVM /*pVM*/, PCDBGCVAR /*paArgs*/, unsigned /*cArgs*/, PDBGCVAR /*pResult*/)
3745{
3746 /*
3747 * We can get a TSS selector (number), a far pointer using a TSS selector, or some kind of TSS pointer.
3748 */
3749
3750 /** @todo */
3751 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "dt is not implemented yet, feel free to do it. \n");
3752}
3753
3754
3755/**
3756 * The 'm' command.
3757 *
3758 * @returns VBox status.
3759 * @param pCmd Pointer to the command descriptor (as registered).
3760 * @param pCmdHlp Pointer to command helper functions.
3761 * @param pVM Pointer to the current VM (if any).
3762 * @param paArgs Pointer to (readonly) array of arguments.
3763 * @param cArgs Number of arguments in the array.
3764 */
3765static DECLCALLBACK(int) dbgcCmdMemoryInfo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3766{
3767 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Address: %DV\n", &paArgs[0]);
3768 if (!pVM)
3769 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
3770 int rc1 = pCmdHlp->pfnExec(pCmdHlp, "dpdg %DV", &paArgs[0]);
3771 int rc2 = pCmdHlp->pfnExec(pCmdHlp, "dpdh %DV", &paArgs[0]);
3772 int rc3 = pCmdHlp->pfnExec(pCmdHlp, "dptg %DV", &paArgs[0]);
3773 int rc4 = pCmdHlp->pfnExec(pCmdHlp, "dpth %DV", &paArgs[0]);
3774 if (VBOX_FAILURE(rc1))
3775 return rc1;
3776 if (VBOX_FAILURE(rc2))
3777 return rc2;
3778 if (VBOX_FAILURE(rc3))
3779 return rc3;
3780 NOREF(pCmd); NOREF(cArgs); NOREF(pResult);
3781 return rc4;
3782}
3783
3784
3785/**
3786 * The 'echo' command.
3787 *
3788 * @returns VBox status.
3789 * @param pCmd Pointer to the command descriptor (as registered).
3790 * @param pCmdHlp Pointer to command helper functions.
3791 * @param pVM Pointer to the current VM (if any).
3792 * @param paArgs Pointer to (readonly) array of arguments.
3793 * @param cArgs Number of arguments in the array.
3794 */
3795static DECLCALLBACK(int) dbgcCmdEcho(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3796{
3797 /*
3798 * Loop thru the arguments and print them with one space between.
3799 */
3800 int rc = 0;
3801 for (unsigned i = 0; i < cArgs; i++)
3802 {
3803 if (paArgs[i].enmType == DBGCVAR_TYPE_STRING)
3804 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, i ? " %s" : "%s", paArgs[i].u.pszString);
3805 else
3806 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, i ? " <parser error>" : "<parser error>");
3807 if (VBOX_FAILURE(rc))
3808 return rc;
3809 }
3810 NOREF(pCmd); NOREF(pResult); NOREF(pVM);
3811 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
3812}
3813
3814
3815/**
3816 * The 'echo' command.
3817 *
3818 * @returns VBox status.
3819 * @param pCmd Pointer to the command descriptor (as registered).
3820 * @param pCmdHlp Pointer to command helper functions.
3821 * @param pVM Pointer to the current VM (if any).
3822 * @param paArgs Pointer to (readonly) array of arguments.
3823 * @param cArgs Number of arguments in the array.
3824 */
3825static DECLCALLBACK(int) dbgcCmdRunScript(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3826{
3827 /* check that the parser did what it's supposed to do. */
3828 if ( cArgs != 1
3829 || paArgs[0].enmType != DBGCVAR_TYPE_STRING)
3830 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "parser error\n");
3831
3832 /*
3833 * Try open the script.
3834 */
3835 const char *pszFilename = paArgs[0].u.pszString;
3836 FILE *pFile = fopen(pszFilename, "r");
3837 if (!pFile)
3838 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Failed to open '%s'.\n", pszFilename);
3839
3840 /*
3841 * Execute it line by line.
3842 */
3843 int rc = 0;
3844 unsigned iLine = 0;
3845 char szLine[8192];
3846 while (fgets(szLine, sizeof(szLine), pFile))
3847 {
3848 /* check that the line isn't too long. */
3849 char *pszEnd = strchr(szLine, '\0');
3850 if (pszEnd == &szLine[sizeof(szLine) - 1])
3851 {
3852 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "runscript error: Line #%u is too long\n", iLine);
3853 break;
3854 }
3855 iLine++;
3856
3857 /* strip leading blanks and check for comment / blank line. */
3858 char *psz = RTStrStripL(szLine);
3859 if ( *psz == '\0'
3860 || *psz == '\n'
3861 || *psz == '#')
3862 continue;
3863
3864 /* strip trailing blanks and check for empty line (\r case). */
3865 while ( pszEnd > psz
3866 && isspace(pszEnd[-1])) /* isspace includes \n and \r normally. */
3867 *--pszEnd = '\0';
3868
3869 /** @todo check for Control-C / Cancel at this point... */
3870
3871 /*
3872 * Execute the command.
3873 *
3874 * This is a bit wasteful with scratch space btw., can fix it later.
3875 * The whole return code crap should be fixed too, so that it's possible
3876 * to know whether a command succeeded (VBOX_SUCCESS()) or failed, and
3877 * more importantly why it failed.
3878 */
3879 rc = pCmdHlp->pfnExec(pCmdHlp, "%s", psz);
3880 if (VBOX_FAILURE(rc))
3881 {
3882 if (rc == VERR_BUFFER_OVERFLOW)
3883 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "runscript error: Line #%u is too long (exec overflowed)\n", iLine);
3884 break;
3885 }
3886 if (rc == VWRN_DBGC_CMD_PENDING)
3887 {
3888 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "runscript error: VWRN_DBGC_CMD_PENDING on line #%u, script terminated\n", iLine);
3889 break;
3890 }
3891 }
3892
3893 fclose(pFile);
3894
3895 NOREF(pCmd); NOREF(pResult); NOREF(pVM);
3896 return rc;
3897}
3898
3899
3900
3901/**
3902 * Print formatted string.
3903 *
3904 * @param pHlp Pointer to this structure.
3905 * @param pszFormat The format string.
3906 * @param ... Arguments.
3907 */
3908static DECLCALLBACK(void) dbgcCmdInfo_Printf(PCDBGFINFOHLP pHlp, const char *pszFormat, ...)
3909{
3910 PDBGCCMDHLP pCmdHlp = *(PDBGCCMDHLP *)(pHlp + 1);
3911 va_list args;
3912 va_start(args, pszFormat);
3913 pCmdHlp->pfnPrintfV(pCmdHlp, NULL, pszFormat, args);
3914 va_end(args);
3915}
3916
3917
3918/**
3919 * Print formatted string.
3920 *
3921 * @param pHlp Pointer to this structure.
3922 * @param pszFormat The format string.
3923 * @param args Argument list.
3924 */
3925static DECLCALLBACK(void) dbgcCmdInfo_PrintfV(PCDBGFINFOHLP pHlp, const char *pszFormat, va_list args)
3926{
3927 PDBGCCMDHLP pCmdHlp = *(PDBGCCMDHLP *)(pHlp + 1);
3928 pCmdHlp->pfnPrintfV(pCmdHlp, NULL, pszFormat, args);
3929}
3930
3931
3932/**
3933 * The 'info' command.
3934 *
3935 * @returns VBox status.
3936 * @param pCmd Pointer to the command descriptor (as registered).
3937 * @param pCmdHlp Pointer to command helper functions.
3938 * @param pVM Pointer to the current VM (if any).
3939 * @param paArgs Pointer to (readonly) array of arguments.
3940 * @param cArgs Number of arguments in the array.
3941 */
3942static DECLCALLBACK(int) dbgcCmdInfo(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3943{
3944 /*
3945 * Validate input.
3946 */
3947 if ( cArgs < 1
3948 || cArgs > 2
3949 || paArgs[0].enmType != DBGCVAR_TYPE_STRING
3950 || paArgs[cArgs - 1].enmType != DBGCVAR_TYPE_STRING)
3951 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "internal error: The parser doesn't do its job properly yet.. quote the string.\n");
3952 if (!pVM)
3953 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: No VM.\n");
3954
3955 /*
3956 * Dump it.
3957 */
3958 struct
3959 {
3960 DBGFINFOHLP Hlp;
3961 PDBGCCMDHLP pCmdHlp;
3962 } Hlp = { { dbgcCmdInfo_Printf, dbgcCmdInfo_PrintfV }, pCmdHlp };
3963 int rc = DBGFR3Info(pVM, paArgs[0].u.pszString, cArgs == 2 ? paArgs[1].u.pszString : NULL, &Hlp.Hlp);
3964 if (VBOX_FAILURE(rc))
3965 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3Info()\n");
3966
3967 NOREF(pCmd); NOREF(pResult);
3968 return 0;
3969}
3970
3971
3972/**
3973 * The 'log' command.
3974 *
3975 * @returns VBox status.
3976 * @param pCmd Pointer to the command descriptor (as registered).
3977 * @param pCmdHlp Pointer to command helper functions.
3978 * @param pVM Pointer to the current VM (if any).
3979 * @param paArgs Pointer to (readonly) array of arguments.
3980 * @param cArgs Number of arguments in the array.
3981 */
3982static DECLCALLBACK(int) dbgcCmdLog(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
3983{
3984 int rc = DBGFR3LogModifyGroups(pVM, paArgs[0].u.pszString);
3985 if (VBOX_SUCCESS(rc))
3986 return VINF_SUCCESS;
3987 NOREF(pCmd); NOREF(cArgs); NOREF(pResult);
3988 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3LogModifyGroups(%p,'%s')\n", pVM, paArgs[0].u.pszString);
3989}
3990
3991
3992/**
3993 * The 'logdest' command.
3994 *
3995 * @returns VBox status.
3996 * @param pCmd Pointer to the command descriptor (as registered).
3997 * @param pCmdHlp Pointer to command helper functions.
3998 * @param pVM Pointer to the current VM (if any).
3999 * @param paArgs Pointer to (readonly) array of arguments.
4000 * @param cArgs Number of arguments in the array.
4001 */
4002static DECLCALLBACK(int) dbgcCmdLogDest(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4003{
4004 int rc = DBGFR3LogModifyDestinations(pVM, paArgs[0].u.pszString);
4005 if (VBOX_SUCCESS(rc))
4006 return VINF_SUCCESS;
4007 NOREF(pCmd); NOREF(cArgs); NOREF(pResult);
4008 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3LogModifyDestinations(%p,'%s')\n", pVM, paArgs[0].u.pszString);
4009}
4010
4011
4012/**
4013 * The 'logflags' command.
4014 *
4015 * @returns VBox status.
4016 * @param pCmd Pointer to the command descriptor (as registered).
4017 * @param pCmdHlp Pointer to command helper functions.
4018 * @param pVM Pointer to the current VM (if any).
4019 * @param paArgs Pointer to (readonly) array of arguments.
4020 * @param cArgs Number of arguments in the array.
4021 */
4022static DECLCALLBACK(int) dbgcCmdLogFlags(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4023{
4024 int rc = DBGFR3LogModifyFlags(pVM, paArgs[0].u.pszString);
4025 if (VBOX_SUCCESS(rc))
4026 return VINF_SUCCESS;
4027 NOREF(pCmd); NOREF(cArgs); NOREF(pResult);
4028 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3LogModifyFlags(%p,'%s')\n", pVM, paArgs[0].u.pszString);
4029}
4030
4031
4032/**
4033 * The 'format' command.
4034 *
4035 * @returns VBox status.
4036 * @param pCmd Pointer to the command descriptor (as registered).
4037 * @param pCmdHlp Pointer to command helper functions.
4038 * @param pVM Pointer to the current VM (if any).
4039 * @param paArgs Pointer to (readonly) array of arguments.
4040 * @param cArgs Number of arguments in the array.
4041 */
4042static DECLCALLBACK(int) dbgcCmdFormat(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4043{
4044 LogFlow(("dbgcCmdFormat\n"));
4045 static const char *apszRangeDesc[] =
4046 {
4047 "none", "bytes", "elements"
4048 };
4049 int rc;
4050
4051 for (unsigned iArg = 0; iArg < cArgs; iArg++)
4052 {
4053 switch (paArgs[iArg].enmType)
4054 {
4055 case DBGCVAR_TYPE_UNKNOWN:
4056 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4057 "Unknown variable type!\n");
4058 break;
4059 case DBGCVAR_TYPE_GC_FLAT:
4060 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4061 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4062 "Guest flat address: %%%08x range %lld %s\n",
4063 paArgs[iArg].u.GCFlat,
4064 paArgs[iArg].u64Range,
4065 apszRangeDesc[paArgs[iArg].enmRangeType]);
4066 else
4067 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4068 "Guest flat address: %%%08x\n",
4069 paArgs[iArg].u.GCFlat);
4070 break;
4071 case DBGCVAR_TYPE_GC_FAR:
4072 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4073 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4074 "Guest far address: %04x:%08x range %lld %s\n",
4075 paArgs[iArg].u.GCFar.sel,
4076 paArgs[iArg].u.GCFar.off,
4077 paArgs[iArg].u64Range,
4078 apszRangeDesc[paArgs[iArg].enmRangeType]);
4079 else
4080 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4081 "Guest far address: %04x:%08x\n",
4082 paArgs[iArg].u.GCFar.sel,
4083 paArgs[iArg].u.GCFar.off);
4084 break;
4085 case DBGCVAR_TYPE_GC_PHYS:
4086 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4087 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4088 "Guest physical address: %%%%%08x range %lld %s\n",
4089 paArgs[iArg].u.GCPhys,
4090 paArgs[iArg].u64Range,
4091 apszRangeDesc[paArgs[iArg].enmRangeType]);
4092 else
4093 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4094 "Guest physical address: %%%%%08x\n",
4095 paArgs[iArg].u.GCPhys);
4096 break;
4097 case DBGCVAR_TYPE_HC_FLAT:
4098 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4099 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4100 "Host flat address: %%%08x range %lld %s\n",
4101 paArgs[iArg].u.pvHCFlat,
4102 paArgs[iArg].u64Range,
4103 apszRangeDesc[paArgs[iArg].enmRangeType]);
4104 else
4105 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4106 "Host flat address: %%%08x\n",
4107 paArgs[iArg].u.pvHCFlat);
4108 break;
4109 case DBGCVAR_TYPE_HC_FAR:
4110 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4111 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4112 "Host far address: %04x:%08x range %lld %s\n",
4113 paArgs[iArg].u.HCFar.sel,
4114 paArgs[iArg].u.HCFar.off,
4115 paArgs[iArg].u64Range,
4116 apszRangeDesc[paArgs[iArg].enmRangeType]);
4117 else
4118 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4119 "Host far address: %04x:%08x\n",
4120 paArgs[iArg].u.HCFar.sel,
4121 paArgs[iArg].u.HCFar.off);
4122 break;
4123 case DBGCVAR_TYPE_HC_PHYS:
4124 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4125 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4126 "Host physical address: %VHp range %lld %s\n",
4127 paArgs[iArg].u.HCPhys,
4128 paArgs[iArg].u64Range,
4129 apszRangeDesc[paArgs[iArg].enmRangeType]);
4130 else
4131 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4132 "Host physical address: %VHp\n",
4133 paArgs[iArg].u.HCPhys);
4134 break;
4135
4136 case DBGCVAR_TYPE_STRING:
4137 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4138 "String, %lld bytes long: %s\n",
4139 paArgs[iArg].u64Range,
4140 paArgs[iArg].u.pszString);
4141 break;
4142
4143 case DBGCVAR_TYPE_NUMBER:
4144 if (paArgs[iArg].enmRangeType != DBGCVAR_RANGE_NONE)
4145 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4146 "Number: hex %llx dec 0i%lld oct 0t%llo range %lld %s\n",
4147 paArgs[iArg].u.u64Number,
4148 paArgs[iArg].u.u64Number,
4149 paArgs[iArg].u.u64Number,
4150 paArgs[iArg].u64Range,
4151 apszRangeDesc[paArgs[iArg].enmRangeType]);
4152 else
4153 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4154 "Number: hex %llx dec 0i%lld oct 0t%llo\n",
4155 paArgs[iArg].u.u64Number,
4156 paArgs[iArg].u.u64Number,
4157 paArgs[iArg].u.u64Number);
4158 break;
4159
4160 default:
4161 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4162 "Invalid argument type %d\n",
4163 paArgs[iArg].enmType);
4164 break;
4165 }
4166 } /* arg loop */
4167
4168 NOREF(pCmd); NOREF(pVM); NOREF(pResult);
4169 return 0;
4170}
4171
4172
4173/**
4174 * List near symbol.
4175 *
4176 * @returns VBox status code.
4177 * @param pCmdHlp Pointer to command helper functions.
4178 * @param pVM Pointer to the current VM (if any).
4179 * @param pArg Pointer to the address or symbol to lookup.
4180 */
4181static int dbgcDoListNear(PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR pArg, PDBGCVAR pResult)
4182{
4183 dbgcVarSetGCFlat(pResult, 0);
4184
4185 DBGFSYMBOL Symbol;
4186 int rc;
4187 if (pArg->enmType == DBGCVAR_TYPE_SYMBOL)
4188 {
4189 /*
4190 * Lookup the symbol address.
4191 */
4192 rc = DBGFR3SymbolByName(pVM, pArg->u.pszString, &Symbol);
4193 if (VBOX_FAILURE(rc))
4194 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3SymbolByName(, %s,)\n", pArg->u.pszString);
4195
4196 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%VGv %s\n", (RTGCUINTPTR)Symbol.Value, Symbol.szName); /** @todo remove the RTUINGCPTR cast once DBGF got correct interfaces! */
4197 dbgcVarSetGCFlatByteRange(pResult, Symbol.Value, Symbol.cb);
4198 }
4199 else
4200 {
4201 /*
4202 * Convert it to a flat GC address and lookup that address.
4203 */
4204 DBGCVAR AddrVar;
4205 rc = pCmdHlp->pfnEval(pCmdHlp, &AddrVar, "%%(%DV)", pArg);
4206 if (VBOX_FAILURE(rc))
4207 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "%%(%DV)\n", pArg);
4208
4209 dbgcVarSetVar(pResult, &AddrVar);
4210
4211 RTGCINTPTR offDisp = 0;
4212 rc = DBGFR3SymbolByAddr(pVM, AddrVar.u.GCFlat, &offDisp, &Symbol);
4213 if (VBOX_FAILURE(rc))
4214 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGFR3SymbolByAddr(, %VGv,,)\n", AddrVar.u.GCFlat);
4215
4216 if (!offDisp)
4217 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV %s", &AddrVar, Symbol.szName);
4218 else if (offDisp > 0)
4219 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV %s + %RGv", &AddrVar, Symbol.szName, offDisp);
4220 else
4221 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%DV %s - %RGv", &AddrVar, Symbol.szName, -offDisp);
4222 if ((RTGCINTPTR)Symbol.cb > -offDisp)
4223 {
4224 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, " LB %RGv\n", Symbol.cb + offDisp);
4225 dbgcVarSetByteRange(pResult, Symbol.cb + offDisp);
4226 }
4227 else
4228 {
4229 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "\n");
4230 dbgcVarSetNoRange(pResult);
4231 }
4232 }
4233
4234 return rc;
4235}
4236
4237
4238/**
4239 * The 'ln' (listnear) command.
4240 *
4241 * @returns VBox status.
4242 * @param pCmd Pointer to the command descriptor (as registered).
4243 * @param pCmdHlp Pointer to command helper functions.
4244 * @param pVM Pointer to the current VM (if any).
4245 * @param paArgs Pointer to (readonly) array of arguments.
4246 * @param cArgs Number of arguments in the array.
4247 */
4248static DECLCALLBACK(int) dbgcCmdListNear(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4249{
4250 dbgcVarSetGCFlat(pResult, 0);
4251 if (!cArgs)
4252 {
4253 /*
4254 * Current cs:eip symbol.
4255 */
4256 DBGCVAR AddrVar;
4257 int rc = pCmdHlp->pfnEval(pCmdHlp, &AddrVar, "%%(cs:eip)");
4258 if (VBOX_FAILURE(rc))
4259 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "%%(cs:eip)\n");
4260 return dbgcDoListNear(pCmdHlp, pVM, &AddrVar, pResult);
4261 }
4262
4263 /*
4264 * Iterate arguments.
4265 */
4266 for (unsigned iArg = 0; iArg < cArgs; iArg++)
4267 {
4268 int rc = dbgcDoListNear(pCmdHlp, pVM, &paArgs[iArg], pResult);
4269 if (VBOX_FAILURE(rc))
4270 return rc;
4271 }
4272
4273 NOREF(pCmd); NOREF(pResult);
4274 return VINF_SUCCESS;
4275}
4276
4277
4278/**
4279 * The 'loadsyms' command.
4280 *
4281 * @returns VBox status.
4282 * @param pCmd Pointer to the command descriptor (as registered).
4283 * @param pCmdHlp Pointer to command helper functions.
4284 * @param pVM Pointer to the current VM (if any).
4285 * @param paArgs Pointer to (readonly) array of arguments.
4286 * @param cArgs Number of arguments in the array.
4287 */
4288static DECLCALLBACK(int) dbgcCmdLoadSyms(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4289{
4290 /*
4291 * Validate the parsing and make sense of the input.
4292 * This is a mess as usual because we don't trust the parser yet.
4293 */
4294 if ( cArgs < 1
4295 || paArgs[0].enmType != DBGCVAR_TYPE_STRING)
4296 {
4297 AssertMsgFailed(("Parse error, first argument required to be string!\n"));
4298 return VERR_PARSE_INCORRECT_ARG_TYPE;
4299 }
4300 DBGCVAR AddrVar;
4301 RTGCUINTPTR Delta = 0;
4302 const char *pszModule = NULL;
4303 RTGCUINTPTR ModuleAddress = 0;
4304 unsigned cbModule = 0;
4305 if (cArgs > 1)
4306 {
4307 unsigned iArg = 1;
4308 if (paArgs[iArg].enmType == DBGCVAR_TYPE_NUMBER)
4309 {
4310 Delta = (RTGCUINTPTR)paArgs[iArg].u.u64Number;
4311 iArg++;
4312 }
4313 if (iArg < cArgs)
4314 {
4315 if (paArgs[iArg].enmType != DBGCVAR_TYPE_STRING)
4316 {
4317 AssertMsgFailed(("Parse error, module argument required to be string!\n"));
4318 return VERR_PARSE_INCORRECT_ARG_TYPE;
4319 }
4320 pszModule = paArgs[iArg].u.pszString;
4321 iArg++;
4322 if (iArg < cArgs)
4323 {
4324 if (DBGCVAR_ISPOINTER(paArgs[iArg].enmType))
4325 {
4326 AssertMsgFailed(("Parse error, module argument required to be GC pointer!\n"));
4327 return VERR_PARSE_INCORRECT_ARG_TYPE;
4328 }
4329 int rc = pCmdHlp->pfnEval(pCmdHlp, &AddrVar, "%%(%Dv)", &paArgs[iArg]);
4330 if (VBOX_FAILURE(rc))
4331 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "Module address cast %%(%Dv) failed.", &paArgs[iArg]);
4332 ModuleAddress = paArgs[iArg].u.GCFlat;
4333 iArg++;
4334 if (iArg < cArgs)
4335 {
4336 if (paArgs[iArg].enmType != DBGCVAR_TYPE_NUMBER)
4337 {
4338 AssertMsgFailed(("Parse error, module argument required to be an interger!\n"));
4339 return VERR_PARSE_INCORRECT_ARG_TYPE;
4340 }
4341 cbModule = (unsigned)paArgs[iArg].u.u64Number;
4342 iArg++;
4343 if (iArg < cArgs)
4344 {
4345 AssertMsgFailed(("Parse error, too many arguments!\n"));
4346 return VERR_PARSE_TOO_MANY_ARGUMENTS;
4347 }
4348 }
4349 }
4350 }
4351 }
4352
4353 /*
4354 * Call the debug info manager about this loading...
4355 */
4356 int rc = DBGFR3ModuleLoad(pVM, paArgs[0].u.pszString, Delta, pszModule, ModuleAddress, cbModule);
4357 if (VBOX_FAILURE(rc))
4358 return pCmdHlp->pfnVBoxError(pCmdHlp, rc, "DBGInfoSymbolLoad(, '%s', %VGv, '%s', %VGv, 0)\n",
4359 paArgs[0].u.pszString, Delta, pszModule, ModuleAddress);
4360
4361 NOREF(pCmd); NOREF(pResult);
4362 return VINF_SUCCESS;
4363}
4364
4365
4366/**
4367 * The 'set' command.
4368 *
4369 * @returns VBox status.
4370 * @param pCmd Pointer to the command descriptor (as registered).
4371 * @param pCmdHlp Pointer to command helper functions.
4372 * @param pVM Pointer to the current VM (if any).
4373 * @param paArgs Pointer to (readonly) array of arguments.
4374 * @param cArgs Number of arguments in the array.
4375 */
4376static DECLCALLBACK(int) dbgcCmdSet(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4377{
4378 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
4379
4380 /* parse sanity check. */
4381 AssertMsg(paArgs[0].enmType == DBGCVAR_TYPE_STRING, ("expected string not %d as first arg!\n", paArgs[0].enmType));
4382 if (paArgs[0].enmType != DBGCVAR_TYPE_STRING)
4383 return VERR_PARSE_INCORRECT_ARG_TYPE;
4384
4385
4386 /*
4387 * A variable must start with an alpha chars and only contain alpha numerical chars.
4388 */
4389 const char *pszVar = paArgs[0].u.pszString;
4390 if (!isalpha(*pszVar) || *pszVar == '_')
4391 return pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4392 "syntax error: Invalid variable name '%s'. Variable names must match regex '[_a-zA-Z][_a-zA-Z0-9*'!", paArgs[0].u.pszString);
4393
4394 while (isalnum(*pszVar) || *pszVar == '_')
4395 *pszVar++;
4396 if (*pszVar)
4397 return pCmdHlp->pfnPrintf(pCmdHlp, NULL,
4398 "syntax error: Invalid variable name '%s'. Variable names must match regex '[_a-zA-Z][_a-zA-Z0-9*]'!", paArgs[0].u.pszString);
4399
4400
4401 /*
4402 * Calc variable size.
4403 */
4404 size_t cbVar = (size_t)paArgs[0].u64Range + sizeof(DBGCNAMEDVAR);
4405 if (paArgs[1].enmType == DBGCVAR_TYPE_STRING)
4406 cbVar += 1 + (size_t)paArgs[1].u64Range;
4407
4408 /*
4409 * Look for existing one.
4410 */
4411 pszVar = paArgs[0].u.pszString;
4412 for (unsigned iVar = 0; iVar < pDbgc->cVars; iVar++)
4413 {
4414 if (!strcmp(pszVar, pDbgc->papVars[iVar]->szName))
4415 {
4416 /*
4417 * Update existing variable.
4418 */
4419 void *pv = RTMemRealloc(pDbgc->papVars[iVar], cbVar);
4420 if (!pv)
4421 return VERR_PARSE_NO_MEMORY;
4422 PDBGCNAMEDVAR pVar = pDbgc->papVars[iVar] = (PDBGCNAMEDVAR)pv;
4423
4424 pVar->Var = paArgs[1];
4425 memcpy(pVar->szName, paArgs[0].u.pszString, (size_t)paArgs[0].u64Range + 1);
4426 if (paArgs[1].enmType == DBGCVAR_TYPE_STRING)
4427 pVar->Var.u.pszString = (char *)memcpy(&pVar->szName[paArgs[0].u64Range + 1], paArgs[1].u.pszString, (size_t)paArgs[1].u64Range + 1);
4428 return 0;
4429 }
4430 }
4431
4432 /*
4433 * Allocate another.
4434 */
4435 PDBGCNAMEDVAR pVar = (PDBGCNAMEDVAR)RTMemAlloc(cbVar);
4436
4437 pVar->Var = paArgs[1];
4438 memcpy(pVar->szName, pszVar, (size_t)paArgs[0].u64Range + 1);
4439 if (paArgs[1].enmType == DBGCVAR_TYPE_STRING)
4440 pVar->Var.u.pszString = (char *)memcpy(&pVar->szName[paArgs[0].u64Range + 1], paArgs[1].u.pszString, (size_t)paArgs[1].u64Range + 1);
4441
4442 /* need to reallocate the pointer array too? */
4443 if (!(pDbgc->cVars % 0x20))
4444 {
4445 void *pv = RTMemRealloc(pDbgc->papVars, (pDbgc->cVars + 0x20) * sizeof(pDbgc->papVars[0]));
4446 if (!pv)
4447 {
4448 RTMemFree(pVar);
4449 return VERR_PARSE_NO_MEMORY;
4450 }
4451 pDbgc->papVars = (PDBGCNAMEDVAR *)pv;
4452 }
4453 pDbgc->papVars[pDbgc->cVars++] = pVar;
4454
4455 NOREF(pCmd); NOREF(pVM); NOREF(cArgs); NOREF(pResult);
4456 return 0;
4457}
4458
4459
4460/**
4461 * The 'unset' command.
4462 *
4463 * @returns VBox status.
4464 * @param pCmd Pointer to the command descriptor (as registered).
4465 * @param pCmdHlp Pointer to command helper functions.
4466 * @param pVM Pointer to the current VM (if any).
4467 * @param paArgs Pointer to (readonly) array of arguments.
4468 * @param cArgs Number of arguments in the array.
4469 */
4470static DECLCALLBACK(int) dbgcCmdUnset(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4471{
4472 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
4473
4474 /*
4475 * Don't trust the parser.
4476 */
4477 for (unsigned i = 0; i < cArgs; i++)
4478 if (paArgs[i].enmType != DBGCVAR_TYPE_STRING)
4479 {
4480 AssertMsgFailed(("expected strings only. (arg=%d)!\n", i));
4481 return VERR_PARSE_INCORRECT_ARG_TYPE;
4482 }
4483
4484 /*
4485 * Iterate the variables and unset them.
4486 */
4487 for (unsigned iArg = 0; iArg < cArgs; iArg++)
4488 {
4489 const char *pszVar = paArgs[iArg].u.pszString;
4490
4491 /*
4492 * Look up the variable.
4493 */
4494 for (unsigned iVar = 0; iVar < pDbgc->cVars; iVar++)
4495 {
4496 if (!strcmp(pszVar, pDbgc->papVars[iVar]->szName))
4497 {
4498 /*
4499 * Shuffle the array removing this entry.
4500 */
4501 void *pvFree = pDbgc->papVars[iVar];
4502 if (iVar + 1 < pDbgc->cVars)
4503 memmove(&pDbgc->papVars[iVar],
4504 &pDbgc->papVars[iVar + 1],
4505 (pDbgc->cVars - iVar - 1) * sizeof(pDbgc->papVars[0]));
4506 pDbgc->papVars[--pDbgc->cVars] = NULL;
4507
4508 RTMemFree(pvFree);
4509 }
4510 } /* lookup */
4511 } /* arg loop */
4512
4513 NOREF(pCmd); NOREF(pVM); NOREF(pResult);
4514 return 0;
4515}
4516
4517
4518/**
4519 * The 'loadvars' command.
4520 *
4521 * @returns VBox status.
4522 * @param pCmd Pointer to the command descriptor (as registered).
4523 * @param pCmdHlp Pointer to command helper functions.
4524 * @param pVM Pointer to the current VM (if any).
4525 * @param paArgs Pointer to (readonly) array of arguments.
4526 * @param cArgs Number of arguments in the array.
4527 */
4528static DECLCALLBACK(int) dbgcCmdLoadVars(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4529{
4530 /*
4531 * Don't trust the parser.
4532 */
4533 if ( cArgs != 1
4534 || paArgs[0].enmType != DBGCVAR_TYPE_STRING)
4535 {
4536 AssertMsgFailed(("Expected one string exactly!\n"));
4537 return VERR_PARSE_INCORRECT_ARG_TYPE;
4538 }
4539
4540 /*
4541 * Iterate the variables and unset them.
4542 */
4543 FILE *pFile = fopen(paArgs[0].u.pszString, "r");
4544 if (pFile)
4545 {
4546 char szLine[4096];
4547 while (fgets(szLine, sizeof(szLine), pFile))
4548 {
4549 /* Strip it. */
4550 char *psz = szLine;
4551 while (isblank(*psz))
4552 psz++;
4553 int i = strlen(psz) - 1;
4554 while (i >= 0 && isspace(psz[i]))
4555 psz[i--] ='\0';
4556 /* Execute it if not comment or empty line. */
4557 if ( *psz != '\0'
4558 && *psz != '#'
4559 && *psz != ';')
4560 {
4561 pCmdHlp->pfnPrintf(pCmdHlp, NULL, "dbg: set %s", psz);
4562 pCmdHlp->pfnExec(pCmdHlp, "set %s", psz);
4563 }
4564 }
4565 fclose(pFile);
4566 }
4567 else
4568 return pCmdHlp->pfnPrintf(pCmdHlp, NULL, "Failed to open file '%s'.\n", paArgs[0].u.pszString);
4569
4570 NOREF(pCmd); NOREF(pVM); NOREF(pResult);
4571 return 0;
4572}
4573
4574
4575/**
4576 * The 'showvars' command.
4577 *
4578 * @returns VBox status.
4579 * @param pCmd Pointer to the command descriptor (as registered).
4580 * @param pCmdHlp Pointer to command helper functions.
4581 * @param pVM Pointer to the current VM (if any).
4582 * @param paArgs Pointer to (readonly) array of arguments.
4583 * @param cArgs Number of arguments in the array.
4584 */
4585static DECLCALLBACK(int) dbgcCmdShowVars(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4586{
4587 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
4588
4589 for (unsigned iVar = 0; iVar < pDbgc->cVars; iVar++)
4590 {
4591 int rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "%-20s ", &pDbgc->papVars[iVar]->szName);
4592 if (!rc)
4593 rc = dbgcCmdFormat(pCmd, pCmdHlp, pVM, &pDbgc->papVars[iVar]->Var, 1, NULL);
4594 if (rc)
4595 return rc;
4596 }
4597
4598 NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
4599 return 0;
4600}
4601
4602
4603/**
4604 * The 'harakiri' command.
4605 *
4606 * @returns VBox status.
4607 * @param pCmd Pointer to the command descriptor (as registered).
4608 * @param pCmdHlp Pointer to command helper functions.
4609 * @param pVM Pointer to the current VM (if any).
4610 * @param paArgs Pointer to (readonly) array of arguments.
4611 * @param cArgs Number of arguments in the array.
4612 */
4613static DECLCALLBACK(int) dbgcCmdHarakiri(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR paArgs, unsigned cArgs, PDBGCVAR pResult)
4614{
4615 Log(("dbgcCmdHarakiri\n"));
4616 for (;;)
4617 exit(126);
4618 NOREF(pCmd); NOREF(pCmdHlp); NOREF(pVM); NOREF(paArgs); NOREF(cArgs); NOREF(pResult);
4619}
4620
4621
4622
4623
4624
4625//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
4626//
4627//
4628// B u l t i n S y m b o l s
4629//
4630//
4631//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
4632
4633
4634
4635/**
4636 * Get builtin register symbol.
4637 *
4638 * The uUser is special for these symbol descriptors. See the SYMREG_* \#defines.
4639 *
4640 * @returns 0 on success.
4641 * @returns VBox evaluation / parsing error code on failure.
4642 * The caller does the bitching.
4643 * @param pSymDesc Pointer to the symbol descriptor.
4644 * @param pCmdHlp Pointer to the command callback structure.
4645 * @param enmType The result type.
4646 * @param pResult Where to store the result.
4647 */
4648static DECLCALLBACK(int) dbgcSymGetReg(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, DBGCVARTYPE enmType, PDBGCVAR pResult)
4649{
4650 LogFlow(("dbgcSymSetReg: pSymDesc->pszName=%d\n", pSymDesc->pszName));
4651
4652 /*
4653 * pVM is required.
4654 */
4655 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
4656 Assert(pDbgc->pVM);
4657
4658 /*
4659 * Get the right CPU context.
4660 */
4661 PCPUMCTX pCtx;
4662 int rc;
4663 if (!(pSymDesc->uUser & SYMREG_FLAGS_HYPER))
4664 rc = CPUMQueryGuestCtxPtr(pDbgc->pVM, &pCtx);
4665 else
4666 rc = CPUMQueryHyperCtxPtr(pDbgc->pVM, &pCtx);
4667 if (VBOX_FAILURE(rc))
4668 return rc;
4669
4670 /*
4671 * Get the value.
4672 */
4673 void *pvValue = (char *)pCtx + SYMREG_OFFSET(pSymDesc->uUser);
4674 uint64_t u64;
4675 switch (SYMREG_SIZE(pSymDesc->uUser))
4676 {
4677 case 1: u64 = *(uint8_t *)pvValue; break;
4678 case 2: u64 = *(uint16_t *)pvValue; break;
4679 case 4: u64 = *(uint32_t *)pvValue; break;
4680 case 6: u64 = *(uint32_t *)pvValue | ((uint64_t)*(uint16_t *)((char *)pvValue + sizeof(uint32_t)) << 32); break;
4681 case 8: u64 = *(uint64_t *)pvValue; break;
4682 default:
4683 return VERR_PARSE_NOT_IMPLEMENTED;
4684 }
4685
4686 /*
4687 * Construct the desired result.
4688 */
4689 if (enmType == DBGCVAR_TYPE_ANY)
4690 enmType = DBGCVAR_TYPE_NUMBER;
4691 pResult->pDesc = NULL;
4692 pResult->pNext = NULL;
4693 pResult->enmType = enmType;
4694 pResult->enmRangeType = DBGCVAR_RANGE_NONE;
4695 pResult->u64Range = 0;
4696
4697 switch (enmType)
4698 {
4699 case DBGCVAR_TYPE_GC_FLAT:
4700 pResult->u.GCFlat = (RTGCPTR)u64;
4701 break;
4702
4703 case DBGCVAR_TYPE_GC_FAR:
4704 switch (SYMREG_SIZE(pSymDesc->uUser))
4705 {
4706 case 4:
4707 if (!(pSymDesc->uUser & SYMREG_FLAGS_HIGH_SEL))
4708 {
4709 pResult->u.GCFar.off = (uint16_t)u64;
4710 pResult->u.GCFar.sel = (uint16_t)(u64 >> 16);
4711 }
4712 else
4713 {
4714 pResult->u.GCFar.sel = (uint16_t)u64;
4715 pResult->u.GCFar.off = (uint16_t)(u64 >> 16);
4716 }
4717 break;
4718 case 6:
4719 if (!(pSymDesc->uUser & SYMREG_FLAGS_HIGH_SEL))
4720 {
4721 pResult->u.GCFar.off = (uint32_t)u64;
4722 pResult->u.GCFar.sel = (uint16_t)(u64 >> 32);
4723 }
4724 else
4725 {
4726 pResult->u.GCFar.sel = (uint32_t)u64;
4727 pResult->u.GCFar.off = (uint16_t)(u64 >> 32);
4728 }
4729 break;
4730
4731 default:
4732 return VERR_PARSE_BAD_RESULT_TYPE;
4733 }
4734 break;
4735
4736 case DBGCVAR_TYPE_GC_PHYS:
4737 pResult->u.GCPhys = (RTGCPHYS)u64;
4738 break;
4739
4740 case DBGCVAR_TYPE_HC_FLAT:
4741 pResult->u.pvHCFlat = (void *)(uintptr_t)u64;
4742 break;
4743
4744 case DBGCVAR_TYPE_HC_FAR:
4745 switch (SYMREG_SIZE(pSymDesc->uUser))
4746 {
4747 case 4:
4748 if (!(pSymDesc->uUser & SYMREG_FLAGS_HIGH_SEL))
4749 {
4750 pResult->u.HCFar.off = (uint16_t)u64;
4751 pResult->u.HCFar.sel = (uint16_t)(u64 >> 16);
4752 }
4753 else
4754 {
4755 pResult->u.HCFar.sel = (uint16_t)u64;
4756 pResult->u.HCFar.off = (uint16_t)(u64 >> 16);
4757 }
4758 break;
4759 case 6:
4760 if (!(pSymDesc->uUser & SYMREG_FLAGS_HIGH_SEL))
4761 {
4762 pResult->u.HCFar.off = (uint32_t)u64;
4763 pResult->u.HCFar.sel = (uint16_t)(u64 >> 32);
4764 }
4765 else
4766 {
4767 pResult->u.HCFar.sel = (uint32_t)u64;
4768 pResult->u.HCFar.off = (uint16_t)(u64 >> 32);
4769 }
4770 break;
4771
4772 default:
4773 return VERR_PARSE_BAD_RESULT_TYPE;
4774 }
4775 break;
4776
4777 case DBGCVAR_TYPE_HC_PHYS:
4778 pResult->u.GCPhys = (RTGCPHYS)u64;
4779 break;
4780
4781 case DBGCVAR_TYPE_NUMBER:
4782 pResult->u.u64Number = u64;
4783 break;
4784
4785 case DBGCVAR_TYPE_STRING:
4786 case DBGCVAR_TYPE_UNKNOWN:
4787 default:
4788 return VERR_PARSE_BAD_RESULT_TYPE;
4789
4790 }
4791
4792 return 0;
4793}
4794
4795
4796/**
4797 * Set builtin register symbol.
4798 *
4799 * The uUser is special for these symbol descriptors. See the SYMREG_* #defines.
4800 *
4801 * @returns 0 on success.
4802 * @returns VBox evaluation / parsing error code on failure.
4803 * The caller does the bitching.
4804 * @param pSymDesc Pointer to the symbol descriptor.
4805 * @param pCmdHlp Pointer to the command callback structure.
4806 * @param pValue The value to assign the symbol.
4807 */
4808static DECLCALLBACK(int) dbgcSymSetReg(PCDBGCSYM pSymDesc, PDBGCCMDHLP pCmdHlp, PCDBGCVAR pValue)
4809{
4810 LogFlow(("dbgcSymSetReg: pSymDesc->pszName=%d\n", pSymDesc->pszName));
4811
4812 /*
4813 * pVM is required.
4814 */
4815 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
4816 Assert(pDbgc->pVM);
4817
4818 /*
4819 * Get the right CPU context.
4820 */
4821 PCPUMCTX pCtx;
4822 int rc;
4823 if (!(pSymDesc->uUser & SYMREG_FLAGS_HYPER))
4824 rc = CPUMQueryGuestCtxPtr(pDbgc->pVM, &pCtx);
4825 else
4826 rc = CPUMQueryHyperCtxPtr(pDbgc->pVM, &pCtx);
4827 if (VBOX_FAILURE(rc))
4828 return rc;
4829
4830 /*
4831 * Check the new value.
4832 */
4833 if (pValue->enmType != DBGCVAR_TYPE_NUMBER)
4834 return VERR_PARSE_ARGUMENT_TYPE_MISMATCH;
4835
4836 /*
4837 * Set the value.
4838 */
4839 void *pvValue = (char *)pCtx + SYMREG_OFFSET(pSymDesc->uUser);
4840 switch (SYMREG_SIZE(pSymDesc->uUser))
4841 {
4842 case 1:
4843 *(uint8_t *)pvValue = (uint8_t)pValue->u.u64Number;
4844 break;
4845 case 2:
4846 *(uint16_t *)pvValue = (uint16_t)pValue->u.u64Number;
4847 break;
4848 case 4:
4849 *(uint32_t *)pvValue = (uint32_t)pValue->u.u64Number;
4850 break;
4851 case 6:
4852 *(uint32_t *)pvValue = (uint32_t)pValue->u.u64Number;
4853 ((uint16_t *)pvValue)[3] = (uint16_t)(pValue->u.u64Number >> 32);
4854 break;
4855 case 8:
4856 *(uint64_t *)pvValue = pValue->u.u64Number;
4857 break;
4858 default:
4859 return VERR_PARSE_NOT_IMPLEMENTED;
4860 }
4861
4862 return VINF_SUCCESS;
4863}
4864
4865
4866
4867
4868
4869
4870
4871//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
4872//
4873//
4874// O p e r a t o r s
4875//
4876//
4877//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
4878
4879
4880/**
4881 * Minus (unary).
4882 *
4883 * @returns 0 on success.
4884 * @returns VBox evaluation / parsing error code on failure.
4885 * The caller does the bitching.
4886 * @param pDbgc Debugger console instance data.
4887 * @param pArg The argument.
4888 * @param pResult Where to store the result.
4889 */
4890static DECLCALLBACK(int) dbgcOpMinus(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
4891{
4892// LogFlow(("dbgcOpMinus\n"));
4893 *pResult = *pArg;
4894 switch (pArg->enmType)
4895 {
4896 case DBGCVAR_TYPE_GC_FLAT:
4897 pResult->u.GCFlat = -(RTGCINTPTR)pResult->u.GCFlat;
4898 break;
4899 case DBGCVAR_TYPE_GC_FAR:
4900 pResult->u.GCFar.off = -(int32_t)pResult->u.GCFar.off;
4901 break;
4902 case DBGCVAR_TYPE_GC_PHYS:
4903 pResult->u.GCPhys = (RTGCPHYS) -(int64_t)pResult->u.GCPhys;
4904 break;
4905 case DBGCVAR_TYPE_HC_FLAT:
4906 pResult->u.pvHCFlat = (void *) -(intptr_t)pResult->u.pvHCFlat;
4907 break;
4908 case DBGCVAR_TYPE_HC_FAR:
4909 pResult->u.HCFar.off = -(int32_t)pResult->u.HCFar.off;
4910 break;
4911 case DBGCVAR_TYPE_HC_PHYS:
4912 pResult->u.HCPhys = (RTHCPHYS) -(int64_t)pResult->u.HCPhys;
4913 break;
4914 case DBGCVAR_TYPE_NUMBER:
4915 pResult->u.u64Number = -(int64_t)pResult->u.u64Number;
4916 break;
4917
4918 case DBGCVAR_TYPE_UNKNOWN:
4919 case DBGCVAR_TYPE_STRING:
4920 default:
4921 return VERR_PARSE_INCORRECT_ARG_TYPE;
4922 }
4923 NOREF(pDbgc);
4924 return 0;
4925}
4926
4927
4928/**
4929 * Pluss (unary).
4930 *
4931 * @returns 0 on success.
4932 * @returns VBox evaluation / parsing error code on failure.
4933 * The caller does the bitching.
4934 * @param pDbgc Debugger console instance data.
4935 * @param pArg The argument.
4936 * @param pResult Where to store the result.
4937 */
4938static DECLCALLBACK(int) dbgcOpPluss(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
4939{
4940// LogFlow(("dbgcOpPluss\n"));
4941 *pResult = *pArg;
4942 switch (pArg->enmType)
4943 {
4944 case DBGCVAR_TYPE_GC_FLAT:
4945 case DBGCVAR_TYPE_GC_FAR:
4946 case DBGCVAR_TYPE_GC_PHYS:
4947 case DBGCVAR_TYPE_HC_FLAT:
4948 case DBGCVAR_TYPE_HC_FAR:
4949 case DBGCVAR_TYPE_HC_PHYS:
4950 case DBGCVAR_TYPE_NUMBER:
4951 break;
4952
4953 case DBGCVAR_TYPE_UNKNOWN:
4954 case DBGCVAR_TYPE_STRING:
4955 default:
4956 return VERR_PARSE_INCORRECT_ARG_TYPE;
4957 }
4958 NOREF(pDbgc);
4959 return 0;
4960}
4961
4962
4963/**
4964 * Boolean not (unary).
4965 *
4966 * @returns 0 on success.
4967 * @returns VBox evaluation / parsing error code on failure.
4968 * The caller does the bitching.
4969 * @param pDbgc Debugger console instance data.
4970 * @param pArg The argument.
4971 * @param pResult Where to store the result.
4972 */
4973static DECLCALLBACK(int) dbgcOpBooleanNot(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
4974{
4975// LogFlow(("dbgcOpBooleanNot\n"));
4976 *pResult = *pArg;
4977 switch (pArg->enmType)
4978 {
4979 case DBGCVAR_TYPE_GC_FLAT:
4980 pResult->u.u64Number = !pResult->u.GCFlat;
4981 break;
4982 case DBGCVAR_TYPE_GC_FAR:
4983 pResult->u.u64Number = !pResult->u.GCFar.off && pResult->u.GCFar.sel <= 3;
4984 break;
4985 case DBGCVAR_TYPE_GC_PHYS:
4986 pResult->u.u64Number = !pResult->u.GCPhys;
4987 break;
4988 case DBGCVAR_TYPE_HC_FLAT:
4989 pResult->u.u64Number = !pResult->u.pvHCFlat;
4990 break;
4991 case DBGCVAR_TYPE_HC_FAR:
4992 pResult->u.u64Number = !pResult->u.HCFar.off && pResult->u.HCFar.sel <= 3;
4993 break;
4994 case DBGCVAR_TYPE_HC_PHYS:
4995 pResult->u.u64Number = !pResult->u.HCPhys;
4996 break;
4997 case DBGCVAR_TYPE_NUMBER:
4998 pResult->u.u64Number = !pResult->u.u64Number;
4999 break;
5000 case DBGCVAR_TYPE_STRING:
5001 pResult->u.u64Number = !pResult->u64Range;
5002 break;
5003
5004 case DBGCVAR_TYPE_UNKNOWN:
5005 default:
5006 return VERR_PARSE_INCORRECT_ARG_TYPE;
5007 }
5008 pResult->enmType = DBGCVAR_TYPE_NUMBER;
5009 NOREF(pDbgc);
5010 return 0;
5011}
5012
5013
5014/**
5015 * Bitwise not (unary).
5016 *
5017 * @returns 0 on success.
5018 * @returns VBox evaluation / parsing error code on failure.
5019 * The caller does the bitching.
5020 * @param pDbgc Debugger console instance data.
5021 * @param pArg The argument.
5022 * @param pResult Where to store the result.
5023 */
5024static DECLCALLBACK(int) dbgcOpBitwiseNot(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5025{
5026// LogFlow(("dbgcOpBitwiseNot\n"));
5027 *pResult = *pArg;
5028 switch (pArg->enmType)
5029 {
5030 case DBGCVAR_TYPE_GC_FLAT:
5031 pResult->u.GCFlat = ~pResult->u.GCFlat;
5032 break;
5033 case DBGCVAR_TYPE_GC_FAR:
5034 pResult->u.GCFar.off = ~pResult->u.GCFar.off;
5035 break;
5036 case DBGCVAR_TYPE_GC_PHYS:
5037 pResult->u.GCPhys = ~pResult->u.GCPhys;
5038 break;
5039 case DBGCVAR_TYPE_HC_FLAT:
5040 pResult->u.pvHCFlat = (void *)~(uintptr_t)pResult->u.pvHCFlat;
5041 break;
5042 case DBGCVAR_TYPE_HC_FAR:
5043 pResult->u.HCFar.off= ~pResult->u.HCFar.off;
5044 break;
5045 case DBGCVAR_TYPE_HC_PHYS:
5046 pResult->u.HCPhys = ~pResult->u.HCPhys;
5047 break;
5048 case DBGCVAR_TYPE_NUMBER:
5049 pResult->u.u64Number = ~pResult->u.u64Number;
5050 break;
5051
5052 case DBGCVAR_TYPE_UNKNOWN:
5053 case DBGCVAR_TYPE_STRING:
5054 default:
5055 return VERR_PARSE_INCORRECT_ARG_TYPE;
5056 }
5057 NOREF(pDbgc);
5058 return 0;
5059}
5060
5061
5062/**
5063 * Reference variable (unary).
5064 *
5065 * @returns 0 on success.
5066 * @returns VBox evaluation / parsing error code on failure.
5067 * The caller does the bitching.
5068 * @param pDbgc Debugger console instance data.
5069 * @param pArg The argument.
5070 * @param pResult Where to store the result.
5071 */
5072static DECLCALLBACK(int) dbgcOpVar(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5073{
5074// LogFlow(("dbgcOpVar: %s\n", pArg->u.pszString));
5075 /*
5076 * Parse sanity.
5077 */
5078 if (pArg->enmType != DBGCVAR_TYPE_STRING)
5079 return VERR_PARSE_INCORRECT_ARG_TYPE;
5080
5081 /*
5082 * Lookup the variable.
5083 */
5084 const char *pszVar = pArg->u.pszString;
5085 for (unsigned iVar = 0; iVar < pDbgc->cVars; iVar++)
5086 {
5087 if (!strcmp(pszVar, pDbgc->papVars[iVar]->szName))
5088 {
5089 *pResult = pDbgc->papVars[iVar]->Var;
5090 return 0;
5091 }
5092 }
5093
5094 return VERR_PARSE_VARIABLE_NOT_FOUND;
5095}
5096
5097
5098/**
5099 * Flat address (unary).
5100 *
5101 * @returns VINF_SUCCESS on success.
5102 * @returns VBox evaluation / parsing error code on failure.
5103 * The caller does the bitching.
5104 * @param pDbgc Debugger console instance data.
5105 * @param pArg The argument.
5106 * @param pResult Where to store the result.
5107 */
5108static DECLCALLBACK(int) dbgcOpAddrFlat(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5109{
5110// LogFlow(("dbgcOpAddrFlat\n"));
5111 int rc;
5112 *pResult = *pArg;
5113
5114 switch (pArg->enmType)
5115 {
5116 case DBGCVAR_TYPE_GC_FLAT:
5117 return VINF_SUCCESS;
5118
5119 case DBGCVAR_TYPE_GC_FAR:
5120 {
5121 Assert(pDbgc->pVM);
5122 DBGFADDRESS Address;
5123 rc = DBGFR3AddrFromSelOff(pDbgc->pVM, &Address, pArg->u.GCFar.sel, pArg->u.GCFar.off);
5124 if (VBOX_SUCCESS(rc))
5125 {
5126 pResult->enmType = DBGCVAR_TYPE_GC_FLAT;
5127 pResult->u.GCFlat = Address.FlatPtr;
5128 return VINF_SUCCESS;
5129 }
5130 return VERR_PARSE_CONVERSION_FAILED;
5131 }
5132
5133 case DBGCVAR_TYPE_GC_PHYS:
5134 //rc = MMR3PhysGCPhys2GCVirtEx(pDbgc->pVM, pResult->u.GCPhys, ..., &pResult->u.GCFlat); - yea, sure.
5135 return VERR_PARSE_INCORRECT_ARG_TYPE;
5136
5137 case DBGCVAR_TYPE_HC_FLAT:
5138 return VINF_SUCCESS;
5139
5140 case DBGCVAR_TYPE_HC_FAR:
5141 return VERR_PARSE_INCORRECT_ARG_TYPE;
5142
5143 case DBGCVAR_TYPE_HC_PHYS:
5144 Assert(pDbgc->pVM);
5145 pResult->enmType = DBGCVAR_TYPE_HC_FLAT;
5146 rc = MMR3HCPhys2HCVirt(pDbgc->pVM, pResult->u.HCPhys, &pResult->u.pvHCFlat);
5147 if (VBOX_SUCCESS(rc))
5148 return VINF_SUCCESS;
5149 return VERR_PARSE_CONVERSION_FAILED;
5150
5151 case DBGCVAR_TYPE_NUMBER:
5152 pResult->enmType = DBGCVAR_TYPE_GC_FLAT;
5153 pResult->u.GCFlat = (RTGCPTR)pResult->u.u64Number;
5154 return VINF_SUCCESS;
5155
5156 case DBGCVAR_TYPE_STRING:
5157 return dbgcSymbolGet(pDbgc, pArg->u.pszString, DBGCVAR_TYPE_GC_FLAT, pResult);
5158
5159 case DBGCVAR_TYPE_UNKNOWN:
5160 default:
5161 return VERR_PARSE_INCORRECT_ARG_TYPE;
5162 }
5163}
5164
5165
5166/**
5167 * Physical address (unary).
5168 *
5169 * @returns 0 on success.
5170 * @returns VBox evaluation / parsing error code on failure.
5171 * The caller does the bitching.
5172 * @param pDbgc Debugger console instance data.
5173 * @param pArg The argument.
5174 * @param pResult Where to store the result.
5175 */
5176static DECLCALLBACK(int) dbgcOpAddrPhys(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5177{
5178// LogFlow(("dbgcOpAddrPhys\n"));
5179 int rc;
5180
5181 *pResult = *pArg;
5182 switch (pArg->enmType)
5183 {
5184 case DBGCVAR_TYPE_GC_FLAT:
5185 Assert(pDbgc->pVM);
5186 pResult->enmType = DBGCVAR_TYPE_GC_PHYS;
5187 rc = PGMPhysGCPtr2GCPhys(pDbgc->pVM, pArg->u.GCFlat, &pResult->u.GCPhys);
5188 if (VBOX_SUCCESS(rc))
5189 return 0;
5190 /** @todo more memory types! */
5191 return VERR_PARSE_CONVERSION_FAILED;
5192
5193 case DBGCVAR_TYPE_GC_FAR:
5194 {
5195 Assert(pDbgc->pVM);
5196 DBGFADDRESS Address;
5197 rc = DBGFR3AddrFromSelOff(pDbgc->pVM, &Address, pArg->u.GCFar.sel, pArg->u.GCFar.off);
5198 if (VBOX_SUCCESS(rc))
5199 {
5200 pResult->enmType = DBGCVAR_TYPE_GC_PHYS;
5201 rc = PGMPhysGCPtr2GCPhys(pDbgc->pVM, Address.FlatPtr, &pResult->u.GCPhys);
5202 if (VBOX_SUCCESS(rc))
5203 return 0;
5204 /** @todo more memory types! */
5205 }
5206 return VERR_PARSE_CONVERSION_FAILED;
5207 }
5208
5209 case DBGCVAR_TYPE_GC_PHYS:
5210 return 0;
5211
5212 case DBGCVAR_TYPE_HC_FLAT:
5213 Assert(pDbgc->pVM);
5214 pResult->enmType = DBGCVAR_TYPE_GC_PHYS;
5215 rc = PGMR3DbgHCPtr2GCPhys(pDbgc->pVM, pArg->u.pvHCFlat, &pResult->u.GCPhys);
5216 if (VBOX_SUCCESS(rc))
5217 return 0;
5218 /** @todo more memory types! */
5219 return VERR_PARSE_CONVERSION_FAILED;
5220
5221 case DBGCVAR_TYPE_HC_FAR:
5222 return VERR_PARSE_INCORRECT_ARG_TYPE;
5223
5224 case DBGCVAR_TYPE_HC_PHYS:
5225 return 0;
5226
5227 case DBGCVAR_TYPE_NUMBER:
5228 pResult->enmType = DBGCVAR_TYPE_GC_PHYS;
5229 pResult->u.GCPhys = (RTGCPHYS)pResult->u.u64Number;
5230 return 0;
5231
5232 case DBGCVAR_TYPE_STRING:
5233 return dbgcSymbolGet(pDbgc, pArg->u.pszString, DBGCVAR_TYPE_GC_PHYS, pResult);
5234
5235 case DBGCVAR_TYPE_UNKNOWN:
5236 default:
5237 return VERR_PARSE_INCORRECT_ARG_TYPE;
5238 }
5239 return 0;
5240}
5241
5242
5243/**
5244 * Physical host address (unary).
5245 *
5246 * @returns 0 on success.
5247 * @returns VBox evaluation / parsing error code on failure.
5248 * The caller does the bitching.
5249 * @param pDbgc Debugger console instance data.
5250 * @param pArg The argument.
5251 * @param pResult Where to store the result.
5252 */
5253static DECLCALLBACK(int) dbgcOpAddrHostPhys(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5254{
5255// LogFlow(("dbgcOpAddrPhys\n"));
5256 int rc;
5257
5258 *pResult = *pArg;
5259 switch (pArg->enmType)
5260 {
5261 case DBGCVAR_TYPE_GC_FLAT:
5262 Assert(pDbgc->pVM);
5263 pResult->enmType = DBGCVAR_TYPE_HC_PHYS;
5264 rc = PGMPhysGCPtr2HCPhys(pDbgc->pVM, pArg->u.GCFlat, &pResult->u.HCPhys);
5265 if (VBOX_SUCCESS(rc))
5266 return 0;
5267 /** @todo more memory types. */
5268 return VERR_PARSE_CONVERSION_FAILED;
5269
5270 case DBGCVAR_TYPE_GC_FAR:
5271 {
5272 Assert(pDbgc->pVM);
5273 DBGFADDRESS Address;
5274 rc = DBGFR3AddrFromSelOff(pDbgc->pVM, &Address, pArg->u.GCFar.sel, pArg->u.GCFar.off);
5275 if (VBOX_SUCCESS(rc))
5276 {
5277 pResult->enmType = DBGCVAR_TYPE_HC_PHYS;
5278 rc = PGMPhysGCPtr2HCPhys(pDbgc->pVM, Address.FlatPtr, &pResult->u.HCPhys);
5279 if (VBOX_SUCCESS(rc))
5280 return 0;
5281 /** @todo more memory types. */
5282 }
5283 return VERR_PARSE_CONVERSION_FAILED;
5284 }
5285
5286 case DBGCVAR_TYPE_GC_PHYS:
5287 Assert(pDbgc->pVM);
5288 pResult->enmType = DBGCVAR_TYPE_HC_PHYS;
5289 rc = PGMPhysGCPhys2HCPhys(pDbgc->pVM, pArg->u.GCFlat, &pResult->u.HCPhys);
5290 if (VBOX_SUCCESS(rc))
5291 return 0;
5292 return VERR_PARSE_CONVERSION_FAILED;
5293
5294 case DBGCVAR_TYPE_HC_FLAT:
5295 Assert(pDbgc->pVM);
5296 pResult->enmType = DBGCVAR_TYPE_HC_PHYS;
5297 rc = PGMR3DbgHCPtr2HCPhys(pDbgc->pVM, pArg->u.pvHCFlat, &pResult->u.HCPhys);
5298 if (VBOX_SUCCESS(rc))
5299 return 0;
5300 /** @todo more memory types! */
5301 return VERR_PARSE_CONVERSION_FAILED;
5302
5303 case DBGCVAR_TYPE_HC_FAR:
5304 return VERR_PARSE_INCORRECT_ARG_TYPE;
5305
5306 case DBGCVAR_TYPE_HC_PHYS:
5307 return 0;
5308
5309 case DBGCVAR_TYPE_NUMBER:
5310 pResult->enmType = DBGCVAR_TYPE_HC_PHYS;
5311 pResult->u.HCPhys = (RTGCPHYS)pResult->u.u64Number;
5312 return 0;
5313
5314 case DBGCVAR_TYPE_STRING:
5315 return dbgcSymbolGet(pDbgc, pArg->u.pszString, DBGCVAR_TYPE_HC_PHYS, pResult);
5316
5317 case DBGCVAR_TYPE_UNKNOWN:
5318 default:
5319 return VERR_PARSE_INCORRECT_ARG_TYPE;
5320 }
5321 return 0;
5322}
5323
5324
5325/**
5326 * Host address (unary).
5327 *
5328 * @returns 0 on success.
5329 * @returns VBox evaluation / parsing error code on failure.
5330 * The caller does the bitching.
5331 * @param pDbgc Debugger console instance data.
5332 * @param pArg The argument.
5333 * @param pResult Where to store the result.
5334 */
5335static DECLCALLBACK(int) dbgcOpAddrHost(PDBGC pDbgc, PCDBGCVAR pArg, PDBGCVAR pResult)
5336{
5337// LogFlow(("dbgcOpAddrHost\n"));
5338 int rc;
5339
5340 *pResult = *pArg;
5341 switch (pArg->enmType)
5342 {
5343 case DBGCVAR_TYPE_GC_FLAT:
5344 Assert(pDbgc->pVM);
5345 pResult->enmType = DBGCVAR_TYPE_HC_FLAT;
5346 rc = PGMPhysGCPtr2HCPtr(pDbgc->pVM, pArg->u.GCFlat, &pResult->u.pvHCFlat);
5347 if (VBOX_SUCCESS(rc))
5348 return 0;
5349 /** @todo more memory types. */
5350 return VERR_PARSE_CONVERSION_FAILED;
5351
5352 case DBGCVAR_TYPE_GC_FAR:
5353 {
5354 Assert(pDbgc->pVM);
5355 DBGFADDRESS Address;
5356 rc = DBGFR3AddrFromSelOff(pDbgc->pVM, &Address, pArg->u.GCFar.sel, pArg->u.GCFar.off);
5357 if (VBOX_SUCCESS(rc))
5358 {
5359 pResult->enmType = DBGCVAR_TYPE_HC_FLAT;
5360 rc = PGMPhysGCPtr2HCPtr(pDbgc->pVM, Address.FlatPtr, &pResult->u.pvHCFlat);
5361 if (VBOX_SUCCESS(rc))
5362 return 0;
5363 /** @todo more memory types. */
5364 }
5365 return VERR_PARSE_CONVERSION_FAILED;
5366 }
5367
5368 case DBGCVAR_TYPE_GC_PHYS:
5369 Assert(pDbgc->pVM);
5370 pResult->enmType = DBGCVAR_TYPE_HC_FLAT;
5371 rc = PGMPhysGCPhys2HCPtr(pDbgc->pVM, pArg->u.GCPhys, 1, &pResult->u.pvHCFlat);
5372 if (VBOX_SUCCESS(rc))
5373 return 0;
5374 return VERR_PARSE_CONVERSION_FAILED;
5375
5376 case DBGCVAR_TYPE_HC_FLAT:
5377 return 0;
5378
5379 case DBGCVAR_TYPE_HC_FAR:
5380 case DBGCVAR_TYPE_HC_PHYS:
5381 /** @todo !*/
5382 return VERR_PARSE_CONVERSION_FAILED;
5383
5384 case DBGCVAR_TYPE_NUMBER:
5385 pResult->enmType = DBGCVAR_TYPE_HC_FLAT;
5386 pResult->u.pvHCFlat = (void *)(uintptr_t)pResult->u.u64Number;
5387 return 0;
5388
5389 case DBGCVAR_TYPE_STRING:
5390 return dbgcSymbolGet(pDbgc, pArg->u.pszString, DBGCVAR_TYPE_HC_FLAT, pResult);
5391
5392 case DBGCVAR_TYPE_UNKNOWN:
5393 default:
5394 return VERR_PARSE_INCORRECT_ARG_TYPE;
5395 }
5396}
5397
5398/**
5399 * Bitwise not (unary).
5400 *
5401 * @returns 0 on success.
5402 * @returns VBox evaluation / parsing error code on failure.
5403 * The caller does the bitching.
5404 * @param pDbgc Debugger console instance data.
5405 * @param pArg The argument.
5406 * @param pResult Where to store the result.
5407 */
5408static DECLCALLBACK(int) dbgcOpAddrFar(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5409{
5410// LogFlow(("dbgcOpAddrFar\n"));
5411 int rc;
5412
5413 switch (pArg1->enmType)
5414 {
5415 case DBGCVAR_TYPE_STRING:
5416 rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_NUMBER, pResult);
5417 if (VBOX_FAILURE(rc))
5418 return rc;
5419 break;
5420 case DBGCVAR_TYPE_NUMBER:
5421 *pResult = *pArg1;
5422 break;
5423
5424 case DBGCVAR_TYPE_GC_FLAT:
5425 case DBGCVAR_TYPE_GC_FAR:
5426 case DBGCVAR_TYPE_GC_PHYS:
5427 case DBGCVAR_TYPE_HC_FLAT:
5428 case DBGCVAR_TYPE_HC_FAR:
5429 case DBGCVAR_TYPE_HC_PHYS:
5430 case DBGCVAR_TYPE_UNKNOWN:
5431 default:
5432 return VERR_PARSE_INCORRECT_ARG_TYPE;
5433 }
5434 pResult->u.GCFar.sel = (RTSEL)pResult->u.u64Number;
5435
5436 /* common code for the two types we support. */
5437 switch (pArg2->enmType)
5438 {
5439 case DBGCVAR_TYPE_GC_FLAT:
5440 pResult->u.GCFar.off = pArg2->u.GCFlat;
5441 pResult->enmType = DBGCVAR_TYPE_GC_FAR;
5442 break;
5443
5444 case DBGCVAR_TYPE_HC_FLAT:
5445 pResult->u.HCFar.off = pArg2->u.GCFlat;
5446 pResult->enmType = DBGCVAR_TYPE_GC_FAR;
5447 break;
5448
5449 case DBGCVAR_TYPE_NUMBER:
5450 pResult->u.GCFar.off = (RTGCPTR)pArg2->u.u64Number;
5451 pResult->enmType = DBGCVAR_TYPE_GC_FAR;
5452 break;
5453
5454 case DBGCVAR_TYPE_STRING:
5455 {
5456 DBGCVAR Var;
5457 rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_NUMBER, &Var);
5458 if (VBOX_FAILURE(rc))
5459 return rc;
5460 pResult->u.GCFar.off = (RTGCPTR)Var.u.u64Number;
5461 pResult->enmType = DBGCVAR_TYPE_GC_FAR;
5462 break;
5463 }
5464
5465 case DBGCVAR_TYPE_GC_FAR:
5466 case DBGCVAR_TYPE_GC_PHYS:
5467 case DBGCVAR_TYPE_HC_FAR:
5468 case DBGCVAR_TYPE_HC_PHYS:
5469 case DBGCVAR_TYPE_UNKNOWN:
5470 default:
5471 return VERR_PARSE_INCORRECT_ARG_TYPE;
5472 }
5473 return 0;
5474
5475}
5476
5477
5478/**
5479 * Multiplication operator (binary).
5480 *
5481 * @returns 0 on success.
5482 * @returns VBox evaluation / parsing error code on failure.
5483 * The caller does the bitching.
5484 * @param pDbgc Debugger console instance data.
5485 * @param pArg1 The first argument.
5486 * @param pArg2 The 2nd argument.
5487 * @param pResult Where to store the result.
5488 */
5489static DECLCALLBACK(int) dbgcOpMult(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5490{
5491// LogFlow(("dbgcOpMult\n"));
5492 int rc;
5493
5494 /*
5495 * Switch the factors so we preserve pointers, far pointers are considered more
5496 * important that physical and flat pointers.
5497 */
5498 if ( DBGCVAR_ISPOINTER(pArg2->enmType)
5499 && ( !DBGCVAR_ISPOINTER(pArg1->enmType)
5500 || ( DBGCVAR_IS_FAR_PTR(pArg2->enmType)
5501 && !DBGCVAR_IS_FAR_PTR(pArg1->enmType))))
5502 {
5503 PCDBGCVAR pTmp = pArg1;
5504 pArg2 = pArg1;
5505 pArg1 = pTmp;
5506 }
5507
5508 /*
5509 * Convert the 2nd number into a number we use multiply the first with.
5510 */
5511 DBGCVAR Factor2 = *pArg2;
5512 if ( Factor2.enmType == DBGCVAR_TYPE_STRING
5513 || Factor2.enmType == DBGCVAR_TYPE_SYMBOL)
5514 {
5515 rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_NUMBER, &Factor2);
5516 if (VBOX_FAILURE(rc))
5517 return rc;
5518 }
5519 uint64_t u64Factor;
5520 switch (Factor2.enmType)
5521 {
5522 case DBGCVAR_TYPE_GC_FLAT: u64Factor = Factor2.u.GCFlat; break;
5523 case DBGCVAR_TYPE_GC_FAR: u64Factor = Factor2.u.GCFar.off; break;
5524 case DBGCVAR_TYPE_GC_PHYS: u64Factor = Factor2.u.GCPhys; break;
5525 case DBGCVAR_TYPE_HC_FLAT: u64Factor = (uintptr_t)Factor2.u.pvHCFlat; break;
5526 case DBGCVAR_TYPE_HC_FAR: u64Factor = Factor2.u.HCFar.off; break;
5527 case DBGCVAR_TYPE_HC_PHYS: u64Factor = Factor2.u.HCPhys; break;
5528 case DBGCVAR_TYPE_NUMBER: u64Factor = Factor2.u.u64Number; break;
5529 default:
5530 return VERR_PARSE_INCORRECT_ARG_TYPE;
5531 }
5532
5533 /*
5534 * Fix symbols in the 1st factor.
5535 */
5536 *pResult = *pArg1;
5537 if ( pResult->enmType == DBGCVAR_TYPE_STRING
5538 || pResult->enmType == DBGCVAR_TYPE_SYMBOL)
5539 {
5540 rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_ANY, pResult);
5541 if (VBOX_FAILURE(rc))
5542 return rc;
5543 }
5544
5545 /*
5546 * Do the multiplication.
5547 */
5548 switch (pArg1->enmType)
5549 {
5550 case DBGCVAR_TYPE_GC_FLAT: pResult->u.GCFlat *= u64Factor; break;
5551 case DBGCVAR_TYPE_GC_FAR: pResult->u.GCFar.off *= u64Factor; break;
5552 case DBGCVAR_TYPE_GC_PHYS: pResult->u.GCPhys *= u64Factor; break;
5553 case DBGCVAR_TYPE_HC_FLAT:
5554 pResult->u.pvHCFlat = (void *)(uintptr_t)((uintptr_t)pResult->u.pvHCFlat * u64Factor);
5555 break;
5556 case DBGCVAR_TYPE_HC_FAR: pResult->u.HCFar.off *= u64Factor; break;
5557 case DBGCVAR_TYPE_HC_PHYS: pResult->u.HCPhys *= u64Factor; break;
5558 case DBGCVAR_TYPE_NUMBER: pResult->u.u64Number *= u64Factor; break;
5559 default:
5560 return VERR_PARSE_INCORRECT_ARG_TYPE;
5561 }
5562 return 0;
5563}
5564
5565
5566/**
5567 * Division operator (binary).
5568 *
5569 * @returns 0 on success.
5570 * @returns VBox evaluation / parsing error code on failure.
5571 * The caller does the bitching.
5572 * @param pDbgc Debugger console instance data.
5573 * @param pArg1 The first argument.
5574 * @param pArg2 The 2nd argument.
5575 * @param pResult Where to store the result.
5576 */
5577static DECLCALLBACK(int) dbgcOpDiv(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5578{
5579 LogFlow(("dbgcOpDiv\n"));
5580 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
5581 return -1;
5582}
5583
5584
5585/**
5586 * Modulus operator (binary).
5587 *
5588 * @returns 0 on success.
5589 * @returns VBox evaluation / parsing error code on failure.
5590 * The caller does the bitching.
5591 * @param pDbgc Debugger console instance data.
5592 * @param pArg1 The first argument.
5593 * @param pArg2 The 2nd argument.
5594 * @param pResult Where to store the result.
5595 */
5596static DECLCALLBACK(int) dbgcOpMod(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5597{
5598 LogFlow(("dbgcOpMod\n"));
5599 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
5600 return -1;
5601}
5602
5603
5604/**
5605 * Addition operator (binary).
5606 *
5607 * @returns 0 on success.
5608 * @returns VBox evaluation / parsing error code on failure.
5609 * The caller does the bitching.
5610 * @param pDbgc Debugger console instance data.
5611 * @param pArg1 The first argument.
5612 * @param pArg2 The 2nd argument.
5613 * @param pResult Where to store the result.
5614 */
5615static DECLCALLBACK(int) dbgcOpAdd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5616{
5617// LogFlow(("dbgcOpAdd\n"));
5618
5619 /*
5620 * An addition operation will return (when possible) the left side type in the
5621 * expression. We make an omission for numbers, where we'll take the right side
5622 * type instead. An expression where only the left hand side is a string we'll
5623 * use the right hand type assuming that the string is a symbol.
5624 */
5625 if ( (pArg1->enmType == DBGCVAR_TYPE_NUMBER && pArg2->enmType != DBGCVAR_TYPE_STRING)
5626 || (pArg1->enmType == DBGCVAR_TYPE_STRING && pArg2->enmType != DBGCVAR_TYPE_STRING))
5627 {
5628 PCDBGCVAR pTmp = pArg2;
5629 pArg2 = pArg1;
5630 pArg1 = pTmp;
5631 }
5632 DBGCVAR Sym1, Sym2;
5633 if (pArg1->enmType == DBGCVAR_TYPE_STRING)
5634 {
5635 int rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_ANY, &Sym1);
5636 if (VBOX_FAILURE(rc))
5637 return rc;
5638 pArg1 = &Sym1;
5639
5640 rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_ANY, &Sym2);
5641 if (VBOX_FAILURE(rc))
5642 return rc;
5643 pArg2 = &Sym2;
5644 }
5645
5646 int rc;
5647 DBGCVAR Var;
5648 DBGCVAR Var2;
5649 switch (pArg1->enmType)
5650 {
5651 /*
5652 * GC Flat
5653 */
5654 case DBGCVAR_TYPE_GC_FLAT:
5655 switch (pArg2->enmType)
5656 {
5657 case DBGCVAR_TYPE_HC_FLAT:
5658 case DBGCVAR_TYPE_HC_FAR:
5659 case DBGCVAR_TYPE_HC_PHYS:
5660 return VERR_PARSE_INVALID_OPERATION;
5661 default:
5662 *pResult = *pArg1;
5663 rc = dbgcOpAddrFlat(pDbgc, pArg2, &Var);
5664 if (VBOX_FAILURE(rc))
5665 return rc;
5666 pResult->u.GCFlat += pArg2->u.GCFlat;
5667 break;
5668 }
5669 break;
5670
5671 /*
5672 * GC Far
5673 */
5674 case DBGCVAR_TYPE_GC_FAR:
5675 switch (pArg2->enmType)
5676 {
5677 case DBGCVAR_TYPE_HC_FLAT:
5678 case DBGCVAR_TYPE_HC_FAR:
5679 case DBGCVAR_TYPE_HC_PHYS:
5680 return VERR_PARSE_INVALID_OPERATION;
5681 case DBGCVAR_TYPE_NUMBER:
5682 *pResult = *pArg1;
5683 pResult->u.GCFar.off += (RTGCPTR)pArg2->u.u64Number;
5684 break;
5685 default:
5686 rc = dbgcOpAddrFlat(pDbgc, pArg1, pResult);
5687 if (VBOX_FAILURE(rc))
5688 return rc;
5689 rc = dbgcOpAddrFlat(pDbgc, pArg2, &Var);
5690 if (VBOX_FAILURE(rc))
5691 return rc;
5692 pResult->u.GCFlat += pArg2->u.GCFlat;
5693 break;
5694 }
5695 break;
5696
5697 /*
5698 * GC Phys
5699 */
5700 case DBGCVAR_TYPE_GC_PHYS:
5701 switch (pArg2->enmType)
5702 {
5703 case DBGCVAR_TYPE_HC_FLAT:
5704 case DBGCVAR_TYPE_HC_FAR:
5705 case DBGCVAR_TYPE_HC_PHYS:
5706 return VERR_PARSE_INVALID_OPERATION;
5707 default:
5708 *pResult = *pArg1;
5709 rc = dbgcOpAddrPhys(pDbgc, pArg2, &Var);
5710 if (VBOX_FAILURE(rc))
5711 return rc;
5712 if (Var.enmType != DBGCVAR_TYPE_GC_PHYS)
5713 return VERR_PARSE_INVALID_OPERATION;
5714 pResult->u.GCPhys += Var.u.GCPhys;
5715 break;
5716 }
5717 break;
5718
5719 /*
5720 * HC Flat
5721 */
5722 case DBGCVAR_TYPE_HC_FLAT:
5723 *pResult = *pArg1;
5724 rc = dbgcOpAddrHost(pDbgc, pArg2, &Var2);
5725 if (VBOX_FAILURE(rc))
5726 return rc;
5727 rc = dbgcOpAddrFlat(pDbgc, &Var2, &Var);
5728 if (VBOX_FAILURE(rc))
5729 return rc;
5730 pResult->u.pvHCFlat = (char *)pResult->u.pvHCFlat + (uintptr_t)Var.u.pvHCFlat;
5731 break;
5732
5733 /*
5734 * HC Far
5735 */
5736 case DBGCVAR_TYPE_HC_FAR:
5737 switch (pArg2->enmType)
5738 {
5739 case DBGCVAR_TYPE_NUMBER:
5740 *pResult = *pArg1;
5741 pResult->u.HCFar.off += (uintptr_t)pArg2->u.u64Number;
5742 break;
5743
5744 default:
5745 rc = dbgcOpAddrFlat(pDbgc, pArg1, pResult);
5746 if (VBOX_FAILURE(rc))
5747 return rc;
5748 rc = dbgcOpAddrHost(pDbgc, pArg2, &Var2);
5749 if (VBOX_FAILURE(rc))
5750 return rc;
5751 rc = dbgcOpAddrFlat(pDbgc, &Var2, &Var);
5752 if (VBOX_FAILURE(rc))
5753 return rc;
5754 pResult->u.pvHCFlat = (char *)pResult->u.pvHCFlat + (uintptr_t)Var.u.pvHCFlat;
5755 break;
5756 }
5757 break;
5758
5759 /*
5760 * HC Phys
5761 */
5762 case DBGCVAR_TYPE_HC_PHYS:
5763 *pResult = *pArg1;
5764 rc = dbgcOpAddrHostPhys(pDbgc, pArg2, &Var);
5765 if (VBOX_FAILURE(rc))
5766 return rc;
5767 pResult->u.HCPhys += Var.u.HCPhys;
5768 break;
5769
5770 /*
5771 * Numbers (see start of function)
5772 */
5773 case DBGCVAR_TYPE_NUMBER:
5774 *pResult = *pArg1;
5775 switch (pArg2->enmType)
5776 {
5777 case DBGCVAR_TYPE_SYMBOL:
5778 case DBGCVAR_TYPE_STRING:
5779 rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_NUMBER, &Var);
5780 if (VBOX_FAILURE(rc))
5781 return rc;
5782 case DBGCVAR_TYPE_NUMBER:
5783 pResult->u.u64Number += pArg2->u.u64Number;
5784 break;
5785 default:
5786 return VERR_PARSE_INVALID_OPERATION;
5787 }
5788 break;
5789
5790 default:
5791 return VERR_PARSE_INVALID_OPERATION;
5792
5793 }
5794 return 0;
5795}
5796
5797
5798/**
5799 * Subtration operator (binary).
5800 *
5801 * @returns 0 on success.
5802 * @returns VBox evaluation / parsing error code on failure.
5803 * The caller does the bitching.
5804 * @param pDbgc Debugger console instance data.
5805 * @param pArg1 The first argument.
5806 * @param pArg2 The 2nd argument.
5807 * @param pResult Where to store the result.
5808 */
5809static DECLCALLBACK(int) dbgcOpSub(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
5810{
5811// LogFlow(("dbgcOpSub\n"));
5812
5813 /*
5814 * An subtraction operation will return the left side type in the expression.
5815 * However, if the left hand side is a number and the right hand a pointer of
5816 * some kind we'll convert the left hand side to the same type as the right hand.
5817 * Any strings will be attempted resolved as symbols.
5818 */
5819 DBGCVAR Sym1, Sym2;
5820 if ( pArg2->enmType == DBGCVAR_TYPE_STRING
5821 && ( pArg1->enmType == DBGCVAR_TYPE_NUMBER
5822 || pArg1->enmType == DBGCVAR_TYPE_STRING))
5823 {
5824 int rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_ANY, &Sym2);
5825 if (VBOX_FAILURE(rc))
5826 return rc;
5827 pArg2 = &Sym2;
5828 }
5829
5830 if (pArg1->enmType == DBGCVAR_TYPE_STRING)
5831 {
5832 DBGCVARTYPE enmType;
5833 switch (pArg2->enmType)
5834 {
5835 case DBGCVAR_TYPE_NUMBER:
5836 enmType = DBGCVAR_TYPE_ANY;
5837 break;
5838 case DBGCVAR_TYPE_GC_FLAT:
5839 case DBGCVAR_TYPE_GC_PHYS:
5840 case DBGCVAR_TYPE_HC_FLAT:
5841 case DBGCVAR_TYPE_HC_PHYS:
5842 enmType = pArg2->enmType;
5843 break;
5844 case DBGCVAR_TYPE_GC_FAR:
5845 enmType = DBGCVAR_TYPE_GC_FLAT;
5846 break;
5847 case DBGCVAR_TYPE_HC_FAR:
5848 enmType = DBGCVAR_TYPE_HC_FLAT;
5849 break;
5850
5851 default:
5852 case DBGCVAR_TYPE_STRING:
5853 AssertMsgFailed(("Can't happen\n"));
5854 enmType = DBGCVAR_TYPE_STRING;
5855 break;
5856 }
5857 if (enmType != DBGCVAR_TYPE_STRING)
5858 {
5859 int rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_ANY, &Sym1);
5860 if (VBOX_FAILURE(rc))
5861 return rc;
5862 pArg1 = &Sym1;
5863 }
5864 }
5865 else if (pArg1->enmType == DBGCVAR_TYPE_NUMBER)
5866 {
5867 PFNDBGCOPUNARY pOp = NULL;
5868 switch (pArg2->enmType)
5869 {
5870 case DBGCVAR_TYPE_GC_FAR:
5871 case DBGCVAR_TYPE_GC_FLAT:
5872 pOp = dbgcOpAddrFlat;
5873 break;
5874 case DBGCVAR_TYPE_GC_PHYS:
5875 pOp = dbgcOpAddrPhys;
5876 break;
5877 case DBGCVAR_TYPE_HC_FAR:
5878 case DBGCVAR_TYPE_HC_FLAT:
5879 pOp = dbgcOpAddrHost;
5880 break;
5881 case DBGCVAR_TYPE_HC_PHYS:
5882 pOp = dbgcOpAddrHostPhys;
5883 break;
5884 case DBGCVAR_TYPE_NUMBER:
5885 break;
5886 default:
5887 case DBGCVAR_TYPE_STRING:
5888 AssertMsgFailed(("Can't happen\n"));
5889 break;
5890 }
5891 if (pOp)
5892 {
5893 int rc = pOp(pDbgc, pArg1, &Sym1);
5894 if (VBOX_FAILURE(rc))
5895 return rc;
5896 pArg1 = &Sym1;
5897 }
5898 }
5899
5900
5901 /*
5902 * Normal processing.
5903 */
5904 int rc;
5905 DBGCVAR Var;
5906 DBGCVAR Var2;
5907 switch (pArg1->enmType)
5908 {
5909 /*
5910 * GC Flat
5911 */
5912 case DBGCVAR_TYPE_GC_FLAT:
5913 switch (pArg2->enmType)
5914 {
5915 case DBGCVAR_TYPE_HC_FLAT:
5916 case DBGCVAR_TYPE_HC_FAR:
5917 case DBGCVAR_TYPE_HC_PHYS:
5918 return VERR_PARSE_INVALID_OPERATION;
5919 default:
5920 *pResult = *pArg1;
5921 rc = dbgcOpAddrFlat(pDbgc, pArg2, &Var);
5922 if (VBOX_FAILURE(rc))
5923 return rc;
5924 pResult->u.GCFlat -= pArg2->u.GCFlat;
5925 break;
5926 }
5927 break;
5928
5929 /*
5930 * GC Far
5931 */
5932 case DBGCVAR_TYPE_GC_FAR:
5933 switch (pArg2->enmType)
5934 {
5935 case DBGCVAR_TYPE_HC_FLAT:
5936 case DBGCVAR_TYPE_HC_FAR:
5937 case DBGCVAR_TYPE_HC_PHYS:
5938 return VERR_PARSE_INVALID_OPERATION;
5939 case DBGCVAR_TYPE_NUMBER:
5940 *pResult = *pArg1;
5941 pResult->u.GCFar.off -= (RTGCPTR)pArg2->u.u64Number;
5942 break;
5943 default:
5944 rc = dbgcOpAddrFlat(pDbgc, pArg1, pResult);
5945 if (VBOX_FAILURE(rc))
5946 return rc;
5947 rc = dbgcOpAddrFlat(pDbgc, pArg2, &Var);
5948 if (VBOX_FAILURE(rc))
5949 return rc;
5950 pResult->u.GCFlat -= pArg2->u.GCFlat;
5951 break;
5952 }
5953 break;
5954
5955 /*
5956 * GC Phys
5957 */
5958 case DBGCVAR_TYPE_GC_PHYS:
5959 switch (pArg2->enmType)
5960 {
5961 case DBGCVAR_TYPE_HC_FLAT:
5962 case DBGCVAR_TYPE_HC_FAR:
5963 case DBGCVAR_TYPE_HC_PHYS:
5964 return VERR_PARSE_INVALID_OPERATION;
5965 default:
5966 *pResult = *pArg1;
5967 rc = dbgcOpAddrPhys(pDbgc, pArg2, &Var);
5968 if (VBOX_FAILURE(rc))
5969 return rc;
5970 if (Var.enmType != DBGCVAR_TYPE_GC_PHYS)
5971 return VERR_PARSE_INVALID_OPERATION;
5972 pResult->u.GCPhys -= Var.u.GCPhys;
5973 break;
5974 }
5975 break;
5976
5977 /*
5978 * HC Flat
5979 */
5980 case DBGCVAR_TYPE_HC_FLAT:
5981 *pResult = *pArg1;
5982 rc = dbgcOpAddrHost(pDbgc, pArg2, &Var2);
5983 if (VBOX_FAILURE(rc))
5984 return rc;
5985 rc = dbgcOpAddrFlat(pDbgc, &Var2, &Var);
5986 if (VBOX_FAILURE(rc))
5987 return rc;
5988 pResult->u.pvHCFlat = (char *)pResult->u.pvHCFlat - (uintptr_t)Var.u.pvHCFlat;
5989 break;
5990
5991 /*
5992 * HC Far
5993 */
5994 case DBGCVAR_TYPE_HC_FAR:
5995 switch (pArg2->enmType)
5996 {
5997 case DBGCVAR_TYPE_NUMBER:
5998 *pResult = *pArg1;
5999 pResult->u.HCFar.off -= (uintptr_t)pArg2->u.u64Number;
6000 break;
6001
6002 default:
6003 rc = dbgcOpAddrFlat(pDbgc, pArg1, pResult);
6004 if (VBOX_FAILURE(rc))
6005 return rc;
6006 rc = dbgcOpAddrHost(pDbgc, pArg2, &Var2);
6007 if (VBOX_FAILURE(rc))
6008 return rc;
6009 rc = dbgcOpAddrFlat(pDbgc, &Var2, &Var);
6010 if (VBOX_FAILURE(rc))
6011 return rc;
6012 pResult->u.pvHCFlat = (char *)pResult->u.pvHCFlat - (uintptr_t)Var.u.pvHCFlat;
6013 break;
6014 }
6015 break;
6016
6017 /*
6018 * HC Phys
6019 */
6020 case DBGCVAR_TYPE_HC_PHYS:
6021 *pResult = *pArg1;
6022 rc = dbgcOpAddrHostPhys(pDbgc, pArg2, &Var);
6023 if (VBOX_FAILURE(rc))
6024 return rc;
6025 pResult->u.HCPhys -= Var.u.HCPhys;
6026 break;
6027
6028 /*
6029 * Numbers (see start of function)
6030 */
6031 case DBGCVAR_TYPE_NUMBER:
6032 *pResult = *pArg1;
6033 switch (pArg2->enmType)
6034 {
6035 case DBGCVAR_TYPE_SYMBOL:
6036 case DBGCVAR_TYPE_STRING:
6037 rc = dbgcSymbolGet(pDbgc, pArg2->u.pszString, DBGCVAR_TYPE_NUMBER, &Var);
6038 if (VBOX_FAILURE(rc))
6039 return rc;
6040 case DBGCVAR_TYPE_NUMBER:
6041 pResult->u.u64Number -= pArg2->u.u64Number;
6042 break;
6043 default:
6044 return VERR_PARSE_INVALID_OPERATION;
6045 }
6046 break;
6047
6048 default:
6049 return VERR_PARSE_INVALID_OPERATION;
6050
6051 }
6052 return 0;
6053}
6054
6055
6056/**
6057 * Bitwise shift left operator (binary).
6058 *
6059 * @returns 0 on success.
6060 * @returns VBox evaluation / parsing error code on failure.
6061 * The caller does the bitching.
6062 * @param pDbgc Debugger console instance data.
6063 * @param pArg1 The first argument.
6064 * @param pArg2 The 2nd argument.
6065 * @param pResult Where to store the result.
6066 */
6067static DECLCALLBACK(int) dbgcOpBitwiseShiftLeft(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6068{
6069 LogFlow(("dbgcOpBitwiseShiftLeft\n"));
6070 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6071 return -1;
6072}
6073
6074
6075/**
6076 * Bitwise shift right operator (binary).
6077 *
6078 * @returns 0 on success.
6079 * @returns VBox evaluation / parsing error code on failure.
6080 * The caller does the bitching.
6081 * @param pDbgc Debugger console instance data.
6082 * @param pArg1 The first argument.
6083 * @param pArg2 The 2nd argument.
6084 * @param pResult Where to store the result.
6085 */
6086static DECLCALLBACK(int) dbgcOpBitwiseShiftRight(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6087{
6088 LogFlow(("dbgcOpBitwiseShiftRight\n"));
6089 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6090 return -1;
6091}
6092
6093
6094/**
6095 * Bitwise and operator (binary).
6096 *
6097 * @returns 0 on success.
6098 * @returns VBox evaluation / parsing error code on failure.
6099 * The caller does the bitching.
6100 * @param pDbgc Debugger console instance data.
6101 * @param pArg1 The first argument.
6102 * @param pArg2 The 2nd argument.
6103 * @param pResult Where to store the result.
6104 */
6105static DECLCALLBACK(int) dbgcOpBitwiseAnd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6106{
6107 LogFlow(("dbgcOpBitwiseAnd\n"));
6108 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6109 return -1;
6110}
6111
6112
6113/**
6114 * Bitwise exclusive or operator (binary).
6115 *
6116 * @returns 0 on success.
6117 * @returns VBox evaluation / parsing error code on failure.
6118 * The caller does the bitching.
6119 * @param pDbgc Debugger console instance data.
6120 * @param pArg1 The first argument.
6121 * @param pArg2 The 2nd argument.
6122 * @param pResult Where to store the result.
6123 */
6124static DECLCALLBACK(int) dbgcOpBitwiseXor(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6125{
6126 LogFlow(("dbgcOpBitwiseXor\n"));
6127 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6128 return -1;
6129}
6130
6131
6132/**
6133 * Bitwise inclusive or operator (binary).
6134 *
6135 * @returns 0 on success.
6136 * @returns VBox evaluation / parsing error code on failure.
6137 * The caller does the bitching.
6138 * @param pDbgc Debugger console instance data.
6139 * @param pArg1 The first argument.
6140 * @param pArg2 The 2nd argument.
6141 * @param pResult Where to store the result.
6142 */
6143static DECLCALLBACK(int) dbgcOpBitwiseOr(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6144{
6145 LogFlow(("dbgcOpBitwiseOr\n"));
6146 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6147 return -1;
6148}
6149
6150
6151/**
6152 * Boolean and operator (binary).
6153 *
6154 * @returns 0 on success.
6155 * @returns VBox evaluation / parsing error code on failure.
6156 * The caller does the bitching.
6157 * @param pDbgc Debugger console instance data.
6158 * @param pArg1 The first argument.
6159 * @param pArg2 The 2nd argument.
6160 * @param pResult Where to store the result.
6161 */
6162static DECLCALLBACK(int) dbgcOpBooleanAnd(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6163{
6164 LogFlow(("dbgcOpBooleanAnd\n"));
6165 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6166 return -1;
6167}
6168
6169
6170/**
6171 * Boolean or operator (binary).
6172 *
6173 * @returns 0 on success.
6174 * @returns VBox evaluation / parsing error code on failure.
6175 * The caller does the bitching.
6176 * @param pDbgc Debugger console instance data.
6177 * @param pArg1 The first argument.
6178 * @param pArg2 The 2nd argument.
6179 * @param pResult Where to store the result.
6180 */
6181static DECLCALLBACK(int) dbgcOpBooleanOr(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6182{
6183 LogFlow(("dbgcOpBooleanOr\n"));
6184 NOREF(pDbgc); NOREF(pArg1); NOREF(pArg2); NOREF(pResult);
6185 return -1;
6186}
6187
6188
6189/**
6190 * Range to operator (binary).
6191 *
6192 * @returns 0 on success.
6193 * @returns VBox evaluation / parsing error code on failure.
6194 * The caller does the bitching.
6195 * @param pDbgc Debugger console instance data.
6196 * @param pArg1 The first argument.
6197 * @param pArg2 The 2nd argument.
6198 * @param pResult Where to store the result.
6199 */
6200static DECLCALLBACK(int) dbgcOpRangeLength(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6201{
6202// LogFlow(("dbgcOpRangeLength\n"));
6203 /*
6204 * Make result. Strings needs to be resolved into symbols.
6205 */
6206 if (pArg1->enmType == DBGCVAR_TYPE_STRING)
6207 {
6208 int rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_ANY, pResult);
6209 if (VBOX_FAILURE(rc))
6210 return rc;
6211 }
6212 else
6213 *pResult = *pArg1;
6214
6215 /*
6216 * Convert 2nd argument to element count.
6217 */
6218 pResult->enmRangeType = DBGCVAR_RANGE_ELEMENTS;
6219 switch (pArg2->enmType)
6220 {
6221 case DBGCVAR_TYPE_NUMBER:
6222 pResult->u64Range = pArg2->u.u64Number;
6223 break;
6224
6225 case DBGCVAR_TYPE_STRING:
6226 {
6227 int rc = dbgcSymbolGet(pDbgc, pArg1->u.pszString, DBGCVAR_TYPE_NUMBER, pResult);
6228 if (VBOX_FAILURE(rc))
6229 return rc;
6230 pResult->u64Range = pArg2->u.u64Number;
6231 break;
6232 }
6233
6234 default:
6235 return VERR_PARSE_INVALID_OPERATION;
6236 }
6237
6238 return VINF_SUCCESS;
6239}
6240
6241
6242/**
6243 * Range to operator (binary).
6244 *
6245 * @returns 0 on success.
6246 * @returns VBox evaluation / parsing error code on failure.
6247 * The caller does the bitching.
6248 * @param pDbgc Debugger console instance data.
6249 * @param pArg1 The first argument.
6250 * @param pArg2 The 2nd argument.
6251 * @param pResult Where to store the result.
6252 */
6253static DECLCALLBACK(int) dbgcOpRangeLengthBytes(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6254{
6255// LogFlow(("dbgcOpRangeLengthBytes\n"));
6256 int rc = dbgcOpRangeLength(pDbgc, pArg1, pArg2, pResult);
6257 if (VBOX_SUCCESS(rc))
6258 pResult->enmRangeType = DBGCVAR_RANGE_BYTES;
6259 return rc;
6260}
6261
6262
6263/**
6264 * Range to operator (binary).
6265 *
6266 * @returns 0 on success.
6267 * @returns VBox evaluation / parsing error code on failure.
6268 * The caller does the bitching.
6269 * @param pDbgc Debugger console instance data.
6270 * @param pArg1 The first argument.
6271 * @param pArg2 The 2nd argument.
6272 * @param pResult Where to store the result.
6273 */
6274static DECLCALLBACK(int) dbgcOpRangeTo(PDBGC pDbgc, PCDBGCVAR pArg1, PCDBGCVAR pArg2, PDBGCVAR pResult)
6275{
6276// LogFlow(("dbgcOpRangeTo\n"));
6277 /*
6278 * Calc number of bytes between the two args.
6279 */
6280 DBGCVAR Diff;
6281 int rc = dbgcOpSub(pDbgc, pArg2, pArg1, &Diff);
6282 if (VBOX_FAILURE(rc))
6283 return rc;
6284
6285 /*
6286 * Use the diff as the range of Arg1.
6287 */
6288 *pResult = *pArg1;
6289 pResult->enmRangeType = DBGCVAR_RANGE_BYTES;
6290 switch (Diff.enmType)
6291 {
6292 case DBGCVAR_TYPE_GC_FLAT:
6293 pResult->u64Range = (RTGCUINTPTR)Diff.u.GCFlat;
6294 break;
6295 case DBGCVAR_TYPE_GC_PHYS:
6296 pResult->u64Range = Diff.u.GCPhys;
6297 break;
6298 case DBGCVAR_TYPE_HC_FLAT:
6299 pResult->u64Range = (uintptr_t)Diff.u.pvHCFlat;
6300 break;
6301 case DBGCVAR_TYPE_HC_PHYS:
6302 pResult->u64Range = Diff.u.HCPhys;
6303 break;
6304 case DBGCVAR_TYPE_NUMBER:
6305 pResult->u64Range = Diff.u.u64Number;
6306 break;
6307
6308 case DBGCVAR_TYPE_GC_FAR:
6309 case DBGCVAR_TYPE_STRING:
6310 case DBGCVAR_TYPE_HC_FAR:
6311 default:
6312 AssertMsgFailed(("Impossible!\n"));
6313 return VERR_PARSE_INVALID_OPERATION;
6314 }
6315
6316 return 0;
6317}
6318
6319
6320
6321
6322
6323/**
6324 * Output callback.
6325 *
6326 * @returns number of bytes written.
6327 * @param pvArg User argument.
6328 * @param pachChars Pointer to an array of utf-8 characters.
6329 * @param cbChars Number of bytes in the character array pointed to by pachChars.
6330 */
6331static DECLCALLBACK(size_t) dbgcFormatOutput(void *pvArg, const char *pachChars, size_t cbChars)
6332{
6333 PDBGC pDbgc = (PDBGC)pvArg;
6334 if (cbChars)
6335 {
6336 int rc = pDbgc->pBack->pfnWrite(pDbgc->pBack, pachChars, cbChars, NULL);
6337 if (VBOX_FAILURE(rc))
6338 {
6339 pDbgc->rcOutput = rc;
6340 cbChars = 0;
6341 }
6342 }
6343
6344 return cbChars;
6345}
6346
6347
6348
6349
6350
6351
6352
6353
6354
6355//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
6356//
6357//
6358// C a l l b a c k H e l p e r s
6359//
6360//
6361//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
6362
6363
6364
6365/**
6366 * Command helper for writing text to the debug console.
6367 *
6368 * @returns VBox status.
6369 * @param pCmdHlp Pointer to the command callback structure.
6370 * @param pvBuf What to write.
6371 * @param cbBuf Number of bytes to write.
6372 * @param pcbWritten Where to store the number of bytes actually written.
6373 * If NULL the entire buffer must be successfully written.
6374 */
6375static DECLCALLBACK(int) dbgcHlpWrite(PDBGCCMDHLP pCmdHlp, const void *pvBuf, size_t cbBuf, size_t *pcbWritten)
6376{
6377 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6378 return pDbgc->pBack->pfnWrite(pDbgc->pBack, pvBuf, cbBuf, pcbWritten);
6379}
6380
6381
6382/**
6383 * Command helper for writing formatted text to the debug console.
6384 *
6385 * @returns VBox status.
6386 * @param pCmdHlp Pointer to the command callback structure.
6387 * @param pcb Where to store the number of bytes written.
6388 * @param pszFormat The format string.
6389 * This is using the log formatter, so it's format extensions can be used.
6390 * @param ... Arguments specified in the format string.
6391 */
6392static DECLCALLBACK(int) dbgcHlpPrintf(PDBGCCMDHLP pCmdHlp, size_t *pcbWritten, const char *pszFormat, ...)
6393{
6394 /*
6395 * Do the formatting and output.
6396 */
6397 va_list args;
6398 va_start(args, pszFormat);
6399 int rc = pCmdHlp->pfnPrintfV(pCmdHlp, pcbWritten, pszFormat, args);
6400 va_end(args);
6401
6402 return rc;
6403}
6404
6405/**
6406 * Callback to format non-standard format specifiers.
6407 *
6408 * @returns The number of bytes formatted.
6409 * @param pvArg Formatter argument.
6410 * @param pfnOutput Pointer to output function.
6411 * @param pvArgOutput Argument for the output function.
6412 * @param ppszFormat Pointer to the format string pointer. Advance this till the char
6413 * after the format specifier.
6414 * @param pArgs Pointer to the argument list. Use this to fetch the arguments.
6415 * @param cchWidth Format Width. -1 if not specified.
6416 * @param cchPrecision Format Precision. -1 if not specified.
6417 * @param fFlags Flags (RTSTR_NTFS_*).
6418 * @param chArgSize The argument size specifier, 'l' or 'L'.
6419 */
6420static DECLCALLBACK(int) dbgcStringFormatter(void *pvArg, PFNRTSTROUTPUT pfnOutput, void *pvArgOutput,
6421 const char **ppszFormat, va_list *pArgs, int cchWidth, int cchPrecision, unsigned fFlags, char chArgSize)
6422{
6423 NOREF(cchWidth); NOREF(cchPrecision); NOREF(fFlags); NOREF(chArgSize); NOREF(pvArg);
6424 if (**ppszFormat != 'D')
6425 {
6426 (*ppszFormat)++;
6427 return 0;
6428 }
6429
6430 (*ppszFormat)++;
6431 switch (**ppszFormat)
6432 {
6433 /*
6434 * Print variable without range.
6435 * The argument is a const pointer to the variable.
6436 */
6437 case 'V':
6438 {
6439 (*ppszFormat)++;
6440 PCDBGCVAR pVar = va_arg(*pArgs, PCDBGCVAR);
6441 switch (pVar->enmType)
6442 {
6443 case DBGCVAR_TYPE_GC_FLAT:
6444 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%%VGv", pVar->u.GCFlat);
6445 case DBGCVAR_TYPE_GC_FAR:
6446 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%04x:%08x", pVar->u.GCFar.sel, pVar->u.GCFar.off);
6447 case DBGCVAR_TYPE_GC_PHYS:
6448 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%%%%VGp", pVar->u.GCPhys);
6449 case DBGCVAR_TYPE_HC_FLAT:
6450 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%#%VHv", (uintptr_t)pVar->u.pvHCFlat);
6451 case DBGCVAR_TYPE_HC_FAR:
6452 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "#%04x:%08x", pVar->u.HCFar.sel, pVar->u.HCFar.off);
6453 case DBGCVAR_TYPE_HC_PHYS:
6454 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "#%%%%%VHp", pVar->u.HCPhys);
6455 case DBGCVAR_TYPE_STRING:
6456 return pfnOutput(pvArgOutput, pVar->u.pszString, (size_t)pVar->u64Range);
6457 case DBGCVAR_TYPE_NUMBER:
6458 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%llx", pVar->u.u64Number);
6459
6460 case DBGCVAR_TYPE_UNKNOWN:
6461 default:
6462 return pfnOutput(pvArgOutput, "??", 2);
6463 }
6464 }
6465
6466 /*
6467 * Print variable with range.
6468 * The argument is a const pointer to the variable.
6469 */
6470 case 'v':
6471 {
6472 (*ppszFormat)++;
6473 PCDBGCVAR pVar = va_arg(*pArgs, PCDBGCVAR);
6474
6475 char szRange[32];
6476 switch (pVar->enmRangeType)
6477 {
6478 case DBGCVAR_RANGE_NONE:
6479 szRange[0] = '\0';
6480 break;
6481 case DBGCVAR_RANGE_ELEMENTS:
6482 RTStrPrintf(szRange, sizeof(szRange), " L %llx", pVar->u64Range);
6483 break;
6484 case DBGCVAR_RANGE_BYTES:
6485 RTStrPrintf(szRange, sizeof(szRange), " LB %llx", pVar->u64Range);
6486 break;
6487 }
6488
6489 switch (pVar->enmType)
6490 {
6491 case DBGCVAR_TYPE_GC_FLAT:
6492 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%%VGv%s", pVar->u.GCFlat, szRange);
6493 case DBGCVAR_TYPE_GC_FAR:
6494 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%04x:%08x%s", pVar->u.GCFar.sel, pVar->u.GCFar.off, szRange);
6495 case DBGCVAR_TYPE_GC_PHYS:
6496 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%%%%VGp%s", pVar->u.GCPhys, szRange);
6497 case DBGCVAR_TYPE_HC_FLAT:
6498 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%%#%VHv%s", (uintptr_t)pVar->u.pvHCFlat, szRange);
6499 case DBGCVAR_TYPE_HC_FAR:
6500 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "#%04x:%08x%s", pVar->u.HCFar.sel, pVar->u.HCFar.off, szRange);
6501 case DBGCVAR_TYPE_HC_PHYS:
6502 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "#%%%%%VHp%s", pVar->u.HCPhys, szRange);
6503 case DBGCVAR_TYPE_STRING:
6504 return pfnOutput(pvArgOutput, pVar->u.pszString, (size_t)pVar->u64Range);
6505 case DBGCVAR_TYPE_NUMBER:
6506 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%llx%s", pVar->u.u64Number, szRange);
6507
6508 case DBGCVAR_TYPE_UNKNOWN:
6509 default:
6510 return pfnOutput(pvArgOutput, "??", 2);
6511 }
6512 }
6513
6514 default:
6515 AssertMsgFailed(("Invalid format type '%s'!\n", **ppszFormat));
6516 return 0;
6517 }
6518}
6519
6520
6521/**
6522 * Command helper for writing formatted text to the debug console.
6523 *
6524 * @returns VBox status.
6525 * @param pCmdHlp Pointer to the command callback structure.
6526 * @param pcb Where to store the number of bytes written.
6527 * @param pszFormat The format string.
6528 * This is using the log formatter, so it's format extensions can be used.
6529 * @param args Arguments specified in the format string.
6530 */
6531static DECLCALLBACK(int) dbgcHlpPrintfV(PDBGCCMDHLP pCmdHlp, size_t *pcbWritten, const char *pszFormat, va_list args)
6532{
6533 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6534
6535 /*
6536 * Do the formatting and output.
6537 */
6538 pDbgc->rcOutput = 0;
6539 size_t cb = RTStrFormatV(dbgcFormatOutput, pDbgc, dbgcStringFormatter, pDbgc, pszFormat, args);
6540
6541 if (pcbWritten)
6542 *pcbWritten = cb;
6543
6544 return pDbgc->rcOutput;
6545}
6546
6547
6548/**
6549 * Reports an error from a DBGF call.
6550 *
6551 * @returns VBox status code appropriate to return from a command.
6552 * @param pCmdHlp Pointer to command helpers.
6553 * @param rc The VBox status code returned by a DBGF call.
6554 * @param pszFormat Format string for additional messages. Can be NULL.
6555 * @param ... Format arguments, optional.
6556 */
6557static DECLCALLBACK(int) dbgcHlpVBoxErrorV(PDBGCCMDHLP pCmdHlp, int rc, const char *pszFormat, va_list args)
6558{
6559 switch (rc)
6560 {
6561 case VINF_SUCCESS:
6562 break;
6563
6564 default:
6565 rc = pCmdHlp->pfnPrintf(pCmdHlp, NULL, "error: %Vrc: %s", rc, pszFormat ? " " : "\n");
6566 if (VBOX_SUCCESS(rc) && pszFormat)
6567 rc = pCmdHlp->pfnPrintfV(pCmdHlp, NULL, pszFormat, args);
6568 break;
6569 }
6570 return rc;
6571}
6572
6573
6574/**
6575 * Reports an error from a DBGF call.
6576 *
6577 * @returns VBox status code appropriate to return from a command.
6578 * @param pCmdHlp Pointer to command helpers.
6579 * @param rc The VBox status code returned by a DBGF call.
6580 * @param pszFormat Format string for additional messages. Can be NULL.
6581 * @param ... Format arguments, optional.
6582 */
6583static DECLCALLBACK(int) dbgcHlpVBoxError(PDBGCCMDHLP pCmdHlp, int rc, const char *pszFormat, ...)
6584{
6585 va_list args;
6586 va_start(args, pszFormat);
6587 int rcRet = pCmdHlp->pfnVBoxErrorV(pCmdHlp, rc, pszFormat, args);
6588 va_end(args);
6589 return rcRet;
6590}
6591
6592
6593/**
6594 * Command helper for reading memory specified by a DBGC variable.
6595 *
6596 * @returns VBox status code appropriate to return from a command.
6597 * @param pCmdHlp Pointer to the command callback structure.
6598 * @param pVM VM handle if GC or physical HC address.
6599 * @param pvBuffer Where to store the read data.
6600 * @param cbRead Number of bytes to read.
6601 * @param pVarPointer DBGC variable specifying where to start reading.
6602 * @param pcbRead Where to store the number of bytes actually read.
6603 * This optional, but it's useful when read GC virtual memory where a
6604 * page in the requested range might not be present.
6605 * If not specified not-present failure or end of a HC physical page
6606 * will cause failure.
6607 */
6608static DECLCALLBACK(int) dbgcHlpMemRead(PDBGCCMDHLP pCmdHlp, PVM pVM, void *pvBuffer, size_t cbRead, PCDBGCVAR pVarPointer, size_t *pcbRead)
6609{
6610 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6611
6612 /*
6613 * Dummy check.
6614 */
6615 if (cbRead == 0)
6616 {
6617 if (*pcbRead)
6618 *pcbRead = 0;
6619 return VINF_SUCCESS;
6620 }
6621
6622 /*
6623 * Convert Far addresses getting size and the correct base address.
6624 * Getting and checking the size is what makes this messy and slow.
6625 */
6626 DBGCVAR Var = *pVarPointer;
6627 switch (pVarPointer->enmType)
6628 {
6629 case DBGCVAR_TYPE_GC_FAR:
6630 {
6631 /* Use DBGFR3AddrFromSelOff for the conversion. */
6632 Assert(pDbgc->pVM);
6633 DBGFADDRESS Address;
6634 int rc = DBGFR3AddrFromSelOff(pDbgc->pVM, &Address, Var.u.GCFar.sel, Var.u.GCFar.off);
6635 if (VBOX_FAILURE(rc))
6636 return rc;
6637
6638 /* don't bother with flat selectors (for now). */
6639 if (!DBGFADDRESS_IS_FLAT(&Address))
6640 {
6641 SELMSELINFO SelInfo;
6642 rc = SELMR3GetSelectorInfo(pDbgc->pVM, Address.Sel, &SelInfo);
6643 if (VBOX_SUCCESS(rc))
6644 {
6645 RTGCUINTPTR cb; /* -1 byte */
6646 if (SELMSelInfoIsExpandDown(&SelInfo))
6647 {
6648 if ( !SelInfo.Raw.Gen.u1Granularity
6649 && Address.off > UINT16_C(0xffff))
6650 return VERR_OUT_OF_SELECTOR_BOUNDS;
6651 if (Address.off <= SelInfo.cbLimit)
6652 return VERR_OUT_OF_SELECTOR_BOUNDS;
6653 cb = (SelInfo.Raw.Gen.u1Granularity ? UINT32_C(0xffffffff) : UINT32_C(0xffff)) - Address.off;
6654 }
6655 else
6656 {
6657 if (Address.off > SelInfo.cbLimit)
6658 return VERR_OUT_OF_SELECTOR_BOUNDS;
6659 cb = SelInfo.cbLimit - Address.off;
6660 }
6661 if (cbRead - 1 > cb)
6662 {
6663 if (!pcbRead)
6664 return VERR_OUT_OF_SELECTOR_BOUNDS;
6665 cbRead = cb + 1;
6666 }
6667 }
6668
6669 Var.enmType = DBGCVAR_TYPE_GC_FLAT;
6670 Var.u.GCFlat = Address.FlatPtr;
6671 }
6672 break;
6673 }
6674
6675 case DBGCVAR_TYPE_GC_FLAT:
6676 case DBGCVAR_TYPE_GC_PHYS:
6677 case DBGCVAR_TYPE_HC_FLAT:
6678 case DBGCVAR_TYPE_HC_PHYS:
6679 break;
6680
6681 case DBGCVAR_TYPE_HC_FAR: /* not supported yet! */
6682 default:
6683 return VERR_NOT_IMPLEMENTED;
6684 }
6685
6686
6687
6688 /*
6689 * Copy page by page.
6690 */
6691 size_t cbLeft = cbRead;
6692 for (;;)
6693 {
6694 /*
6695 * Calc read size.
6696 */
6697 size_t cb = RT_MIN(PAGE_SIZE, cbLeft);
6698 switch (pVarPointer->enmType)
6699 {
6700 case DBGCVAR_TYPE_GC_FLAT: cb = RT_MIN(cb, PAGE_SIZE - (Var.u.GCFlat & PAGE_OFFSET_MASK)); break;
6701 case DBGCVAR_TYPE_GC_PHYS: cb = RT_MIN(cb, PAGE_SIZE - (Var.u.GCPhys & PAGE_OFFSET_MASK)); break;
6702 case DBGCVAR_TYPE_HC_FLAT: cb = RT_MIN(cb, PAGE_SIZE - ((uintptr_t)Var.u.pvHCFlat & PAGE_OFFSET_MASK)); break;
6703 case DBGCVAR_TYPE_HC_PHYS: cb = RT_MIN(cb, PAGE_SIZE - ((size_t)Var.u.HCPhys & PAGE_OFFSET_MASK)); break; /* size_t: MSC has braindead loss of data warnings! */
6704 default: break;
6705 }
6706
6707 /*
6708 * Perform read.
6709 */
6710 int rc;
6711 switch (Var.enmType)
6712 {
6713 case DBGCVAR_TYPE_GC_FLAT:
6714 rc = MMR3ReadGCVirt(pVM, pvBuffer, Var.u.GCFlat, cb);
6715 break;
6716 case DBGCVAR_TYPE_GC_PHYS:
6717 rc = PGMPhysReadGCPhys(pVM, pvBuffer, Var.u.GCPhys, cb);
6718 break;
6719
6720 case DBGCVAR_TYPE_HC_PHYS:
6721 case DBGCVAR_TYPE_HC_FLAT:
6722 case DBGCVAR_TYPE_HC_FAR:
6723 {
6724 DBGCVAR Var2;
6725 rc = dbgcOpAddrFlat(pDbgc, &Var, &Var2);
6726 if (VBOX_SUCCESS(rc))
6727 {
6728 /** @todo protect this!!! */
6729 memcpy(pvBuffer, Var2.u.pvHCFlat, cb);
6730 rc = 0;
6731 }
6732 else
6733 rc = VERR_INVALID_POINTER;
6734 break;
6735 }
6736
6737 default:
6738 rc = VERR_PARSE_INCORRECT_ARG_TYPE;
6739 }
6740
6741 /*
6742 * Check for failure.
6743 */
6744 if (VBOX_FAILURE(rc))
6745 {
6746 if (pcbRead && (*pcbRead = cbRead - cbLeft) > 0)
6747 return VINF_SUCCESS;
6748 return rc;
6749 }
6750
6751 /*
6752 * Next.
6753 */
6754 cbLeft -= cb;
6755 if (!cbLeft)
6756 break;
6757 pvBuffer = (char *)pvBuffer + cb;
6758 rc = pCmdHlp->pfnEval(pCmdHlp, &Var, "%DV + %d", &Var, cb);
6759 if (VBOX_FAILURE(rc))
6760 {
6761 if (pcbRead && (*pcbRead = cbRead - cbLeft) > 0)
6762 return VINF_SUCCESS;
6763 return rc;
6764 }
6765 }
6766
6767 /*
6768 * Done
6769 */
6770 if (pcbRead)
6771 *pcbRead = cbRead;
6772 return 0;
6773}
6774
6775/**
6776 * Command helper for writing memory specified by a DBGC variable.
6777 *
6778 * @returns VBox status code appropriate to return from a command.
6779 * @param pCmdHlp Pointer to the command callback structure.
6780 * @param pVM VM handle if GC or physical HC address.
6781 * @param pvBuffer What to write.
6782 * @param cbWrite Number of bytes to write.
6783 * @param pVarPointer DBGC variable specifying where to start reading.
6784 * @param pcbWritten Where to store the number of bytes written.
6785 * This is optional. If NULL be aware that some of the buffer
6786 * might have been written to the specified address.
6787 */
6788static DECLCALLBACK(int) dbgcHlpMemWrite(PDBGCCMDHLP pCmdHlp, PVM pVM, const void *pvBuffer, size_t cbWrite, PCDBGCVAR pVarPointer, size_t *pcbWritten)
6789{
6790 NOREF(pCmdHlp); NOREF(pVM); NOREF(pvBuffer); NOREF(cbWrite); NOREF(pVarPointer); NOREF(pcbWritten);
6791 return VERR_NOT_IMPLEMENTED;
6792}
6793
6794
6795/**
6796 * Evaluates an expression.
6797 * (Hopefully the parser and functions are fully reentrant.)
6798 *
6799 * @returns VBox status code appropriate to return from a command.
6800 * @param pCmdHlp Pointer to the command callback structure.
6801 * @param pResult Where to store the result.
6802 * @param pszExpr The expression. Format string with the format DBGC extensions.
6803 * @param ... Format arguments.
6804 */
6805static DECLCALLBACK(int) dbgcHlpEval(PDBGCCMDHLP pCmdHlp, PDBGCVAR pResult, const char *pszExpr, ...)
6806{
6807 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6808
6809 /*
6810 * Format the expression.
6811 */
6812 char szExprFormatted[2048];
6813 va_list args;
6814 va_start(args, pszExpr);
6815 size_t cb = RTStrPrintfExV(dbgcStringFormatter, pDbgc, szExprFormatted, sizeof(szExprFormatted), pszExpr, args);
6816 va_end(args);
6817 /* ignore overflows. */
6818
6819 return dbgcEvalSub(pDbgc, &szExprFormatted[0], cb, pResult);
6820}
6821
6822
6823/**
6824 * Executes one command expression.
6825 * (Hopefully the parser and functions are fully reentrant.)
6826 *
6827 * @returns VBox status code appropriate to return from a command.
6828 * @param pCmdHlp Pointer to the command callback structure.
6829 * @param pszExpr The expression. Format string with the format DBGC extensions.
6830 * @param ... Format arguments.
6831 */
6832static DECLCALLBACK(int) dbgcHlpExec(PDBGCCMDHLP pCmdHlp, const char *pszExpr, ...)
6833{
6834 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6835 /* Save the scratch state. */
6836 char *pszScratch = pDbgc->pszScratch;
6837 unsigned iArg = pDbgc->iArg;
6838
6839 /*
6840 * Format the expression.
6841 */
6842 va_list args;
6843 va_start(args, pszExpr);
6844 size_t cbScratch = sizeof(pDbgc->achScratch) - (pDbgc->pszScratch - &pDbgc->achScratch[0]);
6845 size_t cb = RTStrPrintfExV(dbgcStringFormatter, pDbgc, pDbgc->pszScratch, cbScratch, pszExpr, args);
6846 va_end(args);
6847 if (cb >= cbScratch)
6848 return VERR_BUFFER_OVERFLOW;
6849
6850 /*
6851 * Execute the command.
6852 * We save and restore the arg index and scratch buffer pointer.
6853 */
6854 pDbgc->pszScratch = pDbgc->pszScratch + cb + 1;
6855 int rc = dbgcProcessCommand(pDbgc, pszScratch, cb);
6856
6857 /* Restore the scratch state. */
6858 pDbgc->iArg = iArg;
6859 pDbgc->pszScratch = pszScratch;
6860
6861 return rc;
6862}
6863
6864
6865/**
6866 * Converts a DBGC variable to a DBGF address structure.
6867 *
6868 * @returns VBox status code.
6869 * @param pCmdHlp Pointer to the command callback structure.
6870 * @param pVar The variable to convert.
6871 * @param pAddress The target address.
6872 */
6873static DECLCALLBACK(int) dbgcHlpVarToDbgfAddr(PDBGCCMDHLP pCmdHlp, PCDBGCVAR pVar, PDBGFADDRESS pAddress)
6874{
6875 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6876 return dbgcVarToDbgfAddr(pDbgc, pVar, pAddress);
6877}
6878
6879
6880/**
6881 * Converts a DBGC variable to a boolean.
6882 *
6883 * @returns VBox status code.
6884 * @param pCmdHlp Pointer to the command callback structure.
6885 * @param pVar The variable to convert.
6886 * @param pf Where to store the boolean.
6887 */
6888static DECLCALLBACK(int) dbgcHlpVarToBool(PDBGCCMDHLP pCmdHlp, PCDBGCVAR pVar, bool *pf)
6889{
6890 PDBGC pDbgc = DBGC_CMDHLP2DBGC(pCmdHlp);
6891 NOREF(pDbgc);
6892
6893 switch (pVar->enmType)
6894 {
6895 case DBGCVAR_TYPE_STRING:
6896 /** @todo add strcasecmp / stricmp wrappers to iprt/string.h. */
6897 if ( !strcmp(pVar->u.pszString, "true")
6898 || !strcmp(pVar->u.pszString, "True")
6899 || !strcmp(pVar->u.pszString, "TRUE")
6900 || !strcmp(pVar->u.pszString, "on")
6901 || !strcmp(pVar->u.pszString, "On")
6902 || !strcmp(pVar->u.pszString, "oN")
6903 || !strcmp(pVar->u.pszString, "ON")
6904 || !strcmp(pVar->u.pszString, "enabled")
6905 || !strcmp(pVar->u.pszString, "Enabled")
6906 || !strcmp(pVar->u.pszString, "DISABLED"))
6907 {
6908 *pf = true;
6909 return VINF_SUCCESS;
6910 }
6911 if ( !strcmp(pVar->u.pszString, "false")
6912 || !strcmp(pVar->u.pszString, "False")
6913 || !strcmp(pVar->u.pszString, "FALSE")
6914 || !strcmp(pVar->u.pszString, "off")
6915 || !strcmp(pVar->u.pszString, "Off")
6916 || !strcmp(pVar->u.pszString, "OFF")
6917 || !strcmp(pVar->u.pszString, "disabled")
6918 || !strcmp(pVar->u.pszString, "Disabled")
6919 || !strcmp(pVar->u.pszString, "DISABLED"))
6920 {
6921 *pf = false;
6922 return VINF_SUCCESS;
6923 }
6924 return VERR_PARSE_INCORRECT_ARG_TYPE; /** @todo better error code! */
6925
6926 case DBGCVAR_TYPE_GC_FLAT:
6927 case DBGCVAR_TYPE_GC_PHYS:
6928 case DBGCVAR_TYPE_HC_FLAT:
6929 case DBGCVAR_TYPE_HC_PHYS:
6930 case DBGCVAR_TYPE_NUMBER:
6931 *pf = pVar->u.u64Number != 0;
6932 return VINF_SUCCESS;
6933
6934 case DBGCVAR_TYPE_HC_FAR:
6935 case DBGCVAR_TYPE_GC_FAR:
6936 case DBGCVAR_TYPE_SYMBOL:
6937 default:
6938 return VERR_PARSE_INCORRECT_ARG_TYPE;
6939 }
6940}
6941
6942
6943
6944
6945
6946//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
6947//
6948//
6949// V a r i a b l e M a n i p u l a t i o n
6950//
6951//
6952//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
6953
6954
6955
6956/** @todo move me!*/
6957static void dbgcVarSetGCFlat(PDBGCVAR pVar, RTGCPTR GCFlat)
6958{
6959 if (pVar)
6960 {
6961 pVar->enmType = DBGCVAR_TYPE_GC_FLAT;
6962 pVar->u.GCFlat = GCFlat;
6963 pVar->enmRangeType = DBGCVAR_RANGE_NONE;
6964 pVar->u64Range = 0;
6965 }
6966}
6967
6968
6969/** @todo move me!*/
6970static void dbgcVarSetGCFlatByteRange(PDBGCVAR pVar, RTGCPTR GCFlat, uint64_t cb)
6971{
6972 if (pVar)
6973 {
6974 pVar->enmType = DBGCVAR_TYPE_GC_FLAT;
6975 pVar->u.GCFlat = GCFlat;
6976 pVar->enmRangeType = DBGCVAR_RANGE_BYTES;
6977 pVar->u64Range = cb;
6978 }
6979}
6980
6981
6982/** @todo move me!*/
6983static void dbgcVarSetVar(PDBGCVAR pVar, PCDBGCVAR pVar2)
6984{
6985 if (pVar)
6986 {
6987 if (pVar2)
6988 *pVar = *pVar2;
6989 else
6990 {
6991 pVar->enmType = DBGCVAR_TYPE_UNKNOWN;
6992 memset(&pVar->u, 0, sizeof(pVar->u));
6993 pVar->enmRangeType = DBGCVAR_RANGE_NONE;
6994 pVar->u64Range = 0;
6995 }
6996 }
6997}
6998
6999
7000/** @todo move me!*/
7001static void dbgcVarSetByteRange(PDBGCVAR pVar, uint64_t cb)
7002{
7003 if (pVar)
7004 {
7005 pVar->enmRangeType = DBGCVAR_RANGE_BYTES;
7006 pVar->u64Range = cb;
7007 }
7008}
7009
7010
7011/** @todo move me!*/
7012static void dbgcVarSetNoRange(PDBGCVAR pVar)
7013{
7014 if (pVar)
7015 {
7016 pVar->enmRangeType = DBGCVAR_RANGE_NONE;
7017 pVar->u64Range = 0;
7018 }
7019}
7020
7021
7022/**
7023 * Converts a DBGC variable to a DBGF address.
7024 *
7025 * @returns VBox status code.
7026 * @param pDbgc The DBGC instance.
7027 * @param pVar The variable.
7028 * @param pAddress Where to store the address.
7029 */
7030static int dbgcVarToDbgfAddr(PDBGC pDbgc, PCDBGCVAR pVar, PDBGFADDRESS pAddress)
7031{
7032 AssertReturn(pVar, VERR_INVALID_PARAMETER);
7033 switch (pVar->enmType)
7034 {
7035 case DBGCVAR_TYPE_GC_FLAT:
7036 DBGFR3AddrFromFlat(pDbgc->pVM, pAddress, pVar->u.GCFlat);
7037 return VINF_SUCCESS;
7038
7039 case DBGCVAR_TYPE_NUMBER:
7040 DBGFR3AddrFromFlat(pDbgc->pVM, pAddress, (RTGCUINTPTR)pVar->u.u64Number);
7041 return VINF_SUCCESS;
7042
7043 case DBGCVAR_TYPE_GC_FAR:
7044 return DBGFR3AddrFromSelOff(pDbgc->pVM, pAddress, pVar->u.GCFar.sel, pVar->u.GCFar.sel);
7045
7046 case DBGCVAR_TYPE_STRING:
7047 case DBGCVAR_TYPE_SYMBOL:
7048 {
7049 DBGCVAR Var;
7050 int rc = pDbgc->CmdHlp.pfnEval(&pDbgc->CmdHlp, &Var, "%%(%DV)", pVar);
7051 if (VBOX_FAILURE(rc))
7052 return rc;
7053 return dbgcVarToDbgfAddr(pDbgc, &Var, pAddress);
7054 }
7055
7056 case DBGCVAR_TYPE_GC_PHYS:
7057 case DBGCVAR_TYPE_HC_FLAT:
7058 case DBGCVAR_TYPE_HC_FAR:
7059 case DBGCVAR_TYPE_HC_PHYS:
7060 default:
7061 return VERR_PARSE_CONVERSION_FAILED;
7062 }
7063}
7064
7065
7066
7067//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
7068//
7069//
7070// B r e a k p o i n t M a n a g e m e n t
7071//
7072//
7073//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
7074
7075
7076/**
7077 * Adds a breakpoint to the DBGC breakpoint list.
7078 */
7079static int dbgcBpAdd(PDBGC pDbgc, RTUINT iBp, const char *pszCmd)
7080{
7081 /*
7082 * Check if it already exists.
7083 */
7084 PDBGCBP pBp = dbgcBpGet(pDbgc, iBp);
7085 if (pBp)
7086 return VERR_DBGC_BP_EXISTS;
7087
7088 /*
7089 * Add the breakpoint.
7090 */
7091 if (pszCmd)
7092 pszCmd = RTStrStripL(pszCmd);
7093 size_t cchCmd = pszCmd ? strlen(pszCmd) : 0;
7094 pBp = (PDBGCBP)RTMemAlloc(RT_OFFSETOF(DBGCBP, szCmd[cchCmd + 1]));
7095 if (!pBp)
7096 return VERR_NO_MEMORY;
7097 if (cchCmd)
7098 memcpy(pBp->szCmd, pszCmd, cchCmd + 1);
7099 else
7100 pBp->szCmd[0] = '\0';
7101 pBp->cchCmd = cchCmd;
7102 pBp->iBp = iBp;
7103 pBp->pNext = pDbgc->pFirstBp;
7104 pDbgc->pFirstBp = pBp;
7105
7106 return VINF_SUCCESS;
7107}
7108
7109/**
7110 * Updates the a breakpoint.
7111 *
7112 * @returns VBox status code.
7113 * @param pDbgc The DBGC instance.
7114 * @param iBp The breakpoint to update.
7115 * @param pszCmd The new command.
7116 */
7117static int dbgcBpUpdate(PDBGC pDbgc, RTUINT iBp, const char *pszCmd)
7118{
7119 /*
7120 * Find the breakpoint.
7121 */
7122 PDBGCBP pBp = dbgcBpGet(pDbgc, iBp);
7123 if (!pBp)
7124 return VERR_DBGC_BP_NOT_FOUND;
7125
7126 /*
7127 * Do we need to reallocate?
7128 */
7129 if (pszCmd)
7130 pszCmd = RTStrStripL(pszCmd);
7131 if (!pszCmd || !*pszCmd)
7132 pBp->szCmd[0] = '\0';
7133 else
7134 {
7135 size_t cchCmd = strlen(pszCmd);
7136 if (strlen(pBp->szCmd) >= cchCmd)
7137 {
7138 memcpy(pBp->szCmd, pszCmd, cchCmd + 1);
7139 pBp->cchCmd = cchCmd;
7140 }
7141 else
7142 {
7143 /*
7144 * Yes, let's do it the simple way...
7145 */
7146 int rc = dbgcBpDelete(pDbgc, iBp);
7147 AssertRC(rc);
7148 return dbgcBpAdd(pDbgc, iBp, pszCmd);
7149 }
7150 }
7151 return VINF_SUCCESS;
7152}
7153
7154
7155/**
7156 * Deletes a breakpoint.
7157 *
7158 * @returns VBox status code.
7159 * @param pDbgc The DBGC instance.
7160 * @param iBp The breakpoint to delete.
7161 */
7162static int dbgcBpDelete(PDBGC pDbgc, RTUINT iBp)
7163{
7164 /*
7165 * Search thru the list, when found unlink and free it.
7166 */
7167 PDBGCBP pBpPrev = NULL;
7168 PDBGCBP pBp = pDbgc->pFirstBp;
7169 for (; pBp; pBp = pBp->pNext)
7170 {
7171 if (pBp->iBp == iBp)
7172 {
7173 if (pBpPrev)
7174 pBpPrev->pNext = pBp->pNext;
7175 else
7176 pDbgc->pFirstBp = pBp->pNext;
7177 RTMemFree(pBp);
7178 return VINF_SUCCESS;
7179 }
7180 pBpPrev = pBp;
7181 }
7182
7183 return VERR_DBGC_BP_NOT_FOUND;
7184}
7185
7186
7187/**
7188 * Get a breakpoint.
7189 *
7190 * @returns Pointer to the breakpoint.
7191 * @returns NULL if the breakpoint wasn't found.
7192 * @param pDbgc The DBGC instance.
7193 * @param iBp The breakpoint to get.
7194 */
7195static PDBGCBP dbgcBpGet(PDBGC pDbgc, RTUINT iBp)
7196{
7197 /*
7198 * Enumerate the list.
7199 */
7200 PDBGCBP pBp = pDbgc->pFirstBp;
7201 for (; pBp; pBp = pBp->pNext)
7202 if (pBp->iBp == iBp)
7203 return pBp;
7204 return NULL;
7205}
7206
7207
7208/**
7209 * Executes the command of a breakpoint.
7210 *
7211 * @returns VINF_DBGC_BP_NO_COMMAND if there is no command associated with the breakpoint.
7212 * @returns VERR_DBGC_BP_NOT_FOUND if the breakpoint wasn't found.
7213 * @returns VERR_BUFFER_OVERFLOW if the is not enough space in the scratch buffer for the command.
7214 * @returns VBox status code from dbgcProcessCommand() other wise.
7215 * @param pDbgc The DBGC instance.
7216 * @param iBp The breakpoint to execute.
7217 */
7218static int dbgcBpExec(PDBGC pDbgc, RTUINT iBp)
7219{
7220 /*
7221 * Find the breakpoint.
7222 */
7223 PDBGCBP pBp = dbgcBpGet(pDbgc, iBp);
7224 if (!pBp)
7225 return VERR_DBGC_BP_NOT_FOUND;
7226
7227 /*
7228 * Anything to do?
7229 */
7230 if (!pBp->cchCmd)
7231 return VINF_DBGC_BP_NO_COMMAND;
7232
7233 /*
7234 * Execute the command.
7235 * This means copying it to the scratch buffer and process it as if it
7236 * were user input. We must save and restore the state of the scratch buffer.
7237 */
7238 /* Save the scratch state. */
7239 char *pszScratch = pDbgc->pszScratch;
7240 unsigned iArg = pDbgc->iArg;
7241
7242 /* Copy the command to the scratch buffer. */
7243 size_t cbScratch = sizeof(pDbgc->achScratch) - (pDbgc->pszScratch - &pDbgc->achScratch[0]);
7244 if (pBp->cchCmd >= cbScratch)
7245 return VERR_BUFFER_OVERFLOW;
7246 memcpy(pDbgc->pszScratch, pBp->szCmd, pBp->cchCmd + 1);
7247
7248 /* Execute the command. */
7249 pDbgc->pszScratch = pDbgc->pszScratch + pBp->cchCmd + 1;
7250 int rc = dbgcProcessCommand(pDbgc, pszScratch, pBp->cchCmd);
7251
7252 /* Restore the scratch state. */
7253 pDbgc->iArg = iArg;
7254 pDbgc->pszScratch = pszScratch;
7255
7256 return rc;
7257}
7258
7259
7260
7261
7262
7263//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
7264//
7265//
7266// I n p u t , p a r s i n g a n d l o g g i n g
7267//
7268//
7269//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//
7270
7271
7272
7273/**
7274 * Prints any log lines from the log buffer.
7275 *
7276 * The caller must not call function this unless pDbgc->fLog is set.
7277 *
7278 * @returns VBox status. (output related)
7279 * @param pDbgc Debugger console instance data.
7280 */
7281static int dbgcProcessLog(PDBGC pDbgc)
7282{
7283 /** @todo */
7284 NOREF(pDbgc);
7285 return 0;
7286}
7287
7288
7289
7290/**
7291 * Handle input buffer overflow.
7292 *
7293 * Will read any available input looking for a '\n' to reset the buffer on.
7294 *
7295 * @returns VBox status.
7296 * @param pDbgc Debugger console instance data.
7297 */
7298static int dbgcInputOverflow(PDBGC pDbgc)
7299{
7300 /*
7301 * Assert overflow status and reset the input buffer.
7302 */
7303 if (!pDbgc->fInputOverflow)
7304 {
7305 pDbgc->fInputOverflow = true;
7306 pDbgc->iRead = pDbgc->iWrite = 0;
7307 pDbgc->cInputLines = 0;
7308 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "Input overflow!!\n");
7309 }
7310
7311 /*
7312 * Eat input till no more or there is a '\n'.
7313 * When finding a '\n' we'll continue normal processing.
7314 */
7315 while (pDbgc->pBack->pfnInput(pDbgc->pBack, 0))
7316 {
7317 size_t cbRead;
7318 int rc = pDbgc->pBack->pfnRead(pDbgc->pBack, &pDbgc->achInput[0], sizeof(pDbgc->achInput) - 1, &cbRead);
7319 if (VBOX_FAILURE(rc))
7320 return rc;
7321 char *psz = (char *)memchr(&pDbgc->achInput[0], '\n', cbRead);
7322 if (psz)
7323 {
7324 pDbgc->fInputOverflow = false;
7325 pDbgc->iRead = psz - &pDbgc->achInput[0] + 1;
7326 pDbgc->iWrite = cbRead;
7327 pDbgc->cInputLines = 0;
7328 break;
7329 }
7330 }
7331
7332 return 0;
7333}
7334
7335
7336
7337/**
7338 * Read input and do some preprocessing.
7339 *
7340 * @returns VBox status.
7341 * In addition to the iWrite and achInput, cInputLines is maintained.
7342 * In case of an input overflow the fInputOverflow flag will be set.
7343 * @param pDbgc Debugger console instance data.
7344 */
7345static int dbgcInputRead(PDBGC pDbgc)
7346{
7347 /*
7348 * We have ready input.
7349 * Read it till we don't have any or we have a full input buffer.
7350 */
7351 int rc = 0;
7352 do
7353 {
7354 /*
7355 * More available buffer space?
7356 */
7357 size_t cbLeft;
7358 if (pDbgc->iWrite > pDbgc->iRead)
7359 cbLeft = sizeof(pDbgc->achInput) - pDbgc->iWrite - (pDbgc->iRead == 0);
7360 else
7361 cbLeft = pDbgc->iRead - pDbgc->iWrite - 1;
7362 if (!cbLeft)
7363 {
7364 /* overflow? */
7365 if (!pDbgc->cInputLines)
7366 rc = dbgcInputOverflow(pDbgc);
7367 break;
7368 }
7369
7370 /*
7371 * Read one char and interpret it.
7372 */
7373 char achRead[128];
7374 size_t cbRead;
7375 rc = pDbgc->pBack->pfnRead(pDbgc->pBack, &achRead[0], RT_MIN(cbLeft, sizeof(achRead)), &cbRead);
7376 if (VBOX_FAILURE(rc))
7377 return rc;
7378 char *psz = &achRead[0];
7379 while (cbRead-- > 0)
7380 {
7381 char ch = *psz++;
7382 switch (ch)
7383 {
7384 /*
7385 * Ignore.
7386 */
7387 case '\0':
7388 case '\r':
7389 case '\a':
7390 break;
7391
7392 /*
7393 * Backspace.
7394 */
7395 case '\b':
7396 Log2(("DBGC: backspace\n"));
7397 if (pDbgc->iRead != pDbgc->iWrite)
7398 {
7399 unsigned iWriteUndo = pDbgc->iWrite;
7400 if (pDbgc->iWrite)
7401 pDbgc->iWrite--;
7402 else
7403 pDbgc->iWrite = sizeof(pDbgc->achInput) - 1;
7404
7405 if (pDbgc->achInput[pDbgc->iWrite] == '\n')
7406 pDbgc->iWrite = iWriteUndo;
7407 }
7408 break;
7409
7410 /*
7411 * Add char to buffer.
7412 */
7413 case '\t':
7414 case '\n':
7415 case ';':
7416 switch (ch)
7417 {
7418 case '\t': ch = ' '; break;
7419 case '\n': pDbgc->cInputLines++; break;
7420 }
7421 default:
7422 Log2(("DBGC: ch=%02x\n", (unsigned char)ch));
7423 pDbgc->achInput[pDbgc->iWrite] = ch;
7424 if (++pDbgc->iWrite >= sizeof(pDbgc->achInput))
7425 pDbgc->iWrite = 0;
7426 break;
7427 }
7428 }
7429
7430 /* Terminate it to make it easier to read in the debugger. */
7431 pDbgc->achInput[pDbgc->iWrite] = '\0';
7432 } while (pDbgc->pBack->pfnInput(pDbgc->pBack, 0));
7433
7434 return rc;
7435}
7436
7437
7438/**
7439 * Finds a builtin symbol.
7440 * @returns Pointer to symbol descriptor on success.
7441 * @returns NULL on failure.
7442 * @param pDbgc The debug console instance.
7443 * @param pszSymbol The symbol name.
7444 */
7445static PCDBGCSYM dbgcLookupRegisterSymbol(PDBGC pDbgc, const char *pszSymbol)
7446{
7447 for (unsigned iSym = 0; iSym < ELEMENTS(g_aSyms); iSym++)
7448 if (!strcmp(pszSymbol, g_aSyms[iSym].pszName))
7449 return &g_aSyms[iSym];
7450
7451 /** @todo externally registered symbols. */
7452 NOREF(pDbgc);
7453 return NULL;
7454}
7455
7456
7457/**
7458 * Resolves a symbol (or tries to do so at least).
7459 *
7460 * @returns 0 on success.
7461 * @returns VBox status on failure.
7462 * @param pDbgc The debug console instance.
7463 * @param pszSymbol The symbol name.
7464 * @param enmType The result type.
7465 * @param pResult Where to store the result.
7466 */
7467static int dbgcSymbolGet(PDBGC pDbgc, const char *pszSymbol, DBGCVARTYPE enmType, PDBGCVAR pResult)
7468{
7469 /*
7470 * Builtin?
7471 */
7472 PCDBGCSYM pSymDesc = dbgcLookupRegisterSymbol(pDbgc, pszSymbol);
7473 if (pSymDesc)
7474 {
7475 if (!pSymDesc->pfnGet)
7476 return VERR_PARSE_WRITEONLY_SYMBOL;
7477 return pSymDesc->pfnGet(pSymDesc, &pDbgc->CmdHlp, enmType, pResult);
7478 }
7479
7480
7481 /*
7482 * Ask PDM.
7483 */
7484 /** @todo resolve symbols using PDM. */
7485
7486
7487 /*
7488 * Ask the debug info manager.
7489 */
7490 DBGFSYMBOL Symbol;
7491 int rc = DBGFR3SymbolByName(pDbgc->pVM, pszSymbol, &Symbol);
7492 if (VBOX_SUCCESS(rc))
7493 {
7494 /*
7495 * Default return is a flat gc address.
7496 */
7497 memset(pResult, 0, sizeof(*pResult));
7498 pResult->enmRangeType = Symbol.cb ? DBGCVAR_RANGE_BYTES : DBGCVAR_RANGE_NONE;
7499 pResult->u64Range = Symbol.cb;
7500 pResult->enmType = DBGCVAR_TYPE_GC_FLAT;
7501 pResult->u.GCFlat = Symbol.Value;
7502 DBGCVAR VarTmp;
7503 switch (enmType)
7504 {
7505 /* nothing to do. */
7506 case DBGCVAR_TYPE_GC_FLAT:
7507 case DBGCVAR_TYPE_GC_FAR:
7508 case DBGCVAR_TYPE_ANY:
7509 return VINF_SUCCESS;
7510
7511 /* simply make it numeric. */
7512 case DBGCVAR_TYPE_NUMBER:
7513 pResult->enmType = DBGCVAR_TYPE_NUMBER;
7514 pResult->u.u64Number = Symbol.Value;
7515 return VINF_SUCCESS;
7516
7517 /* cast it. */
7518
7519 case DBGCVAR_TYPE_GC_PHYS:
7520 VarTmp = *pResult;
7521 return dbgcOpAddrPhys(pDbgc, &VarTmp, pResult);
7522
7523 case DBGCVAR_TYPE_HC_FAR:
7524 case DBGCVAR_TYPE_HC_FLAT:
7525 VarTmp = *pResult;
7526 return dbgcOpAddrHost(pDbgc, &VarTmp, pResult);
7527
7528 case DBGCVAR_TYPE_HC_PHYS:
7529 VarTmp = *pResult;
7530 return dbgcOpAddrHostPhys(pDbgc, &VarTmp, pResult);
7531
7532 default:
7533 AssertMsgFailed(("Internal error enmType=%d\n", enmType));
7534 return VERR_INVALID_PARAMETER;
7535 }
7536 }
7537
7538 return VERR_PARSE_NOT_IMPLEMENTED;
7539}
7540
7541
7542
7543/**
7544 * Finds a routine.
7545 *
7546 * @returns Pointer to the command descriptor.
7547 * If the request was for an external command, the caller is responsible for
7548 * unlocking the external command list.
7549 * @returns NULL if not found.
7550 * @param pDbgc The debug console instance.
7551 * @param pachName Pointer to the routine string (not terminated).
7552 * @param cchName Length of the routine name.
7553 * @param fExternal Whether or not the routine is external.
7554 */
7555static PCDBGCCMD dbgcRoutineLookup(PDBGC pDbgc, const char *pachName, size_t cchName, bool fExternal)
7556{
7557 if (!fExternal)
7558 {
7559 for (unsigned iCmd = 0; iCmd < ELEMENTS(g_aCmds); iCmd++)
7560 {
7561 if ( !strncmp(pachName, g_aCmds[iCmd].pszCmd, cchName)
7562 && !g_aCmds[iCmd].pszCmd[cchName])
7563 return &g_aCmds[iCmd];
7564 }
7565 }
7566 else
7567 {
7568 DBGCEXTCMDS_LOCK_RD();
7569 for (PDBGCEXTCMDS pExtCmds = g_pExtCmdsHead; pExtCmds; pExtCmds = pExtCmds->pNext)
7570 {
7571 for (unsigned iCmd = 0; iCmd < pExtCmds->cCmds; iCmd++)
7572 {
7573 if ( !strncmp(pachName, pExtCmds->paCmds[iCmd].pszCmd, cchName)
7574 && !pExtCmds->paCmds[iCmd].pszCmd[cchName])
7575 return &pExtCmds->paCmds[iCmd];
7576 }
7577 }
7578 DBGCEXTCMDS_UNLOCK_RD();
7579 }
7580
7581 NOREF(pDbgc);
7582 return NULL;
7583}
7584
7585
7586/**
7587 * Searches for an operator descriptor which matches the start of
7588 * the expression given us.
7589 *
7590 * @returns Pointer to the operator on success.
7591 * @param pDbgc The debug console instance.
7592 * @param pszExpr Pointer to the expression string which might start with an operator.
7593 * @param fPreferBinary Whether to favour binary or unary operators.
7594 * Caller must assert that it's the disired type! Both types will still
7595 * be returned, this is only for resolving duplicates.
7596 * @param chPrev The previous char. Some operators requires a blank in front of it.
7597 */
7598static PCDBGCOP dbgcOperatorLookup(PDBGC pDbgc, const char *pszExpr, bool fPreferBinary, char chPrev)
7599{
7600 PCDBGCOP pOp = NULL;
7601 for (unsigned iOp = 0; iOp < ELEMENTS(g_aOps); iOp++)
7602 {
7603 if ( g_aOps[iOp].szName[0] == pszExpr[0]
7604 && (!g_aOps[iOp].szName[1] || g_aOps[iOp].szName[1] == pszExpr[1])
7605 && (!g_aOps[iOp].szName[2] || g_aOps[iOp].szName[2] == pszExpr[2]))
7606 {
7607 /*
7608 * Check that we don't mistake it for some other operator which have more chars.
7609 */
7610 unsigned j;
7611 for (j = iOp + 1; j < ELEMENTS(g_aOps); j++)
7612 if ( g_aOps[j].cchName > g_aOps[iOp].cchName
7613 && g_aOps[j].szName[0] == pszExpr[0]
7614 && (!g_aOps[j].szName[1] || g_aOps[j].szName[1] == pszExpr[1])
7615 && (!g_aOps[j].szName[2] || g_aOps[j].szName[2] == pszExpr[2]) )
7616 break;
7617 if (j < ELEMENTS(g_aOps))
7618 continue; /* we'll catch it later. (for theoretical +,++,+++ cases.) */
7619 pOp = &g_aOps[iOp];
7620
7621 /*
7622 * Prefered type?
7623 */
7624 if (g_aOps[iOp].fBinary == fPreferBinary)
7625 break;
7626 }
7627 }
7628
7629 if (pOp)
7630 Log2(("dbgcOperatorLookup: pOp=%p %s\n", pOp, pOp->szName));
7631 NOREF(pDbgc); NOREF(chPrev);
7632 return pOp;
7633}
7634
7635
7636/**
7637 * Initalizes g_bmOperatorChars.
7638 */
7639static void dbgcInitOpCharBitMap(void)
7640{
7641 memset(g_bmOperatorChars, 0, sizeof(g_bmOperatorChars));
7642 for (unsigned iOp = 0; iOp < RT_ELEMENTS(g_aOps); iOp++)
7643 ASMBitSet(&g_bmOperatorChars[0], (uint8_t)g_aOps[iOp].szName[0]);
7644}
7645
7646
7647/**
7648 * Checks whether the character may be the start of an operator.
7649 *
7650 * @returns true/false.
7651 * @param ch The character.
7652 */
7653DECLINLINE(bool) dbgcIsOpChar(char ch)
7654{
7655 return ASMBitTest(&g_bmOperatorChars[0], (uint8_t)ch);
7656}
7657
7658
7659static int dbgcEvalSubString(PDBGC pDbgc, char *pszExpr, size_t cchExpr, PDBGCVAR pArg)
7660{
7661 Log2(("dbgcEvalSubString: cchExpr=%d pszExpr=%s\n", cchExpr, pszExpr));
7662
7663 /*
7664 * Removing any quoting and escapings.
7665 */
7666 char ch = *pszExpr;
7667 if (ch == '"' || ch == '\'' || ch == '`')
7668 {
7669 if (pszExpr[--cchExpr] != ch)
7670 return VERR_PARSE_UNBALANCED_QUOTE;
7671 cchExpr--;
7672 pszExpr++;
7673
7674 /** @todo string unescaping. */
7675 }
7676 pszExpr[cchExpr] = '\0';
7677
7678 /*
7679 * Make the argument.
7680 */
7681 pArg->pDesc = NULL;
7682 pArg->pNext = NULL;
7683 pArg->enmType = DBGCVAR_TYPE_STRING;
7684 pArg->u.pszString = pszExpr;
7685 pArg->enmRangeType = DBGCVAR_RANGE_BYTES;
7686 pArg->u64Range = cchExpr;
7687
7688 NOREF(pDbgc);
7689 return 0;
7690}
7691
7692
7693static int dbgcEvalSubNum(char *pszExpr, unsigned uBase, PDBGCVAR pArg)
7694{
7695 Log2(("dbgcEvalSubNum: uBase=%d pszExpr=%s\n", uBase, pszExpr));
7696 /*
7697 * Convert to number.
7698 */
7699 uint64_t u64 = 0;
7700 char ch;
7701 while ((ch = *pszExpr) != '\0')
7702 {
7703 uint64_t u64Prev = u64;
7704 unsigned u = ch - '0';
7705 if (u < 10 && u < uBase)
7706 u64 = u64 * uBase + u;
7707 else if (ch >= 'a' && (u = ch - ('a' - 10)) < uBase)
7708 u64 = u64 * uBase + u;
7709 else if (ch >= 'A' && (u = ch - ('A' - 10)) < uBase)
7710 u64 = u64 * uBase + u;
7711 else
7712 return VERR_PARSE_INVALID_NUMBER;
7713
7714 /* check for overflow - ARG!!! How to detect overflow correctly!?!?!? */
7715 if (u64Prev != u64 / uBase)
7716 return VERR_PARSE_NUMBER_TOO_BIG;
7717
7718 /* next */
7719 pszExpr++;
7720 }
7721
7722 /*
7723 * Initialize the argument.
7724 */
7725 pArg->pDesc = NULL;
7726 pArg->pNext = NULL;
7727 pArg->enmType = DBGCVAR_TYPE_NUMBER;
7728 pArg->u.u64Number = u64;
7729 pArg->enmRangeType = DBGCVAR_RANGE_NONE;
7730 pArg->u64Range = 0;
7731
7732 return 0;
7733}
7734
7735
7736/**
7737 * Match variable and variable descriptor, promoting the variable if necessary.
7738 *
7739 * @returns VBox status code.
7740 * @param pDbgc Debug console instanace.
7741 * @param pVar Variable.
7742 * @param pVarDesc Variable descriptor.
7743 */
7744static int dbgcEvalSubMatchVar(PDBGC pDbgc, PDBGCVAR pVar, PCDBGCVARDESC pVarDesc)
7745{
7746 /*
7747 * (If match or promoted to match, return, else break.)
7748 */
7749 switch (pVarDesc->enmCategory)
7750 {
7751 /*
7752 * Anything goes
7753 */
7754 case DBGCVAR_CAT_ANY:
7755 return VINF_SUCCESS;
7756
7757 /*
7758 * Pointer with and without range.
7759 * We can try resolve strings and symbols as symbols and
7760 * promote numbers to flat GC pointers.
7761 */
7762 case DBGCVAR_CAT_POINTER_NO_RANGE:
7763 if (pVar->enmRangeType != DBGCVAR_RANGE_NONE)
7764 return VERR_PARSE_NO_RANGE_ALLOWED;
7765 /* fallthru */
7766 case DBGCVAR_CAT_POINTER:
7767 switch (pVar->enmType)
7768 {
7769 case DBGCVAR_TYPE_GC_FLAT:
7770 case DBGCVAR_TYPE_GC_FAR:
7771 case DBGCVAR_TYPE_GC_PHYS:
7772 case DBGCVAR_TYPE_HC_FLAT:
7773 case DBGCVAR_TYPE_HC_FAR:
7774 case DBGCVAR_TYPE_HC_PHYS:
7775 return VINF_SUCCESS;
7776
7777 case DBGCVAR_TYPE_SYMBOL:
7778 case DBGCVAR_TYPE_STRING:
7779 {
7780 DBGCVAR Var;
7781 int rc = dbgcSymbolGet(pDbgc, pVar->u.pszString, DBGCVAR_TYPE_GC_FLAT, &Var);
7782 if (VBOX_SUCCESS(rc))
7783 {
7784 /* deal with range */
7785 if (pVar->enmRangeType != DBGCVAR_RANGE_NONE)
7786 {
7787 Var.enmRangeType = pVar->enmRangeType;
7788 Var.u64Range = pVar->u64Range;
7789 }
7790 else if (pVarDesc->enmCategory == DBGCVAR_CAT_POINTER_NO_RANGE)
7791 Var.enmRangeType = DBGCVAR_RANGE_NONE;
7792 *pVar = Var;
7793 return rc;
7794 }
7795 break;
7796 }
7797
7798 case DBGCVAR_TYPE_NUMBER:
7799 {
7800 RTGCPTR GCPtr = (RTGCPTR)pVar->u.u64Number;
7801 pVar->enmType = DBGCVAR_TYPE_GC_FLAT;
7802 pVar->u.GCFlat = GCPtr;
7803 return VINF_SUCCESS;
7804 }
7805
7806 default:
7807 break;
7808 }
7809 break;
7810
7811 /*
7812 * GC pointer with and without range.
7813 * We can try resolve strings and symbols as symbols and
7814 * promote numbers to flat GC pointers.
7815 */
7816 case DBGCVAR_CAT_GC_POINTER_NO_RANGE:
7817 if (pVar->enmRangeType != DBGCVAR_RANGE_NONE)
7818 return VERR_PARSE_NO_RANGE_ALLOWED;
7819 /* fallthru */
7820 case DBGCVAR_CAT_GC_POINTER:
7821 switch (pVar->enmType)
7822 {
7823 case DBGCVAR_TYPE_GC_FLAT:
7824 case DBGCVAR_TYPE_GC_FAR:
7825 case DBGCVAR_TYPE_GC_PHYS:
7826 return VINF_SUCCESS;
7827
7828 case DBGCVAR_TYPE_HC_FLAT:
7829 case DBGCVAR_TYPE_HC_FAR:
7830 case DBGCVAR_TYPE_HC_PHYS:
7831 return VERR_PARSE_CONVERSION_FAILED;
7832
7833 case DBGCVAR_TYPE_SYMBOL:
7834 case DBGCVAR_TYPE_STRING:
7835 {
7836 DBGCVAR Var;
7837 int rc = dbgcSymbolGet(pDbgc, pVar->u.pszString, DBGCVAR_TYPE_GC_FLAT, &Var);
7838 if (VBOX_SUCCESS(rc))
7839 {
7840 /* deal with range */
7841 if (pVar->enmRangeType != DBGCVAR_RANGE_NONE)
7842 {
7843 Var.enmRangeType = pVar->enmRangeType;
7844 Var.u64Range = pVar->u64Range;
7845 }
7846 else if (pVarDesc->enmCategory == DBGCVAR_CAT_POINTER_NO_RANGE)
7847 Var.enmRangeType = DBGCVAR_RANGE_NONE;
7848 *pVar = Var;
7849 return rc;
7850 }
7851 break;
7852 }
7853
7854 case DBGCVAR_TYPE_NUMBER:
7855 {
7856 RTGCPTR GCPtr = (RTGCPTR)pVar->u.u64Number;
7857 pVar->enmType = DBGCVAR_TYPE_GC_FLAT;
7858 pVar->u.GCFlat = GCPtr;
7859 return VINF_SUCCESS;
7860 }
7861
7862 default:
7863 break;
7864 }
7865 break;
7866
7867 /*
7868 * Number with or without a range.
7869 * Numbers can be resolved from symbols, but we cannot demote a pointer
7870 * to a number.
7871 */
7872 case DBGCVAR_CAT_NUMBER_NO_RANGE:
7873 if (pVar->enmRangeType != DBGCVAR_RANGE_NONE)
7874 return VERR_PARSE_NO_RANGE_ALLOWED;
7875 /* fallthru */
7876 case DBGCVAR_CAT_NUMBER:
7877 switch (pVar->enmType)
7878 {
7879 case DBGCVAR_TYPE_NUMBER:
7880 return VINF_SUCCESS;
7881
7882 case DBGCVAR_TYPE_SYMBOL:
7883 case DBGCVAR_TYPE_STRING:
7884 {
7885 DBGCVAR Var;
7886 int rc = dbgcSymbolGet(pDbgc, pVar->u.pszString, DBGCVAR_TYPE_NUMBER, &Var);
7887 if (VBOX_SUCCESS(rc))
7888 {
7889 *pVar = Var;
7890 return rc;
7891 }
7892 break;
7893 }
7894 default:
7895 break;
7896 }
7897 break;
7898
7899 /*
7900 * Strings can easily be made from symbols (and of course strings).
7901 * We could consider reformatting the addresses and numbers into strings later...
7902 */
7903 case DBGCVAR_CAT_STRING:
7904 switch (pVar->enmType)
7905 {
7906 case DBGCVAR_TYPE_SYMBOL:
7907 pVar->enmType = DBGCVAR_TYPE_STRING;
7908 /* fallthru */
7909 case DBGCVAR_TYPE_STRING:
7910 return VINF_SUCCESS;
7911 default:
7912 break;
7913 }
7914 break;
7915
7916 /*
7917 * Symol is pretty much the same thing as a string (at least until we actually implement it).
7918 */
7919 case DBGCVAR_CAT_SYMBOL:
7920 switch (pVar->enmType)
7921 {
7922 case DBGCVAR_TYPE_STRING:
7923 pVar->enmType = DBGCVAR_TYPE_SYMBOL;
7924 /* fallthru */
7925 case DBGCVAR_TYPE_SYMBOL:
7926 return VINF_SUCCESS;
7927 default:
7928 break;
7929 }
7930 break;
7931
7932 /*
7933 * Anything else is illegal.
7934 */
7935 default:
7936 AssertMsgFailed(("enmCategory=%d\n", pVar->enmType));
7937 break;
7938 }
7939
7940 return VERR_PARSE_NO_ARGUMENT_MATCH;
7941}
7942
7943
7944/**
7945 * Matches a set of variables with a description set.
7946 *
7947 * This is typically used for routine arguments before a call. The effects in
7948 * addition to the validation, is that some variables might be propagated to
7949 * other types in order to match the description. The following transformations
7950 * are supported:
7951 * - String reinterpreted as a symbol and resolved to a number or pointer.
7952 * - Number to a pointer.
7953 * - Pointer to a number.
7954 * @returns 0 on success with paVars.
7955 * @returns VBox error code for match errors.
7956 */
7957static int dbgcEvalSubMatchVars(PDBGC pDbgc, unsigned cVarsMin, unsigned cVarsMax,
7958 PCDBGCVARDESC paVarDescs, unsigned cVarDescs,
7959 PDBGCVAR paVars, unsigned cVars)
7960{
7961 /*
7962 * Just do basic min / max checks first.
7963 */
7964 if (cVars < cVarsMin)
7965 return VERR_PARSE_TOO_FEW_ARGUMENTS;
7966 if (cVars > cVarsMax)
7967 return VERR_PARSE_TOO_MANY_ARGUMENTS;
7968
7969 /*
7970 * Match the descriptors and actual variables.
7971 */
7972 PCDBGCVARDESC pPrevDesc = NULL;
7973 unsigned cCurDesc = 0;
7974 unsigned iVar = 0;
7975 unsigned iVarDesc = 0;
7976 while (iVar < cVars)
7977 {
7978 /* walk the descriptors */
7979 if (iVarDesc >= cVarDescs)
7980 return VERR_PARSE_TOO_MANY_ARGUMENTS;
7981 if ( ( paVarDescs[iVarDesc].fFlags & DBGCVD_FLAGS_DEP_PREV
7982 && &paVarDescs[iVarDesc - 1] != pPrevDesc)
7983 || cCurDesc >= paVarDescs[iVarDesc].cTimesMax)
7984 {
7985 iVarDesc++;
7986 if (iVarDesc >= cVarDescs)
7987 return VERR_PARSE_TOO_MANY_ARGUMENTS;
7988 cCurDesc = 0;
7989 }
7990
7991 /*
7992 * Skip thru optional arguments until we find something which matches
7993 * or can easily be promoted to what the descriptor want.
7994 */
7995 for (;;)
7996 {
7997 int rc = dbgcEvalSubMatchVar(pDbgc, &paVars[iVar], &paVarDescs[iVarDesc]);
7998 if (VBOX_SUCCESS(rc))
7999 {
8000 paVars[iVar].pDesc = &paVarDescs[iVarDesc];
8001 cCurDesc++;
8002 break;
8003 }
8004
8005 /* can we advance? */
8006 if (paVarDescs[iVarDesc].cTimesMin > cCurDesc)
8007 return VERR_PARSE_ARGUMENT_TYPE_MISMATCH;
8008 if (++iVarDesc >= cVarDescs)
8009 return VERR_PARSE_ARGUMENT_TYPE_MISMATCH;
8010 cCurDesc = 0;
8011 }
8012
8013 /* next var */
8014 iVar++;
8015 }
8016
8017 /*
8018 * Check that the rest of the descriptors are optional.
8019 */
8020 while (iVarDesc < cVarDescs)
8021 {
8022 if (paVarDescs[iVarDesc].cTimesMin > cCurDesc)
8023 return VERR_PARSE_TOO_FEW_ARGUMENTS;
8024 cCurDesc = 0;
8025
8026 /* next */
8027 iVarDesc++;
8028 }
8029
8030 return 0;
8031}
8032
8033
8034/**
8035 * Evaluates one argument with respect to unary operators.
8036 *
8037 * @returns 0 on success. pResult contains the result.
8038 * @returns VBox error code on parse or other evaluation error.
8039 *
8040 * @param pDbgc Debugger console instance data.
8041 * @param pszExpr The expression string.
8042 * @param pResult Where to store the result of the expression evaluation.
8043 */
8044static int dbgcEvalSubUnary(PDBGC pDbgc, char *pszExpr, size_t cchExpr, PDBGCVAR pResult)
8045{
8046 Log2(("dbgcEvalSubUnary: cchExpr=%d pszExpr=%s\n", cchExpr, pszExpr));
8047
8048 /*
8049 * The state of the expression is now such that it will start by zero or more
8050 * unary operators and being followed by an expression of some kind.
8051 * The expression is either plain or in parenthesis.
8052 *
8053 * Being in a lazy, recursive mode today, the parsing is done as simple as possible. :-)
8054 * ASSUME: unary operators are all of equal precedence.
8055 */
8056 int rc = 0;
8057 PCDBGCOP pOp = dbgcOperatorLookup(pDbgc, pszExpr, false, ' ');
8058 if (pOp)
8059 {
8060 /* binary operators means syntax error. */
8061 if (pOp->fBinary)
8062 return VERR_PARSE_UNEXPECTED_OPERATOR;
8063
8064 /*
8065 * If the next expression (the one following the unary operator) is in a
8066 * parenthesis a full eval is needed. If not the unary eval will suffice.
8067 */
8068 /* calc and strip next expr. */
8069 char *pszExpr2 = pszExpr + pOp->cchName;
8070 while (isblank(*pszExpr2))
8071 pszExpr2++;
8072
8073 if (!*pszExpr2)
8074 rc = VERR_PARSE_EMPTY_ARGUMENT;
8075 else
8076 {
8077 DBGCVAR Arg;
8078 if (*pszExpr2 == '(')
8079 rc = dbgcEvalSub(pDbgc, pszExpr2, cchExpr - (pszExpr2 - pszExpr), &Arg);
8080 else
8081 rc = dbgcEvalSubUnary(pDbgc, pszExpr2, cchExpr - (pszExpr2 - pszExpr), &Arg);
8082 if (VBOX_SUCCESS(rc))
8083 rc = pOp->pfnHandlerUnary(pDbgc, &Arg, pResult);
8084 }
8085 }
8086 else
8087 {
8088 /*
8089 * Didn't find any operators, so it we have to check if this can be an
8090 * function call before assuming numeric or string expression.
8091 *
8092 * (ASSUMPTIONS:)
8093 * A function name only contains alphanumerical chars and it can not start
8094 * with a numerical character.
8095 * Immediately following the name is a parenthesis which must over
8096 * the remaining part of the expression.
8097 */
8098 bool fExternal = *pszExpr == '.';
8099 char *pszFun = fExternal ? pszExpr + 1 : pszExpr;
8100 char *pszFunEnd = NULL;
8101 if (pszExpr[cchExpr - 1] == ')' && isalpha(*pszFun))
8102 {
8103 pszFunEnd = pszExpr + 1;
8104 while (*pszFunEnd != '(' && isalnum(*pszFunEnd))
8105 pszFunEnd++;
8106 if (*pszFunEnd != '(')
8107 pszFunEnd = NULL;
8108 }
8109
8110 if (pszFunEnd)
8111 {
8112 /*
8113 * Ok, it's a function call.
8114 */
8115 if (fExternal)
8116 pszExpr++, cchExpr--;
8117 PCDBGCCMD pFun = dbgcRoutineLookup(pDbgc, pszExpr, pszFunEnd - pszExpr, fExternal);
8118 if (!pFun)
8119 return VERR_PARSE_FUNCTION_NOT_FOUND;
8120 if (!pFun->pResultDesc)
8121 return VERR_PARSE_NOT_A_FUNCTION;
8122
8123 /*
8124 * Parse the expression in parenthesis.
8125 */
8126 cchExpr -= pszFunEnd - pszExpr;
8127 pszExpr = pszFunEnd;
8128 /** @todo implement multiple arguments. */
8129 DBGCVAR Arg;
8130 rc = dbgcEvalSub(pDbgc, pszExpr, cchExpr, &Arg);
8131 if (!rc)
8132 {
8133 rc = dbgcEvalSubMatchVars(pDbgc, pFun->cArgsMin, pFun->cArgsMax, pFun->paArgDescs, pFun->cArgDescs, &Arg, 1);
8134 if (!rc)
8135 rc = pFun->pfnHandler(pFun, &pDbgc->CmdHlp, pDbgc->pVM, &Arg, 1, pResult);
8136 }
8137 else if (rc == VERR_PARSE_EMPTY_ARGUMENT && pFun->cArgsMin == 0)
8138 rc = pFun->pfnHandler(pFun, &pDbgc->CmdHlp, pDbgc->pVM, NULL, 0, pResult);
8139 }
8140 else
8141 {
8142 /*
8143 * Didn't find any operators, so it must be a plain expression.
8144 * This might be numeric or a string expression.
8145 */
8146 char ch = pszExpr[0];
8147 char ch2 = pszExpr[1];
8148 if (ch == '0' && (ch2 == 'x' || ch2 == 'X'))
8149 rc = dbgcEvalSubNum(pszExpr + 2, 16, pResult);
8150 else if (ch == '0' && (ch2 == 'i' || ch2 == 'i'))
8151 rc = dbgcEvalSubNum(pszExpr + 2, 10, pResult);
8152 else if (ch == '0' && (ch2 == 't' || ch2 == 'T'))
8153 rc = dbgcEvalSubNum(pszExpr + 2, 8, pResult);
8154 /// @todo 0b doesn't work as a binary prefix, we confuse it with 0bf8:0123 and stuff.
8155 //else if (ch == '0' && (ch2 == 'b' || ch2 == 'b'))
8156 // rc = dbgcEvalSubNum(pszExpr + 2, 2, pResult);
8157 else
8158 {
8159 /*
8160 * Hexadecimal number or a string?
8161 */
8162 char *psz = pszExpr;
8163 while (isxdigit(*psz))
8164 psz++;
8165 if (!*psz)
8166 rc = dbgcEvalSubNum(pszExpr, 16, pResult);
8167 else if ((*psz == 'h' || *psz == 'H') && !psz[1])
8168 {
8169 *psz = '\0';
8170 rc = dbgcEvalSubNum(pszExpr, 16, pResult);
8171 }
8172 else
8173 rc = dbgcEvalSubString(pDbgc, pszExpr, cchExpr, pResult);
8174 }
8175 }
8176 }
8177
8178 return rc;
8179}
8180
8181
8182/**
8183 * Evaluates one argument.
8184 *
8185 * @returns 0 on success. pResult contains the result.
8186 * @returns VBox error code on parse or other evaluation error.
8187 *
8188 * @param pDbgc Debugger console instance data.
8189 * @param pszExpr The expression string.
8190 * @param pResult Where to store the result of the expression evaluation.
8191 */
8192static int dbgcEvalSub(PDBGC pDbgc, char *pszExpr, size_t cchExpr, PDBGCVAR pResult)
8193{
8194 Log2(("dbgcEvalSub: cchExpr=%d pszExpr=%s\n", cchExpr, pszExpr));
8195 /*
8196 * First we need to remove blanks in both ends.
8197 * ASSUMES: There is no quoting unless the entire expression is a string.
8198 */
8199
8200 /* stripping. */
8201 while (cchExpr > 0 && isblank(pszExpr[cchExpr - 1]))
8202 pszExpr[--cchExpr] = '\0';
8203 while (isblank(*pszExpr))
8204 pszExpr++, cchExpr--;
8205 if (!*pszExpr)
8206 return VERR_PARSE_EMPTY_ARGUMENT;
8207
8208 /* it there is any kind of quoting in the expression, it's string meat. */
8209 if (strpbrk(pszExpr, "\"'`"))
8210 return dbgcEvalSubString(pDbgc, pszExpr, cchExpr, pResult);
8211
8212 /*
8213 * Check if there are any parenthesis which needs removing.
8214 */
8215 if (pszExpr[0] == '(' && pszExpr[cchExpr - 1] == ')')
8216 {
8217 do
8218 {
8219 unsigned cPar = 1;
8220 char *psz = pszExpr + 1;
8221 char ch;
8222 while ((ch = *psz) != '\0')
8223 {
8224 if (ch == '(')
8225 cPar++;
8226 else if (ch == ')')
8227 {
8228 if (cPar <= 0)
8229 return VERR_PARSE_UNBALANCED_PARENTHESIS;
8230 cPar--;
8231 if (cPar == 0 && psz[1]) /* If not at end, there's nothing to do. */
8232 break;
8233 }
8234 /* next */
8235 psz++;
8236 }
8237 if (ch)
8238 break;
8239
8240 /* remove the parenthesis. */
8241 pszExpr++;
8242 cchExpr -= 2;
8243 pszExpr[cchExpr] = '\0';
8244
8245 /* strip blanks. */
8246 while (cchExpr > 0 && isblank(pszExpr[cchExpr - 1]))
8247 pszExpr[--cchExpr] = '\0';
8248 while (isblank(*pszExpr))
8249 pszExpr++, cchExpr--;
8250 if (!*pszExpr)
8251 return VERR_PARSE_EMPTY_ARGUMENT;
8252 } while (pszExpr[0] == '(' && pszExpr[cchExpr - 1] == ')');
8253 }
8254
8255 /* tabs to spaces. */
8256 char *psz = pszExpr;
8257 while ((psz = strchr(psz, '\t')) != NULL)
8258 *psz = ' ';
8259
8260 /*
8261 * Now, we need to look for the binary operator with the lowest precedence.
8262 *
8263 * If there are no operators we're left with a simple expression which we
8264 * evaluate with respect to unary operators
8265 */
8266 char *pszOpSplit = NULL;
8267 PCDBGCOP pOpSplit = NULL;
8268 unsigned cBinaryOps = 0;
8269 unsigned cPar = 0;
8270 char ch;
8271 char chPrev = ' ';
8272 bool fBinary = false;
8273 psz = pszExpr;
8274
8275 while ((ch = *psz) != '\0')
8276 {
8277 //Log2(("ch=%c cPar=%d fBinary=%d\n", ch, cPar, fBinary));
8278 /*
8279 * Parenthesis.
8280 */
8281 if (ch == '(')
8282 {
8283 cPar++;
8284 fBinary = false;
8285 }
8286 else if (ch == ')')
8287 {
8288 if (cPar <= 0)
8289 return VERR_PARSE_UNBALANCED_PARENTHESIS;
8290 cPar--;
8291 fBinary = true;
8292 }
8293 /*
8294 * Potential operator.
8295 */
8296 else if (cPar == 0 && !isblank(ch))
8297 {
8298 PCDBGCOP pOp = dbgcIsOpChar(ch)
8299 ? dbgcOperatorLookup(pDbgc, psz, fBinary, chPrev)
8300 : NULL;
8301 if (pOp)
8302 {
8303 /* If not the right kind of operator we've got a syntax error. */
8304 if (pOp->fBinary != fBinary)
8305 return VERR_PARSE_UNEXPECTED_OPERATOR;
8306
8307 /*
8308 * Update the parse state and skip the operator.
8309 */
8310 if (!pOpSplit)
8311 {
8312 pOpSplit = pOp;
8313 pszOpSplit = psz;
8314 cBinaryOps = fBinary;
8315 }
8316 else if (fBinary)
8317 {
8318 cBinaryOps++;
8319 if (pOp->iPrecedence >= pOpSplit->iPrecedence)
8320 {
8321 pOpSplit = pOp;
8322 pszOpSplit = psz;
8323 }
8324 }
8325
8326 psz += pOp->cchName - 1;
8327 fBinary = false;
8328 }
8329 else
8330 fBinary = true;
8331 }
8332
8333 /* next */
8334 psz++;
8335 chPrev = ch;
8336 } /* parse loop. */
8337
8338
8339 /*
8340 * Either we found an operator to divide the expression by
8341 * or we didn't find any. In the first case it's divide and
8342 * conquer. In the latter it's a single expression which
8343 * needs dealing with its unary operators if any.
8344 */
8345 int rc;
8346 if ( cBinaryOps
8347 && pOpSplit->fBinary)
8348 {
8349 /* process 1st sub expression. */
8350 *pszOpSplit = '\0';
8351 DBGCVAR Arg1;
8352 rc = dbgcEvalSub(pDbgc, pszExpr, pszOpSplit - pszExpr, &Arg1);
8353 if (VBOX_SUCCESS(rc))
8354 {
8355 /* process 2nd sub expression. */
8356 char *psz2 = pszOpSplit + pOpSplit->cchName;
8357 DBGCVAR Arg2;
8358 rc = dbgcEvalSub(pDbgc, psz2, cchExpr - (psz2 - pszExpr), &Arg2);
8359 if (VBOX_SUCCESS(rc))
8360 /* apply the operator. */
8361 rc = pOpSplit->pfnHandlerBinary(pDbgc, &Arg1, &Arg2, pResult);
8362 }
8363 }
8364 else if (cBinaryOps)
8365 {
8366 /* process sub expression. */
8367 pszOpSplit += pOpSplit->cchName;
8368 DBGCVAR Arg;
8369 rc = dbgcEvalSub(pDbgc, pszOpSplit, cchExpr - (pszOpSplit - pszExpr), &Arg);
8370 if (VBOX_SUCCESS(rc))
8371 /* apply the operator. */
8372 rc = pOpSplit->pfnHandlerUnary(pDbgc, &Arg, pResult);
8373 }
8374 else
8375 /* plain expression or using unary operators perhaps with paratheses. */
8376 rc = dbgcEvalSubUnary(pDbgc, pszExpr, cchExpr, pResult);
8377
8378 return rc;
8379}
8380
8381
8382/**
8383 * Parses the arguments of one command.
8384 *
8385 * @returns 0 on success.
8386 * @returns VBox error code on parse error with *pcArg containing the argument causing trouble.
8387 * @param pDbgc Debugger console instance data.
8388 * @param pCmd Pointer to the command descriptor.
8389 * @param pszArg Pointer to the arguments to parse.
8390 * @param paArgs Where to store the parsed arguments.
8391 * @param cArgs Size of the paArgs array.
8392 * @param pcArgs Where to store the number of arguments.
8393 * In the event of an error this is used to store the index of the offending argument.
8394 */
8395static int dbgcProcessArguments(PDBGC pDbgc, PCDBGCCMD pCmd, char *pszArgs, PDBGCVAR paArgs, unsigned cArgs, unsigned *pcArgs)
8396{
8397 Log2(("dbgcProcessArguments: pCmd=%s pszArgs='%s'\n", pCmd->pszCmd, pszArgs));
8398 /*
8399 * Check if we have any argument and if the command takes any.
8400 */
8401 *pcArgs = 0;
8402 /* strip leading blanks. */
8403 while (*pszArgs && isblank(*pszArgs))
8404 pszArgs++;
8405 if (!*pszArgs)
8406 {
8407 if (!pCmd->cArgsMin)
8408 return 0;
8409 return VERR_PARSE_TOO_FEW_ARGUMENTS;
8410 }
8411 /** @todo fixme - foo() doesn't work. */
8412 if (!pCmd->cArgsMax)
8413 return VERR_PARSE_TOO_MANY_ARGUMENTS;
8414
8415 /*
8416 * This is a hack, it's "temporary" and should go away "when" the parser is
8417 * modified to match arguments while parsing.
8418 */
8419 if ( pCmd->cArgsMax == 1
8420 && pCmd->cArgsMin == 1
8421 && pCmd->cArgDescs == 1
8422 && pCmd->paArgDescs[0].enmCategory == DBGCVAR_CAT_STRING
8423 && cArgs >= 1)
8424 {
8425 *pcArgs = 1;
8426 RTStrStripR(pszArgs);
8427 return dbgcEvalSubString(pDbgc, pszArgs, strlen(pszArgs), &paArgs[0]);
8428 }
8429
8430
8431 /*
8432 * The parse loop.
8433 */
8434 PDBGCVAR pArg0 = &paArgs[0];
8435 PDBGCVAR pArg = pArg0;
8436 *pcArgs = 0;
8437 do
8438 {
8439 /*
8440 * Can we have another argument?
8441 */
8442 if (*pcArgs >= pCmd->cArgsMax)
8443 return VERR_PARSE_TOO_MANY_ARGUMENTS;
8444 if (pArg >= &paArgs[cArgs])
8445 return VERR_PARSE_ARGUMENT_OVERFLOW;
8446
8447 /*
8448 * Find the end of the argument.
8449 */
8450 int cPar = 0;
8451 char chQuote = '\0';
8452 char *pszEnd = NULL;
8453 char *psz = pszArgs;
8454 char ch;
8455 bool fBinary = false;
8456 for (;;)
8457 {
8458 /*
8459 * Check for the end.
8460 */
8461 if ((ch = *psz) == '\0')
8462 {
8463 if (chQuote)
8464 return VERR_PARSE_UNBALANCED_QUOTE;
8465 if (cPar)
8466 return VERR_PARSE_UNBALANCED_PARENTHESIS;
8467 pszEnd = psz;
8468 break;
8469 }
8470 /*
8471 * When quoted we ignore everything but the quotation char.
8472 * We use the REXX way of escaping the quotation char, i.e. double occurence.
8473 */
8474 else if (ch == '\'' || ch == '"' || ch == '`')
8475 {
8476 if (chQuote)
8477 {
8478 /* end quote? */
8479 if (ch == chQuote)
8480 {
8481 if (psz[1] == ch)
8482 psz++; /* skip the escaped quote char */
8483 else
8484 chQuote = '\0'; /* end of quoted string. */
8485 }
8486 }
8487 else
8488 chQuote = ch; /* open new quote */
8489 }
8490 /*
8491 * Parenthesis can of course be nested.
8492 */
8493 else if (ch == '(')
8494 {
8495 cPar++;
8496 fBinary = false;
8497 }
8498 else if (ch == ')')
8499 {
8500 if (!cPar)
8501 return VERR_PARSE_UNBALANCED_PARENTHESIS;
8502 cPar--;
8503 fBinary = true;
8504 }
8505 else if (!chQuote && !cPar)
8506 {
8507 /*
8508 * Encountering blanks may mean the end of it all. A binary operator
8509 * will force continued parsing.
8510 */
8511 if (isblank(*psz))
8512 {
8513 pszEnd = psz++; /* just in case. */
8514 while (isblank(*psz))
8515 psz++;
8516 PCDBGCOP pOp = dbgcOperatorLookup(pDbgc, psz, fBinary, ' ');
8517 if (!pOp || pOp->fBinary != fBinary)
8518 break; /* the end. */
8519 psz += pOp->cchName;
8520 while (isblank(*psz)) /* skip blanks so we don't get here again */
8521 psz++;
8522 fBinary = false;
8523 continue;
8524 }
8525
8526 /*
8527 * Look for operators without a space up front.
8528 */
8529 if (dbgcIsOpChar(*psz))
8530 {
8531 PCDBGCOP pOp = dbgcOperatorLookup(pDbgc, psz, fBinary, ' ');
8532 if (pOp)
8533 {
8534 if (pOp->fBinary != fBinary)
8535 {
8536 pszEnd = psz;
8537 /** @todo this is a parsing error really. */
8538 break; /* the end. */
8539 }
8540 psz += pOp->cchName;
8541 while (isblank(*psz)) /* skip blanks so we don't get here again */
8542 psz++;
8543 fBinary = false;
8544 continue;
8545 }
8546 }
8547 fBinary = true;
8548 }
8549
8550 /* next char */
8551 psz++;
8552 }
8553 *pszEnd = '\0';
8554 /* (psz = next char to process) */
8555
8556 /*
8557 * Parse and evaluate the argument.
8558 */
8559 int rc = dbgcEvalSub(pDbgc, pszArgs, strlen(pszArgs), pArg);
8560 if (VBOX_FAILURE(rc))
8561 return rc;
8562
8563 /*
8564 * Next.
8565 */
8566 pArg++;
8567 (*pcArgs)++;
8568 pszArgs = psz;
8569 while (*pszArgs && isblank(*pszArgs))
8570 pszArgs++;
8571 } while (*pszArgs);
8572
8573 /*
8574 * Match the arguments.
8575 */
8576 return dbgcEvalSubMatchVars(pDbgc, pCmd->cArgsMin, pCmd->cArgsMax, pCmd->paArgDescs, pCmd->cArgDescs, pArg0, pArg - pArg0);
8577}
8578
8579
8580/**
8581 * Process one command.
8582 *
8583 * @returns VBox status code. Any error indicates the termination of the console session.
8584 * @param pDbgc Debugger console instance data.
8585 * @param pszCmd Pointer to the command.
8586 * @param cchCmd Length of the command.
8587 */
8588static int dbgcProcessCommand(PDBGC pDbgc, char *pszCmd, size_t cchCmd)
8589{
8590 char *pszCmdInput = pszCmd;
8591
8592 /*
8593 * Skip blanks.
8594 */
8595 while (isblank(*pszCmd))
8596 pszCmd++, cchCmd--;
8597
8598 /* external command? */
8599 bool fExternal = *pszCmd == '.';
8600 if (fExternal)
8601 pszCmd++, cchCmd--;
8602
8603 /*
8604 * Find arguments.
8605 */
8606 char *pszArgs = pszCmd;
8607 while (isalnum(*pszArgs))
8608 pszArgs++;
8609 if (*pszArgs && (!isblank(*pszArgs) || pszArgs == pszCmd))
8610 {
8611 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "Syntax error in command '%s'!\n", pszCmdInput);
8612 return 0;
8613 }
8614
8615 /*
8616 * Find the command.
8617 */
8618 PCDBGCCMD pCmd = dbgcRoutineLookup(pDbgc, pszCmd, pszArgs - pszCmd, fExternal);
8619 if (!pCmd || (pCmd->fFlags & DBGCCMD_FLAGS_FUNCTION))
8620 return pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "Unknown command '%s'!\n", pszCmdInput);
8621
8622 /*
8623 * Parse arguments (if any).
8624 */
8625 unsigned cArgs;
8626 int rc = dbgcProcessArguments(pDbgc, pCmd, pszArgs, &pDbgc->aArgs[pDbgc->iArg], ELEMENTS(pDbgc->aArgs) - pDbgc->iArg, &cArgs);
8627
8628 /*
8629 * Execute the command.
8630 */
8631 if (!rc)
8632 {
8633 rc = pCmd->pfnHandler(pCmd, &pDbgc->CmdHlp, pDbgc->pVM, &pDbgc->aArgs[0], cArgs, NULL);
8634 }
8635 else
8636 {
8637 /* report parse / eval error. */
8638 switch (rc)
8639 {
8640 case VERR_PARSE_TOO_FEW_ARGUMENTS:
8641 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8642 "Syntax error: Too few arguments. Minimum is %d for command '%s'.\n", pCmd->cArgsMin, pCmd->pszCmd);
8643 break;
8644 case VERR_PARSE_TOO_MANY_ARGUMENTS:
8645 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8646 "Syntax error: Too many arguments. Maximum is %d for command '%s'.\n", pCmd->cArgsMax, pCmd->pszCmd);
8647 break;
8648 case VERR_PARSE_ARGUMENT_OVERFLOW:
8649 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8650 "Syntax error: Too many arguments.\n");
8651 break;
8652 case VERR_PARSE_UNBALANCED_QUOTE:
8653 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8654 "Syntax error: Unbalanced quote (argument %d).\n", cArgs);
8655 break;
8656 case VERR_PARSE_UNBALANCED_PARENTHESIS:
8657 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8658 "Syntax error: Unbalanced parenthesis (argument %d).\n", cArgs);
8659 break;
8660 case VERR_PARSE_EMPTY_ARGUMENT:
8661 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8662 "Syntax error: An argument or subargument contains nothing useful (argument %d).\n", cArgs);
8663 break;
8664 case VERR_PARSE_UNEXPECTED_OPERATOR:
8665 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8666 "Syntax error: Invalid operator usage (argument %d).\n", cArgs);
8667 break;
8668 case VERR_PARSE_INVALID_NUMBER:
8669 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8670 "Syntax error: Ivalid numeric value (argument %d). If a string was the intention, then quote it.\n", cArgs);
8671 break;
8672 case VERR_PARSE_NUMBER_TOO_BIG:
8673 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8674 "Error: Numeric overflow (argument %d).\n", cArgs);
8675 break;
8676 case VERR_PARSE_INVALID_OPERATION:
8677 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8678 "Error: Invalid operation attempted (argument %d).\n", cArgs);
8679 break;
8680 case VERR_PARSE_FUNCTION_NOT_FOUND:
8681 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8682 "Error: Function not found (argument %d).\n", cArgs);
8683 break;
8684 case VERR_PARSE_NOT_A_FUNCTION:
8685 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8686 "Error: The function specified is not a function (argument %d).\n", cArgs);
8687 break;
8688 case VERR_PARSE_NO_MEMORY:
8689 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8690 "Error: Out memory in the regular heap! Expect odd stuff to happen...\n", cArgs);
8691 break;
8692 case VERR_PARSE_INCORRECT_ARG_TYPE:
8693 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8694 "Error: Incorrect argument type (argument %d?).\n", cArgs);
8695 break;
8696 case VERR_PARSE_VARIABLE_NOT_FOUND:
8697 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8698 "Error: An undefined variable was referenced (argument %d).\n", cArgs);
8699 break;
8700 case VERR_PARSE_CONVERSION_FAILED:
8701 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8702 "Error: A conversion between two types failed (argument %d).\n", cArgs);
8703 break;
8704 case VERR_PARSE_NOT_IMPLEMENTED:
8705 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8706 "Error: You hit a debugger feature which isn't implemented yet (argument %d).\n", cArgs);
8707 break;
8708 case VERR_PARSE_BAD_RESULT_TYPE:
8709 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8710 "Error: Couldn't satisfy a request for a specific result type (argument %d). (Usually applies to symbols)\n", cArgs);
8711 break;
8712 case VERR_PARSE_WRITEONLY_SYMBOL:
8713 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8714 "Error: Cannot get symbol, it's set only (argument %d).\n", cArgs);
8715 break;
8716
8717 default:
8718 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8719 "Error: Unknown error %d!\n", rc);
8720 return rc;
8721 }
8722
8723 /*
8724 * Parse errors are non fatal.
8725 */
8726 if (rc >= VERR_PARSE_FIRST && rc < VERR_PARSE_LAST)
8727 rc = 0;
8728 }
8729
8730 return rc;
8731}
8732
8733
8734/**
8735 * Process all commands current in the buffer.
8736 *
8737 * @returns VBox status code. Any error indicates the termination of the console session.
8738 * @param pDbgc Debugger console instance data.
8739 */
8740static int dbgcProcessCommands(PDBGC pDbgc)
8741{
8742 int rc = 0;
8743 while (pDbgc->cInputLines)
8744 {
8745 /*
8746 * Empty the log buffer if we're hooking the log.
8747 */
8748 if (pDbgc->fLog)
8749 {
8750 rc = dbgcProcessLog(pDbgc);
8751 if (VBOX_FAILURE(rc))
8752 break;
8753 }
8754
8755 if (pDbgc->iRead == pDbgc->iWrite)
8756 {
8757 AssertMsgFailed(("The input buffer is empty while cInputLines=%d!\n", pDbgc->cInputLines));
8758 pDbgc->cInputLines = 0;
8759 return 0;
8760 }
8761
8762 /*
8763 * Copy the command to the parse buffer.
8764 */
8765 char ch;
8766 char *psz = &pDbgc->achInput[pDbgc->iRead];
8767 char *pszTrg = &pDbgc->achScratch[0];
8768 while ((*pszTrg = ch = *psz++) != ';' && ch != '\n' )
8769 {
8770 if (psz == &pDbgc->achInput[sizeof(pDbgc->achInput)])
8771 psz = &pDbgc->achInput[0];
8772
8773 if (psz == &pDbgc->achInput[pDbgc->iWrite])
8774 {
8775 AssertMsgFailed(("The buffer contains no commands while cInputLines=%d!\n", pDbgc->cInputLines));
8776 pDbgc->cInputLines = 0;
8777 return 0;
8778 }
8779
8780 pszTrg++;
8781 }
8782 *pszTrg = '\0';
8783
8784 /*
8785 * Advance the buffer.
8786 */
8787 pDbgc->iRead = psz - &pDbgc->achInput[0];
8788 if (ch == '\n')
8789 pDbgc->cInputLines--;
8790
8791 /*
8792 * Parse and execute this command.
8793 */
8794 pDbgc->pszScratch = psz;
8795 pDbgc->iArg = 0;
8796 rc = dbgcProcessCommand(pDbgc, &pDbgc->achScratch[0], psz - &pDbgc->achScratch[0] - 1);
8797 if (rc)
8798 break;
8799 }
8800
8801 return rc;
8802}
8803
8804
8805/**
8806 * Reads input, parses it and executes commands on '\n'.
8807 *
8808 * @returns VBox status.
8809 * @param pDbgc Debugger console instance data.
8810 */
8811static int dbgcProcessInput(PDBGC pDbgc)
8812{
8813 /*
8814 * We know there's input ready, so let's read it first.
8815 */
8816 int rc = dbgcInputRead(pDbgc);
8817 if (VBOX_FAILURE(rc))
8818 return rc;
8819
8820 /*
8821 * Now execute any ready commands.
8822 */
8823 if (pDbgc->cInputLines)
8824 {
8825 /** @todo this fReady stuff is broken. */
8826 pDbgc->fReady = false;
8827 rc = dbgcProcessCommands(pDbgc);
8828 if (VBOX_SUCCESS(rc) && rc != VWRN_DBGC_CMD_PENDING)
8829 pDbgc->fReady = true;
8830 if ( VBOX_SUCCESS(rc)
8831 && pDbgc->iRead == pDbgc->iWrite
8832 && pDbgc->fReady)
8833 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "VBoxDbg> ");
8834 }
8835
8836 return rc;
8837}
8838
8839
8840/**
8841 * Gets the event context identifier string.
8842 * @returns Read only string.
8843 * @param enmCtx The context.
8844 */
8845static const char *dbgcGetEventCtx(DBGFEVENTCTX enmCtx)
8846{
8847 switch (enmCtx)
8848 {
8849 case DBGFEVENTCTX_RAW: return "raw";
8850 case DBGFEVENTCTX_REM: return "rem";
8851 case DBGFEVENTCTX_HWACCL: return "hwaccl";
8852 case DBGFEVENTCTX_HYPER: return "hyper";
8853 case DBGFEVENTCTX_OTHER: return "other";
8854
8855 case DBGFEVENTCTX_INVALID: return "!Invalid Event Ctx!";
8856 default:
8857 AssertMsgFailed(("enmCtx=%d\n", enmCtx));
8858 return "!Unknown Event Ctx!";
8859 }
8860}
8861
8862
8863/**
8864 * Processes debugger events.
8865 *
8866 * @returns VBox status.
8867 * @param pDbgc DBGC Instance data.
8868 * @param pEvent Pointer to event data.
8869 */
8870static int dbgcProcessEvent(PDBGC pDbgc, PCDBGFEVENT pEvent)
8871{
8872 /*
8873 * Flush log first.
8874 */
8875 if (pDbgc->fLog)
8876 {
8877 int rc = dbgcProcessLog(pDbgc);
8878 if (VBOX_FAILURE(rc))
8879 return rc;
8880 }
8881
8882 /*
8883 * Process the event.
8884 */
8885 pDbgc->pszScratch = &pDbgc->achInput[0];
8886 pDbgc->iArg = 0;
8887 bool fPrintPrompt = true;
8888 int rc = VINF_SUCCESS;
8889 switch (pEvent->enmType)
8890 {
8891 /*
8892 * The first part is events we have initiated with commands.
8893 */
8894 case DBGFEVENT_HALT_DONE:
8895 {
8896 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: VM %p is halted! (%s)\n",
8897 pDbgc->pVM, dbgcGetEventCtx(pEvent->enmCtx));
8898 pDbgc->fRegCtxGuest = true; /* we're always in guest context when halted. */
8899 if (VBOX_SUCCESS(rc))
8900 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8901 break;
8902 }
8903
8904
8905 /*
8906 * The second part is events which can occur at any time.
8907 */
8908 case DBGFEVENT_FATAL_ERROR:
8909 {
8910 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbf event: Fatal error! (%s)\n",
8911 dbgcGetEventCtx(pEvent->enmCtx));
8912 pDbgc->fRegCtxGuest = false; /* fatal errors are always in hypervisor. */
8913 if (VBOX_SUCCESS(rc))
8914 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8915 break;
8916 }
8917
8918 case DBGFEVENT_BREAKPOINT:
8919 case DBGFEVENT_BREAKPOINT_HYPER:
8920 {
8921 bool fRegCtxGuest = pDbgc->fRegCtxGuest;
8922 pDbgc->fRegCtxGuest = pEvent->enmType == DBGFEVENT_BREAKPOINT;
8923
8924 rc = dbgcBpExec(pDbgc, pEvent->u.Bp.iBp);
8925 switch (rc)
8926 {
8927 case VERR_DBGC_BP_NOT_FOUND:
8928 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Unknown breakpoint %u! (%s)\n",
8929 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
8930 break;
8931
8932 case VINF_DBGC_BP_NO_COMMAND:
8933 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Breakpoint %u! (%s)\n",
8934 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
8935 break;
8936
8937 case VINF_BUFFER_OVERFLOW:
8938 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Breakpoint %u! Command too long to execute! (%s)\n",
8939 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
8940 break;
8941
8942 default:
8943 break;
8944 }
8945 if (VBOX_SUCCESS(rc) && DBGFR3IsHalted(pDbgc->pVM))
8946 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8947 else
8948 pDbgc->fRegCtxGuest = fRegCtxGuest;
8949 break;
8950 }
8951
8952 case DBGFEVENT_STEPPED:
8953 case DBGFEVENT_STEPPED_HYPER:
8954 {
8955 pDbgc->fRegCtxGuest = pEvent->enmType == DBGFEVENT_STEPPED;
8956
8957 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Single step! (%s)\n", dbgcGetEventCtx(pEvent->enmCtx));
8958 if (VBOX_SUCCESS(rc))
8959 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8960 break;
8961 }
8962
8963 case DBGFEVENT_ASSERTION_HYPER:
8964 {
8965 pDbgc->fRegCtxGuest = false;
8966
8967 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8968 "\ndbgf event: Hypervisor Assertion! (%s)\n"
8969 "%s"
8970 "%s"
8971 "\n",
8972 dbgcGetEventCtx(pEvent->enmCtx),
8973 pEvent->u.Assert.pszMsg1,
8974 pEvent->u.Assert.pszMsg2);
8975 if (VBOX_SUCCESS(rc))
8976 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8977 break;
8978 }
8979
8980 case DBGFEVENT_DEV_STOP:
8981 {
8982 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8983 "\n"
8984 "dbgf event: DBGFSTOP (%s)\n"
8985 "File: %s\n"
8986 "Line: %d\n"
8987 "Function: %s\n",
8988 dbgcGetEventCtx(pEvent->enmCtx),
8989 pEvent->u.Src.pszFile,
8990 pEvent->u.Src.uLine,
8991 pEvent->u.Src.pszFunction);
8992 if (VBOX_SUCCESS(rc) && pEvent->u.Src.pszMessage && *pEvent->u.Src.pszMessage)
8993 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
8994 "Message: %s\n",
8995 pEvent->u.Src.pszMessage);
8996 if (VBOX_SUCCESS(rc))
8997 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
8998 break;
8999 }
9000
9001
9002 case DBGFEVENT_INVALID_COMMAND:
9003 {
9004 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf/dbgc error: Invalid command event!\n");
9005 fPrintPrompt = !pDbgc->fReady;
9006 break;
9007 }
9008
9009 case DBGFEVENT_TERMINATING:
9010 {
9011 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\nVM is terminating!\n");
9012 rc = VERR_GENERAL_FAILURE;
9013 break;
9014 }
9015
9016
9017 default:
9018 {
9019 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf/dbgc error: Unknown event %d!\n", pEvent->enmType);
9020 fPrintPrompt = !pDbgc->fReady;
9021 break;
9022 }
9023 }
9024
9025 /*
9026 * Prompt, anyone?
9027 */
9028 if (fPrintPrompt && VBOX_SUCCESS(rc))
9029 {
9030 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "VBoxDbg> ");
9031 }
9032
9033 return rc;
9034}
9035
9036
9037
9038
9039
9040/**
9041 * Make a console instance.
9042 *
9043 * This will not return until either an 'exit' command is issued or a error code
9044 * indicating connection loss is encountered.
9045 *
9046 * @returns VINF_SUCCESS if console termination caused by the 'exit' command.
9047 * @returns The VBox status code causing the console termination.
9048 *
9049 * @param pVM VM Handle.
9050 * @param pBack Pointer to the backend structure. This must contain
9051 * a full set of function pointers to service the console.
9052 * @param fFlags Reserved, must be zero.
9053 * @remark A forced termination of the console is easiest done by forcing the
9054 * callbacks to return fatal failures.
9055 */
9056DBGDECL(int) DBGCCreate(PVM pVM, PDBGCBACK pBack, unsigned fFlags)
9057{
9058 /*
9059 * Validate input.
9060 */
9061 AssertReturn(VALID_PTR(pVM), VERR_INVALID_PARAMETER);
9062 AssertReturn(VALID_PTR(pBack), VERR_INVALID_PARAMETER);
9063 AssertMsgReturn(!fFlags, ("%#x", fFlags), VERR_INVALID_PARAMETER);
9064
9065 /*
9066 * Allocate and initialize instance data
9067 */
9068 PDBGC pDbgc = (PDBGC)RTMemAllocZ(sizeof(*pDbgc));
9069 if (!pDbgc)
9070 return VERR_NO_MEMORY;
9071
9072 pDbgc->CmdHlp.pfnWrite = dbgcHlpWrite;
9073 pDbgc->CmdHlp.pfnPrintfV = dbgcHlpPrintfV;
9074 pDbgc->CmdHlp.pfnPrintf = dbgcHlpPrintf;
9075 pDbgc->CmdHlp.pfnVBoxErrorV = dbgcHlpVBoxErrorV;
9076 pDbgc->CmdHlp.pfnVBoxError = dbgcHlpVBoxError;
9077 pDbgc->CmdHlp.pfnMemRead = dbgcHlpMemRead;
9078 pDbgc->CmdHlp.pfnMemWrite = dbgcHlpMemWrite;
9079 pDbgc->CmdHlp.pfnEval = dbgcHlpEval;
9080 pDbgc->CmdHlp.pfnExec = dbgcHlpExec;
9081 pDbgc->CmdHlp.pfnVarToDbgfAddr = dbgcHlpVarToDbgfAddr;
9082 pDbgc->CmdHlp.pfnVarToBool = dbgcHlpVarToBool;
9083 pDbgc->pBack = pBack;
9084 pDbgc->pszScratch = &pDbgc->achScratch[0];
9085 pDbgc->fRegTerse = true;
9086 pDbgc->fRegCtxGuest = true;
9087 pDbgc->fLog = false;
9088 pDbgc->iRead = 0;
9089 pDbgc->iWrite = 0;
9090 pDbgc->fReady = true;
9091 pDbgc->fInputOverflow = false;
9092 pDbgc->cInputLines = 0;
9093
9094 dbgcInitOpCharBitMap();
9095
9096 /*
9097 * Print welcome message.
9098 */
9099 int rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
9100 "Welcome to the VirtualBox Debugger!\n");
9101 if (VBOX_FAILURE(rc))
9102 goto l_failure;
9103
9104 /*
9105 * Attach to the VM.
9106 */
9107 rc = DBGFR3Attach(pVM);
9108 if (VBOX_FAILURE(rc))
9109 {
9110 rc = pDbgc->CmdHlp.pfnVBoxError(&pDbgc->CmdHlp, rc, "When trying to attach to VM %p\n", pDbgc->pVM);
9111 goto l_failure;
9112 }
9113 pDbgc->pVM = pVM;
9114
9115 /*
9116 * Print commandline and auto select result.
9117 */
9118 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
9119 "Current VM is %08x\n" /** @todo get and print the VM name! */
9120 "VBoxDbg> ",
9121 pDbgc->pVM);
9122 if (VBOX_FAILURE(rc))
9123 goto l_failure;
9124
9125 /*
9126 * Main Debugger Loop.
9127 *
9128 * This loop will either block on waiting for input or on waiting on
9129 * debug events. If we're forwarding the log we cannot wait for long
9130 * before we must flush the log.
9131 */
9132 for (rc = 0;;)
9133 {
9134 if (pDbgc->pVM && DBGFR3CanWait(pDbgc->pVM))
9135 {
9136 /*
9137 * Wait for a debug event.
9138 */
9139 PCDBGFEVENT pEvent;
9140 rc = DBGFR3EventWait(pDbgc->pVM, pDbgc->fLog ? 1 : 32, &pEvent);
9141 if (VBOX_SUCCESS(rc))
9142 {
9143 rc = dbgcProcessEvent(pDbgc, pEvent);
9144 if (VBOX_FAILURE(rc))
9145 break;
9146 }
9147 else if (rc != VERR_TIMEOUT)
9148 break;
9149
9150 /*
9151 * Check for input.
9152 */
9153 if (pBack->pfnInput(pDbgc->pBack, 0))
9154 {
9155 rc = dbgcProcessInput(pDbgc);
9156 if (VBOX_FAILURE(rc))
9157 break;
9158 }
9159 }
9160 else
9161 {
9162 /*
9163 * Wait for input. If Logging is enabled we'll only wait very briefly.
9164 */
9165 if (pBack->pfnInput(pDbgc->pBack, pDbgc->fLog ? 1 : 1000))
9166 {
9167 rc = dbgcProcessInput(pDbgc);
9168 if (VBOX_FAILURE(rc))
9169 break;
9170 }
9171 }
9172
9173 /*
9174 * Forward log output.
9175 */
9176 if (pDbgc->fLog)
9177 {
9178 rc = dbgcProcessLog(pDbgc);
9179 if (VBOX_FAILURE(rc))
9180 break;
9181 }
9182 }
9183
9184
9185l_failure:
9186 /*
9187 * Cleanup console debugger session.
9188 */
9189 /* Disable log hook. */
9190 if (pDbgc->fLog)
9191 {
9192
9193 }
9194
9195 /* Detach from the VM. */
9196 if (pDbgc->pVM)
9197 DBGFR3Detach(pDbgc->pVM);
9198
9199 /* finally, free the instance memory. */
9200 RTMemFree(pDbgc);
9201
9202 return rc;
9203}
9204
9205
9206
9207/**
9208 * Register one or more external commands.
9209 *
9210 * @returns VBox status.
9211 * @param paCommands Pointer to an array of command descriptors.
9212 * The commands must be unique. It's not possible
9213 * to register the same commands more than once.
9214 * @param cCommands Number of commands.
9215 */
9216DBGDECL(int) DBGCRegisterCommands(PCDBGCCMD paCommands, unsigned cCommands)
9217{
9218 /*
9219 * Lock the list.
9220 */
9221 DBGCEXTCMDS_LOCK_WR();
9222 PDBGCEXTCMDS pCur = g_pExtCmdsHead;
9223 while (pCur)
9224 {
9225 if (paCommands == pCur->paCmds)
9226 {
9227 DBGCEXTCMDS_UNLOCK_WR();
9228 AssertMsgFailed(("Attempt at re-registering %d command(s)!\n", cCommands));
9229 return VWRN_DBGC_ALREADY_REGISTERED;
9230 }
9231 pCur = pCur->pNext;
9232 }
9233
9234 /*
9235 * Allocate new chunk.
9236 */
9237 int rc = 0;
9238 pCur = (PDBGCEXTCMDS)RTMemAlloc(sizeof(*pCur));
9239 if (pCur)
9240 {
9241 pCur->cCmds = cCommands;
9242 pCur->paCmds = paCommands;
9243 pCur->pNext = g_pExtCmdsHead;
9244 g_pExtCmdsHead = pCur;
9245 }
9246 else
9247 rc = VERR_NO_MEMORY;
9248 DBGCEXTCMDS_UNLOCK_WR();
9249
9250 return rc;
9251}
9252
9253
9254/**
9255 * Deregister one or more external commands previously registered by
9256 * DBGCRegisterCommands().
9257 *
9258 * @returns VBox status.
9259 * @param paCommands Pointer to an array of command descriptors
9260 * as given to DBGCRegisterCommands().
9261 * @param cCommands Number of commands.
9262 */
9263DBGDECL(int) DBGCDeregisterCommands(PCDBGCCMD paCommands, unsigned cCommands)
9264{
9265 /*
9266 * Lock the list.
9267 */
9268 DBGCEXTCMDS_LOCK_WR();
9269 PDBGCEXTCMDS pPrev = NULL;
9270 PDBGCEXTCMDS pCur = g_pExtCmdsHead;
9271 while (pCur)
9272 {
9273 if (paCommands == pCur->paCmds)
9274 {
9275 if (pPrev)
9276 pPrev->pNext = pCur->pNext;
9277 else
9278 g_pExtCmdsHead = pCur->pNext;
9279 DBGCEXTCMDS_UNLOCK_WR();
9280
9281 RTMemFree(pCur);
9282 return VINF_SUCCESS;
9283 }
9284 pPrev = pCur;
9285 pCur = pCur->pNext;
9286 }
9287 DBGCEXTCMDS_UNLOCK_WR();
9288
9289 NOREF(cCommands);
9290 return VERR_DBGC_COMMANDS_NOT_REGISTERED;
9291}
9292
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