diff options
Diffstat (limited to 'src/VBox/VMM/VMMR0/DBGFR0Bp.cpp')
-rw-r--r-- | src/VBox/VMM/VMMR0/DBGFR0Bp.cpp | 599 |
1 files changed, 599 insertions, 0 deletions
diff --git a/src/VBox/VMM/VMMR0/DBGFR0Bp.cpp b/src/VBox/VMM/VMMR0/DBGFR0Bp.cpp new file mode 100644 index 00000000..36f1d91f --- /dev/null +++ b/src/VBox/VMM/VMMR0/DBGFR0Bp.cpp @@ -0,0 +1,599 @@ +/* $Id: DBGFR0Bp.cpp $ */ +/** @file + * DBGF - Debugger Facility, R0 breakpoint management part. + */ + +/* + * Copyright (C) 2020-2022 Oracle and/or its affiliates. + * + * This file is part of VirtualBox base platform packages, as + * available from https://www.virtualbox.org. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, in version 3 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses>. + * + * SPDX-License-Identifier: GPL-3.0-only + */ + + +/********************************************************************************************************************************* +* Header Files * +*********************************************************************************************************************************/ +#define LOG_GROUP LOG_GROUP_DBGF +#include "DBGFInternal.h" +#include <VBox/vmm/gvm.h> +#include <VBox/vmm/gvmm.h> +#include <VBox/vmm/vmm.h> + +#include <VBox/log.h> +#include <VBox/sup.h> +#include <iprt/asm.h> +#include <iprt/assert.h> +#include <iprt/errcore.h> +#include <iprt/ctype.h> +#include <iprt/mem.h> +#include <iprt/memobj.h> +#include <iprt/process.h> +#include <iprt/string.h> + +#include "dtrace/VBoxVMM.h" + + +/********************************************************************************************************************************* +* Internal Functions * +*********************************************************************************************************************************/ + +/** + * Used by DBGFR0InitPerVM() to initialize the breakpoint manager. + * + * @returns nothing. + * @param pGVM The global (ring-0) VM structure. + */ +DECLHIDDEN(void) dbgfR0BpInit(PGVM pGVM) +{ + pGVM->dbgfr0.s.hMemObjBpOwners = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.hMapObjBpOwners = NIL_RTR0MEMOBJ; + //pGVM->dbgfr0.s.paBpOwnersR0 = NULL; + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpChunks); i++) + { + PDBGFBPCHUNKR0 pBpChunk = &pGVM->dbgfr0.s.aBpChunks[i]; + + pBpChunk->hMemObj = NIL_RTR0MEMOBJ; + pBpChunk->hMapObj = NIL_RTR0MEMOBJ; + //pBpChunk->paBpBaseSharedR0 = NULL; + //pBpChunk->paBpBaseR0Only = NULL; + } + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpL2TblChunks); i++) + { + PDBGFBPL2TBLCHUNKR0 pL2Chunk = &pGVM->dbgfr0.s.aBpL2TblChunks[i]; + + pL2Chunk->hMemObj = NIL_RTR0MEMOBJ; + pL2Chunk->hMapObj = NIL_RTR0MEMOBJ; + //pL2Chunk->paBpL2TblBaseSharedR0 = NULL; + } + + pGVM->dbgfr0.s.hMemObjBpLocL1 = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.hMapObjBpLocL1 = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.hMemObjBpLocPortIo = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.hMapObjBpLocPortIo = NIL_RTR0MEMOBJ; + //pGVM->dbgfr0.s.paBpLocL1R0 = NULL; + //pGVM->dbgfr0.s.paBpLocPortIoR0 = NULL; + //pGVM->dbgfr0.s.fInit = false; +} + + +/** + * Used by DBGFR0CleanupVM to destroy the breakpoint manager. + * + * This is done during VM cleanup so that we're sure there are no active threads + * using the breakpoint code. + * + * @param pGVM The global (ring-0) VM structure. + */ +DECLHIDDEN(void) dbgfR0BpDestroy(PGVM pGVM) +{ + if (pGVM->dbgfr0.s.hMemObjBpOwners != NIL_RTR0MEMOBJ) + { + Assert(pGVM->dbgfr0.s.hMapObjBpOwners != NIL_RTR0MEMOBJ); + AssertPtr(pGVM->dbgfr0.s.paBpOwnersR0); + + RTR0MEMOBJ hMemObj = pGVM->dbgfr0.s.hMapObjBpOwners; + pGVM->dbgfr0.s.hMapObjBpOwners = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + + hMemObj = pGVM->dbgfr0.s.hMemObjBpOwners; + pGVM->dbgfr0.s.hMemObjBpOwners = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + } + + if (pGVM->dbgfr0.s.fInit) + { + Assert(pGVM->dbgfr0.s.hMemObjBpLocL1 != NIL_RTR0MEMOBJ); + AssertPtr(pGVM->dbgfr0.s.paBpLocL1R0); + + /* + * Free all allocated memory and ring-3 mapping objects. + */ + RTR0MEMOBJ hMemObj = pGVM->dbgfr0.s.hMemObjBpLocL1; + pGVM->dbgfr0.s.hMemObjBpLocL1 = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.paBpLocL1R0 = NULL; + RTR0MemObjFree(hMemObj, true); + + if (pGVM->dbgfr0.s.paBpLocPortIoR0) + { + Assert(pGVM->dbgfr0.s.hMemObjBpLocPortIo != NIL_RTR0MEMOBJ); + Assert(pGVM->dbgfr0.s.hMapObjBpLocPortIo != NIL_RTR0MEMOBJ); + + hMemObj = pGVM->dbgfr0.s.hMapObjBpLocPortIo; + pGVM->dbgfr0.s.hMapObjBpLocPortIo = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + + hMemObj = pGVM->dbgfr0.s.hMemObjBpLocPortIo; + pGVM->dbgfr0.s.hMemObjBpLocPortIo = NIL_RTR0MEMOBJ; + pGVM->dbgfr0.s.paBpLocPortIoR0 = NULL; + RTR0MemObjFree(hMemObj, true); + } + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpChunks); i++) + { + PDBGFBPCHUNKR0 pBpChunk = &pGVM->dbgfr0.s.aBpChunks[i]; + + if (pBpChunk->hMemObj != NIL_RTR0MEMOBJ) + { + Assert(pBpChunk->hMapObj != NIL_RTR0MEMOBJ); + + pBpChunk->paBpBaseSharedR0 = NULL; + pBpChunk->paBpBaseR0Only = NULL; + + hMemObj = pBpChunk->hMapObj; + pBpChunk->hMapObj = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + + hMemObj = pBpChunk->hMemObj; + pBpChunk->hMemObj = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + } + } + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpL2TblChunks); i++) + { + PDBGFBPL2TBLCHUNKR0 pL2Chunk = &pGVM->dbgfr0.s.aBpL2TblChunks[i]; + + if (pL2Chunk->hMemObj != NIL_RTR0MEMOBJ) + { + Assert(pL2Chunk->hMapObj != NIL_RTR0MEMOBJ); + + pL2Chunk->paBpL2TblBaseSharedR0 = NULL; + + hMemObj = pL2Chunk->hMapObj; + pL2Chunk->hMapObj = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + + hMemObj = pL2Chunk->hMemObj; + pL2Chunk->hMemObj = NIL_RTR0MEMOBJ; + RTR0MemObjFree(hMemObj, true); + } + } + + pGVM->dbgfr0.s.fInit = false; + } +#ifdef RT_STRICT + else + { + Assert(pGVM->dbgfr0.s.hMemObjBpLocL1 == NIL_RTR0MEMOBJ); + Assert(!pGVM->dbgfr0.s.paBpLocL1R0); + + Assert(pGVM->dbgfr0.s.hMemObjBpLocPortIo == NIL_RTR0MEMOBJ); + Assert(!pGVM->dbgfr0.s.paBpLocPortIoR0); + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpChunks); i++) + { + PDBGFBPCHUNKR0 pBpChunk = &pGVM->dbgfr0.s.aBpChunks[i]; + + Assert(pBpChunk->hMemObj == NIL_RTR0MEMOBJ); + Assert(pBpChunk->hMapObj == NIL_RTR0MEMOBJ); + Assert(!pBpChunk->paBpBaseSharedR0); + Assert(!pBpChunk->paBpBaseR0Only); + } + + for (uint32_t i = 0; i < RT_ELEMENTS(pGVM->dbgfr0.s.aBpL2TblChunks); i++) + { + PDBGFBPL2TBLCHUNKR0 pL2Chunk = &pGVM->dbgfr0.s.aBpL2TblChunks[i]; + + Assert(pL2Chunk->hMemObj == NIL_RTR0MEMOBJ); + Assert(pL2Chunk->hMapObj == NIL_RTR0MEMOBJ); + Assert(!pL2Chunk->paBpL2TblBaseSharedR0); + } + } +#endif +} + + +/** + * Worker for DBGFR0BpInitReqHandler() that does the actual initialization. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param ppaBpLocL1R3 Where to return the ring-3 L1 lookup table address on success. + * @thread EMT(0) + */ +static int dbgfR0BpInitWorker(PGVM pGVM, R3PTRTYPE(volatile uint32_t *) *ppaBpLocL1R3) +{ + /* + * Figure out how much memory we need for the L1 lookup table and allocate it. + */ + uint32_t const cbL1Loc = RT_ALIGN_32(UINT16_MAX * sizeof(uint32_t), HOST_PAGE_SIZE); + + RTR0MEMOBJ hMemObj; + int rc = RTR0MemObjAllocPage(&hMemObj, cbL1Loc, false /*fExecutable*/); + if (RT_FAILURE(rc)) + return rc; + RT_BZERO(RTR0MemObjAddress(hMemObj), cbL1Loc); + + /* Map it. */ + RTR0MEMOBJ hMapObj; + rc = RTR0MemObjMapUserEx(&hMapObj, hMemObj, (RTR3PTR)-1, 0, RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf(), + 0 /*offSub*/, cbL1Loc); + if (RT_SUCCESS(rc)) + { + pGVM->dbgfr0.s.hMemObjBpLocL1 = hMemObj; + pGVM->dbgfr0.s.hMapObjBpLocL1 = hMapObj; + pGVM->dbgfr0.s.paBpLocL1R0 = (volatile uint32_t *)RTR0MemObjAddress(hMemObj); + + /* + * We're done. + */ + *ppaBpLocL1R3 = RTR0MemObjAddressR3(hMapObj); + pGVM->dbgfr0.s.fInit = true; + return rc; + } + + RTR0MemObjFree(hMemObj, true); + return rc; +} + + +/** + * Worker for DBGFR0BpPortIoInitReqHandler() that does the actual initialization. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param ppaBpLocPortIoR3 Where to return the ring-3 L1 lookup table address on success. + * @thread EMT(0) + */ +static int dbgfR0BpPortIoInitWorker(PGVM pGVM, R3PTRTYPE(volatile uint32_t *) *ppaBpLocPortIoR3) +{ + /* + * Figure out how much memory we need for the I/O port breakpoint lookup table and allocate it. + */ + uint32_t const cbPortIoLoc = RT_ALIGN_32(UINT16_MAX * sizeof(uint32_t), HOST_PAGE_SIZE); + + RTR0MEMOBJ hMemObj; + int rc = RTR0MemObjAllocPage(&hMemObj, cbPortIoLoc, false /*fExecutable*/); + if (RT_FAILURE(rc)) + return rc; + RT_BZERO(RTR0MemObjAddress(hMemObj), cbPortIoLoc); + + /* Map it. */ + RTR0MEMOBJ hMapObj; + rc = RTR0MemObjMapUserEx(&hMapObj, hMemObj, (RTR3PTR)-1, 0, RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf(), + 0 /*offSub*/, cbPortIoLoc); + if (RT_SUCCESS(rc)) + { + pGVM->dbgfr0.s.hMemObjBpLocPortIo = hMemObj; + pGVM->dbgfr0.s.hMapObjBpLocPortIo = hMapObj; + pGVM->dbgfr0.s.paBpLocPortIoR0 = (volatile uint32_t *)RTR0MemObjAddress(hMemObj); + + /* + * We're done. + */ + *ppaBpLocPortIoR3 = RTR0MemObjAddressR3(hMapObj); + return rc; + } + + RTR0MemObjFree(hMemObj, true); + return rc; +} + + +/** + * Worker for DBGFR0BpOwnerInitReqHandler() that does the actual initialization. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param ppaBpOwnerR3 Where to return the ring-3 breakpoint owner table base address on success. + * @thread EMT(0) + */ +static int dbgfR0BpOwnerInitWorker(PGVM pGVM, R3PTRTYPE(void *) *ppaBpOwnerR3) +{ + /* + * Figure out how much memory we need for the owner tables and allocate it. + */ + uint32_t const cbBpOwnerR0 = RT_ALIGN_32(DBGF_BP_OWNER_COUNT_MAX * sizeof(DBGFBPOWNERINTR0), HOST_PAGE_SIZE); + uint32_t const cbBpOwnerR3 = RT_ALIGN_32(DBGF_BP_OWNER_COUNT_MAX * sizeof(DBGFBPOWNERINT), HOST_PAGE_SIZE); + uint32_t const cbTotal = RT_ALIGN_32(cbBpOwnerR0 + cbBpOwnerR3, HOST_PAGE_SIZE); + + RTR0MEMOBJ hMemObj; + int rc = RTR0MemObjAllocPage(&hMemObj, cbTotal, false /*fExecutable*/); + if (RT_FAILURE(rc)) + return rc; + RT_BZERO(RTR0MemObjAddress(hMemObj), cbTotal); + + /* Map it. */ + RTR0MEMOBJ hMapObj; + rc = RTR0MemObjMapUserEx(&hMapObj, hMemObj, (RTR3PTR)-1, 0, RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf(), + cbBpOwnerR0 /*offSub*/, cbBpOwnerR3); + if (RT_SUCCESS(rc)) + { + pGVM->dbgfr0.s.hMemObjBpOwners = hMemObj; + pGVM->dbgfr0.s.hMapObjBpOwners = hMapObj; + pGVM->dbgfr0.s.paBpOwnersR0 = (PDBGFBPOWNERINTR0)RTR0MemObjAddress(hMemObj); + + /* + * We're done. + */ + *ppaBpOwnerR3 = RTR0MemObjAddressR3(hMapObj); + return rc; + } + + RTR0MemObjFree(hMemObj, true); + return rc; +} + + +/** + * Worker for DBGFR0BpChunkAllocReqHandler() that does the actual chunk allocation. + * + * Allocates a memory object and divides it up as follows: + * @verbatim + -------------------------------------- + ring-0 chunk data + -------------------------------------- + page alignment padding + -------------------------------------- + shared chunk data + -------------------------------------- + @endverbatim + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param idChunk The chunk ID to allocate. + * @param ppBpChunkBaseR3 Where to return the ring-3 chunk base address on success. + * @thread EMT(0) + */ +static int dbgfR0BpChunkAllocWorker(PGVM pGVM, uint32_t idChunk, R3PTRTYPE(void *) *ppBpChunkBaseR3) +{ + /* + * Figure out how much memory we need for the chunk and allocate it. + */ + uint32_t const cbRing0 = RT_ALIGN_32(DBGF_BP_COUNT_PER_CHUNK * sizeof(DBGFBPINTR0), HOST_PAGE_SIZE); + uint32_t const cbShared = RT_ALIGN_32(DBGF_BP_COUNT_PER_CHUNK * sizeof(DBGFBPINT), HOST_PAGE_SIZE); + uint32_t const cbTotal = cbRing0 + cbShared; + + RTR0MEMOBJ hMemObj; + int rc = RTR0MemObjAllocPage(&hMemObj, cbTotal, false /*fExecutable*/); + if (RT_FAILURE(rc)) + return rc; + RT_BZERO(RTR0MemObjAddress(hMemObj), cbTotal); + + /* Map it. */ + RTR0MEMOBJ hMapObj; + rc = RTR0MemObjMapUserEx(&hMapObj, hMemObj, (RTR3PTR)-1, 0, RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf(), + cbRing0 /*offSub*/, cbTotal - cbRing0); + if (RT_SUCCESS(rc)) + { + PDBGFBPCHUNKR0 pBpChunkR0 = &pGVM->dbgfr0.s.aBpChunks[idChunk]; + + pBpChunkR0->hMemObj = hMemObj; + pBpChunkR0->hMapObj = hMapObj; + pBpChunkR0->paBpBaseR0Only = (PDBGFBPINTR0)RTR0MemObjAddress(hMemObj); + pBpChunkR0->paBpBaseSharedR0 = (PDBGFBPINT)&pBpChunkR0->paBpBaseR0Only[DBGF_BP_COUNT_PER_CHUNK]; + + /* + * We're done. + */ + *ppBpChunkBaseR3 = RTR0MemObjAddressR3(hMapObj); + return rc; + } + + RTR0MemObjFree(hMemObj, true); + return rc; +} + + +/** + * Worker for DBGFR0BpL2TblChunkAllocReqHandler() that does the actual chunk allocation. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param idChunk The chunk ID to allocate. + * @param ppL2ChunkBaseR3 Where to return the ring-3 chunk base address on success. + * @thread EMT(0) + */ +static int dbgfR0BpL2TblChunkAllocWorker(PGVM pGVM, uint32_t idChunk, R3PTRTYPE(void *) *ppL2ChunkBaseR3) +{ + /* + * Figure out how much memory we need for the chunk and allocate it. + */ + uint32_t const cbTotal = RT_ALIGN_32(DBGF_BP_L2_TBL_ENTRIES_PER_CHUNK * sizeof(DBGFBPL2ENTRY), HOST_PAGE_SIZE); + + RTR0MEMOBJ hMemObj; + int rc = RTR0MemObjAllocPage(&hMemObj, cbTotal, false /*fExecutable*/); + if (RT_FAILURE(rc)) + return rc; + RT_BZERO(RTR0MemObjAddress(hMemObj), cbTotal); + + /* Map it. */ + RTR0MEMOBJ hMapObj; + rc = RTR0MemObjMapUserEx(&hMapObj, hMemObj, (RTR3PTR)-1, 0, RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf(), + 0 /*offSub*/, cbTotal); + if (RT_SUCCESS(rc)) + { + PDBGFBPL2TBLCHUNKR0 pL2ChunkR0 = &pGVM->dbgfr0.s.aBpL2TblChunks[idChunk]; + + pL2ChunkR0->hMemObj = hMemObj; + pL2ChunkR0->hMapObj = hMapObj; + pL2ChunkR0->paBpL2TblBaseSharedR0 = (PDBGFBPL2ENTRY)RTR0MemObjAddress(hMemObj); + + /* + * We're done. + */ + *ppL2ChunkBaseR3 = RTR0MemObjAddressR3(hMapObj); + return rc; + } + + RTR0MemObjFree(hMemObj, true); + return rc; +} + + +/** + * Used by ring-3 DBGF to fully initialize the breakpoint manager for operation. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param pReq Pointer to the request buffer. + * @thread EMT(0) + */ +VMMR0_INT_DECL(int) DBGFR0BpInitReqHandler(PGVM pGVM, PDBGFBPINITREQ pReq) +{ + LogFlow(("DBGFR0BpInitReqHandler:\n")); + + /* + * Validate the request. + */ + AssertReturn(pReq->Hdr.cbReq == sizeof(*pReq), VERR_INVALID_PARAMETER); + + int rc = GVMMR0ValidateGVMandEMT(pGVM, 0); + AssertRCReturn(rc, rc); + + AssertReturn(!pGVM->dbgfr0.s.fInit, VERR_WRONG_ORDER); + + return dbgfR0BpInitWorker(pGVM, &pReq->paBpLocL1R3); +} + + +/** + * Used by ring-3 DBGF to initialize the breakpoint manager for port I/O breakpoint operation. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param pReq Pointer to the request buffer. + * @thread EMT(0) + */ +VMMR0_INT_DECL(int) DBGFR0BpPortIoInitReqHandler(PGVM pGVM, PDBGFBPINITREQ pReq) +{ + LogFlow(("DBGFR0BpPortIoInitReqHandler:\n")); + + /* + * Validate the request. + */ + AssertReturn(pReq->Hdr.cbReq == sizeof(*pReq), VERR_INVALID_PARAMETER); + + int rc = GVMMR0ValidateGVMandEMT(pGVM, 0); + AssertRCReturn(rc, rc); + + AssertReturn(pGVM->dbgfr0.s.fInit, VERR_WRONG_ORDER); + AssertReturn(!pGVM->dbgfr0.s.paBpLocPortIoR0, VERR_WRONG_ORDER); + + return dbgfR0BpPortIoInitWorker(pGVM, &pReq->paBpLocL1R3); +} + + +/** + * Used by ring-3 DBGF to initialize the breakpoint owner table for operation. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param pReq Pointer to the request buffer. + * @thread EMT(0) + */ +VMMR0_INT_DECL(int) DBGFR0BpOwnerInitReqHandler(PGVM pGVM, PDBGFBPOWNERINITREQ pReq) +{ + LogFlow(("DBGFR0BpOwnerInitReqHandler:\n")); + + /* + * Validate the request. + */ + AssertReturn(pReq->Hdr.cbReq == sizeof(*pReq), VERR_INVALID_PARAMETER); + + int rc = GVMMR0ValidateGVMandEMT(pGVM, 0); + AssertRCReturn(rc, rc); + + AssertReturn(!pGVM->dbgfr0.s.paBpOwnersR0, VERR_WRONG_ORDER); + + return dbgfR0BpOwnerInitWorker(pGVM, &pReq->paBpOwnerR3); +} + + +/** + * Used by ring-3 DBGF to allocate a given chunk in the global breakpoint table. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param pReq Pointer to the request buffer. + * @thread EMT(0) + */ +VMMR0_INT_DECL(int) DBGFR0BpChunkAllocReqHandler(PGVM pGVM, PDBGFBPCHUNKALLOCREQ pReq) +{ + LogFlow(("DBGFR0BpChunkAllocReqHandler:\n")); + + /* + * Validate the request. + */ + AssertReturn(pReq->Hdr.cbReq == sizeof(*pReq), VERR_INVALID_PARAMETER); + + uint32_t const idChunk = pReq->idChunk; + AssertReturn(idChunk < DBGF_BP_CHUNK_COUNT, VERR_INVALID_PARAMETER); + + int rc = GVMMR0ValidateGVMandEMT(pGVM, 0); + AssertRCReturn(rc, rc); + + AssertReturn(pGVM->dbgfr0.s.fInit, VERR_WRONG_ORDER); + AssertReturn(pGVM->dbgfr0.s.aBpChunks[idChunk].hMemObj == NIL_RTR0MEMOBJ, VERR_INVALID_PARAMETER); + + return dbgfR0BpChunkAllocWorker(pGVM, idChunk, &pReq->pChunkBaseR3); +} + + +/** + * Used by ring-3 DBGF to allocate a given chunk in the global L2 lookup table. + * + * @returns VBox status code. + * @param pGVM The global (ring-0) VM structure. + * @param pReq Pointer to the request buffer. + * @thread EMT(0) + */ +VMMR0_INT_DECL(int) DBGFR0BpL2TblChunkAllocReqHandler(PGVM pGVM, PDBGFBPL2TBLCHUNKALLOCREQ pReq) +{ + LogFlow(("DBGFR0BpL2TblChunkAllocReqHandler:\n")); + + /* + * Validate the request. + */ + AssertReturn(pReq->Hdr.cbReq == sizeof(*pReq), VERR_INVALID_PARAMETER); + + uint32_t const idChunk = pReq->idChunk; + AssertReturn(idChunk < DBGF_BP_L2_TBL_CHUNK_COUNT, VERR_INVALID_PARAMETER); + + int rc = GVMMR0ValidateGVMandEMT(pGVM, 0); + AssertRCReturn(rc, rc); + + AssertReturn(pGVM->dbgfr0.s.fInit, VERR_WRONG_ORDER); + AssertReturn(pGVM->dbgfr0.s.aBpL2TblChunks[idChunk].hMemObj == NIL_RTR0MEMOBJ, VERR_INVALID_PARAMETER); + + return dbgfR0BpL2TblChunkAllocWorker(pGVM, idChunk, &pReq->pChunkBaseR3); +} + |