VirtualBox

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

Last change on this file since 46144 was 46137, checked in by vboxsync, 12 years ago

Missing file, reverting 85830.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 34.4 KB
Line 
1/* $Id: DBGConsole.cpp 46137 2013-05-17 08:09:59Z vboxsync $ */
2/** @file
3 * DBGC - Debugger Console.
4 */
5
6/*
7 * Copyright (C) 2006-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/** @page pg_dbgc DBGC - The Debug Console
20 *
21 * The debugger console is an early attempt to make some interactive
22 * debugging facilities for the VirtualBox VMM. It was initially only
23 * accessible thru a telnet session in debug builds. Later it was hastily built
24 * into the VBoxDbg module with a very simple Qt wrapper around it.
25 *
26 * The current state is that it's by default shipped with all standard
27 * VirtualBox builds. The GUI component is by default accessible in all
28 * non-release builds, while release builds require extra data, environment or
29 * command line options to make it visible.
30 *
31 * Now, even if we ship it with all standard builds we would like it to remain
32 * an optional feature that can be omitted when building VirtualBox. Therefore,
33 * all external code interfacing DBGC need to be enclosed in
34 * \#ifdef VBOX_WITH_DEBUGGER blocks. This is mandatory for components that
35 * register external commands.
36 *
37 *
38 * @section sec_dbgc_op Operation
39 *
40 * The console will process commands in a manner similar to the OS/2 and Windows
41 * kernel debuggers. This means ';' is a command separator and that when
42 * possible we'll use the same command names as these two uses. As an
43 * alternative we intent to provide a set of gdb-like commands as well and let
44 * the user decide which should take precedence.
45 *
46 *
47 * @subsection sec_dbg_op_numbers Numbers
48 *
49 * Numbers are hexadecimal unless specified with a prefix indicating
50 * elsewise. Prefixes:
51 * - '0x' - hexadecimal.
52 * - '0n' - decimal
53 * - '0t' - octal.
54 * - '0y' - binary.
55 *
56 * Some of the prefixes are a bit uncommon, the reason for this that the
57 * typical binary prefix '0b' can also be a hexadecimal value since no prefix or
58 * suffix is required for such values. Ditto for '0n' and '0' for decimal and
59 * octal.
60 *
61 * The '`' can be used in the numeric value to separate parts as the user
62 * wishes. Generally, though the debugger may use it in output as thousand
63 * separator in decimal numbers and 32-bit separator in hex numbers.
64 *
65 * For historical reasons, a 'h' suffix is suffered on hex numbers. Unlike most
66 * assemblers, a leading 0 before a-f is not required with the 'h' suffix.
67 *
68 * The prefix '0i' can be used instead of '0n', as it was the early decimal
69 * prefix employed by DBGC. It's being deprecated and may be removed later.
70 *
71 *
72 * @subsection sec_dbg_op_strings Strings and Symbols
73 *
74 * The debugger will try to guess, convert or promote what the type of an
75 * argument to a command, function or operator based on the input description of
76 * the receiver. If the user wants to make it clear to the debugger that
77 * something is a string, put it inside double quotes. Symbols should use
78 * single quotes, though we're current still a bit flexible on this point.
79 *
80 * If you need to put a quote character inside the quoted text, you escape it by
81 * repating it once: echo "printf(""hello world"");"
82 *
83 *
84 * @subsection sec_dbg_op_address Addressing modes
85 *
86 * - Default is flat. For compatibility '%' also means flat.
87 * - Segmented addresses are specified selector:offset.
88 * - Physical addresses are specified using '%%'.
89 * - The default target for the addressing is the guest context, the '#'
90 * will override this and set it to the host.
91 * Note that several operations won't work on host addresses.
92 *
93 * The '%', '%%' and '#' prefixes is implemented as unary operators, while ':'
94 * is a binary operator. Operator precedence takes care of evaluation order.
95 *
96 *
97 * @subsection sec_dbg_op_c_operators C/C++ Operators
98 *
99 * Most unary and binary arithmetic, comparison, logical and bitwise C/C++
100 * operators are supported by the debugger, with the same precedence rules of
101 * course. There is one notable change made due to the unary '%' and '%%'
102 * operators, and that is that the modulo (remainder) operator is called 'mod'
103 * instead of '%'. This saves a lot of trouble separating argument.
104 *
105 * There are no assignment operators. Instead some simple global variable space
106 * is provided thru the 'set' and 'unset' commands and the unary '$' operator.
107 *
108 *
109 * @subsection sec_dbg_op_registers Registers
110 *
111 * All registers and their sub-fields exposed by the DBGF API are accessible via
112 * the '\@' operator. A few CPU register are accessible directly (as symbols)
113 * without using the '\@' operator. Hypervisor registers are accessible by
114 * prefixing the register name with a dot ('.').
115 *
116 *
117 * @subsection sec_dbg_op_commands Commands
118 *
119 * Commands names are case sensitive. By convention they are lower cased, starts
120 * with a letter but may contain digits and underscores afterwards. Operators
121 * are not allowed in the name (not even part of it), as we would risk
122 * misunderstanding it otherwise.
123 *
124 * Commands returns a status code.
125 *
126 * The '.' prefix indicates the set of external commands. External commands are
127 * command registered by VMM components.
128 *
129 *
130 * @subsection sec_dbg_op_functions Functions
131 *
132 * Functions are similar to commands, but return a variable and can only be used
133 * as part of an expression making up the argument of a command, function,
134 * operator or language statement (if we get around to implement that).
135 *
136 *
137 * @section sec_dbgc_logging Logging
138 *
139 * The idea is to be able to pass thru debug and release logs to the console
140 * if the user so wishes. This feature requires some kind of hook into the
141 * logger instance and while this was sketched it hasn't yet been implemented
142 * (dbgcProcessLog and DBGC::fLog).
143 *
144 * This feature has not materialized and probably never will.
145 *
146 *
147 * @section sec_dbgc_linking Linking and API
148 *
149 * The DBGC code is linked into the VBoxVMM module.
150 *
151 * IMachineDebugger may one day be extended with a DBGC interface so we can work
152 * with DBGC remotely without requiring TCP. Some questions about callbacks
153 * (for output) and security (you may wish to restrict users from debugging a
154 * VM) needs to be answered first though.
155 */
156
157
158/*******************************************************************************
159* Header Files *
160*******************************************************************************/
161#define LOG_GROUP LOG_GROUP_DBGC
162#include <VBox/dbg.h>
163#include <VBox/vmm/dbgf.h>
164#include <VBox/vmm/vmapi.h> /* VMR3GetVM() */
165#include <VBox/vmm/hm.h> /* HMR3IsEnabled */
166#include <VBox/err.h>
167#include <VBox/log.h>
168
169#include <iprt/asm.h>
170#include <iprt/assert.h>
171#include <iprt/mem.h>
172#include <iprt/string.h>
173
174#include "DBGCInternal.h"
175#include "DBGPlugIns.h"
176
177
178/*******************************************************************************
179* Internal Functions *
180*******************************************************************************/
181static int dbgcProcessLog(PDBGC pDbgc);
182
183
184/**
185 * Resolves a symbol (or tries to do so at least).
186 *
187 * @returns 0 on success.
188 * @returns VBox status on failure.
189 * @param pDbgc The debug console instance.
190 * @param pszSymbol The symbol name.
191 * @param enmType The result type. Specifying DBGCVAR_TYPE_GC_FAR may
192 * cause failure, avoid it.
193 * @param pResult Where to store the result.
194 */
195int dbgcSymbolGet(PDBGC pDbgc, const char *pszSymbol, DBGCVARTYPE enmType, PDBGCVAR pResult)
196{
197 int rc;
198
199 /*
200 * Builtin?
201 */
202 PCDBGCSYM pSymDesc = dbgcLookupRegisterSymbol(pDbgc, pszSymbol);
203 if (pSymDesc)
204 {
205 if (!pSymDesc->pfnGet)
206 return VERR_DBGC_PARSE_WRITEONLY_SYMBOL;
207 return pSymDesc->pfnGet(pSymDesc, &pDbgc->CmdHlp, enmType, pResult);
208 }
209
210 /*
211 * A typical register? (Guest only)
212 */
213 static const char s_szSixLetterRegisters[] =
214 "rflags;eflags;"
215 ;
216 static const char s_szThreeLetterRegisters[] =
217 "eax;rax;" "r10;" "r8d;r8w;r8b;" "cr0;" "dr0;"
218 "ebx;rbx;" "r11;" "r9d;r9w;r8b;" "dr1;"
219 "ecx;rcx;" "r12;" "cr2;" "dr2;"
220 "edx;rdx;" "r13;" "cr3;" "dr3;"
221 "edi;rdi;dil;" "r14;" "cr4;" "dr4;"
222 "esi;rsi;sil;" "r15;" "cr8;"
223 "ebp;rbp;"
224 "esp;rsp;" "dr6;"
225 "rip;eip;" "dr7;"
226 "efl;"
227 ;
228 static const char s_szTwoLetterRegisters[] =
229 "ax;al;ah;" "r8;"
230 "bx;bl;bh;" "r9;"
231 "cx;cl;ch;" "cs;"
232 "dx;dl;dh;" "ds;"
233 "di;" "es;"
234 "si;" "fs;"
235 "bp;" "gs;"
236 "sp;" "ss;"
237 "ip;"
238 ;
239 size_t const cchSymbol = strlen(pszSymbol);
240 if ( (cchSymbol == 2 && strstr(s_szTwoLetterRegisters, pszSymbol))
241 || (cchSymbol == 3 && strstr(s_szThreeLetterRegisters, pszSymbol))
242 || (cchSymbol == 6 && strstr(s_szSixLetterRegisters, pszSymbol)))
243 {
244 if (!strchr(pszSymbol, ';'))
245 {
246 DBGCVAR Var;
247 DBGCVAR_INIT_SYMBOL(&Var, pszSymbol);
248 rc = dbgcOpRegister(pDbgc, &Var, DBGCVAR_CAT_ANY, pResult);
249 if (RT_SUCCESS(rc))
250 return DBGCCmdHlpConvert(&pDbgc->CmdHlp, pResult, enmType, false /*fConvSyms*/, pResult);
251 }
252 }
253
254 /*
255 * Ask PDM.
256 */
257 /** @todo resolve symbols using PDM. */
258
259 /*
260 * Ask the debug info manager.
261 */
262 RTDBGSYMBOL Symbol;
263 rc = DBGFR3AsSymbolByName(pDbgc->pUVM, pDbgc->hDbgAs, pszSymbol, &Symbol, NULL);
264 if (RT_SUCCESS(rc))
265 {
266 /*
267 * Default return is a flat gc address.
268 */
269 DBGCVAR_INIT_GC_FLAT(pResult, Symbol.Value);
270 if (Symbol.cb)
271 DBGCVAR_SET_RANGE(pResult, DBGCVAR_RANGE_BYTES, Symbol.cb);
272
273 switch (enmType)
274 {
275 /* nothing to do. */
276 case DBGCVAR_TYPE_GC_FLAT:
277 case DBGCVAR_TYPE_ANY:
278 return VINF_SUCCESS;
279
280 /* impossible at the moment. */
281 case DBGCVAR_TYPE_GC_FAR:
282 return VERR_DBGC_PARSE_CONVERSION_FAILED;
283
284 /* simply make it numeric. */
285 case DBGCVAR_TYPE_NUMBER:
286 pResult->enmType = DBGCVAR_TYPE_NUMBER;
287 pResult->u.u64Number = Symbol.Value;
288 return VINF_SUCCESS;
289
290 /* cast it. */
291 case DBGCVAR_TYPE_GC_PHYS:
292 case DBGCVAR_TYPE_HC_FLAT:
293 case DBGCVAR_TYPE_HC_PHYS:
294 return DBGCCmdHlpConvert(&pDbgc->CmdHlp, pResult, enmType, false /*fConvSyms*/, pResult);
295
296 default:
297 AssertMsgFailed(("Internal error enmType=%d\n", enmType));
298 return VERR_INVALID_PARAMETER;
299 }
300 }
301
302 return VERR_DBGC_PARSE_NOT_IMPLEMENTED;
303}
304
305
306/**
307 * Process all commands currently in the buffer.
308 *
309 * @returns VBox status code. Any error indicates the termination of the console session.
310 * @param pDbgc Debugger console instance data.
311 * @param fNoExecute Indicates that no commands should actually be executed.
312 */
313static int dbgcProcessCommands(PDBGC pDbgc, bool fNoExecute)
314{
315 /** @todo Replace this with a sh/ksh/csh/rexx like toplevel language that
316 * allows doing function, loops, if, cases, and such. */
317 int rc = VINF_SUCCESS;
318 while (pDbgc->cInputLines)
319 {
320 /*
321 * Empty the log buffer if we're hooking the log.
322 */
323 if (pDbgc->fLog)
324 {
325 rc = dbgcProcessLog(pDbgc);
326 if (RT_FAILURE(rc))
327 break;
328 }
329
330 if (pDbgc->iRead == pDbgc->iWrite)
331 {
332 AssertMsgFailed(("The input buffer is empty while cInputLines=%d!\n", pDbgc->cInputLines));
333 pDbgc->cInputLines = 0;
334 return 0;
335 }
336
337 /*
338 * Copy the command to the parse buffer.
339 */
340 char ch;
341 char *psz = &pDbgc->achInput[pDbgc->iRead];
342 char *pszTrg = &pDbgc->achScratch[0];
343 while ((*pszTrg = ch = *psz++) != ';' && ch != '\n' )
344 {
345 if (psz == &pDbgc->achInput[sizeof(pDbgc->achInput)])
346 psz = &pDbgc->achInput[0];
347
348 if (psz == &pDbgc->achInput[pDbgc->iWrite])
349 {
350 AssertMsgFailed(("The buffer contains no commands while cInputLines=%d!\n", pDbgc->cInputLines));
351 pDbgc->cInputLines = 0;
352 return 0;
353 }
354
355 pszTrg++;
356 }
357 *pszTrg = '\0';
358
359 /*
360 * Advance the buffer.
361 */
362 pDbgc->iRead = psz - &pDbgc->achInput[0];
363 if (ch == '\n')
364 pDbgc->cInputLines--;
365
366 /*
367 * Parse and execute this command.
368 */
369 pDbgc->pszScratch = pszTrg + 1;
370 pDbgc->iArg = 0;
371 rc = dbgcEvalCommand(pDbgc, &pDbgc->achScratch[0], pszTrg - &pDbgc->achScratch[0] - 1, fNoExecute);
372 if ( rc == VERR_DBGC_QUIT
373 || rc == VWRN_DBGC_CMD_PENDING)
374 break;
375 rc = VINF_SUCCESS; /* ignore other statuses */
376 }
377
378 return rc;
379}
380
381
382/**
383 * Handle input buffer overflow.
384 *
385 * Will read any available input looking for a '\n' to reset the buffer on.
386 *
387 * @returns VBox status.
388 * @param pDbgc Debugger console instance data.
389 */
390static int dbgcInputOverflow(PDBGC pDbgc)
391{
392 /*
393 * Assert overflow status and reset the input buffer.
394 */
395 if (!pDbgc->fInputOverflow)
396 {
397 pDbgc->fInputOverflow = true;
398 pDbgc->iRead = pDbgc->iWrite = 0;
399 pDbgc->cInputLines = 0;
400 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "Input overflow!!\n");
401 }
402
403 /*
404 * Eat input till no more or there is a '\n'.
405 * When finding a '\n' we'll continue normal processing.
406 */
407 while (pDbgc->pBack->pfnInput(pDbgc->pBack, 0))
408 {
409 size_t cbRead;
410 int rc = pDbgc->pBack->pfnRead(pDbgc->pBack, &pDbgc->achInput[0], sizeof(pDbgc->achInput) - 1, &cbRead);
411 if (RT_FAILURE(rc))
412 return rc;
413 char *psz = (char *)memchr(&pDbgc->achInput[0], '\n', cbRead);
414 if (psz)
415 {
416 pDbgc->fInputOverflow = false;
417 pDbgc->iRead = psz - &pDbgc->achInput[0] + 1;
418 pDbgc->iWrite = (unsigned)cbRead;
419 pDbgc->cInputLines = 0;
420 break;
421 }
422 }
423
424 return 0;
425}
426
427
428/**
429 * Read input and do some preprocessing.
430 *
431 * @returns VBox status.
432 * In addition to the iWrite and achInput, cInputLines is maintained.
433 * In case of an input overflow the fInputOverflow flag will be set.
434 * @param pDbgc Debugger console instance data.
435 */
436static int dbgcInputRead(PDBGC pDbgc)
437{
438 /*
439 * We have ready input.
440 * Read it till we don't have any or we have a full input buffer.
441 */
442 int rc = 0;
443 do
444 {
445 /*
446 * More available buffer space?
447 */
448 size_t cbLeft;
449 if (pDbgc->iWrite > pDbgc->iRead)
450 cbLeft = sizeof(pDbgc->achInput) - pDbgc->iWrite - (pDbgc->iRead == 0);
451 else
452 cbLeft = pDbgc->iRead - pDbgc->iWrite - 1;
453 if (!cbLeft)
454 {
455 /* overflow? */
456 if (!pDbgc->cInputLines)
457 rc = dbgcInputOverflow(pDbgc);
458 break;
459 }
460
461 /*
462 * Read one char and interpret it.
463 */
464 char achRead[128];
465 size_t cbRead;
466 rc = pDbgc->pBack->pfnRead(pDbgc->pBack, &achRead[0], RT_MIN(cbLeft, sizeof(achRead)), &cbRead);
467 if (RT_FAILURE(rc))
468 return rc;
469 char *psz = &achRead[0];
470 while (cbRead-- > 0)
471 {
472 char ch = *psz++;
473 switch (ch)
474 {
475 /*
476 * Ignore.
477 */
478 case '\0':
479 case '\r':
480 case '\a':
481 break;
482
483 /*
484 * Backspace.
485 */
486 case '\b':
487 Log2(("DBGC: backspace\n"));
488 if (pDbgc->iRead != pDbgc->iWrite)
489 {
490 unsigned iWriteUndo = pDbgc->iWrite;
491 if (pDbgc->iWrite)
492 pDbgc->iWrite--;
493 else
494 pDbgc->iWrite = sizeof(pDbgc->achInput) - 1;
495
496 if (pDbgc->achInput[pDbgc->iWrite] == '\n')
497 pDbgc->iWrite = iWriteUndo;
498 }
499 break;
500
501 /*
502 * Add char to buffer.
503 */
504 case '\t':
505 case '\n':
506 case ';':
507 switch (ch)
508 {
509 case '\t': ch = ' '; break;
510 case '\n': pDbgc->cInputLines++; break;
511 }
512 default:
513 Log2(("DBGC: ch=%02x\n", (unsigned char)ch));
514 pDbgc->achInput[pDbgc->iWrite] = ch;
515 if (++pDbgc->iWrite >= sizeof(pDbgc->achInput))
516 pDbgc->iWrite = 0;
517 break;
518 }
519 }
520
521 /* Terminate it to make it easier to read in the debugger. */
522 pDbgc->achInput[pDbgc->iWrite] = '\0';
523 } while (pDbgc->pBack->pfnInput(pDbgc->pBack, 0));
524
525 return rc;
526}
527
528
529/**
530 * Reads input, parses it and executes commands on '\n'.
531 *
532 * @returns VBox status.
533 * @param pDbgc Debugger console instance data.
534 * @param fNoExecute Indicates that no commands should actually be executed.
535 */
536int dbgcProcessInput(PDBGC pDbgc, bool fNoExecute)
537{
538 /*
539 * We know there's input ready, so let's read it first.
540 */
541 int rc = dbgcInputRead(pDbgc);
542 if (RT_FAILURE(rc))
543 return rc;
544
545 /*
546 * Now execute any ready commands.
547 */
548 if (pDbgc->cInputLines)
549 {
550 pDbgc->pBack->pfnSetReady(pDbgc->pBack, false);
551 pDbgc->fReady = false;
552 rc = dbgcProcessCommands(pDbgc, fNoExecute);
553 if (RT_SUCCESS(rc) && rc != VWRN_DBGC_CMD_PENDING)
554 pDbgc->fReady = true;
555
556 if ( RT_SUCCESS(rc)
557 && pDbgc->iRead == pDbgc->iWrite
558 && pDbgc->fReady)
559 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "VBoxDbg> ");
560
561 if ( RT_SUCCESS(rc)
562 && pDbgc->fReady)
563 pDbgc->pBack->pfnSetReady(pDbgc->pBack, true);
564 }
565 else
566 /* Received nonsense; just skip it. */
567 pDbgc->iRead = pDbgc->iWrite;
568
569 return rc;
570}
571
572
573/**
574 * Gets the event context identifier string.
575 * @returns Read only string.
576 * @param enmCtx The context.
577 */
578static const char *dbgcGetEventCtx(DBGFEVENTCTX enmCtx)
579{
580 switch (enmCtx)
581 {
582 case DBGFEVENTCTX_RAW: return "raw";
583 case DBGFEVENTCTX_REM: return "rem";
584 case DBGFEVENTCTX_HM: return "hwaccl";
585 case DBGFEVENTCTX_HYPER: return "hyper";
586 case DBGFEVENTCTX_OTHER: return "other";
587
588 case DBGFEVENTCTX_INVALID: return "!Invalid Event Ctx!";
589 default:
590 AssertMsgFailed(("enmCtx=%d\n", enmCtx));
591 return "!Unknown Event Ctx!";
592 }
593}
594
595
596/**
597 * Processes debugger events.
598 *
599 * @returns VBox status.
600 * @param pDbgc DBGC Instance data.
601 * @param pEvent Pointer to event data.
602 */
603static int dbgcProcessEvent(PDBGC pDbgc, PCDBGFEVENT pEvent)
604{
605 /*
606 * Flush log first.
607 */
608 if (pDbgc->fLog)
609 {
610 int rc = dbgcProcessLog(pDbgc);
611 if (RT_FAILURE(rc))
612 return rc;
613 }
614
615 /*
616 * Process the event.
617 */
618 pDbgc->pszScratch = &pDbgc->achInput[0];
619 pDbgc->iArg = 0;
620 bool fPrintPrompt = true;
621 int rc = VINF_SUCCESS;
622 switch (pEvent->enmType)
623 {
624 /*
625 * The first part is events we have initiated with commands.
626 */
627 case DBGFEVENT_HALT_DONE:
628 {
629 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: VM %p is halted! (%s)\n",
630 pDbgc->pVM, dbgcGetEventCtx(pEvent->enmCtx));
631 pDbgc->fRegCtxGuest = true; /* we're always in guest context when halted. */
632 if (RT_SUCCESS(rc))
633 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
634 break;
635 }
636
637
638 /*
639 * The second part is events which can occur at any time.
640 */
641 case DBGFEVENT_FATAL_ERROR:
642 {
643 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbf event: Fatal error! (%s)\n",
644 dbgcGetEventCtx(pEvent->enmCtx));
645 pDbgc->fRegCtxGuest = false; /* fatal errors are always in hypervisor. */
646 if (RT_SUCCESS(rc))
647 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
648 break;
649 }
650
651 case DBGFEVENT_BREAKPOINT:
652 case DBGFEVENT_BREAKPOINT_HYPER:
653 {
654 bool fRegCtxGuest = pDbgc->fRegCtxGuest;
655 pDbgc->fRegCtxGuest = pEvent->enmType == DBGFEVENT_BREAKPOINT;
656
657 rc = dbgcBpExec(pDbgc, pEvent->u.Bp.iBp);
658 switch (rc)
659 {
660 case VERR_DBGC_BP_NOT_FOUND:
661 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Unknown breakpoint %u! (%s)\n",
662 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
663 break;
664
665 case VINF_DBGC_BP_NO_COMMAND:
666 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Breakpoint %u! (%s)\n",
667 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
668 break;
669
670 case VINF_BUFFER_OVERFLOW:
671 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Breakpoint %u! Command too long to execute! (%s)\n",
672 pEvent->u.Bp.iBp, dbgcGetEventCtx(pEvent->enmCtx));
673 break;
674
675 default:
676 break;
677 }
678 if (RT_SUCCESS(rc) && DBGFR3IsHalted(pDbgc->pUVM))
679 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
680 else
681 pDbgc->fRegCtxGuest = fRegCtxGuest;
682 break;
683 }
684
685 case DBGFEVENT_STEPPED:
686 case DBGFEVENT_STEPPED_HYPER:
687 {
688 pDbgc->fRegCtxGuest = pEvent->enmType == DBGFEVENT_STEPPED;
689
690 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf event: Single step! (%s)\n", dbgcGetEventCtx(pEvent->enmCtx));
691 if (RT_SUCCESS(rc))
692 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
693 break;
694 }
695
696 case DBGFEVENT_ASSERTION_HYPER:
697 {
698 pDbgc->fRegCtxGuest = false;
699
700 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
701 "\ndbgf event: Hypervisor Assertion! (%s)\n"
702 "%s"
703 "%s"
704 "\n",
705 dbgcGetEventCtx(pEvent->enmCtx),
706 pEvent->u.Assert.pszMsg1,
707 pEvent->u.Assert.pszMsg2);
708 if (RT_SUCCESS(rc))
709 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
710 break;
711 }
712
713 case DBGFEVENT_DEV_STOP:
714 {
715 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
716 "\n"
717 "dbgf event: DBGFSTOP (%s)\n"
718 "File: %s\n"
719 "Line: %d\n"
720 "Function: %s\n",
721 dbgcGetEventCtx(pEvent->enmCtx),
722 pEvent->u.Src.pszFile,
723 pEvent->u.Src.uLine,
724 pEvent->u.Src.pszFunction);
725 if (RT_SUCCESS(rc) && pEvent->u.Src.pszMessage && *pEvent->u.Src.pszMessage)
726 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
727 "Message: %s\n",
728 pEvent->u.Src.pszMessage);
729 if (RT_SUCCESS(rc))
730 rc = pDbgc->CmdHlp.pfnExec(&pDbgc->CmdHlp, "r");
731 break;
732 }
733
734
735 case DBGFEVENT_INVALID_COMMAND:
736 {
737 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf/dbgc error: Invalid command event!\n");
738 break;
739 }
740
741 case DBGFEVENT_POWERING_OFF:
742 {
743 pDbgc->fReady = false;
744 pDbgc->pBack->pfnSetReady(pDbgc->pBack, false);
745 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\nVM is powering off!\n");
746 fPrintPrompt = false;
747 rc = VERR_GENERAL_FAILURE;
748 break;
749 }
750
751
752 default:
753 {
754 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\ndbgf/dbgc error: Unknown event %d!\n", pEvent->enmType);
755 break;
756 }
757 }
758
759 /*
760 * Prompt, anyone?
761 */
762 if (fPrintPrompt && RT_SUCCESS(rc))
763 {
764 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "VBoxDbg> ");
765 pDbgc->fReady = true;
766 if (RT_SUCCESS(rc))
767 pDbgc->pBack->pfnSetReady(pDbgc->pBack, true);
768 }
769
770 return rc;
771}
772
773
774/**
775 * Prints any log lines from the log buffer.
776 *
777 * The caller must not call function this unless pDbgc->fLog is set.
778 *
779 * @returns VBox status. (output related)
780 * @param pDbgc Debugger console instance data.
781 */
782static int dbgcProcessLog(PDBGC pDbgc)
783{
784 /** @todo */
785 NOREF(pDbgc);
786 return 0;
787}
788
789/** @callback_method_impl{FNRTDBGCFGLOG} */
790static DECLCALLBACK(void) dbgcDbgCfgLogCallback(RTDBGCFG hDbgCfg, uint32_t iLevel, const char *pszMsg, void *pvUser)
791{
792 /** @todo Add symbol noise setting. */
793 NOREF(hDbgCfg); NOREF(iLevel);
794 PDBGC pDbgc = (PDBGC)pvUser;
795 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "%s", pszMsg);
796}
797
798
799/**
800 * Run the debugger console.
801 *
802 * @returns VBox status.
803 * @param pDbgc Pointer to the debugger console instance data.
804 */
805int dbgcRun(PDBGC pDbgc)
806{
807 /*
808 * We're ready for commands now.
809 */
810 pDbgc->fReady = true;
811 pDbgc->pBack->pfnSetReady(pDbgc->pBack, true);
812
813 /*
814 * Main Debugger Loop.
815 *
816 * This loop will either block on waiting for input or on waiting on
817 * debug events. If we're forwarding the log we cannot wait for long
818 * before we must flush the log.
819 */
820 int rc;
821 for (;;)
822 {
823 rc = VERR_SEM_OUT_OF_TURN;
824 if (pDbgc->pUVM)
825 rc = DBGFR3QueryWaitable(pDbgc->pUVM);
826
827 if (RT_SUCCESS(rc))
828 {
829 /*
830 * Wait for a debug event.
831 */
832 PCDBGFEVENT pEvent;
833 rc = DBGFR3EventWait(pDbgc->pUVM, pDbgc->fLog ? 1 : 32, &pEvent);
834 if (RT_SUCCESS(rc))
835 {
836 rc = dbgcProcessEvent(pDbgc, pEvent);
837 if (RT_FAILURE(rc))
838 break;
839 }
840 else if (rc != VERR_TIMEOUT)
841 break;
842
843 /*
844 * Check for input.
845 */
846 if (pDbgc->pBack->pfnInput(pDbgc->pBack, 0))
847 {
848 rc = dbgcProcessInput(pDbgc, false /* fNoExecute */);
849 if (RT_FAILURE(rc))
850 break;
851 }
852 }
853 else if (rc == VERR_SEM_OUT_OF_TURN)
854 {
855 /*
856 * Wait for input. If Logging is enabled we'll only wait very briefly.
857 */
858 if (pDbgc->pBack->pfnInput(pDbgc->pBack, pDbgc->fLog ? 1 : 1000))
859 {
860 rc = dbgcProcessInput(pDbgc, false /* fNoExecute */);
861 if (RT_FAILURE(rc))
862 break;
863 }
864 }
865 else
866 break;
867
868 /*
869 * Forward log output.
870 */
871 if (pDbgc->fLog)
872 {
873 rc = dbgcProcessLog(pDbgc);
874 if (RT_FAILURE(rc))
875 break;
876 }
877 }
878
879 return rc;
880}
881
882
883/**
884 * Creates a a new instance.
885 *
886 * @returns VBox status code.
887 * @param ppDbgc Where to store the pointer to the instance data.
888 * @param pBack Pointer to the backend.
889 * @param fFlags The flags.
890 */
891int dbgcCreate(PDBGC *ppDbgc, PDBGCBACK pBack, unsigned fFlags)
892{
893 /*
894 * Validate input.
895 */
896 AssertPtrReturn(pBack, VERR_INVALID_POINTER);
897 AssertMsgReturn(!fFlags, ("%#x", fFlags), VERR_INVALID_PARAMETER);
898
899 /*
900 * Allocate and initialize.
901 */
902 PDBGC pDbgc = (PDBGC)RTMemAllocZ(sizeof(*pDbgc));
903 if (!pDbgc)
904 return VERR_NO_MEMORY;
905
906 dbgcInitCmdHlp(pDbgc);
907 pDbgc->pBack = pBack;
908 pDbgc->pVM = NULL;
909 pDbgc->pUVM = NULL;
910 pDbgc->idCpu = 0;
911 pDbgc->hDbgAs = DBGF_AS_GLOBAL;
912 pDbgc->pszEmulation = "CodeView/WinDbg";
913 pDbgc->paEmulationCmds = &g_aCmdsCodeView[0];
914 pDbgc->cEmulationCmds = g_cCmdsCodeView;
915 pDbgc->paEmulationFuncs = &g_aFuncsCodeView[0];
916 pDbgc->cEmulationFuncs = g_cFuncsCodeView;
917 //pDbgc->fLog = false;
918 pDbgc->fRegCtxGuest = true;
919 pDbgc->fRegTerse = true;
920 //pDbgc->cPagingHierarchyDumps = 0;
921 //pDbgc->DisasmPos = {0};
922 //pDbgc->SourcePos = {0};
923 //pDbgc->DumpPos = {0};
924 pDbgc->pLastPos = &pDbgc->DisasmPos;
925 //pDbgc->cbDumpElement = 0;
926 //pDbgc->cVars = 0;
927 //pDbgc->paVars = NULL;
928 //pDbgc->pPlugInHead = NULL;
929 //pDbgc->pFirstBp = NULL;
930 //pDbgc->abSearch = {0};
931 //pDbgc->cbSearch = 0;
932 pDbgc->cbSearchUnit = 1;
933 pDbgc->cMaxSearchHits = 1;
934 //pDbgc->SearchAddr = {0};
935 //pDbgc->cbSearchRange = 0;
936
937 //pDbgc->uInputZero = 0;
938 //pDbgc->iRead = 0;
939 //pDbgc->iWrite = 0;
940 //pDbgc->cInputLines = 0;
941 //pDbgc->fInputOverflow = false;
942 pDbgc->fReady = true;
943 pDbgc->pszScratch = &pDbgc->achScratch[0];
944 //pDbgc->iArg = 0;
945 //pDbgc->rcOutput = 0;
946 //pDbgc->rcCmd = 0;
947
948 dbgcEvalInit();
949
950 *ppDbgc = pDbgc;
951 return VINF_SUCCESS;
952}
953
954/**
955 * Destroys a DBGC instance created by dbgcCreate.
956 *
957 * @param pDbgc Pointer to the debugger console instance data.
958 */
959void dbgcDestroy(PDBGC pDbgc)
960{
961 AssertPtr(pDbgc);
962
963 /* Disable log hook. */
964 if (pDbgc->fLog)
965 {
966
967 }
968
969 /* Unload all plug-ins. */
970 dbgcPlugInUnloadAll(pDbgc);
971
972 /* Detach from the VM. */
973 if (pDbgc->pUVM)
974 DBGFR3Detach(pDbgc->pUVM);
975
976 /* finally, free the instance memory. */
977 RTMemFree(pDbgc);
978}
979
980
981/**
982 * Make a console instance.
983 *
984 * This will not return until either an 'exit' command is issued or a error code
985 * indicating connection loss is encountered.
986 *
987 * @returns VINF_SUCCESS if console termination caused by the 'exit' command.
988 * @returns The VBox status code causing the console termination.
989 *
990 * @param pUVM The user mode VM handle.
991 * @param pBack Pointer to the backend structure. This must contain
992 * a full set of function pointers to service the console.
993 * @param fFlags Reserved, must be zero.
994 * @remark A forced termination of the console is easiest done by forcing the
995 * callbacks to return fatal failures.
996 */
997DBGDECL(int) DBGCCreate(PUVM pUVM, PDBGCBACK pBack, unsigned fFlags)
998{
999 /*
1000 * Validate input.
1001 */
1002 AssertPtrNullReturn(pUVM, VERR_INVALID_VM_HANDLE);
1003 PVM pVM = NULL;
1004 if (pUVM)
1005 {
1006 pVM = VMR3GetVM(pUVM);
1007 AssertPtrReturn(pVM, VERR_INVALID_VM_HANDLE);
1008 }
1009
1010 /*
1011 * Allocate and initialize instance data
1012 */
1013 PDBGC pDbgc;
1014 int rc = dbgcCreate(&pDbgc, pBack, fFlags);
1015 if (RT_FAILURE(rc))
1016 return rc;
1017 if (!HMR3IsEnabled(pUVM))
1018 pDbgc->hDbgAs = DBGF_AS_RC_AND_GC_GLOBAL;
1019
1020 /*
1021 * Print welcome message.
1022 */
1023 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
1024 "Welcome to the VirtualBox Debugger!\n");
1025
1026 /*
1027 * Attach to the specified VM.
1028 */
1029 if (RT_SUCCESS(rc) && pUVM)
1030 {
1031 rc = DBGFR3Attach(pUVM);
1032 if (RT_SUCCESS(rc))
1033 {
1034 pDbgc->pVM = pVM;
1035 pDbgc->pUVM = pUVM;
1036 pDbgc->idCpu = 0;
1037 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL,
1038 "Current VM is %08x, CPU #%u\n" /** @todo get and print the VM name! */
1039 , pDbgc->pVM, pDbgc->idCpu);
1040 }
1041 else
1042 rc = pDbgc->CmdHlp.pfnVBoxError(&pDbgc->CmdHlp, rc, "When trying to attach to VM %p\n", pDbgc->pVM);
1043 }
1044
1045 /*
1046 * Load plugins.
1047 */
1048 if (RT_SUCCESS(rc))
1049 {
1050 if (pVM)
1051 dbgcPlugInAutoLoad(pDbgc);
1052 rc = pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "VBoxDbg> ");
1053 if (RT_SUCCESS(rc))
1054 {
1055 /*
1056 * Set debug config log callback.
1057 */
1058 RTDBGCFG hDbgCfg = DBGFR3AsGetConfig(pUVM);
1059 if ( hDbgCfg != NIL_RTDBGCFG
1060 && RTDbgCfgRetain(hDbgCfg) != UINT32_MAX)
1061 {
1062 int rc2 = RTDbgCfgSetLogCallback(hDbgCfg, dbgcDbgCfgLogCallback, pDbgc);
1063 if (RT_FAILURE(rc2))
1064 {
1065 hDbgCfg = NIL_RTDBGCFG;
1066 RTDbgCfgRelease(hDbgCfg);
1067 }
1068 }
1069 else
1070 hDbgCfg = NIL_RTDBGCFG;
1071
1072
1073 /*
1074 * Run the debugger main loop.
1075 */
1076 rc = dbgcRun(pDbgc);
1077
1078
1079 /*
1080 * Remove debug config log callback.
1081 */
1082 if (hDbgCfg != NIL_RTDBGCFG)
1083 {
1084 RTDbgCfgSetLogCallback(hDbgCfg, NULL, NULL);
1085 RTDbgCfgRelease(hDbgCfg);
1086 }
1087 }
1088 }
1089 else
1090 pDbgc->CmdHlp.pfnPrintf(&pDbgc->CmdHlp, NULL, "\nDBGCCreate error: %Rrc\n", rc);
1091
1092
1093 /*
1094 * Cleanup console debugger session.
1095 */
1096 dbgcDestroy(pDbgc);
1097 return rc == VERR_DBGC_QUIT ? VINF_SUCCESS : rc;
1098}
1099
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