VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/DBGFBp.cpp@ 42418

Last change on this file since 42418 was 41965, checked in by vboxsync, 13 years ago

VMM: ran scm. Mostly svn:keywords changes (adding Revision).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 26.8 KB
Line 
1/* $Id: DBGFBp.cpp 41965 2012-06-29 02:52:49Z vboxsync $ */
2/** @file
3 * DBGF - Debugger Facility, Breakpoint Management.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DBGF
23#include <VBox/vmm/dbgf.h>
24#include <VBox/vmm/selm.h>
25#ifdef VBOX_WITH_REM
26# include <VBox/vmm/rem.h>
27#else
28# include <VBox/vmm/iem.h>
29#endif
30#include "DBGFInternal.h"
31#include <VBox/vmm/vm.h>
32#include <VBox/vmm/mm.h>
33#include <VBox/err.h>
34#include <VBox/log.h>
35#include <iprt/assert.h>
36#include <iprt/string.h>
37
38
39/*******************************************************************************
40* Internal Functions *
41*******************************************************************************/
42RT_C_DECLS_BEGIN
43static DECLCALLBACK(int) dbgfR3BpSetReg(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable,
44 uint8_t u8Type, uint8_t cb, uint32_t *piBp);
45static DECLCALLBACK(int) dbgfR3BpSetInt3(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable, uint32_t *piBp);
46static DECLCALLBACK(int) dbgfR3BpSetREM(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable, uint32_t *piBp);
47static DECLCALLBACK(int) dbgfR3BpClear(PVM pVM, uint32_t iBp);
48static DECLCALLBACK(int) dbgfR3BpEnable(PVM pVM, uint32_t iBp);
49static DECLCALLBACK(int) dbgfR3BpDisable(PVM pVM, uint32_t iBp);
50static DECLCALLBACK(int) dbgfR3BpEnum(PVM pVM, PFNDBGFBPENUM pfnCallback, void *pvUser);
51static int dbgfR3BpRegArm(PVM pVM, PDBGFBP pBp);
52static int dbgfR3BpRegDisarm(PVM pVM, PDBGFBP pBp);
53static int dbgfR3BpInt3Arm(PVM pVM, PDBGFBP pBp);
54static int dbgfR3BpInt3Disarm(PVM pVM, PDBGFBP pBp);
55RT_C_DECLS_END
56
57
58
59/**
60 * Initialize the breakpoint stuff.
61 *
62 * @returns VINF_SUCCESS
63 * @param pVM Pointer to the VM.
64 */
65int dbgfR3BpInit(PVM pVM)
66{
67 /*
68 * Init structures.
69 */
70 unsigned i;
71 for (i = 0; i < RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints); i++)
72 {
73 pVM->dbgf.s.aHwBreakpoints[i].iBp = i;
74 pVM->dbgf.s.aHwBreakpoints[i].enmType = DBGFBPTYPE_FREE;
75 pVM->dbgf.s.aHwBreakpoints[i].u.Reg.iReg = i;
76 }
77
78 for (i = 0; i < RT_ELEMENTS(pVM->dbgf.s.aBreakpoints); i++)
79 {
80 pVM->dbgf.s.aBreakpoints[i].iBp = i + RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints);
81 pVM->dbgf.s.aBreakpoints[i].enmType = DBGFBPTYPE_FREE;
82 }
83
84 /*
85 * Register saved state.
86 */
87 /** @todo */
88
89 return VINF_SUCCESS;
90}
91
92
93
94/**
95 * Allocate a breakpoint.
96 *
97 * @returns Pointer to the allocated breakpoint.
98 * @returns NULL if we're out of breakpoints.
99 * @param pVM Pointer to the VM.
100 * @param enmType The type to allocate.
101 */
102static PDBGFBP dbgfR3BpAlloc(PVM pVM, DBGFBPTYPE enmType)
103{
104 /*
105 * Determine which array to search.
106 */
107 unsigned cBps;
108 uint32_t *pcBpsCur;
109 PDBGFBP paBps;
110 switch (enmType)
111 {
112 case DBGFBPTYPE_REG:
113 cBps = RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints);
114 paBps = &pVM->dbgf.s.aHwBreakpoints[0];
115 pcBpsCur = &pVM->dbgf.s.cHwBreakpoints;
116 break;
117
118 case DBGFBPTYPE_INT3:
119 case DBGFBPTYPE_REM:
120 cBps = RT_ELEMENTS(pVM->dbgf.s.aBreakpoints);
121 paBps = &pVM->dbgf.s.aBreakpoints[0];
122 pcBpsCur = &pVM->dbgf.s.cBreakpoints;
123 break;
124
125 default:
126 AssertMsgFailed(("enmType=%d\n", enmType));
127 return NULL;
128 }
129
130 /*
131 * Search.
132 */
133 for (unsigned iBp = 0; iBp < cBps; iBp++)
134 if (paBps[iBp].enmType == DBGFBPTYPE_FREE)
135 {
136 ++*pcBpsCur;
137 paBps[iBp].cHits = 0;
138 paBps[iBp].enmType = enmType;
139 return &paBps[iBp];
140 }
141
142 LogFlow(("dbgfR3BpAlloc: returns NULL - we're out of breakpoint slots! %u/%u\n", *pcBpsCur, cBps));
143 return NULL;
144}
145
146
147/**
148 * Get a breakpoint give by breakpoint id.
149 *
150 * @returns Pointer to the allocated breakpoint.
151 * @returns NULL if the breakpoint is invalid.
152 * @param pVM Pointer to the VM.
153 * @param iBp The breakpoint id.
154 */
155static PDBGFBP dbgfR3BpGet(PVM pVM, uint32_t iBp)
156{
157 /* Find it. */
158 PDBGFBP pBp;
159 if (iBp < RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints))
160 pBp = &pVM->dbgf.s.aHwBreakpoints[iBp];
161 else
162 {
163 iBp -= RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints);
164 if (iBp >= RT_ELEMENTS(pVM->dbgf.s.aBreakpoints))
165 return NULL;
166 pBp = &pVM->dbgf.s.aBreakpoints[iBp];
167 }
168
169 /* check if it's valid. */
170 switch (pBp->enmType)
171 {
172 case DBGFBPTYPE_FREE:
173 return NULL;
174
175 case DBGFBPTYPE_REG:
176 case DBGFBPTYPE_INT3:
177 case DBGFBPTYPE_REM:
178 break;
179
180 default:
181 AssertMsgFailed(("Invalid enmType=%d!\n", pBp->enmType));
182 return NULL;
183 }
184
185 return pBp;
186}
187
188
189/**
190 * Get a breakpoint give by address.
191 *
192 * @returns Pointer to the allocated breakpoint.
193 * @returns NULL if the breakpoint is invalid.
194 * @param pVM Pointer to the VM.
195 * @param enmType The breakpoint type.
196 * @param GCPtr The breakpoint address.
197 */
198static PDBGFBP dbgfR3BpGetByAddr(PVM pVM, DBGFBPTYPE enmType, RTGCUINTPTR GCPtr)
199{
200 /*
201 * Determine which array to search.
202 */
203 unsigned cBps;
204 PDBGFBP paBps;
205 switch (enmType)
206 {
207 case DBGFBPTYPE_REG:
208 cBps = RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints);
209 paBps = &pVM->dbgf.s.aHwBreakpoints[0];
210 break;
211
212 case DBGFBPTYPE_INT3:
213 case DBGFBPTYPE_REM:
214 cBps = RT_ELEMENTS(pVM->dbgf.s.aBreakpoints);
215 paBps = &pVM->dbgf.s.aBreakpoints[0];
216 break;
217
218 default:
219 AssertMsgFailed(("enmType=%d\n", enmType));
220 return NULL;
221 }
222
223 /*
224 * Search.
225 */
226 for (unsigned iBp = 0; iBp < cBps; iBp++)
227 {
228 if ( paBps[iBp].enmType == enmType
229 && paBps[iBp].GCPtr == GCPtr)
230 return &paBps[iBp];
231 }
232
233 return NULL;
234}
235
236
237/**
238 * Frees a breakpoint.
239 *
240 * @param pVM Pointer to the VM.
241 * @param pBp The breakpoint to free.
242 */
243static void dbgfR3BpFree(PVM pVM, PDBGFBP pBp)
244{
245 switch (pBp->enmType)
246 {
247 case DBGFBPTYPE_FREE:
248 AssertMsgFailed(("Already freed!\n"));
249 return;
250
251 case DBGFBPTYPE_REG:
252 Assert(pVM->dbgf.s.cHwBreakpoints > 0);
253 pVM->dbgf.s.cHwBreakpoints--;
254 break;
255
256 case DBGFBPTYPE_INT3:
257 case DBGFBPTYPE_REM:
258 Assert(pVM->dbgf.s.cBreakpoints > 0);
259 pVM->dbgf.s.cBreakpoints--;
260 break;
261
262 default:
263 AssertMsgFailed(("Invalid enmType=%d!\n", pBp->enmType));
264 return;
265
266 }
267 pBp->enmType = DBGFBPTYPE_FREE;
268}
269
270
271/**
272 * Sets a breakpoint (int 3 based).
273 *
274 * @returns VBox status code.
275 * @param pVM Pointer to the VM.
276 * @param pAddress The address of the breakpoint.
277 * @param iHitTrigger The hit count at which the breakpoint start triggering.
278 * Use 0 (or 1) if it's gonna trigger at once.
279 * @param iHitDisable The hit count which disables the breakpoint.
280 * Use ~(uint64_t) if it's never gonna be disabled.
281 * @param piBp Where to store the breakpoint id. (optional)
282 * @thread Any thread.
283 */
284VMMR3DECL(int) DBGFR3BpSet(PVM pVM, PCDBGFADDRESS pAddress, uint64_t iHitTrigger, uint64_t iHitDisable, uint32_t *piBp)
285{
286 /*
287 * This must be done on EMT.
288 */
289 /** @todo SMP? */
290 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpSetInt3, 5, pVM, pAddress, &iHitTrigger, &iHitDisable, piBp);
291 LogFlow(("DBGFR3BpSet: returns %Rrc\n", rc));
292 return rc;
293}
294
295
296/**
297 * Sets a breakpoint (int 3 based).
298 *
299 * @returns VBox status code.
300 * @param pVM Pointer to the VM.
301 * @param pAddress The address of the breakpoint.
302 * @param piHitTrigger The hit count at which the breakpoint start triggering.
303 * Use 0 (or 1) if it's gonna trigger at once.
304 * @param piHitDisable The hit count which disables the breakpoint.
305 * Use ~(uint64_t) if it's never gonna be disabled.
306 * @param piBp Where to store the breakpoint id. (optional)
307 * @thread Any thread.
308 */
309static DECLCALLBACK(int) dbgfR3BpSetInt3(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable, uint32_t *piBp)
310{
311 /*
312 * Validate input.
313 */
314 if (!DBGFR3AddrIsValid(pVM, pAddress))
315 return VERR_INVALID_PARAMETER;
316 if (*piHitTrigger > *piHitDisable)
317 return VERR_INVALID_PARAMETER;
318 AssertMsgReturn(!piBp || VALID_PTR(piBp), ("piBp=%p\n", piBp), VERR_INVALID_POINTER);
319 if (piBp)
320 *piBp = ~0;
321
322 /*
323 * Check if the breakpoint already exists.
324 */
325 PDBGFBP pBp = dbgfR3BpGetByAddr(pVM, DBGFBPTYPE_INT3, pAddress->FlatPtr);
326 if (pBp)
327 {
328 int rc = VINF_SUCCESS;
329 if (!pBp->fEnabled)
330 rc = dbgfR3BpInt3Arm(pVM, pBp);
331 if (RT_SUCCESS(rc))
332 {
333 rc = VINF_DBGF_BP_ALREADY_EXIST;
334 if (piBp)
335 *piBp = pBp->iBp;
336 }
337 return rc;
338 }
339
340 /*
341 * Allocate and initialize the bp.
342 */
343 pBp = dbgfR3BpAlloc(pVM, DBGFBPTYPE_INT3);
344 if (!pBp)
345 return VERR_DBGF_NO_MORE_BP_SLOTS;
346 pBp->GCPtr = pAddress->FlatPtr;
347 pBp->iHitTrigger = *piHitTrigger;
348 pBp->iHitDisable = *piHitDisable;
349 pBp->fEnabled = true;
350
351 /*
352 * Now ask REM to set the breakpoint.
353 */
354 int rc = dbgfR3BpInt3Arm(pVM, pBp);
355 if (RT_SUCCESS(rc))
356 {
357 if (piBp)
358 *piBp = pBp->iBp;
359 }
360 else
361 dbgfR3BpFree(pVM, pBp);
362
363 return rc;
364}
365
366
367/**
368 * Arms an int 3 breakpoint.
369 * This is used to implement both DBGFR3BpSetReg() and DBGFR3BpEnable().
370 *
371 * @returns VBox status code.
372 * @param pVM Pointer to the VM.
373 * @param pBp The breakpoint.
374 */
375static int dbgfR3BpInt3Arm(PVM pVM, PDBGFBP pBp)
376{
377 /** @todo should actually use physical address here! */
378
379 /* @todo SMP support! */
380 VMCPUID idCpu = 0;
381
382 /*
383 * Save current byte and write int3 instruction.
384 */
385 DBGFADDRESS Addr;
386 DBGFR3AddrFromFlat(pVM, &Addr, pBp->GCPtr);
387 int rc = DBGFR3MemRead(pVM, idCpu, &Addr, &pBp->u.Int3.bOrg, 1);
388 if (RT_SUCCESS(rc))
389 {
390 static const uint8_t s_bInt3 = 0xcc;
391 rc = DBGFR3MemWrite(pVM, idCpu, &Addr, &s_bInt3, 1);
392 }
393 return rc;
394}
395
396
397/**
398 * Disarms an int 3 breakpoint.
399 * This is used to implement both DBGFR3BpClear() and DBGFR3BpDisable().
400 *
401 * @returns VBox status code.
402 * @param pVM Pointer to the VM.
403 * @param pBp The breakpoint.
404 */
405static int dbgfR3BpInt3Disarm(PVM pVM, PDBGFBP pBp)
406{
407 /* @todo SMP support! */
408 VMCPUID idCpu = 0;
409
410 /*
411 * Check that the current byte is the int3 instruction, and restore the original one.
412 * We currently ignore invalid bytes.
413 */
414 DBGFADDRESS Addr;
415 DBGFR3AddrFromFlat(pVM, &Addr, pBp->GCPtr);
416 uint8_t bCurrent;
417 int rc = DBGFR3MemRead(pVM, idCpu, &Addr, &bCurrent, 1);
418 if (bCurrent == 0xcc)
419 rc = DBGFR3MemWrite(pVM, idCpu, &Addr, &pBp->u.Int3.bOrg, 1);
420 return rc;
421}
422
423
424/**
425 * Sets a register breakpoint.
426 *
427 * @returns VBox status code.
428 * @param pVM Pointer to the VM.
429 * @param pAddress The address of the breakpoint.
430 * @param iHitTrigger The hit count at which the breakpoint start triggering.
431 * Use 0 (or 1) if it's gonna trigger at once.
432 * @param iHitDisable The hit count which disables the breakpoint.
433 * Use ~(uint64_t) if it's never gonna be disabled.
434 * @param fType The access type (one of the X86_DR7_RW_* defines).
435 * @param cb The access size - 1,2,4 or 8 (the latter is AMD64 long mode only.
436 * Must be 1 if fType is X86_DR7_RW_EO.
437 * @param piBp Where to store the breakpoint id. (optional)
438 * @thread Any thread.
439 */
440VMMR3DECL(int) DBGFR3BpSetReg(PVM pVM, PCDBGFADDRESS pAddress, uint64_t iHitTrigger, uint64_t iHitDisable,
441 uint8_t fType, uint8_t cb, uint32_t *piBp)
442{
443 /** @todo SMP - broadcast, VT-x/AMD-V. */
444 /*
445 * This must be done on EMT.
446 */
447 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpSetReg, 7, pVM, pAddress, &iHitTrigger, &iHitDisable, fType, cb, piBp);
448 LogFlow(("DBGFR3BpSetReg: returns %Rrc\n", rc));
449 return rc;
450
451}
452
453
454/**
455 * Sets a register breakpoint.
456 *
457 * @returns VBox status code.
458 * @param pVM Pointer to the VM.
459 * @param pAddress The address of the breakpoint.
460 * @param piHitTrigger The hit count at which the breakpoint start triggering.
461 * Use 0 (or 1) if it's gonna trigger at once.
462 * @param piHitDisable The hit count which disables the breakpoint.
463 * Use ~(uint64_t) if it's never gonna be disabled.
464 * @param fType The access type (one of the X86_DR7_RW_* defines).
465 * @param cb The access size - 1,2,4 or 8 (the latter is AMD64 long mode only.
466 * Must be 1 if fType is X86_DR7_RW_EO.
467 * @param piBp Where to store the breakpoint id. (optional)
468 * @thread EMT
469 * @internal
470 */
471static DECLCALLBACK(int) dbgfR3BpSetReg(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable,
472 uint8_t fType, uint8_t cb, uint32_t *piBp)
473{
474 /*
475 * Validate input.
476 */
477 if (!DBGFR3AddrIsValid(pVM, pAddress))
478 return VERR_INVALID_PARAMETER;
479 if (*piHitTrigger > *piHitDisable)
480 return VERR_INVALID_PARAMETER;
481 AssertMsgReturn(!piBp || VALID_PTR(piBp), ("piBp=%p\n", piBp), VERR_INVALID_POINTER);
482 if (piBp)
483 *piBp = ~0;
484 switch (fType)
485 {
486 case X86_DR7_RW_EO:
487 if (cb == 1)
488 break;
489 AssertMsgFailed(("fType=%#x cb=%d != 1\n", fType, cb));
490 return VERR_INVALID_PARAMETER;
491 case X86_DR7_RW_IO:
492 case X86_DR7_RW_RW:
493 case X86_DR7_RW_WO:
494 break;
495 default:
496 AssertMsgFailed(("fType=%#x\n", fType));
497 return VERR_INVALID_PARAMETER;
498 }
499 switch (cb)
500 {
501 case 1:
502 case 2:
503 case 4:
504 break;
505 default:
506 AssertMsgFailed(("cb=%#x\n", cb));
507 return VERR_INVALID_PARAMETER;
508 }
509
510 /*
511 * Check if the breakpoint already exists.
512 */
513 PDBGFBP pBp = dbgfR3BpGetByAddr(pVM, DBGFBPTYPE_REG, pAddress->FlatPtr);
514 if ( pBp
515 && pBp->u.Reg.cb == cb
516 && pBp->u.Reg.fType == fType)
517 {
518 int rc = VINF_SUCCESS;
519 if (!pBp->fEnabled)
520 rc = dbgfR3BpRegArm(pVM, pBp);
521 if (RT_SUCCESS(rc))
522 {
523 rc = VINF_DBGF_BP_ALREADY_EXIST;
524 if (piBp)
525 *piBp = pBp->iBp;
526 }
527 return rc;
528 }
529
530 /*
531 * Allocate and initialize the bp.
532 */
533 pBp = dbgfR3BpAlloc(pVM, DBGFBPTYPE_REG);
534 if (!pBp)
535 return VERR_DBGF_NO_MORE_BP_SLOTS;
536 pBp->GCPtr = pAddress->FlatPtr;
537 pBp->iHitTrigger = *piHitTrigger;
538 pBp->iHitDisable = *piHitDisable;
539 pBp->fEnabled = true;
540 Assert(pBp->iBp == pBp->u.Reg.iReg);
541 pBp->u.Reg.fType = fType;
542 pBp->u.Reg.cb = cb;
543
544 /*
545 * Arm the breakpoint.
546 */
547 int rc = dbgfR3BpRegArm(pVM, pBp);
548 if (RT_SUCCESS(rc))
549 {
550 if (piBp)
551 *piBp = pBp->iBp;
552 }
553 else
554 dbgfR3BpFree(pVM, pBp);
555
556 return rc;
557}
558
559
560/**
561 * Arms a debug register breakpoint.
562 * This is used to implement both DBGFR3BpSetReg() and DBGFR3BpEnable().
563 *
564 * @returns VBox status code.
565 * @param pVM Pointer to the VM.
566 * @param pBp The breakpoint.
567 */
568static int dbgfR3BpRegArm(PVM pVM, PDBGFBP pBp)
569{
570 /* @todo SMP support! */
571 PVMCPU pVCpu = &pVM->aCpus[0];
572
573 Assert(pBp->fEnabled);
574 return CPUMRecalcHyperDRx(pVCpu);
575}
576
577
578/**
579 * Disarms a debug register breakpoint.
580 * This is used to implement both DBGFR3BpClear() and DBGFR3BpDisable().
581 *
582 * @returns VBox status code.
583 * @param pVM Pointer to the VM.
584 * @param pBp The breakpoint.
585 */
586static int dbgfR3BpRegDisarm(PVM pVM, PDBGFBP pBp)
587{
588 /** @todo SMP support! */
589 PVMCPU pVCpu = &pVM->aCpus[0];
590
591 Assert(!pBp->fEnabled);
592 return CPUMRecalcHyperDRx(pVCpu);
593}
594
595
596/**
597 * Sets a recompiler breakpoint.
598 *
599 * @returns VBox status code.
600 * @param pVM Pointer to the VM.
601 * @param pAddress The address of the breakpoint.
602 * @param iHitTrigger The hit count at which the breakpoint start triggering.
603 * Use 0 (or 1) if it's gonna trigger at once.
604 * @param iHitDisable The hit count which disables the breakpoint.
605 * Use ~(uint64_t) if it's never gonna be disabled.
606 * @param piBp Where to store the breakpoint id. (optional)
607 * @thread Any thread.
608 */
609VMMR3DECL(int) DBGFR3BpSetREM(PVM pVM, PCDBGFADDRESS pAddress, uint64_t iHitTrigger, uint64_t iHitDisable, uint32_t *piBp)
610{
611 /*
612 * This must be done on EMT.
613 */
614 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpSetREM, 5, pVM, pAddress, &iHitTrigger, &iHitDisable, piBp);
615 LogFlow(("DBGFR3BpSetREM: returns %Rrc\n", rc));
616 return rc;
617}
618
619
620/**
621 * EMT worker for DBGFR3BpSetREM().
622 *
623 * @returns VBox status code.
624 * @param pVM Pointer to the VM.
625 * @param pAddress The address of the breakpoint.
626 * @param piHitTrigger The hit count at which the breakpoint start triggering.
627 * Use 0 (or 1) if it's gonna trigger at once.
628 * @param piHitDisable The hit count which disables the breakpoint.
629 * Use ~(uint64_t) if it's never gonna be disabled.
630 * @param piBp Where to store the breakpoint id. (optional)
631 * @thread EMT
632 * @internal
633 */
634static DECLCALLBACK(int) dbgfR3BpSetREM(PVM pVM, PCDBGFADDRESS pAddress, uint64_t *piHitTrigger, uint64_t *piHitDisable, uint32_t *piBp)
635{
636 /*
637 * Validate input.
638 */
639 if (!DBGFR3AddrIsValid(pVM, pAddress))
640 return VERR_INVALID_PARAMETER;
641 if (*piHitTrigger > *piHitDisable)
642 return VERR_INVALID_PARAMETER;
643 AssertMsgReturn(!piBp || VALID_PTR(piBp), ("piBp=%p\n", piBp), VERR_INVALID_POINTER);
644 if (piBp)
645 *piBp = ~0;
646
647
648 /*
649 * Check if the breakpoint already exists.
650 */
651 PDBGFBP pBp = dbgfR3BpGetByAddr(pVM, DBGFBPTYPE_REM, pAddress->FlatPtr);
652 if (pBp)
653 {
654 int rc = VINF_SUCCESS;
655 if (!pBp->fEnabled)
656#ifdef VBOX_WITH_REM
657 rc = REMR3BreakpointSet(pVM, pBp->GCPtr);
658#else
659 rc = IEMBreakpointSet(pVM, pBp->GCPtr);
660#endif
661 if (RT_SUCCESS(rc))
662 {
663 rc = VINF_DBGF_BP_ALREADY_EXIST;
664 if (piBp)
665 *piBp = pBp->iBp;
666 }
667 return rc;
668 }
669
670 /*
671 * Allocate and initialize the bp.
672 */
673 pBp = dbgfR3BpAlloc(pVM, DBGFBPTYPE_REM);
674 if (!pBp)
675 return VERR_DBGF_NO_MORE_BP_SLOTS;
676 pBp->GCPtr = pAddress->FlatPtr;
677 pBp->iHitTrigger = *piHitTrigger;
678 pBp->iHitDisable = *piHitDisable;
679 pBp->fEnabled = true;
680
681 /*
682 * Now ask REM to set the breakpoint.
683 */
684#ifdef VBOX_WITH_REM
685 int rc = REMR3BreakpointSet(pVM, pAddress->FlatPtr);
686#else
687 int rc = IEMBreakpointSet(pVM, pAddress->FlatPtr);
688#endif
689 if (RT_SUCCESS(rc))
690 {
691 if (piBp)
692 *piBp = pBp->iBp;
693 }
694 else
695 dbgfR3BpFree(pVM, pBp);
696
697 return rc;
698}
699
700
701/**
702 * Clears a breakpoint.
703 *
704 * @returns VBox status code.
705 * @param pVM Pointer to the VM.
706 * @param iBp The id of the breakpoint which should be removed (cleared).
707 * @thread Any thread.
708 */
709VMMR3DECL(int) DBGFR3BpClear(PVM pVM, uint32_t iBp)
710{
711 /*
712 * This must be done on EMT.
713 */
714 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpClear, 2, pVM, iBp);
715 LogFlow(("DBGFR3BpClear: returns %Rrc\n", rc));
716 return rc;
717}
718
719
720/**
721 * EMT worker for DBGFR3BpClear().
722 *
723 * @returns VBox status code.
724 * @param pVM Pointer to the VM.
725 * @param iBp The id of the breakpoint which should be removed (cleared).
726 * @thread EMT
727 * @internal
728 */
729static DECLCALLBACK(int) dbgfR3BpClear(PVM pVM, uint32_t iBp)
730{
731 /*
732 * Validate input.
733 */
734 PDBGFBP pBp = dbgfR3BpGet(pVM, iBp);
735 if (!pBp)
736 return VERR_DBGF_BP_NOT_FOUND;
737
738 /*
739 * Disarm the breakpoint if it's enabled.
740 */
741 if (pBp->fEnabled)
742 {
743 pBp->fEnabled = false;
744 int rc;
745 switch (pBp->enmType)
746 {
747 case DBGFBPTYPE_REG:
748 rc = dbgfR3BpRegDisarm(pVM, pBp);
749 break;
750
751 case DBGFBPTYPE_INT3:
752 rc = dbgfR3BpInt3Disarm(pVM, pBp);
753 break;
754
755 case DBGFBPTYPE_REM:
756#ifdef VBOX_WITH_REM
757 rc = REMR3BreakpointClear(pVM, pBp->GCPtr);
758#else
759 rc = IEMBreakpointClear(pVM, pBp->GCPtr);
760#endif
761 break;
762
763 default:
764 AssertMsgFailedReturn(("Invalid enmType=%d!\n", pBp->enmType), VERR_IPE_NOT_REACHED_DEFAULT_CASE);
765 }
766 AssertRCReturn(rc, rc);
767 }
768
769 /*
770 * Free the breakpoint.
771 */
772 dbgfR3BpFree(pVM, pBp);
773 return VINF_SUCCESS;
774}
775
776
777/**
778 * Enables a breakpoint.
779 *
780 * @returns VBox status code.
781 * @param pVM Pointer to the VM.
782 * @param iBp The id of the breakpoint which should be enabled.
783 * @thread Any thread.
784 */
785VMMR3DECL(int) DBGFR3BpEnable(PVM pVM, uint32_t iBp)
786{
787 /*
788 * This must be done on EMT.
789 */
790 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpEnable, 2, pVM, iBp);
791 LogFlow(("DBGFR3BpEnable: returns %Rrc\n", rc));
792 return rc;
793}
794
795
796/**
797 * EMT worker for DBGFR3BpEnable().
798 *
799 * @returns VBox status code.
800 * @param pVM Pointer to the VM.
801 * @param iBp The id of the breakpoint which should be enabled.
802 * @thread EMT
803 * @internal
804 */
805static DECLCALLBACK(int) dbgfR3BpEnable(PVM pVM, uint32_t iBp)
806{
807 /*
808 * Validate input.
809 */
810 PDBGFBP pBp = dbgfR3BpGet(pVM, iBp);
811 if (!pBp)
812 return VERR_DBGF_BP_NOT_FOUND;
813
814 /*
815 * Already enabled?
816 */
817 if (pBp->fEnabled)
818 return VINF_DBGF_BP_ALREADY_ENABLED;
819
820 /*
821 * Remove the breakpoint.
822 */
823 int rc;
824 pBp->fEnabled = true;
825 switch (pBp->enmType)
826 {
827 case DBGFBPTYPE_REG:
828 rc = dbgfR3BpRegArm(pVM, pBp);
829 break;
830
831 case DBGFBPTYPE_INT3:
832 rc = dbgfR3BpInt3Arm(pVM, pBp);
833 break;
834
835 case DBGFBPTYPE_REM:
836#ifdef VBOX_WITH_REM
837 rc = REMR3BreakpointSet(pVM, pBp->GCPtr);
838#else
839 rc = IEMBreakpointSet(pVM, pBp->GCPtr);
840#endif
841 break;
842
843 default:
844 AssertMsgFailedReturn(("Invalid enmType=%d!\n", pBp->enmType), VERR_IPE_NOT_REACHED_DEFAULT_CASE);
845 }
846 if (RT_FAILURE(rc))
847 pBp->fEnabled = false;
848
849 return rc;
850}
851
852
853/**
854 * Disables a breakpoint.
855 *
856 * @returns VBox status code.
857 * @param pVM Pointer to the VM.
858 * @param iBp The id of the breakpoint which should be disabled.
859 * @thread Any thread.
860 */
861VMMR3DECL(int) DBGFR3BpDisable(PVM pVM, uint32_t iBp)
862{
863 /*
864 * This must be done on EMT.
865 */
866 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpDisable, 2, pVM, iBp);
867 LogFlow(("DBGFR3BpDisable: returns %Rrc\n", rc));
868 return rc;
869}
870
871
872/**
873 * EMT worker for DBGFR3BpDisable().
874 *
875 * @returns VBox status code.
876 * @param pVM Pointer to the VM.
877 * @param iBp The id of the breakpoint which should be disabled.
878 * @thread EMT
879 * @internal
880 */
881static DECLCALLBACK(int) dbgfR3BpDisable(PVM pVM, uint32_t iBp)
882{
883 /*
884 * Validate input.
885 */
886 PDBGFBP pBp = dbgfR3BpGet(pVM, iBp);
887 if (!pBp)
888 return VERR_DBGF_BP_NOT_FOUND;
889
890 /*
891 * Already enabled?
892 */
893 if (!pBp->fEnabled)
894 return VINF_DBGF_BP_ALREADY_DISABLED;
895
896 /*
897 * Remove the breakpoint.
898 */
899 pBp->fEnabled = false;
900 int rc;
901 switch (pBp->enmType)
902 {
903 case DBGFBPTYPE_REG:
904 rc = dbgfR3BpRegDisarm(pVM, pBp);
905 break;
906
907 case DBGFBPTYPE_INT3:
908 rc = dbgfR3BpInt3Disarm(pVM, pBp);
909 break;
910
911 case DBGFBPTYPE_REM:
912#ifdef VBOX_WITH_REM
913 rc = REMR3BreakpointClear(pVM, pBp->GCPtr);
914#else
915 rc = IEMBreakpointClear(pVM, pBp->GCPtr);
916#endif
917 break;
918
919 default:
920 AssertMsgFailedReturn(("Invalid enmType=%d!\n", pBp->enmType), VERR_IPE_NOT_REACHED_DEFAULT_CASE);
921 }
922
923 return rc;
924}
925
926
927/**
928 * Enumerate the breakpoints.
929 *
930 * @returns VBox status code.
931 * @param pVM Pointer to the VM.
932 * @param pfnCallback The callback function.
933 * @param pvUser The user argument to pass to the callback.
934 * @thread Any thread but the callback will be called from EMT.
935 */
936VMMR3DECL(int) DBGFR3BpEnum(PVM pVM, PFNDBGFBPENUM pfnCallback, void *pvUser)
937{
938 /*
939 * This must be done on EMT.
940 */
941 int rc = VMR3ReqPriorityCallWait(pVM, VMCPUID_ANY, (PFNRT)dbgfR3BpEnum, 3, pVM, pfnCallback, pvUser);
942 LogFlow(("DBGFR3BpClear: returns %Rrc\n", rc));
943 return rc;
944}
945
946
947/**
948 * EMT worker for DBGFR3BpEnum().
949 *
950 * @returns VBox status code.
951 * @param pVM Pointer to the VM.
952 * @param pfnCallback The callback function.
953 * @param pvUser The user argument to pass to the callback.
954 * @thread EMT
955 * @internal
956 */
957static DECLCALLBACK(int) dbgfR3BpEnum(PVM pVM, PFNDBGFBPENUM pfnCallback, void *pvUser)
958{
959 /*
960 * Validate input.
961 */
962 AssertMsgReturn(VALID_PTR(pfnCallback), ("pfnCallback=%p\n", pfnCallback), VERR_INVALID_POINTER);
963
964 /*
965 * Enumerate the hardware breakpoints.
966 */
967 unsigned i;
968 for (i = 0; i < RT_ELEMENTS(pVM->dbgf.s.aHwBreakpoints); i++)
969 if (pVM->dbgf.s.aHwBreakpoints[i].enmType != DBGFBPTYPE_FREE)
970 {
971 int rc = pfnCallback(pVM, pvUser, &pVM->dbgf.s.aHwBreakpoints[i]);
972 if (RT_FAILURE(rc))
973 return rc;
974 }
975
976 /*
977 * Enumerate the other breakpoints.
978 */
979 for (i = 0; i < RT_ELEMENTS(pVM->dbgf.s.aBreakpoints); i++)
980 if (pVM->dbgf.s.aBreakpoints[i].enmType != DBGFBPTYPE_FREE)
981 {
982 int rc = pfnCallback(pVM, pvUser, &pVM->dbgf.s.aBreakpoints[i]);
983 if (RT_FAILURE(rc))
984 return rc;
985 }
986
987 return VINF_SUCCESS;
988}
989
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