VirtualBox

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

Last change on this file since 35752 was 35632, checked in by vboxsync, 14 years ago

DBGC: Simplified the status code handling around dbgcEvalCommand.

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