VirtualBox

source: vbox/trunk/src/VBox/VMM/PDMDevice.cpp@ 12935

Last change on this file since 12935 was 12807, checked in by vboxsync, 16 years ago

PDM - VMMDevHeap: PDM_DEVHLP_VERSION minor increment, GCPhysVMMDevHeap init, spaces and doxygen.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 184.3 KB
Line 
1/* $Id: PDMDevice.cpp 12807 2008-09-29 15:03:55Z vboxsync $ */
2/** @file
3 * PDM - Pluggable Device and Driver Manager, Device parts.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_PDM_DEVICE
27#include "PDMInternal.h"
28#include <VBox/pdm.h>
29#include <VBox/mm.h>
30#include <VBox/pgm.h>
31#include <VBox/iom.h>
32#include <VBox/cfgm.h>
33#include <VBox/rem.h>
34#include <VBox/dbgf.h>
35#include <VBox/vm.h>
36#include <VBox/vmm.h>
37
38#include <VBox/version.h>
39#include <VBox/log.h>
40#include <VBox/err.h>
41#include <iprt/alloc.h>
42#include <iprt/alloca.h>
43#include <iprt/asm.h>
44#include <iprt/assert.h>
45#include <iprt/path.h>
46#include <iprt/semaphore.h>
47#include <iprt/string.h>
48#include <iprt/thread.h>
49
50
51
52/*******************************************************************************
53* Structures and Typedefs *
54*******************************************************************************/
55/**
56 * Internal callback structure pointer.
57 * The main purpose is to define the extra data we associate
58 * with PDMDEVREGCB so we can find the VM instance and so on.
59 */
60typedef struct PDMDEVREGCBINT
61{
62 /** The callback structure. */
63 PDMDEVREGCB Core;
64 /** A bit of padding. */
65 uint32_t u32[4];
66 /** VM Handle. */
67 PVM pVM;
68} PDMDEVREGCBINT, *PPDMDEVREGCBINT;
69typedef const PDMDEVREGCBINT *PCPDMDEVREGCBINT;
70
71
72/*******************************************************************************
73* Internal Functions *
74*******************************************************************************/
75__BEGIN_DECLS
76static DECLCALLBACK(int) pdmR3DevReg_Register(PPDMDEVREGCB pCallbacks, PCPDMDEVREG pDevReg);
77static DECLCALLBACK(void *) pdmR3DevReg_MMHeapAlloc(PPDMDEVREGCB pCallbacks, size_t cb);
78static DECLCALLBACK(bool) pdmR3DevHlpQueueConsumer(PVM pVM, PPDMQUEUEITEMCORE pItem);
79
80/* VSlick regex:
81search : \om/\*\*.+?\*\/\nDECLCALLBACKMEMBER\(([^,]*), *pfn([^)]*)\)\(
82replace: \/\*\* @copydoc PDMDEVHLP::pfn\2 \*\/\nstatic DECLCALLBACK\(\1\) pdmR3DevHlp_\2\(
83 */
84
85/** @name R3 DevHlp
86 * @{
87 */
88static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTHCPTR pvUser, PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn, PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc);
89static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterGC(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTRCPTR pvUser, const char *pszOut, const char *pszIn, const char *pszOutStr, const char *pszInStr, const char *pszDesc);
90static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterR0(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTR0PTR pvUser, const char *pszOut, const char *pszIn, const char *pszOutStr, const char *pszInStr, const char *pszDesc);
91static DECLCALLBACK(int) pdmR3DevHlp_IOPortDeregister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts);
92static DECLCALLBACK(int) pdmR3DevHlp_MMIORegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTHCPTR pvUser,
93 PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, PFNIOMMMIOFILL pfnFill,
94 const char *pszDesc);
95static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterGC(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTGCPTR pvUser,
96 const char *pszWrite, const char *pszRead, const char *pszFill,
97 const char *pszDesc);
98static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterR0(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTR0PTR pvUser,
99 const char *pszWrite, const char *pszRead, const char *pszFill,
100 const char *pszDesc);
101static DECLCALLBACK(int) pdmR3DevHlp_MMIODeregister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange);
102static DECLCALLBACK(int) pdmR3DevHlp_ROMRegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, const void *pvBinary, bool fShadow, const char *pszDesc);
103static DECLCALLBACK(int) pdmR3DevHlp_SSMRegister(PPDMDEVINS pDevIns, const char *pszName, uint32_t u32Instance, uint32_t u32Version, size_t cbGuess,
104 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
105 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone);
106static DECLCALLBACK(int) pdmR3DevHlp_TMTimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback, const char *pszDesc, PPTMTIMERR3 ppTimer);
107static DECLCALLBACK(PTMTIMERR3) pdmR3DevHlp_TMTimerCreateExternal(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMEREXT pfnCallback, void *pvUser, const char *pszDesc);
108static DECLCALLBACK(int) pdmR3DevHlp_PCIRegister(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev);
109static DECLCALLBACK(int) pdmR3DevHlp_PCIIORegionRegister(PPDMDEVINS pDevIns, int iRegion, uint32_t cbRegion, PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback);
110static DECLCALLBACK(void) pdmR3DevHlp_PCISetConfigCallbacks(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
111 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld);
112static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel);
113static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel);
114static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel);
115static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel);
116static DECLCALLBACK(int) pdmR3DevHlp_DriverAttach(PPDMDEVINS pDevIns, RTUINT iLun, PPDMIBASE pBaseInterface, PPDMIBASE *ppBaseInterface, const char *pszDesc);
117static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAlloc(PPDMDEVINS pDevIns, size_t cb);
118static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAllocZ(PPDMDEVINS pDevIns, size_t cb);
119static DECLCALLBACK(void) pdmR3DevHlp_MMHeapFree(PPDMDEVINS pDevIns, void *pv);
120static DECLCALLBACK(int) pdmR3DevHlp_VMSetError(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...);
121static DECLCALLBACK(int) pdmR3DevHlp_VMSetErrorV(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va);
122static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeError(PPDMDEVINS pDevIns, bool fFatal, const char *pszErrorID, const char *pszFormat, ...);
123static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeErrorV(PPDMDEVINS pDevIns, bool fFatal, const char *pszErrorID, const char *pszFormat, va_list va);
124static DECLCALLBACK(bool) pdmR3DevHlp_AssertEMT(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction);
125static DECLCALLBACK(bool) pdmR3DevHlp_AssertOther(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction);
126static DECLCALLBACK(int) pdmR3DevHlp_DBGFStopV(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction, const char *pszFormat, va_list args);
127static DECLCALLBACK(int) pdmR3DevHlp_DBGFInfoRegister(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler);
128static DECLCALLBACK(void) pdmR3DevHlp_STAMRegister(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName, STAMUNIT enmUnit, const char *pszDesc);
129static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterF(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc, const char *pszName, ...);
130static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterV(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc, const char *pszName, va_list args);
131static DECLCALLBACK(int) pdmR3DevHlp_CritSectInit(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, const char *pszName);
132static DECLCALLBACK(PRTTIMESPEC) pdmR3DevHlp_UTCNow(PPDMDEVINS pDevIns, PRTTIMESPEC pTime);
133static DECLCALLBACK(int) pdmR3DevHlp_PDMThreadCreate(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
134 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName);
135
136static DECLCALLBACK(PVM) pdmR3DevHlp_GetVM(PPDMDEVINS pDevIns);
137static DECLCALLBACK(int) pdmR3DevHlp_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3);
138static DECLCALLBACK(int) pdmR3DevHlp_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3);
139static DECLCALLBACK(int) pdmR3DevHlp_APICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3);
140static DECLCALLBACK(int) pdmR3DevHlp_IOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3);
141static DECLCALLBACK(int) pdmR3DevHlp_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp);
142static DECLCALLBACK(int) pdmR3DevHlp_RTCRegister(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp);
143static DECLCALLBACK(int) pdmR3DevHlp_PDMQueueCreate(PPDMDEVINS pDevIns, RTUINT cbItem, RTUINT cItems, uint32_t cMilliesInterval, PFNPDMQUEUEDEV pfnCallback, bool fGCEnabled, PPDMQUEUE *ppQueue);
144static DECLCALLBACK(void) pdmR3DevHlp_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead);
145static DECLCALLBACK(void) pdmR3DevHlp_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite);
146static DECLCALLBACK(int) pdmR3DevHlp_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb);
147static DECLCALLBACK(int) pdmR3DevHlp_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb);
148static DECLCALLBACK(int) pdmR3DevHlp_PhysReserve(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, const char *pszDesc);
149static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPtr2GCPhys(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys);
150static DECLCALLBACK(VMSTATE) pdmR3DevHlp_VMState(PPDMDEVINS pDevIns);
151static DECLCALLBACK(bool) pdmR3DevHlp_A20IsEnabled(PPDMDEVINS pDevIns);
152static DECLCALLBACK(void) pdmR3DevHlp_A20Set(PPDMDEVINS pDevIns, bool fEnable);
153static DECLCALLBACK(int) pdmR3DevHlp_VMReset(PPDMDEVINS pDevIns);
154static DECLCALLBACK(int) pdmR3DevHlp_VMPowerOff(PPDMDEVINS pDevIns);
155static DECLCALLBACK(int) pdmR3DevHlp_VMSuspend(PPDMDEVINS pDevIns);
156static DECLCALLBACK(int) pdmR3DevHlp_LockVM(PPDMDEVINS pDevIns);
157static DECLCALLBACK(int) pdmR3DevHlp_UnlockVM(PPDMDEVINS pDevIns);
158static DECLCALLBACK(bool) pdmR3DevHlp_AssertVMLock(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction);
159static DECLCALLBACK(int) pdmR3DevHlp_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser);
160static DECLCALLBACK(int) pdmR3DevHlp_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead);
161static DECLCALLBACK(int) pdmR3DevHlp_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten);
162static DECLCALLBACK(int) pdmR3DevHlp_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel);
163static DECLCALLBACK(uint8_t) pdmR3DevHlp_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel);
164static DECLCALLBACK(void) pdmR3DevHlp_DMASchedule(PPDMDEVINS pDevIns);
165static DECLCALLBACK(int) pdmR3DevHlp_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value);
166static DECLCALLBACK(int) pdmR3DevHlp_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value);
167static DECLCALLBACK(void) pdmR3DevHlp_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
168 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx);
169static DECLCALLBACK(int) pdmR3DevHlp_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange);
170static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc);
171static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion);
172static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys);
173static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys);
174static DECLCALLBACK(int) pdmR3DevHlp_MMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb, const char *pszDesc, PRTRCPTR pRCPtr);
175static DECLCALLBACK(int) pdmR3DevHlp_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize);
176static DECLCALLBACK(int) pdmR3DevHlp_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys);
177
178static DECLCALLBACK(PVM) pdmR3DevHlp_Untrusted_GetVM(PPDMDEVINS pDevIns);
179static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3);
180static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3);
181static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_APICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3);
182static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_IOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3);
183static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp);
184static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead);
185static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite);
186static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb);
187static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb);
188static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysReserve(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, const char *pszDesc);
189static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_Obsolete_Phys2HCVirt(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, PRTHCPTR ppvHC);
190static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_Obsolete_PhysGCPtr2HCPtr(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTHCPTR pHCPtr);
191
192static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_A20IsEnabled(PPDMDEVINS pDevIns);
193static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_A20Set(PPDMDEVINS pDevIns, bool fEnable);
194static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMReset(PPDMDEVINS pDevIns);
195static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMPowerOff(PPDMDEVINS pDevIns);
196static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspend(PPDMDEVINS pDevIns);
197static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_LockVM(PPDMDEVINS pDevIns);
198static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_UnlockVM(PPDMDEVINS pDevIns);
199static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_AssertVMLock(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction);
200static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser);
201static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead);
202static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten);
203static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel);
204static DECLCALLBACK(uint8_t) pdmR3DevHlp_Untrusted_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel);
205static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_DMASchedule(PPDMDEVINS pDevIns);
206static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value);
207static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value);
208static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
209 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx);
210static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange);
211static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc);
212static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion);
213static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys);
214static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys);
215static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb, const char *pszDesc, PRTRCPTR pRCPtr);
216static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize);
217static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys);
218/** @} */
219
220
221/** @name HC PIC Helpers
222 * @{
223 */
224static DECLCALLBACK(void) pdmR3PicHlp_SetInterruptFF(PPDMDEVINS pDevIns);
225static DECLCALLBACK(void) pdmR3PicHlp_ClearInterruptFF(PPDMDEVINS pDevIns);
226static DECLCALLBACK(int) pdmR3PicHlp_Lock(PPDMDEVINS pDevIns, int rc);
227static DECLCALLBACK(void) pdmR3PicHlp_Unlock(PPDMDEVINS pDevIns);
228static DECLCALLBACK(PCPDMPICHLPRC) pdmR3PicHlp_GetRCHelpers(PPDMDEVINS pDevIns);
229static DECLCALLBACK(PCPDMPICHLPR0) pdmR3PicHlp_GetR0Helpers(PPDMDEVINS pDevIns);
230/** @} */
231
232
233/** @name HC APIC Helpers
234 * @{
235 */
236static DECLCALLBACK(void) pdmR3ApicHlp_SetInterruptFF(PPDMDEVINS pDevIns, VMCPUID idCpu);
237static DECLCALLBACK(void) pdmR3ApicHlp_ClearInterruptFF(PPDMDEVINS pDevIns, VMCPUID idCpu);
238static DECLCALLBACK(void) pdmR3ApicHlp_ChangeFeature(PPDMDEVINS pDevIns, bool fEnabled);
239static DECLCALLBACK(int) pdmR3ApicHlp_Lock(PPDMDEVINS pDevIns, int rc);
240static DECLCALLBACK(void) pdmR3ApicHlp_Unlock(PPDMDEVINS pDevIns);
241static DECLCALLBACK(VMCPUID) pdmR3ApicHlp_GetCpuId(PPDMDEVINS pDevIns);
242static DECLCALLBACK(PCPDMAPICHLPRC) pdmR3ApicHlp_GetRCHelpers(PPDMDEVINS pDevIns);
243static DECLCALLBACK(PCPDMAPICHLPR0) pdmR3ApicHlp_GetR0Helpers(PPDMDEVINS pDevIns);
244/** @} */
245
246
247/** @name HC I/O APIC Helpers
248 * @{
249 */
250static DECLCALLBACK(void) pdmR3IoApicHlp_ApicBusDeliver(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
251 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode);
252static DECLCALLBACK(int) pdmR3IoApicHlp_Lock(PPDMDEVINS pDevIns, int rc);
253static DECLCALLBACK(void) pdmR3IoApicHlp_Unlock(PPDMDEVINS pDevIns);
254static DECLCALLBACK(PCPDMIOAPICHLPRC) pdmR3IoApicHlp_GetRCHelpers(PPDMDEVINS pDevIns);
255static DECLCALLBACK(PCPDMIOAPICHLPR0) pdmR3IoApicHlp_GetR0Helpers(PPDMDEVINS pDevIns);
256/** @} */
257
258
259/** @name HC PCI Bus Helpers
260 * @{
261 */
262static DECLCALLBACK(void) pdmR3PciHlp_IsaSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel);
263static DECLCALLBACK(void) pdmR3PciHlp_IoApicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel);
264static DECLCALLBACK(bool) pdmR3PciHlp_IsMMIO2Base(PPDMDEVINS pDevIns, PPDMDEVINS pOwner, RTGCPHYS GCPhys);
265static DECLCALLBACK(int) pdmR3PciHlp_Lock(PPDMDEVINS pDevIns, int rc);
266static DECLCALLBACK(void) pdmR3PciHlp_Unlock(PPDMDEVINS pDevIns);
267static DECLCALLBACK(PCPDMPCIHLPRC) pdmR3PciHlp_GetRCHelpers(PPDMDEVINS pDevIns);
268static DECLCALLBACK(PCPDMPCIHLPR0) pdmR3PciHlp_GetR0Helpers(PPDMDEVINS pDevIns);
269/** @} */
270
271/** @def PDMDEV_ASSERT_DEVINS
272 * Asserts the validity of the device instance.
273 */
274#ifdef VBOX_STRICT
275# define PDMDEV_ASSERT_DEVINS(pDevIns) \
276 do { \
277 AssertPtr(pDevIns); \
278 Assert(pDevIns->u32Version == PDM_DEVINS_VERSION); \
279 Assert(pDevIns->pvInstanceDataR3 == (void *)&pDevIns->achInstanceData[0]); \
280 } while (0)
281#else
282# define PDMDEV_ASSERT_DEVINS(pDevIns) do { } while (0)
283#endif
284static int pdmR3DevLoad(PVM pVM, PPDMDEVREGCBINT pRegCB, const char *pszFilename, const char *pszName);
285
286
287/*
288 * Allow physical read and writes from any thread
289 */
290#define PDM_PHYS_READWRITE_FROM_ANY_THREAD
291
292__END_DECLS
293
294/*******************************************************************************
295* Global Variables *
296*******************************************************************************/
297/**
298 * The device helper structure for trusted devices.
299 */
300const PDMDEVHLP g_pdmR3DevHlpTrusted =
301{
302 PDM_DEVHLP_VERSION,
303 pdmR3DevHlp_IOPortRegister,
304 pdmR3DevHlp_IOPortRegisterGC,
305 pdmR3DevHlp_IOPortRegisterR0,
306 pdmR3DevHlp_IOPortDeregister,
307 pdmR3DevHlp_MMIORegister,
308 pdmR3DevHlp_MMIORegisterGC,
309 pdmR3DevHlp_MMIORegisterR0,
310 pdmR3DevHlp_MMIODeregister,
311 pdmR3DevHlp_ROMRegister,
312 pdmR3DevHlp_SSMRegister,
313 pdmR3DevHlp_TMTimerCreate,
314 pdmR3DevHlp_TMTimerCreateExternal,
315 pdmR3DevHlp_PCIRegister,
316 pdmR3DevHlp_PCIIORegionRegister,
317 pdmR3DevHlp_PCISetConfigCallbacks,
318 pdmR3DevHlp_PCISetIrq,
319 pdmR3DevHlp_PCISetIrqNoWait,
320 pdmR3DevHlp_ISASetIrq,
321 pdmR3DevHlp_ISASetIrqNoWait,
322 pdmR3DevHlp_DriverAttach,
323 pdmR3DevHlp_MMHeapAlloc,
324 pdmR3DevHlp_MMHeapAllocZ,
325 pdmR3DevHlp_MMHeapFree,
326 pdmR3DevHlp_VMSetError,
327 pdmR3DevHlp_VMSetErrorV,
328 pdmR3DevHlp_VMSetRuntimeError,
329 pdmR3DevHlp_VMSetRuntimeErrorV,
330 pdmR3DevHlp_AssertEMT,
331 pdmR3DevHlp_AssertOther,
332 pdmR3DevHlp_DBGFStopV,
333 pdmR3DevHlp_DBGFInfoRegister,
334 pdmR3DevHlp_STAMRegister,
335 pdmR3DevHlp_STAMRegisterF,
336 pdmR3DevHlp_STAMRegisterV,
337 pdmR3DevHlp_RTCRegister,
338 pdmR3DevHlp_PDMQueueCreate,
339 pdmR3DevHlp_CritSectInit,
340 pdmR3DevHlp_UTCNow,
341 pdmR3DevHlp_PDMThreadCreate,
342 pdmR3DevHlp_PhysGCPtr2GCPhys,
343 pdmR3DevHlp_VMState,
344 0,
345 0,
346 0,
347 0,
348 0,
349 0,
350 0,
351 pdmR3DevHlp_GetVM,
352 pdmR3DevHlp_PCIBusRegister,
353 pdmR3DevHlp_PICRegister,
354 pdmR3DevHlp_APICRegister,
355 pdmR3DevHlp_IOAPICRegister,
356 pdmR3DevHlp_DMACRegister,
357 pdmR3DevHlp_PhysRead,
358 pdmR3DevHlp_PhysWrite,
359 pdmR3DevHlp_PhysReadGCVirt,
360 pdmR3DevHlp_PhysWriteGCVirt,
361 pdmR3DevHlp_PhysReserve,
362 pdmR3DevHlp_Untrusted_Obsolete_Phys2HCVirt,
363 pdmR3DevHlp_Untrusted_Obsolete_PhysGCPtr2HCPtr,
364 pdmR3DevHlp_A20IsEnabled,
365 pdmR3DevHlp_A20Set,
366 pdmR3DevHlp_VMReset,
367 pdmR3DevHlp_VMSuspend,
368 pdmR3DevHlp_VMPowerOff,
369 pdmR3DevHlp_LockVM,
370 pdmR3DevHlp_UnlockVM,
371 pdmR3DevHlp_AssertVMLock,
372 pdmR3DevHlp_DMARegister,
373 pdmR3DevHlp_DMAReadMemory,
374 pdmR3DevHlp_DMAWriteMemory,
375 pdmR3DevHlp_DMASetDREQ,
376 pdmR3DevHlp_DMAGetChannelMode,
377 pdmR3DevHlp_DMASchedule,
378 pdmR3DevHlp_CMOSWrite,
379 pdmR3DevHlp_CMOSRead,
380 pdmR3DevHlp_GetCpuId,
381 pdmR3DevHlp_ROMProtectShadow,
382 pdmR3DevHlp_MMIO2Register,
383 pdmR3DevHlp_MMIO2Deregister,
384 pdmR3DevHlp_MMIO2Map,
385 pdmR3DevHlp_MMIO2Unmap,
386 pdmR3DevHlp_MMHyperMapMMIO2,
387 pdmR3DevHlp_RegisterVMMDevHeap,
388 pdmR3DevHlp_UnregisterVMMDevHeap,
389 PDM_DEVHLP_VERSION /* the end */
390};
391
392
393/**
394 * The device helper structure for non-trusted devices.
395 */
396const PDMDEVHLP g_pdmR3DevHlpUnTrusted =
397{
398 PDM_DEVHLP_VERSION,
399 pdmR3DevHlp_IOPortRegister,
400 pdmR3DevHlp_IOPortRegisterGC,
401 pdmR3DevHlp_IOPortRegisterR0,
402 pdmR3DevHlp_IOPortDeregister,
403 pdmR3DevHlp_MMIORegister,
404 pdmR3DevHlp_MMIORegisterGC,
405 pdmR3DevHlp_MMIORegisterR0,
406 pdmR3DevHlp_MMIODeregister,
407 pdmR3DevHlp_ROMRegister,
408 pdmR3DevHlp_SSMRegister,
409 pdmR3DevHlp_TMTimerCreate,
410 pdmR3DevHlp_TMTimerCreateExternal,
411 pdmR3DevHlp_PCIRegister,
412 pdmR3DevHlp_PCIIORegionRegister,
413 pdmR3DevHlp_PCISetConfigCallbacks,
414 pdmR3DevHlp_PCISetIrq,
415 pdmR3DevHlp_PCISetIrqNoWait,
416 pdmR3DevHlp_ISASetIrq,
417 pdmR3DevHlp_ISASetIrqNoWait,
418 pdmR3DevHlp_DriverAttach,
419 pdmR3DevHlp_MMHeapAlloc,
420 pdmR3DevHlp_MMHeapAllocZ,
421 pdmR3DevHlp_MMHeapFree,
422 pdmR3DevHlp_VMSetError,
423 pdmR3DevHlp_VMSetErrorV,
424 pdmR3DevHlp_VMSetRuntimeError,
425 pdmR3DevHlp_VMSetRuntimeErrorV,
426 pdmR3DevHlp_AssertEMT,
427 pdmR3DevHlp_AssertOther,
428 pdmR3DevHlp_DBGFStopV,
429 pdmR3DevHlp_DBGFInfoRegister,
430 pdmR3DevHlp_STAMRegister,
431 pdmR3DevHlp_STAMRegisterF,
432 pdmR3DevHlp_STAMRegisterV,
433 pdmR3DevHlp_RTCRegister,
434 pdmR3DevHlp_PDMQueueCreate,
435 pdmR3DevHlp_CritSectInit,
436 pdmR3DevHlp_UTCNow,
437 pdmR3DevHlp_PDMThreadCreate,
438 pdmR3DevHlp_PhysGCPtr2GCPhys,
439 pdmR3DevHlp_VMState,
440 0,
441 0,
442 0,
443 0,
444 0,
445 0,
446 0,
447 pdmR3DevHlp_Untrusted_GetVM,
448 pdmR3DevHlp_Untrusted_PCIBusRegister,
449 pdmR3DevHlp_Untrusted_PICRegister,
450 pdmR3DevHlp_Untrusted_APICRegister,
451 pdmR3DevHlp_Untrusted_IOAPICRegister,
452 pdmR3DevHlp_Untrusted_DMACRegister,
453 pdmR3DevHlp_Untrusted_PhysRead,
454 pdmR3DevHlp_Untrusted_PhysWrite,
455 pdmR3DevHlp_Untrusted_PhysReadGCVirt,
456 pdmR3DevHlp_Untrusted_PhysWriteGCVirt,
457 pdmR3DevHlp_Untrusted_PhysReserve,
458 pdmR3DevHlp_Untrusted_Obsolete_Phys2HCVirt,
459 pdmR3DevHlp_Untrusted_Obsolete_PhysGCPtr2HCPtr,
460 pdmR3DevHlp_Untrusted_A20IsEnabled,
461 pdmR3DevHlp_Untrusted_A20Set,
462 pdmR3DevHlp_Untrusted_VMReset,
463 pdmR3DevHlp_Untrusted_VMSuspend,
464 pdmR3DevHlp_Untrusted_VMPowerOff,
465 pdmR3DevHlp_Untrusted_LockVM,
466 pdmR3DevHlp_Untrusted_UnlockVM,
467 pdmR3DevHlp_Untrusted_AssertVMLock,
468 pdmR3DevHlp_Untrusted_DMARegister,
469 pdmR3DevHlp_Untrusted_DMAReadMemory,
470 pdmR3DevHlp_Untrusted_DMAWriteMemory,
471 pdmR3DevHlp_Untrusted_DMASetDREQ,
472 pdmR3DevHlp_Untrusted_DMAGetChannelMode,
473 pdmR3DevHlp_Untrusted_DMASchedule,
474 pdmR3DevHlp_Untrusted_CMOSWrite,
475 pdmR3DevHlp_Untrusted_CMOSRead,
476 pdmR3DevHlp_Untrusted_GetCpuId,
477 pdmR3DevHlp_Untrusted_ROMProtectShadow,
478 pdmR3DevHlp_Untrusted_MMIO2Register,
479 pdmR3DevHlp_Untrusted_MMIO2Deregister,
480 pdmR3DevHlp_Untrusted_MMIO2Map,
481 pdmR3DevHlp_Untrusted_MMIO2Unmap,
482 pdmR3DevHlp_Untrusted_MMHyperMapMMIO2,
483 pdmR3DevHlp_Untrusted_RegisterVMMDevHeap,
484 pdmR3DevHlp_Untrusted_UnregisterVMMDevHeap,
485 PDM_DEVHLP_VERSION /* the end */
486};
487
488
489/**
490 * PIC Device Helpers.
491 */
492const PDMPICHLPR3 g_pdmR3DevPicHlp =
493{
494 PDM_PICHLPR3_VERSION,
495 pdmR3PicHlp_SetInterruptFF,
496 pdmR3PicHlp_ClearInterruptFF,
497 pdmR3PicHlp_Lock,
498 pdmR3PicHlp_Unlock,
499 pdmR3PicHlp_GetRCHelpers,
500 pdmR3PicHlp_GetR0Helpers,
501 PDM_PICHLPR3_VERSION /* the end */
502};
503
504
505/**
506 * APIC Device Helpers.
507 */
508const PDMAPICHLPR3 g_pdmR3DevApicHlp =
509{
510 PDM_APICHLPR3_VERSION,
511 pdmR3ApicHlp_SetInterruptFF,
512 pdmR3ApicHlp_ClearInterruptFF,
513 pdmR3ApicHlp_ChangeFeature,
514 pdmR3ApicHlp_Lock,
515 pdmR3ApicHlp_Unlock,
516 pdmR3ApicHlp_GetCpuId,
517 pdmR3ApicHlp_GetRCHelpers,
518 pdmR3ApicHlp_GetR0Helpers,
519 PDM_APICHLPR3_VERSION /* the end */
520};
521
522
523/**
524 * I/O APIC Device Helpers.
525 */
526const PDMIOAPICHLPR3 g_pdmR3DevIoApicHlp =
527{
528 PDM_IOAPICHLPR3_VERSION,
529 pdmR3IoApicHlp_ApicBusDeliver,
530 pdmR3IoApicHlp_Lock,
531 pdmR3IoApicHlp_Unlock,
532 pdmR3IoApicHlp_GetRCHelpers,
533 pdmR3IoApicHlp_GetR0Helpers,
534 PDM_IOAPICHLPR3_VERSION /* the end */
535};
536
537
538/**
539 * PCI Bus Device Helpers.
540 */
541const PDMPCIHLPR3 g_pdmR3DevPciHlp =
542{
543 PDM_PCIHLPR3_VERSION,
544 pdmR3PciHlp_IsaSetIrq,
545 pdmR3PciHlp_IoApicSetIrq,
546 pdmR3PciHlp_IsMMIO2Base,
547 pdmR3PciHlp_GetRCHelpers,
548 pdmR3PciHlp_GetR0Helpers,
549 pdmR3PciHlp_Lock,
550 pdmR3PciHlp_Unlock,
551 PDM_PCIHLPR3_VERSION, /* the end */
552};
553
554
555/**
556 * DMAC Device Helpers.
557 */
558const PDMDMACHLP g_pdmR3DevDmacHlp =
559{
560 PDM_DMACHLP_VERSION
561};
562
563
564/**
565 * RTC Device Helpers.
566 */
567const PDMRTCHLP g_pdmR3DevRtcHlp =
568{
569 PDM_RTCHLP_VERSION
570};
571
572
573/**
574 * This function will initialize the devices for this VM instance.
575 *
576 *
577 * First of all this mean loading the builtin device and letting them
578 * register themselves. Beyond that any additional device modules are
579 * loaded and called for registration.
580 *
581 * Then the device configuration is enumerated, the instantiation order
582 * is determined, and finally they are instantiated.
583 *
584 * After all device have been successfully instantiated the the primary
585 * PCI Bus device is called to emulate the PCI BIOS, i.e. making the
586 * resource assignments. If there is no PCI device, this step is of course
587 * skipped.
588 *
589 * Finally the init completion routines of the instantiated devices
590 * are called.
591 *
592 * @returns VBox status code.
593 * @param pVM VM Handle.
594 */
595int pdmR3DevInit(PVM pVM)
596{
597 LogFlow(("pdmR3DevInit:\n"));
598
599 AssertRelease(!(RT_OFFSETOF(PDMDEVINS, achInstanceData) & 15));
600 AssertRelease(sizeof(pVM->pdm.s.pDevInstances->Internal.s) <= sizeof(pVM->pdm.s.pDevInstances->Internal.padding));
601
602 /*
603 * Get the GC & R0 devhlps and create the devhlp R3 task queue.
604 */
605 RCPTRTYPE(PCPDMDEVHLPGC) pDevHlpGC;
606 int rc = PDMR3GetSymbolGC(pVM, NULL, "g_pdmGCDevHlp", &pDevHlpGC);
607 AssertReleaseRCReturn(rc, rc);
608
609 R0PTRTYPE(PCPDMDEVHLPR0) pDevHlpR0;
610 rc = PDMR3GetSymbolR0(pVM, NULL, "g_pdmR0DevHlp", &pDevHlpR0);
611 AssertReleaseRCReturn(rc, rc);
612
613 rc = PDMR3QueueCreateInternal(pVM, sizeof(PDMDEVHLPTASK), 8, 0, pdmR3DevHlpQueueConsumer, true, &pVM->pdm.s.pDevHlpQueueHC);
614 AssertRCReturn(rc, rc);
615 pVM->pdm.s.pDevHlpQueueGC = PDMQueueGCPtr(pVM->pdm.s.pDevHlpQueueHC);
616
617
618 /*
619 * Initialize the callback structure.
620 */
621 PDMDEVREGCBINT RegCB;
622 RegCB.Core.u32Version = PDM_DEVREG_CB_VERSION;
623 RegCB.Core.pfnRegister = pdmR3DevReg_Register;
624 RegCB.Core.pfnMMHeapAlloc = pdmR3DevReg_MMHeapAlloc;
625 RegCB.pVM = pVM;
626
627 /*
628 * Load the builtin module
629 */
630 PCFGMNODE pDevicesNode = CFGMR3GetChild(CFGMR3GetRoot(pVM), "PDM/Devices");
631 bool fLoadBuiltin;
632 rc = CFGMR3QueryBool(pDevicesNode, "LoadBuiltin", &fLoadBuiltin);
633 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
634 fLoadBuiltin = true;
635 else if (VBOX_FAILURE(rc))
636 {
637 AssertMsgFailed(("Configuration error: Querying boolean \"LoadBuiltin\" failed with %Vrc\n", rc));
638 return rc;
639 }
640 if (fLoadBuiltin)
641 {
642 /* make filename */
643 char *pszFilename = pdmR3FileR3("VBoxDD", /* fShared = */ true);
644 if (!pszFilename)
645 return VERR_NO_TMP_MEMORY;
646 rc = pdmR3DevLoad(pVM, &RegCB, pszFilename, "VBoxDD");
647 RTMemTmpFree(pszFilename);
648 if (VBOX_FAILURE(rc))
649 return rc;
650
651 /* make filename */
652 pszFilename = pdmR3FileR3("VBoxDD2", /* fShared = */ true);
653 if (!pszFilename)
654 return VERR_NO_TMP_MEMORY;
655 rc = pdmR3DevLoad(pVM, &RegCB, pszFilename, "VBoxDD2");
656 RTMemTmpFree(pszFilename);
657 if (VBOX_FAILURE(rc))
658 return rc;
659 }
660
661 /*
662 * Load additional device modules.
663 */
664 PCFGMNODE pCur;
665 for (pCur = CFGMR3GetFirstChild(pDevicesNode); pCur; pCur = CFGMR3GetNextChild(pCur))
666 {
667 /*
668 * Get the name and path.
669 */
670 char szName[PDMMOD_NAME_LEN];
671 rc = CFGMR3GetName(pCur, &szName[0], sizeof(szName));
672 if (rc == VERR_CFGM_NOT_ENOUGH_SPACE)
673 {
674 AssertMsgFailed(("configuration error: The module name is too long, cchName=%d.\n", CFGMR3GetNameLen(pCur)));
675 return VERR_PDM_MODULE_NAME_TOO_LONG;
676 }
677 else if (VBOX_FAILURE(rc))
678 {
679 AssertMsgFailed(("CFGMR3GetName -> %Vrc.\n", rc));
680 return rc;
681 }
682
683 /* the path is optional, if no path the module name + path is used. */
684 char szFilename[RTPATH_MAX];
685 rc = CFGMR3QueryString(pCur, "Path", &szFilename[0], sizeof(szFilename));
686 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
687 strcpy(szFilename, szName);
688 else if (VBOX_FAILURE(rc))
689 {
690 AssertMsgFailed(("configuration error: Failure to query the module path, rc=%Vrc.\n", rc));
691 return rc;
692 }
693
694 /* prepend path? */
695 if (!RTPathHavePath(szFilename))
696 {
697 char *psz = pdmR3FileR3(szFilename);
698 if (!psz)
699 return VERR_NO_TMP_MEMORY;
700 size_t cch = strlen(psz) + 1;
701 if (cch > sizeof(szFilename))
702 {
703 RTMemTmpFree(psz);
704 AssertMsgFailed(("Filename too long! cch=%d '%s'\n", cch, psz));
705 return VERR_FILENAME_TOO_LONG;
706 }
707 memcpy(szFilename, psz, cch);
708 RTMemTmpFree(psz);
709 }
710
711 /*
712 * Load the module and register it's devices.
713 */
714 rc = pdmR3DevLoad(pVM, &RegCB, szFilename, szName);
715 if (VBOX_FAILURE(rc))
716 return rc;
717 }
718
719#ifdef VBOX_WITH_USB
720 /* ditto for USB Devices. */
721 rc = pdmR3UsbLoadModules(pVM);
722 if (RT_FAILURE(rc))
723 return rc;
724#endif
725
726
727 /*
728 *
729 * Enumerate the device instance configurations
730 * and come up with a instantiation order.
731 *
732 */
733 /* Switch to /Devices, which contains the device instantiations. */
734 pDevicesNode = CFGMR3GetChild(CFGMR3GetRoot(pVM), "Devices");
735
736 /*
737 * Count the device instances.
738 */
739 PCFGMNODE pInstanceNode;
740 unsigned cDevs = 0;
741 for (pCur = CFGMR3GetFirstChild(pDevicesNode); pCur; pCur = CFGMR3GetNextChild(pCur))
742 for (pInstanceNode = CFGMR3GetFirstChild(pCur); pInstanceNode; pInstanceNode = CFGMR3GetNextChild(pInstanceNode))
743 cDevs++;
744 if (!cDevs)
745 {
746 Log(("PDM: No devices were configured!\n"));
747 return VINF_SUCCESS;
748 }
749 Log2(("PDM: cDevs=%d!\n", cDevs));
750
751 /*
752 * Collect info on each device instance.
753 */
754 struct DEVORDER
755 {
756 /** Configuration node. */
757 PCFGMNODE pNode;
758 /** Pointer to device. */
759 PPDMDEV pDev;
760 /** Init order. */
761 uint32_t u32Order;
762 /** VBox instance number. */
763 uint32_t iInstance;
764 } *paDevs = (struct DEVORDER *)alloca(sizeof(paDevs[0]) * (cDevs + 1)); /* (One extra for swapping) */
765 Assert(paDevs);
766 unsigned i = 0;
767 for (pCur = CFGMR3GetFirstChild(pDevicesNode); pCur; pCur = CFGMR3GetNextChild(pCur))
768 {
769 /* Get the device name. */
770 char szName[sizeof(paDevs[0].pDev->pDevReg->szDeviceName)];
771 rc = CFGMR3GetName(pCur, szName, sizeof(szName));
772 AssertMsgRCReturn(rc, ("Configuration error: device name is too long (or something)! rc=%Vrc\n", rc), rc);
773
774 /* Find the device. */
775 PPDMDEV pDev = pdmR3DevLookup(pVM, szName);
776 AssertMsgReturn(pDev, ("Configuration error: device '%s' not found!\n", szName), VERR_PDM_DEVICE_NOT_FOUND);
777
778 /* Configured priority or use default based on device class? */
779 uint32_t u32Order;
780 rc = CFGMR3QueryU32(pCur, "Priority", &u32Order);
781 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
782 {
783 uint32_t u32 = pDev->pDevReg->fClass;
784 for (u32Order = 1; !(u32 & u32Order); u32Order <<= 1)
785 /* nop */;
786 }
787 else
788 AssertMsgRCReturn(rc, ("Configuration error: reading \"Priority\" for the '%s' device failed rc=%Vrc!\n", szName, rc), rc);
789
790 /* Enumerate the device instances. */
791 for (pInstanceNode = CFGMR3GetFirstChild(pCur); pInstanceNode; pInstanceNode = CFGMR3GetNextChild(pInstanceNode))
792 {
793 paDevs[i].pNode = pInstanceNode;
794 paDevs[i].pDev = pDev;
795 paDevs[i].u32Order = u32Order;
796
797 /* Get the instance number. */
798 char szInstance[32];
799 rc = CFGMR3GetName(pInstanceNode, szInstance, sizeof(szInstance));
800 AssertMsgRCReturn(rc, ("Configuration error: instance name is too long (or something)! rc=%Vrc\n", rc), rc);
801 char *pszNext = NULL;
802 rc = RTStrToUInt32Ex(szInstance, &pszNext, 0, &paDevs[i].iInstance);
803 AssertMsgRCReturn(rc, ("Configuration error: RTStrToInt32Ex failed on the instance name '%s'! rc=%Vrc\n", szInstance, rc), rc);
804 AssertMsgReturn(!*pszNext, ("Configuration error: the instance name '%s' isn't all digits. (%s)\n", szInstance, pszNext), VERR_INVALID_PARAMETER);
805
806 /* next instance */
807 i++;
808 }
809 } /* devices */
810 Assert(i == cDevs);
811
812 /*
813 * Sort the device array ascending on u32Order. (bubble)
814 */
815 unsigned c = cDevs - 1;
816 while (c)
817 {
818 unsigned j = 0;
819 for (i = 0; i < c; i++)
820 if (paDevs[i].u32Order > paDevs[i + 1].u32Order)
821 {
822 paDevs[cDevs] = paDevs[i + 1];
823 paDevs[i + 1] = paDevs[i];
824 paDevs[i] = paDevs[cDevs];
825 j = i;
826 }
827 c = j;
828 }
829
830
831 /*
832 *
833 * Instantiate the devices.
834 *
835 */
836 for (i = 0; i < cDevs; i++)
837 {
838 /*
839 * Gather a bit of config.
840 */
841 /* trusted */
842 bool fTrusted;
843 rc = CFGMR3QueryBool(paDevs[i].pNode, "Trusted", &fTrusted);
844 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
845 fTrusted = false;
846 else if (VBOX_FAILURE(rc))
847 {
848 AssertMsgFailed(("configuration error: failed to query boolean \"Trusted\", rc=%Vrc\n", rc));
849 return rc;
850 }
851 /* config node */
852 PCFGMNODE pConfigNode = CFGMR3GetChild(paDevs[i].pNode, "Config");
853 if (!pConfigNode)
854 {
855 rc = CFGMR3InsertNode(paDevs[i].pNode, "Config", &pConfigNode);
856 if (VBOX_FAILURE(rc))
857 {
858 AssertMsgFailed(("Failed to create Config node! rc=%Vrc\n", rc));
859 return rc;
860 }
861 }
862 CFGMR3SetRestrictedRoot(pConfigNode);
863
864 /*
865 * Allocate the device instance.
866 */
867 size_t cb = RT_OFFSETOF(PDMDEVINS, achInstanceData[paDevs[i].pDev->pDevReg->cbInstance]);
868 cb = RT_ALIGN_Z(cb, 16);
869 PPDMDEVINS pDevIns;
870 if (paDevs[i].pDev->pDevReg->fFlags & (PDM_DEVREG_FLAGS_GC | PDM_DEVREG_FLAGS_R0))
871 rc = MMR3HyperAllocOnceNoRel(pVM, cb, 0, MM_TAG_PDM_DEVICE, (void **)&pDevIns);
872 else
873 rc = MMR3HeapAllocZEx(pVM, MM_TAG_PDM_DEVICE, cb, (void **)&pDevIns);
874 if (VBOX_FAILURE(rc))
875 {
876 AssertMsgFailed(("Failed to allocate %d bytes of instance data for device '%s'. rc=%Vrc\n",
877 cb, paDevs[i].pDev->pDevReg->szDeviceName, rc));
878 return rc;
879 }
880
881 /*
882 * Initialize it.
883 */
884 pDevIns->u32Version = PDM_DEVINS_VERSION;
885 //pDevIns->Internal.s.pNextHC = NULL;
886 //pDevIns->Internal.s.pPerDeviceNextHC = NULL;
887 pDevIns->Internal.s.pDevHC = paDevs[i].pDev;
888 pDevIns->Internal.s.pVMHC = pVM;
889 pDevIns->Internal.s.pVMGC = pVM->pVMGC;
890 //pDevIns->Internal.s.pLunsHC = NULL;
891 pDevIns->Internal.s.pCfgHandle = paDevs[i].pNode;
892 //pDevIns->Internal.s.pPciDevice = NULL;
893 //pDevIns->Internal.s.pPciBus = NULL; /** @todo pci bus selection. (in 2008 perhaps) */
894 pDevIns->pDevHlp = fTrusted ? &g_pdmR3DevHlpTrusted : &g_pdmR3DevHlpUnTrusted;
895 pDevIns->pDevHlpGC = pDevHlpGC;
896 pDevIns->pDevHlpR0 = pDevHlpR0;
897 pDevIns->pDevReg = paDevs[i].pDev->pDevReg;
898 pDevIns->pCfgHandle = pConfigNode;
899 pDevIns->iInstance = paDevs[i].iInstance;
900 pDevIns->pvInstanceDataR3 = &pDevIns->achInstanceData[0];
901 pDevIns->pvInstanceDataGC = pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC
902 ? MMHyperHC2GC(pVM, pDevIns->pvInstanceDataR3) : 0;
903 pDevIns->pvInstanceDataR0 = pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_R0
904 ? MMHyperR3ToR0(pVM, pDevIns->pvInstanceDataR3) : 0;
905
906 /*
907 * Link it into all the lists.
908 */
909 /* The global instance FIFO. */
910 PPDMDEVINS pPrev1 = pVM->pdm.s.pDevInstances;
911 if (!pPrev1)
912 pVM->pdm.s.pDevInstances = pDevIns;
913 else
914 {
915 while (pPrev1->Internal.s.pNextHC)
916 pPrev1 = pPrev1->Internal.s.pNextHC;
917 pPrev1->Internal.s.pNextHC = pDevIns;
918 }
919
920 /* The per device instance FIFO. */
921 PPDMDEVINS pPrev2 = paDevs[i].pDev->pInstances;
922 if (!pPrev2)
923 paDevs[i].pDev->pInstances = pDevIns;
924 else
925 {
926 while (pPrev2->Internal.s.pPerDeviceNextHC)
927 pPrev2 = pPrev2->Internal.s.pPerDeviceNextHC;
928 pPrev2->Internal.s.pPerDeviceNextHC = pDevIns;
929 }
930
931 /*
932 * Call the constructor.
933 */
934 Log(("PDM: Constructing device '%s' instance %d...\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
935 rc = pDevIns->pDevReg->pfnConstruct(pDevIns, pDevIns->iInstance, pDevIns->pCfgHandle);
936 if (VBOX_FAILURE(rc))
937 {
938 LogRel(("PDM: Failed to construct '%s'/%d! %Vra\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
939 /* because we're damn lazy right now, we'll say that the destructor will be called even if the constructor fails. */
940 return rc;
941 }
942 } /* for device instances */
943
944#ifdef VBOX_WITH_USB
945 /* ditto for USB Devices. */
946 rc = pdmR3UsbInstantiateDevices(pVM);
947 if (RT_FAILURE(rc))
948 return rc;
949#endif
950
951
952 /*
953 *
954 * PCI BIOS Fake and Init Complete.
955 *
956 */
957 if (pVM->pdm.s.aPciBuses[0].pDevInsR3)
958 {
959 pdmLock(pVM);
960 rc = pVM->pdm.s.aPciBuses[0].pfnFakePCIBIOSR3(pVM->pdm.s.aPciBuses[0].pDevInsR3);
961 pdmUnlock(pVM);
962 if (VBOX_FAILURE(rc))
963 {
964 AssertMsgFailed(("PCI BIOS fake failed rc=%Vrc\n", rc));
965 return rc;
966 }
967 }
968
969 for (PPDMDEVINS pDevIns = pVM->pdm.s.pDevInstances; pDevIns; pDevIns = pDevIns->Internal.s.pNextHC)
970 {
971 if (pDevIns->pDevReg->pfnInitComplete)
972 {
973 rc = pDevIns->pDevReg->pfnInitComplete(pDevIns);
974 if (VBOX_FAILURE(rc))
975 {
976 AssertMsgFailed(("InitComplete on device '%s'/%d failed with rc=%Vrc\n",
977 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
978 return rc;
979 }
980 }
981 }
982
983#ifdef VBOX_WITH_USB
984 /* ditto for USB Devices. */
985 rc = pdmR3UsbVMInitComplete(pVM);
986 if (RT_FAILURE(rc))
987 return rc;
988#endif
989
990 LogFlow(("pdmR3DevInit: returns %Vrc\n", VINF_SUCCESS));
991 return VINF_SUCCESS;
992}
993
994
995/**
996 * Lookups a device structure by name.
997 * @internal
998 */
999PPDMDEV pdmR3DevLookup(PVM pVM, const char *pszName)
1000{
1001 RTUINT cchName = strlen(pszName);
1002 for (PPDMDEV pDev = pVM->pdm.s.pDevs; pDev; pDev = pDev->pNext)
1003 if ( pDev->cchName == cchName
1004 && !strcmp(pDev->pDevReg->szDeviceName, pszName))
1005 return pDev;
1006 return NULL;
1007}
1008
1009
1010/**
1011 * Loads one device module and call the registration entry point.
1012 *
1013 * @returns VBox status code.
1014 * @param pVM VM handle.
1015 * @param pRegCB The registration callback stuff.
1016 * @param pszFilename Module filename.
1017 * @param pszName Module name.
1018 */
1019static int pdmR3DevLoad(PVM pVM, PPDMDEVREGCBINT pRegCB, const char *pszFilename, const char *pszName)
1020{
1021 /*
1022 * Load it.
1023 */
1024 int rc = pdmR3LoadR3U(pVM->pUVM, pszFilename, pszName);
1025 if (VBOX_SUCCESS(rc))
1026 {
1027 /*
1028 * Get the registration export and call it.
1029 */
1030 FNPDMVBOXDEVICESREGISTER *pfnVBoxDevicesRegister;
1031 rc = PDMR3GetSymbolR3(pVM, pszName, "VBoxDevicesRegister", (void **)&pfnVBoxDevicesRegister);
1032 if (VBOX_SUCCESS(rc))
1033 {
1034 Log(("PDM: Calling VBoxDevicesRegister (%p) of %s (%s)\n", pfnVBoxDevicesRegister, pszName, pszFilename));
1035 rc = pfnVBoxDevicesRegister(&pRegCB->Core, VBOX_VERSION);
1036 if (VBOX_SUCCESS(rc))
1037 Log(("PDM: Successfully loaded device module %s (%s).\n", pszName, pszFilename));
1038 else
1039 AssertMsgFailed(("VBoxDevicesRegister failed with rc=%Vrc for module %s (%s)\n", rc, pszName, pszFilename));
1040 }
1041 else
1042 {
1043 AssertMsgFailed(("Failed to locate 'VBoxDevicesRegister' in %s (%s) rc=%Vrc\n", pszName, pszFilename, rc));
1044 if (rc == VERR_SYMBOL_NOT_FOUND)
1045 rc = VERR_PDM_NO_REGISTRATION_EXPORT;
1046 }
1047 }
1048 else
1049 AssertMsgFailed(("Failed to load %s %s!\n", pszFilename, pszName));
1050 return rc;
1051}
1052
1053
1054
1055/**
1056 * Registers a device with the current VM instance.
1057 *
1058 * @returns VBox status code.
1059 * @param pCallbacks Pointer to the callback table.
1060 * @param pDevReg Pointer to the device registration record.
1061 * This data must be permanent and readonly.
1062 */
1063static DECLCALLBACK(int) pdmR3DevReg_Register(PPDMDEVREGCB pCallbacks, PCPDMDEVREG pDevReg)
1064{
1065 /*
1066 * Validate the registration structure.
1067 */
1068 Assert(pDevReg);
1069 if (pDevReg->u32Version != PDM_DEVREG_VERSION)
1070 {
1071 AssertMsgFailed(("Unknown struct version %#x!\n", pDevReg->u32Version));
1072 return VERR_PDM_UNKNOWN_DEVREG_VERSION;
1073 }
1074 if ( !pDevReg->szDeviceName[0]
1075 || strlen(pDevReg->szDeviceName) >= sizeof(pDevReg->szDeviceName))
1076 {
1077 AssertMsgFailed(("Invalid name '%s'\n", pDevReg->szDeviceName));
1078 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1079 }
1080 if ( (pDevReg->fFlags & PDM_DEVREG_FLAGS_GC)
1081 && ( !pDevReg->szGCMod[0]
1082 || strlen(pDevReg->szGCMod) >= sizeof(pDevReg->szGCMod)))
1083 {
1084 AssertMsgFailed(("Invalid GC module name '%s' - (Device %s)\n", pDevReg->szGCMod, pDevReg->szDeviceName));
1085 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1086 }
1087 if ( (pDevReg->fFlags & PDM_DEVREG_FLAGS_R0)
1088 && ( !pDevReg->szR0Mod[0]
1089 || strlen(pDevReg->szR0Mod) >= sizeof(pDevReg->szR0Mod)))
1090 {
1091 AssertMsgFailed(("Invalid R0 module name '%s' - (Device %s)\n", pDevReg->szR0Mod, pDevReg->szDeviceName));
1092 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1093 }
1094 if ((pDevReg->fFlags & PDM_DEVREG_FLAGS_HOST_BITS_MASK) != PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT)
1095 {
1096 AssertMsgFailed(("Invalid host bits flags! fFlags=%#x (Device %s)\n", pDevReg->fFlags, pDevReg->szDeviceName));
1097 return VERR_PDM_INVALID_DEVICE_HOST_BITS;
1098 }
1099 if (!(pDevReg->fFlags & PDM_DEVREG_FLAGS_GUEST_BITS_MASK))
1100 {
1101 AssertMsgFailed(("Invalid guest bits flags! fFlags=%#x (Device %s)\n", pDevReg->fFlags, pDevReg->szDeviceName));
1102 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1103 }
1104 if (!pDevReg->fClass)
1105 {
1106 AssertMsgFailed(("No class! (Device %s)\n", pDevReg->szDeviceName));
1107 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1108 }
1109 if (pDevReg->cMaxInstances <= 0)
1110 {
1111 AssertMsgFailed(("Max instances %u! (Device %s)\n", pDevReg->cMaxInstances, pDevReg->szDeviceName));
1112 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1113 }
1114 if (pDevReg->cbInstance > (RTUINT)(pDevReg->fFlags & (PDM_DEVREG_FLAGS_GC | PDM_DEVREG_FLAGS_R0) ? 96 * _1K : _1M))
1115 {
1116 AssertMsgFailed(("Instance size %d bytes! (Device %s)\n", pDevReg->cbInstance, pDevReg->szDeviceName));
1117 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1118 }
1119 if (!pDevReg->pfnConstruct)
1120 {
1121 AssertMsgFailed(("No constructore! (Device %s)\n", pDevReg->szDeviceName));
1122 return VERR_PDM_INVALID_DEVICE_REGISTRATION;
1123 }
1124 /* Check matching guest bits last without any asserting. Enables trial and error registration. */
1125 if (!(pDevReg->fFlags & PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT))
1126 {
1127 Log(("PDM: Rejected device '%s' because it didn't match the guest bits.\n", pDevReg->szDeviceName));
1128 return VERR_PDM_INVALID_DEVICE_GUEST_BITS;
1129 }
1130
1131 /*
1132 * Check for duplicate and find FIFO entry at the same time.
1133 */
1134 PCPDMDEVREGCBINT pRegCB = (PCPDMDEVREGCBINT)pCallbacks;
1135 PPDMDEV pDevPrev = NULL;
1136 PPDMDEV pDev = pRegCB->pVM->pdm.s.pDevs;
1137 for (; pDev; pDevPrev = pDev, pDev = pDev->pNext)
1138 {
1139 if (!strcmp(pDev->pDevReg->szDeviceName, pDevReg->szDeviceName))
1140 {
1141 AssertMsgFailed(("Device '%s' already exists\n", pDevReg->szDeviceName));
1142 return VERR_PDM_DEVICE_NAME_CLASH;
1143 }
1144 }
1145
1146 /*
1147 * Allocate new device structure and insert it into the list.
1148 */
1149 pDev = (PPDMDEV)MMR3HeapAlloc(pRegCB->pVM, MM_TAG_PDM_DEVICE, sizeof(*pDev));
1150 if (pDev)
1151 {
1152 pDev->pNext = NULL;
1153 pDev->cInstances = 0;
1154 pDev->pInstances = NULL;
1155 pDev->pDevReg = pDevReg;
1156 pDev->cchName = strlen(pDevReg->szDeviceName);
1157
1158 if (pDevPrev)
1159 pDevPrev->pNext = pDev;
1160 else
1161 pRegCB->pVM->pdm.s.pDevs = pDev;
1162 Log(("PDM: Registered device '%s'\n", pDevReg->szDeviceName));
1163 return VINF_SUCCESS;
1164 }
1165 return VERR_NO_MEMORY;
1166}
1167
1168
1169/**
1170 * Allocate memory which is associated with current VM instance
1171 * and automatically freed on it's destruction.
1172 *
1173 * @returns Pointer to allocated memory. The memory is *NOT* zero-ed.
1174 * @param pCallbacks Pointer to the callback table.
1175 * @param cb Number of bytes to allocate.
1176 */
1177static DECLCALLBACK(void *) pdmR3DevReg_MMHeapAlloc(PPDMDEVREGCB pCallbacks, size_t cb)
1178{
1179 Assert(pCallbacks);
1180 Assert(pCallbacks->u32Version == PDM_DEVREG_CB_VERSION);
1181 LogFlow(("pdmR3DevReg_MMHeapAlloc: cb=%#x\n", cb));
1182
1183 void *pv = MMR3HeapAlloc(((PPDMDEVREGCBINT)pCallbacks)->pVM, MM_TAG_PDM_DEVICE_USER, cb);
1184
1185 LogFlow(("pdmR3DevReg_MMHeapAlloc: returns %p\n", pv));
1186 return pv;
1187}
1188
1189
1190/**
1191 * Queue consumer callback for internal component.
1192 *
1193 * @returns Success indicator.
1194 * If false the item will not be removed and the flushing will stop.
1195 * @param pVM The VM handle.
1196 * @param pItem The item to consume. Upon return this item will be freed.
1197 */
1198static DECLCALLBACK(bool) pdmR3DevHlpQueueConsumer(PVM pVM, PPDMQUEUEITEMCORE pItem)
1199{
1200 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)pItem;
1201 LogFlow(("pdmR3DevHlpQueueConsumer: enmOp=%d pDevIns=%p\n", pTask->enmOp, pTask->pDevInsHC));
1202 switch (pTask->enmOp)
1203 {
1204 case PDMDEVHLPTASKOP_ISA_SET_IRQ:
1205 PDMIsaSetIrq(pVM, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
1206 break;
1207
1208 case PDMDEVHLPTASKOP_PCI_SET_IRQ:
1209 pdmR3DevHlp_PCISetIrq(pTask->pDevInsHC, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
1210 break;
1211
1212 case PDMDEVHLPTASKOP_IOAPIC_SET_IRQ:
1213 PDMIoApicSetIrq(pVM, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
1214 break;
1215
1216 default:
1217 AssertReleaseMsgFailed(("Invalid operation %d\n", pTask->enmOp));
1218 break;
1219 }
1220 return true;
1221}
1222
1223
1224/** @copydoc PDMDEVHLP::pfnIOPortRegister */
1225static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTHCPTR pvUser, PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn,
1226 PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc)
1227{
1228 PDMDEV_ASSERT_DEVINS(pDevIns);
1229 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: Port=%#x cPorts=%#x pvUser=%p pfnOut=%p pfnIn=%p pfnOutStr=%p pfnInStr=%p p32_tszDesc=%p:{%s}\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
1230 Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc, pszDesc));
1231 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1232
1233 int rc = IOMR3IOPortRegisterR3(pDevIns->Internal.s.pVMHC, pDevIns, Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc);
1234
1235 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1236 return rc;
1237}
1238
1239
1240/** @copydoc PDMDEVHLP::pfnIOPortRegisterGC */
1241static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterGC(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTRCPTR pvUser,
1242 const char *pszOut, const char *pszIn,
1243 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
1244{
1245 PDMDEV_ASSERT_DEVINS(pDevIns);
1246 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1247 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: Port=%#x cPorts=%#x pvUser=%p pszOut=%p:{%s} pszIn=%p:{%s} pszOutStr=%p:{%s} pszInStr=%p:{%s} pszDesc=%p:{%s}\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
1248 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
1249
1250 /*
1251 * Resolve the functions (one of the can be NULL).
1252 */
1253 int rc = VINF_SUCCESS;
1254 if ( pDevIns->pDevReg->szGCMod[0]
1255 && (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC))
1256 {
1257 RTGCPTR32 GCPtrIn = 0;
1258 if (pszIn)
1259 {
1260 rc = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszIn, &GCPtrIn);
1261 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pDevReg->szGCMod, pszIn));
1262 }
1263 RTGCPTR32 GCPtrOut = 0;
1264 if (pszOut && VBOX_SUCCESS(rc))
1265 {
1266 rc = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszOut, &GCPtrOut);
1267 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pDevReg->szGCMod, pszOut));
1268 }
1269 RTGCPTR32 GCPtrInStr = 0;
1270 if (pszInStr && VBOX_SUCCESS(rc))
1271 {
1272 rc = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszInStr, &GCPtrInStr);
1273 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pDevReg->szGCMod, pszInStr));
1274 }
1275 RTGCPTR32 GCPtrOutStr = 0;
1276 if (pszOutStr && VBOX_SUCCESS(rc))
1277 {
1278 rc = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszOutStr, &GCPtrOutStr);
1279 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pDevReg->szGCMod, pszOutStr));
1280 }
1281
1282 if (VBOX_SUCCESS(rc))
1283 rc = IOMR3IOPortRegisterRC(pDevIns->Internal.s.pVMHC, pDevIns, Port, cPorts, pvUser, GCPtrOut, GCPtrIn, GCPtrOutStr, GCPtrInStr, pszDesc);
1284 }
1285 else
1286 {
1287 AssertMsgFailed(("No GC module for this driver!\n"));
1288 rc = VERR_INVALID_PARAMETER;
1289 }
1290
1291 LogFlow(("pdmR3DevHlp_IOPortRegisterGC: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1292 return rc;
1293}
1294
1295
1296/** @copydoc PDMDEVHLP::pfnIOPortRegisterR0 */
1297static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterR0(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTR0PTR pvUser,
1298 const char *pszOut, const char *pszIn,
1299 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
1300{
1301 PDMDEV_ASSERT_DEVINS(pDevIns);
1302 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1303 LogFlow(("pdmR3DevHlp_IOPortRegisterR0: caller='%s'/%d: Port=%#x cPorts=%#x pvUser=%p pszOut=%p:{%s} pszIn=%p:{%s} pszOutStr=%p:{%s} pszInStr=%p:{%s} pszDesc=%p:{%s}\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
1304 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
1305
1306 /*
1307 * Resolve the functions (one of the can be NULL).
1308 */
1309 int rc = VINF_SUCCESS;
1310 if ( pDevIns->pDevReg->szR0Mod[0]
1311 && (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_R0))
1312 {
1313 R0PTRTYPE(PFNIOMIOPORTIN) pfnR0PtrIn = 0;
1314 if (pszIn)
1315 {
1316 rc = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszIn, &pfnR0PtrIn);
1317 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pDevReg->szR0Mod, pszIn));
1318 }
1319 R0PTRTYPE(PFNIOMIOPORTOUT) pfnR0PtrOut = 0;
1320 if (pszOut && VBOX_SUCCESS(rc))
1321 {
1322 rc = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszOut, &pfnR0PtrOut);
1323 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pDevReg->szR0Mod, pszOut));
1324 }
1325 R0PTRTYPE(PFNIOMIOPORTINSTRING) pfnR0PtrInStr = 0;
1326 if (pszInStr && VBOX_SUCCESS(rc))
1327 {
1328 rc = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszInStr, &pfnR0PtrInStr);
1329 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pDevReg->szR0Mod, pszInStr));
1330 }
1331 R0PTRTYPE(PFNIOMIOPORTOUTSTRING) pfnR0PtrOutStr = 0;
1332 if (pszOutStr && VBOX_SUCCESS(rc))
1333 {
1334 rc = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszOutStr, &pfnR0PtrOutStr);
1335 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pDevReg->szR0Mod, pszOutStr));
1336 }
1337
1338 if (VBOX_SUCCESS(rc))
1339 rc = IOMR3IOPortRegisterR0(pDevIns->Internal.s.pVMHC, pDevIns, Port, cPorts, pvUser, pfnR0PtrOut, pfnR0PtrIn, pfnR0PtrOutStr, pfnR0PtrInStr, pszDesc);
1340 }
1341 else
1342 {
1343 AssertMsgFailed(("No R0 module for this driver!\n"));
1344 rc = VERR_INVALID_PARAMETER;
1345 }
1346
1347 LogFlow(("pdmR3DevHlp_IOPortRegisterR0: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1348 return rc;
1349}
1350
1351
1352/** @copydoc PDMDEVHLP::pfnIOPortDeregister */
1353static DECLCALLBACK(int) pdmR3DevHlp_IOPortDeregister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts)
1354{
1355 PDMDEV_ASSERT_DEVINS(pDevIns);
1356 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1357 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: Port=%#x cPorts=%#x\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
1358 Port, cPorts));
1359
1360 int rc = IOMR3IOPortDeregister(pDevIns->Internal.s.pVMHC, pDevIns, Port, cPorts);
1361
1362 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1363 return rc;
1364}
1365
1366
1367/** @copydoc PDMDEVHLP::pfnMMIORegister */
1368static DECLCALLBACK(int) pdmR3DevHlp_MMIORegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTHCPTR pvUser,
1369 PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, PFNIOMMMIOFILL pfnFill,
1370 const char *pszDesc)
1371{
1372 PDMDEV_ASSERT_DEVINS(pDevIns);
1373 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1374 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x pvUser=%p pfnWrite=%p pfnRead=%p pfnFill=%p pszDesc=%p:{%s}\n",
1375 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill, pszDesc, pszDesc));
1376
1377 int rc = IOMR3MMIORegisterR3(pDevIns->Internal.s.pVMHC, pDevIns, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill, pszDesc);
1378
1379 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1380 return rc;
1381}
1382
1383
1384/** @copydoc PDMDEVHLP::pfnMMIORegisterGC */
1385static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterGC(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTGCPTR pvUser,
1386 const char *pszWrite, const char *pszRead, const char *pszFill,
1387 const char *pszDesc)
1388{
1389 PDMDEV_ASSERT_DEVINS(pDevIns);
1390 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1391 LogFlow(("pdmR3DevHlp_MMIORegisterGC: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
1392 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
1393
1394 /*
1395 * Resolve the functions.
1396 * Not all function have to present, leave it to IOM to enforce this.
1397 */
1398 int rc = VINF_SUCCESS;
1399 if ( pDevIns->pDevReg->szGCMod[0]
1400 && (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC))
1401 {
1402 RTGCPTR32 GCPtrWrite = 0;
1403 if (pszWrite)
1404 rc = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszWrite, &GCPtrWrite);
1405 RTGCPTR32 GCPtrRead = 0;
1406 int rc2 = VINF_SUCCESS;
1407 if (pszRead)
1408 rc2 = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszRead, &GCPtrRead);
1409 RTGCPTR32 GCPtrFill = 0;
1410 int rc3 = VINF_SUCCESS;
1411 if (pszFill)
1412 rc3 = PDMR3GetSymbolGCLazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szGCMod, pszFill, &GCPtrFill);
1413 if (VBOX_SUCCESS(rc) && VBOX_SUCCESS(rc2) && VBOX_SUCCESS(rc3))
1414 rc = IOMR3MMIORegisterRC(pDevIns->Internal.s.pVMHC, pDevIns, GCPhysStart, cbRange, pvUser, GCPtrWrite, GCPtrRead, GCPtrFill);
1415 else
1416 {
1417 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pDevReg->szGCMod, pszWrite));
1418 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pDevReg->szGCMod, pszRead));
1419 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pDevReg->szGCMod, pszFill));
1420 if (VBOX_FAILURE(rc2) && VBOX_SUCCESS(rc))
1421 rc = rc2;
1422 if (VBOX_FAILURE(rc3) && VBOX_SUCCESS(rc))
1423 rc = rc3;
1424 }
1425 }
1426 else
1427 {
1428 AssertMsgFailed(("No GC module for this driver!\n"));
1429 rc = VERR_INVALID_PARAMETER;
1430 }
1431
1432 LogFlow(("pdmR3DevHlp_MMIORegisterGC: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1433 return rc;
1434}
1435
1436/** @copydoc PDMDEVHLP::pfnMMIORegisterR0 */
1437static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterR0(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTR0PTR pvUser,
1438 const char *pszWrite, const char *pszRead, const char *pszFill,
1439 const char *pszDesc)
1440{
1441 PDMDEV_ASSERT_DEVINS(pDevIns);
1442 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1443 LogFlow(("pdmR3DevHlp_MMIORegisterHC: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
1444 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
1445
1446 /*
1447 * Resolve the functions.
1448 * Not all function have to present, leave it to IOM to enforce this.
1449 */
1450 int rc = VINF_SUCCESS;
1451 if ( pDevIns->pDevReg->szR0Mod[0]
1452 && (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_R0))
1453 {
1454 R0PTRTYPE(PFNIOMMMIOWRITE) pfnR0PtrWrite = 0;
1455 if (pszWrite)
1456 rc = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszWrite, &pfnR0PtrWrite);
1457 R0PTRTYPE(PFNIOMMMIOREAD) pfnR0PtrRead = 0;
1458 int rc2 = VINF_SUCCESS;
1459 if (pszRead)
1460 rc2 = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszRead, &pfnR0PtrRead);
1461 R0PTRTYPE(PFNIOMMMIOFILL) pfnR0PtrFill = 0;
1462 int rc3 = VINF_SUCCESS;
1463 if (pszFill)
1464 rc3 = PDMR3GetSymbolR0Lazy(pDevIns->Internal.s.pVMHC, pDevIns->pDevReg->szR0Mod, pszFill, &pfnR0PtrFill);
1465 if (VBOX_SUCCESS(rc) && VBOX_SUCCESS(rc2) && VBOX_SUCCESS(rc3))
1466 rc = IOMR3MMIORegisterR0(pDevIns->Internal.s.pVMHC, pDevIns, GCPhysStart, cbRange, pvUser, pfnR0PtrWrite, pfnR0PtrRead, pfnR0PtrFill);
1467 else
1468 {
1469 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pDevReg->szR0Mod, pszWrite));
1470 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pDevReg->szR0Mod, pszRead));
1471 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pDevReg->szR0Mod, pszFill));
1472 if (VBOX_FAILURE(rc2) && VBOX_SUCCESS(rc))
1473 rc = rc2;
1474 if (VBOX_FAILURE(rc3) && VBOX_SUCCESS(rc))
1475 rc = rc3;
1476 }
1477 }
1478 else
1479 {
1480 AssertMsgFailed(("No R0 module for this driver!\n"));
1481 rc = VERR_INVALID_PARAMETER;
1482 }
1483
1484 LogFlow(("pdmR3DevHlp_MMIORegisterR0: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1485 return rc;
1486}
1487
1488
1489/** @copydoc PDMDEVHLP::pfnMMIODeregister */
1490static DECLCALLBACK(int) pdmR3DevHlp_MMIODeregister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange)
1491{
1492 PDMDEV_ASSERT_DEVINS(pDevIns);
1493 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1494 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x\n",
1495 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange));
1496
1497 int rc = IOMR3MMIODeregister(pDevIns->Internal.s.pVMHC, pDevIns, GCPhysStart, cbRange);
1498
1499 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1500 return rc;
1501}
1502
1503
1504/** @copydoc PDMDEVHLP::pfnROMRegister */
1505static DECLCALLBACK(int) pdmR3DevHlp_ROMRegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, const void *pvBinary, bool fShadow, const char *pszDesc)
1506{
1507 PDMDEV_ASSERT_DEVINS(pDevIns);
1508 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1509 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x pvBinary=%p fShadow=%RTbool pszDesc=%p:{%s}\n",
1510 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange, pvBinary, fShadow, pszDesc, pszDesc));
1511
1512 int rc = MMR3PhysRomRegister(pDevIns->Internal.s.pVMHC, pDevIns, GCPhysStart, cbRange, pvBinary, fShadow, pszDesc);
1513
1514 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1515 return rc;
1516}
1517
1518
1519/** @copydoc PDMDEVHLP::pfnSSMRegister */
1520static DECLCALLBACK(int) pdmR3DevHlp_SSMRegister(PPDMDEVINS pDevIns, const char *pszName, uint32_t u32Instance, uint32_t u32Version, size_t cbGuess,
1521 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
1522 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone)
1523{
1524 PDMDEV_ASSERT_DEVINS(pDevIns);
1525 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1526 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: pszName=%p:{%s} u32Instance=%#x u32Version=#x cbGuess=%#x pfnSavePrep=%p pfnSaveExec=%p pfnSaveDone=%p pszLoadPrep=%p pfnLoadExec=%p pfnLoaddone=%p\n",
1527 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pszName, pszName, u32Instance, u32Version, cbGuess, pfnSavePrep, pfnSaveExec, pfnSaveDone, pfnLoadPrep, pfnLoadExec, pfnLoadDone));
1528
1529 int rc = SSMR3Register(pDevIns->Internal.s.pVMHC, pDevIns, pszName, u32Instance, u32Version, cbGuess,
1530 pfnSavePrep, pfnSaveExec, pfnSaveDone,
1531 pfnLoadPrep, pfnLoadExec, pfnLoadDone);
1532
1533 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1534 return rc;
1535}
1536
1537
1538/** @copydoc PDMDEVHLP::pfnTMTimerCreate */
1539static DECLCALLBACK(int) pdmR3DevHlp_TMTimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback, const char *pszDesc, PPTMTIMERR3 ppTimer)
1540{
1541 PDMDEV_ASSERT_DEVINS(pDevIns);
1542 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1543 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: enmClock=%d pfnCallback=%p pszDesc=%p:{%s} ppTimer=%p\n",
1544 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, enmClock, pfnCallback, pszDesc, pszDesc, ppTimer));
1545
1546 int rc = TMR3TimerCreateDevice(pDevIns->Internal.s.pVMHC, pDevIns, enmClock, pfnCallback, pszDesc, ppTimer);
1547
1548 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1549 return rc;
1550}
1551
1552
1553/** @copydoc PDMDEVHLP::pfnTMTimerCreateExternal */
1554static DECLCALLBACK(PTMTIMERR3) pdmR3DevHlp_TMTimerCreateExternal(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMEREXT pfnCallback, void *pvUser, const char *pszDesc)
1555{
1556 PDMDEV_ASSERT_DEVINS(pDevIns);
1557 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
1558
1559 return TMR3TimerCreateExternal(pDevIns->Internal.s.pVMHC, enmClock, pfnCallback, pvUser, pszDesc);
1560}
1561
1562/** @copydoc PDMDEVHLP::pfnPCIRegister */
1563static DECLCALLBACK(int) pdmR3DevHlp_PCIRegister(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev)
1564{
1565 PDMDEV_ASSERT_DEVINS(pDevIns);
1566 PVM pVM = pDevIns->Internal.s.pVMHC;
1567 VM_ASSERT_EMT(pVM);
1568 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: pPciDev=%p:{.config={%#.256Vhxs}\n",
1569 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pPciDev, pPciDev->config));
1570
1571 /*
1572 * Validate input.
1573 */
1574 if (!pPciDev)
1575 {
1576 Assert(pPciDev);
1577 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc (pPciDev)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1578 return VERR_INVALID_PARAMETER;
1579 }
1580 if (!pPciDev->config[0] && !pPciDev->config[1])
1581 {
1582 Assert(pPciDev->config[0] || pPciDev->config[1]);
1583 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc (vendor)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1584 return VERR_INVALID_PARAMETER;
1585 }
1586 if (pDevIns->Internal.s.pPciDeviceHC)
1587 {
1588 /** @todo the PCI device vs. PDM device designed is a bit flawed if we have to
1589 * support a PDM device with multiple PCI devices. This might become a problem
1590 * when upgrading the chipset for instance...
1591 */
1592 AssertMsgFailed(("Only one PCI device per device is currently implemented!\n"));
1593 return VERR_INTERNAL_ERROR;
1594 }
1595
1596 /*
1597 * Choose the PCI bus for the device.
1598 * This is simple. If the device was configured for a particular bus, it'll
1599 * already have one. If not, we'll just take the first one.
1600 */
1601 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusHC;
1602 if (!pBus)
1603 pBus = pDevIns->Internal.s.pPciBusHC = &pVM->pdm.s.aPciBuses[0];
1604 int rc;
1605 if (pBus)
1606 {
1607 if (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC)
1608 pDevIns->Internal.s.pPciBusGC = MMHyperHC2GC(pVM, pDevIns->Internal.s.pPciBusHC);
1609
1610 /*
1611 * Check the configuration for PCI device and function assignment.
1612 */
1613 int iDev = -1;
1614 uint8_t u8Device;
1615 rc = CFGMR3QueryU8(pDevIns->Internal.s.pCfgHandle, "PCIDeviceNo", &u8Device);
1616 if (VBOX_SUCCESS(rc))
1617 {
1618 if (u8Device > 31)
1619 {
1620 AssertMsgFailed(("Configuration error: PCIDeviceNo=%d, max is 31. (%s/%d)\n",
1621 u8Device, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1622 return VERR_INTERNAL_ERROR;
1623 }
1624
1625 uint8_t u8Function;
1626 rc = CFGMR3QueryU8(pDevIns->Internal.s.pCfgHandle, "PCIFunctionNo", &u8Function);
1627 if (VBOX_FAILURE(rc))
1628 {
1629 AssertMsgFailed(("Configuration error: PCIDeviceNo, but PCIFunctionNo query failed with rc=%Vrc (%s/%d)\n",
1630 rc, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1631 return rc;
1632 }
1633 if (u8Function > 7)
1634 {
1635 AssertMsgFailed(("Configuration error: PCIFunctionNo=%d, max is 7. (%s/%d)\n",
1636 u8Function, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1637 return VERR_INTERNAL_ERROR;
1638 }
1639 iDev = (u8Device << 3) | u8Function;
1640 }
1641 else if (rc != VERR_CFGM_VALUE_NOT_FOUND)
1642 {
1643 AssertMsgFailed(("Configuration error: PCIDeviceNo query failed with rc=%Vrc (%s/%d)\n",
1644 rc, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1645 return rc;
1646 }
1647
1648 /*
1649 * Call the pci bus device to do the actual registration.
1650 */
1651 pdmLock(pVM);
1652 rc = pBus->pfnRegisterR3(pBus->pDevInsR3, pPciDev, pDevIns->pDevReg->szDeviceName, iDev);
1653 pdmUnlock(pVM);
1654 if (VBOX_SUCCESS(rc))
1655 {
1656 pDevIns->Internal.s.pPciDeviceHC = pPciDev;
1657 if (pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC)
1658 pDevIns->Internal.s.pPciDeviceGC = MMHyperHC2GC(pVM, pPciDev);
1659 else
1660 pDevIns->Internal.s.pPciDeviceGC = 0;
1661 pPciDev->pDevIns = pDevIns;
1662 Log(("PDM: Registered device '%s'/%d as PCI device %d on bus %d\n",
1663 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pPciDev->devfn, pDevIns->Internal.s.pPciBusHC->iBus));
1664 }
1665 }
1666 else
1667 {
1668 AssertMsgFailed(("Configuration error: No PCI bus available. This could be related to init order too!\n"));
1669 rc = VERR_PDM_NO_PCI_BUS;
1670 }
1671
1672 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1673 return rc;
1674}
1675
1676
1677/** @copydoc PDMDEVHLP::pfnPCIIORegionRegister */
1678static DECLCALLBACK(int) pdmR3DevHlp_PCIIORegionRegister(PPDMDEVINS pDevIns, int iRegion, uint32_t cbRegion, PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback)
1679{
1680 PDMDEV_ASSERT_DEVINS(pDevIns);
1681 PVM pVM = pDevIns->Internal.s.pVMHC;
1682 VM_ASSERT_EMT(pVM);
1683 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: iRegion=%d cbRegion=%#x enmType=%d pfnCallback=%p\n",
1684 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion, cbRegion, enmType, pfnCallback));
1685
1686 /*
1687 * Validate input.
1688 */
1689 if (iRegion < 0 || iRegion >= PCI_NUM_REGIONS)
1690 {
1691 Assert(iRegion >= 0 && iRegion < PCI_NUM_REGIONS);
1692 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Vrc (iRegion)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1693 return VERR_INVALID_PARAMETER;
1694 }
1695 switch (enmType)
1696 {
1697 case PCI_ADDRESS_SPACE_MEM:
1698 case PCI_ADDRESS_SPACE_IO:
1699 case PCI_ADDRESS_SPACE_MEM_PREFETCH:
1700 break;
1701 default:
1702 AssertMsgFailed(("enmType=%#x is unknown\n", enmType));
1703 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Vrc (enmType)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1704 return VERR_INVALID_PARAMETER;
1705 }
1706 if (!pfnCallback)
1707 {
1708 Assert(pfnCallback);
1709 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Vrc (callback)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1710 return VERR_INVALID_PARAMETER;
1711 }
1712 AssertRelease(VMR3GetState(pVM) != VMSTATE_RUNNING);
1713
1714 /*
1715 * Must have a PCI device registered!
1716 */
1717 int rc;
1718 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceHC;
1719 if (pPciDev)
1720 {
1721 /*
1722 * We're currently restricted to page aligned MMIO regions.
1723 */
1724 if ( (enmType == PCI_ADDRESS_SPACE_MEM || enmType == PCI_ADDRESS_SPACE_MEM_PREFETCH)
1725 && cbRegion != RT_ALIGN_32(cbRegion, PAGE_SIZE))
1726 {
1727 Log(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: aligning cbRegion %#x -> %#x\n",
1728 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, cbRegion, RT_ALIGN_32(cbRegion, PAGE_SIZE)));
1729 cbRegion = RT_ALIGN_32(cbRegion, PAGE_SIZE);
1730 }
1731
1732 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusHC;
1733 Assert(pBus);
1734 pdmLock(pVM);
1735 rc = pBus->pfnIORegionRegisterR3(pBus->pDevInsR3, pPciDev, iRegion, cbRegion, enmType, pfnCallback);
1736 pdmUnlock(pVM);
1737 }
1738 else
1739 {
1740 AssertMsgFailed(("No PCI device registered!\n"));
1741 rc = VERR_PDM_NOT_PCI_DEVICE;
1742 }
1743
1744 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
1745 return rc;
1746}
1747
1748
1749/** @copydoc PDMDEVHLP::pfnPCISetConfigCallbacks */
1750static DECLCALLBACK(void) pdmR3DevHlp_PCISetConfigCallbacks(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
1751 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld)
1752{
1753 PDMDEV_ASSERT_DEVINS(pDevIns);
1754 PVM pVM = pDevIns->Internal.s.pVMHC;
1755 VM_ASSERT_EMT(pVM);
1756 LogFlow(("pdmR3DevHlp_PCISetConfigCallbacks: caller='%s'/%d: pPciDev=%p pfnRead=%p ppfnReadOld=%p pfnWrite=%p ppfnWriteOld=%p\n",
1757 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pPciDev, pfnRead, ppfnReadOld, pfnWrite, ppfnWriteOld));
1758
1759 /*
1760 * Validate input and resolve defaults.
1761 */
1762 AssertPtr(pfnRead);
1763 AssertPtr(pfnWrite);
1764 AssertPtrNull(ppfnReadOld);
1765 AssertPtrNull(ppfnWriteOld);
1766 AssertPtrNull(pPciDev);
1767
1768 if (!pPciDev)
1769 pPciDev = pDevIns->Internal.s.pPciDeviceHC;
1770 AssertReleaseMsg(pPciDev, ("You must register your device first!\n"));
1771 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusHC;
1772 AssertRelease(pBus);
1773 AssertRelease(VMR3GetState(pVM) != VMSTATE_RUNNING);
1774
1775 /*
1776 * Do the job.
1777 */
1778 pdmLock(pVM);
1779 pBus->pfnSetConfigCallbacksR3(pBus->pDevInsR3, pPciDev, pfnRead, ppfnReadOld, pfnWrite, ppfnWriteOld);
1780 pdmUnlock(pVM);
1781
1782 LogFlow(("pdmR3DevHlp_PCISetConfigCallbacks: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1783}
1784
1785
1786/** @copydoc PDMDEVHLP::pfnPCISetIrq */
1787static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1788{
1789 PDMDEV_ASSERT_DEVINS(pDevIns);
1790 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iIrq, iLevel));
1791
1792 /*
1793 * Validate input.
1794 */
1795 /** @todo iIrq and iLevel checks. */
1796
1797 /*
1798 * Must have a PCI device registered!
1799 */
1800 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceHC;
1801 if (pPciDev)
1802 {
1803 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusHC; /** @todo the bus should be associated with the PCI device not the PDM device. */
1804 Assert(pBus);
1805 PVM pVM = pDevIns->Internal.s.pVMHC;
1806 pdmLock(pVM);
1807 pBus->pfnSetIrqR3(pBus->pDevInsR3, pPciDev, iIrq, iLevel);
1808 pdmUnlock(pVM);
1809 }
1810 else
1811 AssertReleaseMsgFailed(("No PCI device registered!\n"));
1812
1813 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1814}
1815
1816
1817/** @copydoc PDMDEVHLP::pfnPCISetIrqNoWait */
1818static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1819{
1820 pdmR3DevHlp_PCISetIrq(pDevIns, iIrq, iLevel);
1821}
1822
1823
1824/** @copydoc PDMDEVHLP::pfnISASetIrq */
1825static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1826{
1827 PDMDEV_ASSERT_DEVINS(pDevIns);
1828 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iIrq, iLevel));
1829
1830 /*
1831 * Validate input.
1832 */
1833 /** @todo iIrq and iLevel checks. */
1834
1835 PVM pVM = pDevIns->Internal.s.pVMHC;
1836 PDMIsaSetIrq(pVM, iIrq, iLevel); /* (The API takes the lock.) */
1837
1838 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1839}
1840
1841
1842/** @copydoc PDMDEVHLP::pfnISASetIrqNoWait */
1843static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1844{
1845 pdmR3DevHlp_ISASetIrq(pDevIns, iIrq, iLevel);
1846}
1847
1848
1849/** @copydoc PDMDEVHLP::pfnDriverAttach */
1850static DECLCALLBACK(int) pdmR3DevHlp_DriverAttach(PPDMDEVINS pDevIns, RTUINT iLun, PPDMIBASE pBaseInterface, PPDMIBASE *ppBaseInterface, const char *pszDesc)
1851{
1852 PDMDEV_ASSERT_DEVINS(pDevIns);
1853 PVM pVM = pDevIns->Internal.s.pVMHC;
1854 VM_ASSERT_EMT(pVM);
1855 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: iLun=%d pBaseInterface=%p ppBaseInterface=%p pszDesc=%p:{%s}\n",
1856 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iLun, pBaseInterface, ppBaseInterface, pszDesc, pszDesc));
1857
1858 /*
1859 * Lookup the LUN, it might already be registered.
1860 */
1861 PPDMLUN pLunPrev = NULL;
1862 PPDMLUN pLun = pDevIns->Internal.s.pLunsHC;
1863 for (; pLun; pLunPrev = pLun, pLun = pLun->pNext)
1864 if (pLun->iLun == iLun)
1865 break;
1866
1867 /*
1868 * Create the LUN if if wasn't found, else check if driver is already attached to it.
1869 */
1870 if (!pLun)
1871 {
1872 if ( !pBaseInterface
1873 || !pszDesc
1874 || !*pszDesc)
1875 {
1876 Assert(pBaseInterface);
1877 Assert(pszDesc || *pszDesc);
1878 return VERR_INVALID_PARAMETER;
1879 }
1880
1881 pLun = (PPDMLUN)MMR3HeapAlloc(pVM, MM_TAG_PDM_LUN, sizeof(*pLun));
1882 if (!pLun)
1883 return VERR_NO_MEMORY;
1884
1885 pLun->iLun = iLun;
1886 pLun->pNext = pLunPrev ? pLunPrev->pNext : NULL;
1887 pLun->pTop = NULL;
1888 pLun->pBottom = NULL;
1889 pLun->pDevIns = pDevIns;
1890 pLun->pszDesc = pszDesc;
1891 pLun->pBase = pBaseInterface;
1892 if (!pLunPrev)
1893 pDevIns->Internal.s.pLunsHC = pLun;
1894 else
1895 pLunPrev->pNext = pLun;
1896 Log(("pdmR3DevHlp_DriverAttach: Registered LUN#%d '%s' with device '%s'/%d.\n",
1897 iLun, pszDesc, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1898 }
1899 else if (pLun->pTop)
1900 {
1901 AssertMsgFailed(("Already attached! The device should keep track of such things!\n"));
1902 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_PDM_DRIVER_ALREADY_ATTACHED));
1903 return VERR_PDM_DRIVER_ALREADY_ATTACHED;
1904 }
1905 Assert(pLun->pBase == pBaseInterface);
1906
1907
1908 /*
1909 * Get the attached driver configuration.
1910 */
1911 int rc;
1912 char szNode[48];
1913 RTStrPrintf(szNode, sizeof(szNode), "LUN#%d", iLun);
1914 PCFGMNODE pNode = CFGMR3GetChild(pDevIns->Internal.s.pCfgHandle, szNode);
1915 if (pNode)
1916 {
1917 char *pszName;
1918 rc = CFGMR3QueryStringAlloc(pNode, "Driver", &pszName);
1919 if (VBOX_SUCCESS(rc))
1920 {
1921 /*
1922 * Find the driver.
1923 */
1924 PPDMDRV pDrv = pdmR3DrvLookup(pVM, pszName);
1925 if (pDrv)
1926 {
1927 /* config node */
1928 PCFGMNODE pConfigNode = CFGMR3GetChild(pNode, "Config");
1929 if (!pConfigNode)
1930 rc = CFGMR3InsertNode(pNode, "Config", &pConfigNode);
1931 if (VBOX_SUCCESS(rc))
1932 {
1933 CFGMR3SetRestrictedRoot(pConfigNode);
1934
1935 /*
1936 * Allocate the driver instance.
1937 */
1938 size_t cb = RT_OFFSETOF(PDMDRVINS, achInstanceData[pDrv->pDrvReg->cbInstance]);
1939 cb = RT_ALIGN_Z(cb, 16);
1940 PPDMDRVINS pNew = (PPDMDRVINS)MMR3HeapAllocZ(pVM, MM_TAG_PDM_DRIVER, cb);
1941 if (pNew)
1942 {
1943 /*
1944 * Initialize the instance structure (declaration order).
1945 */
1946 pNew->u32Version = PDM_DRVINS_VERSION;
1947 //pNew->Internal.s.pUp = NULL;
1948 //pNew->Internal.s.pDown = NULL;
1949 pNew->Internal.s.pLun = pLun;
1950 pNew->Internal.s.pDrv = pDrv;
1951 pNew->Internal.s.pVM = pVM;
1952 //pNew->Internal.s.fDetaching = false;
1953 pNew->Internal.s.pCfgHandle = pNode;
1954 pNew->pDrvHlp = &g_pdmR3DrvHlp;
1955 pNew->pDrvReg = pDrv->pDrvReg;
1956 pNew->pCfgHandle = pConfigNode;
1957 pNew->iInstance = pDrv->cInstances++;
1958 pNew->pUpBase = pBaseInterface;
1959 //pNew->pDownBase = NULL;
1960 //pNew->IBase.pfnQueryInterface = NULL;
1961 pNew->pvInstanceData = &pNew->achInstanceData[0];
1962
1963 /*
1964 * Link with LUN and call the constructor.
1965 */
1966 pLun->pTop = pLun->pBottom = pNew;
1967 rc = pDrv->pDrvReg->pfnConstruct(pNew, pNew->pCfgHandle);
1968 if (VBOX_SUCCESS(rc))
1969 {
1970 MMR3HeapFree(pszName);
1971 *ppBaseInterface = &pNew->IBase;
1972 Log(("PDM: Attached driver '%s'/%d to LUN#%d on device '%s'/%d.\n",
1973 pDrv->pDrvReg->szDriverName, pNew->iInstance, iLun, pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1974 LogFlow(("pdmR3DevHlp_DriverAttach: caller '%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VINF_SUCCESS));
1975 /*
1976 * Might return != VINF_SUCCESS (e.g. VINF_NAT_DNS) */
1977 return rc;
1978 }
1979
1980 /*
1981 * Free the driver.
1982 */
1983 pLun->pTop = pLun->pBottom = NULL;
1984 ASMMemFill32(pNew, cb, 0xdeadd0d0);
1985 MMR3HeapFree(pNew);
1986 pDrv->cInstances--;
1987 }
1988 else
1989 {
1990 AssertMsgFailed(("Failed to allocate %d bytes for instantiating driver '%s'\n", cb, pszName));
1991 rc = VERR_NO_MEMORY;
1992 }
1993 }
1994 else
1995 AssertMsgFailed(("Failed to create Config node! rc=%Vrc\n", rc));
1996 }
1997 else
1998 {
1999 AssertMsgFailed(("Driver '%s' wasn't found!\n", pszName));
2000 rc = VERR_PDM_DRIVER_NOT_FOUND;
2001 }
2002 MMR3HeapFree(pszName);
2003 }
2004 else
2005 {
2006 AssertMsgFailed(("Query for string value of \"Driver\" -> %Vrc\n", rc));
2007 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
2008 rc = VERR_PDM_CFG_MISSING_DRIVER_NAME;
2009 }
2010 }
2011 else
2012 rc = VERR_PDM_NO_ATTACHED_DRIVER;
2013
2014
2015 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2016 return rc;
2017}
2018
2019
2020/** @copydoc PDMDEVHLP::pfnMMHeapAlloc */
2021static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAlloc(PPDMDEVINS pDevIns, size_t cb)
2022{
2023 PDMDEV_ASSERT_DEVINS(pDevIns);
2024 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: cb=%#x\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, cb));
2025
2026 void *pv = MMR3HeapAlloc(pDevIns->Internal.s.pVMHC, MM_TAG_PDM_DEVICE_USER, cb);
2027
2028 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns %p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pv));
2029 return pv;
2030}
2031
2032
2033/** @copydoc PDMDEVHLP::pfnMMHeapAllocZ */
2034static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAllocZ(PPDMDEVINS pDevIns, size_t cb)
2035{
2036 PDMDEV_ASSERT_DEVINS(pDevIns);
2037 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: cb=%#x\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, cb));
2038
2039 void *pv = MMR3HeapAllocZ(pDevIns->Internal.s.pVMHC, MM_TAG_PDM_DEVICE_USER, cb);
2040
2041 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: returns %p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pv));
2042 return pv;
2043}
2044
2045
2046/** @copydoc PDMDEVHLP::pfnMMHeapFree */
2047static DECLCALLBACK(void) pdmR3DevHlp_MMHeapFree(PPDMDEVINS pDevIns, void *pv)
2048{
2049 PDMDEV_ASSERT_DEVINS(pDevIns);
2050 LogFlow(("pdmR3DevHlp_MMHeapFree: caller='%s'/%d: pv=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pv));
2051
2052 MMR3HeapFree(pv);
2053
2054 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
2055}
2056
2057
2058/** @copydoc PDMDEVHLP::pfnVMSetError */
2059static DECLCALLBACK(int) pdmR3DevHlp_VMSetError(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
2060{
2061 PDMDEV_ASSERT_DEVINS(pDevIns);
2062 va_list args;
2063 va_start(args, pszFormat);
2064 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMHC, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc2 == rc); NOREF(rc2);
2065 va_end(args);
2066 return rc;
2067}
2068
2069
2070/** @copydoc PDMDEVHLP::pfnVMSetErrorV */
2071static DECLCALLBACK(int) pdmR3DevHlp_VMSetErrorV(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
2072{
2073 PDMDEV_ASSERT_DEVINS(pDevIns);
2074 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMHC, rc, RT_SRC_POS_ARGS, pszFormat, va); Assert(rc2 == rc); NOREF(rc2);
2075 return rc;
2076}
2077
2078
2079/** @copydoc PDMDEVHLP::pfnVMSetRuntimeError */
2080static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeError(PPDMDEVINS pDevIns, bool fFatal, const char *pszErrorID, const char *pszFormat, ...)
2081{
2082 PDMDEV_ASSERT_DEVINS(pDevIns);
2083 va_list args;
2084 va_start(args, pszFormat);
2085 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMHC, fFatal, pszErrorID, pszFormat, args);
2086 va_end(args);
2087 return rc;
2088}
2089
2090
2091/** @copydoc PDMDEVHLP::pfnVMSetRuntimeErrorV */
2092static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeErrorV(PPDMDEVINS pDevIns, bool fFatal, const char *pszErrorID, const char *pszFormat, va_list va)
2093{
2094 PDMDEV_ASSERT_DEVINS(pDevIns);
2095 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMHC, fFatal, pszErrorID, pszFormat, va);
2096 return rc;
2097}
2098
2099
2100/** @copydoc PDMDEVHLP::pfnAssertEMT */
2101static DECLCALLBACK(bool) pdmR3DevHlp_AssertEMT(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
2102{
2103 PDMDEV_ASSERT_DEVINS(pDevIns);
2104 if (VM_IS_EMT(pDevIns->Internal.s.pVMHC))
2105 return true;
2106
2107 char szMsg[100];
2108 RTStrPrintf(szMsg, sizeof(szMsg), "AssertEMT '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance);
2109 AssertMsg1(szMsg, iLine, pszFile, pszFunction);
2110 AssertBreakpoint();
2111 return false;
2112}
2113
2114
2115/** @copydoc PDMDEVHLP::pfnAssertOther */
2116static DECLCALLBACK(bool) pdmR3DevHlp_AssertOther(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
2117{
2118 PDMDEV_ASSERT_DEVINS(pDevIns);
2119 if (!VM_IS_EMT(pDevIns->Internal.s.pVMHC))
2120 return true;
2121
2122 char szMsg[100];
2123 RTStrPrintf(szMsg, sizeof(szMsg), "AssertOther '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance);
2124 AssertMsg1(szMsg, iLine, pszFile, pszFunction);
2125 AssertBreakpoint();
2126 return false;
2127}
2128
2129
2130/** @copydoc PDMDEVHLP::pfnDBGFStopV */
2131static DECLCALLBACK(int) pdmR3DevHlp_DBGFStopV(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction, const char *pszFormat, va_list args)
2132{
2133 PDMDEV_ASSERT_DEVINS(pDevIns);
2134#ifdef LOG_ENABLED
2135 va_list va2;
2136 va_copy(va2, args);
2137 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: pszFile=%p:{%s} iLine=%d pszFunction=%p:{%s} pszFormat=%p:{%s} (%N)\n",
2138 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pszFile, pszFile, iLine, pszFunction, pszFunction, pszFormat, pszFormat, pszFormat, &va2));
2139 va_end(va2);
2140#endif
2141
2142 PVM pVM = pDevIns->Internal.s.pVMHC;
2143 VM_ASSERT_EMT(pVM);
2144 int rc = DBGFR3EventSrcV(pVM, DBGFEVENT_DEV_STOP, pszFile, iLine, pszFunction, pszFormat, args);
2145
2146 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2147 return rc;
2148}
2149
2150
2151/** @copydoc PDMDEVHLP::pfnDBGFInfoRegister */
2152static DECLCALLBACK(int) pdmR3DevHlp_DBGFInfoRegister(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler)
2153{
2154 PDMDEV_ASSERT_DEVINS(pDevIns);
2155 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: pszName=%p:{%s} pszDesc=%p:{%s} pfnHandler=%p\n",
2156 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pszName, pszName, pszDesc, pszDesc, pfnHandler));
2157
2158 PVM pVM = pDevIns->Internal.s.pVMHC;
2159 VM_ASSERT_EMT(pVM);
2160 int rc = DBGFR3InfoRegisterDevice(pVM, pszName, pszDesc, pfnHandler, pDevIns);
2161
2162 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2163 return rc;
2164}
2165
2166
2167/** @copydoc PDMDEVHLP::pfnSTAMRegister */
2168static DECLCALLBACK(void) pdmR3DevHlp_STAMRegister(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName, STAMUNIT enmUnit, const char *pszDesc)
2169{
2170 PDMDEV_ASSERT_DEVINS(pDevIns);
2171 PVM pVM = pDevIns->Internal.s.pVMHC;
2172 VM_ASSERT_EMT(pVM);
2173
2174 STAM_REG(pVM, pvSample, enmType, pszName, enmUnit, pszDesc);
2175 NOREF(pVM);
2176}
2177
2178
2179
2180/** @copydoc PDMDEVHLP::pfnSTAMRegisterF */
2181static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterF(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility,
2182 STAMUNIT enmUnit, const char *pszDesc, const char *pszName, ...)
2183{
2184 PDMDEV_ASSERT_DEVINS(pDevIns);
2185 PVM pVM = pDevIns->Internal.s.pVMHC;
2186 VM_ASSERT_EMT(pVM);
2187
2188 va_list args;
2189 va_start(args, pszName);
2190 int rc = STAMR3RegisterV(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
2191 va_end(args);
2192 AssertRC(rc);
2193
2194 NOREF(pVM);
2195}
2196
2197
2198/** @copydoc PDMDEVHLP::pfnSTAMRegisterV */
2199static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterV(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility,
2200 STAMUNIT enmUnit, const char *pszDesc, const char *pszName, va_list args)
2201{
2202 PDMDEV_ASSERT_DEVINS(pDevIns);
2203 PVM pVM = pDevIns->Internal.s.pVMHC;
2204 VM_ASSERT_EMT(pVM);
2205
2206 int rc = STAMR3RegisterV(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
2207 AssertRC(rc);
2208
2209 NOREF(pVM);
2210}
2211
2212
2213/** @copydoc PDMDEVHLP::pfnRTCRegister */
2214static DECLCALLBACK(int) pdmR3DevHlp_RTCRegister(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp)
2215{
2216 PDMDEV_ASSERT_DEVINS(pDevIns);
2217 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
2218 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: pRtcReg=%p:{.u32Version=%#x, .pfnWrite=%p, .pfnRead=%p} ppRtcHlp=%p\n",
2219 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pRtcReg, pRtcReg->u32Version, pRtcReg->pfnWrite,
2220 pRtcReg->pfnWrite, ppRtcHlp));
2221
2222 /*
2223 * Validate input.
2224 */
2225 if (pRtcReg->u32Version != PDM_RTCREG_VERSION)
2226 {
2227 AssertMsgFailed(("u32Version=%#x expected %#x\n", pRtcReg->u32Version,
2228 PDM_RTCREG_VERSION));
2229 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Vrc (version)\n",
2230 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2231 return VERR_INVALID_PARAMETER;
2232 }
2233 if ( !pRtcReg->pfnWrite
2234 || !pRtcReg->pfnRead)
2235 {
2236 Assert(pRtcReg->pfnWrite);
2237 Assert(pRtcReg->pfnRead);
2238 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Vrc (callbacks)\n",
2239 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2240 return VERR_INVALID_PARAMETER;
2241 }
2242
2243 if (!ppRtcHlp)
2244 {
2245 Assert(ppRtcHlp);
2246 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Vrc (ppRtcHlp)\n",
2247 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2248 return VERR_INVALID_PARAMETER;
2249 }
2250
2251 /*
2252 * Only one DMA device.
2253 */
2254 PVM pVM = pDevIns->Internal.s.pVMHC;
2255 if (pVM->pdm.s.pRtc)
2256 {
2257 AssertMsgFailed(("Only one RTC device is supported!\n"));
2258 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Vrc\n",
2259 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2260 return VERR_INVALID_PARAMETER;
2261 }
2262
2263 /*
2264 * Allocate and initialize pci bus structure.
2265 */
2266 int rc = VINF_SUCCESS;
2267 PPDMRTC pRtc = (PPDMRTC)MMR3HeapAlloc(pDevIns->Internal.s.pVMHC, MM_TAG_PDM_DEVICE, sizeof(*pRtc));
2268 if (pRtc)
2269 {
2270 pRtc->pDevIns = pDevIns;
2271 pRtc->Reg = *pRtcReg;
2272 pVM->pdm.s.pRtc = pRtc;
2273
2274 /* set the helper pointer. */
2275 *ppRtcHlp = &g_pdmR3DevRtcHlp;
2276 Log(("PDM: Registered RTC device '%s'/%d pDevIns=%p\n",
2277 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
2278 }
2279 else
2280 rc = VERR_NO_MEMORY;
2281
2282 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Vrc\n",
2283 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2284 return rc;
2285}
2286
2287
2288/** @copydoc PDMDEVHLP::pfnPDMQueueCreate */
2289static DECLCALLBACK(int) pdmR3DevHlp_PDMQueueCreate(PPDMDEVINS pDevIns, RTUINT cbItem, RTUINT cItems, uint32_t cMilliesInterval,
2290 PFNPDMQUEUEDEV pfnCallback, bool fGCEnabled, PPDMQUEUE *ppQueue)
2291{
2292 PDMDEV_ASSERT_DEVINS(pDevIns);
2293 LogFlow(("pdmR3DevHlp_PDMQueueCreate: caller='%s'/%d: cbItem=%#x cItems=%#x cMilliesInterval=%u pfnCallback=%p fGCEnabled=%RTbool ppQueue=%p\n",
2294 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, cbItem, cItems, cMilliesInterval, pfnCallback, fGCEnabled, ppQueue));
2295
2296 PVM pVM = pDevIns->Internal.s.pVMHC;
2297 VM_ASSERT_EMT(pVM);
2298 int rc = PDMR3QueueCreateDevice(pVM, pDevIns, cbItem, cItems, cMilliesInterval, pfnCallback, fGCEnabled, ppQueue);
2299
2300 LogFlow(("pdmR3DevHlp_PDMQueueCreate: caller='%s'/%d: returns %Vrc *ppQueue=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc, *ppQueue));
2301 return rc;
2302}
2303
2304
2305/** @copydoc PDMDEVHLP::pfnCritSectInit */
2306static DECLCALLBACK(int) pdmR3DevHlp_CritSectInit(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, const char *pszName)
2307{
2308 PDMDEV_ASSERT_DEVINS(pDevIns);
2309 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: pCritSect=%p pszName=%p:{%s}\n",
2310 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pCritSect, pszName, pszName));
2311
2312 PVM pVM = pDevIns->Internal.s.pVMHC;
2313 VM_ASSERT_EMT(pVM);
2314 int rc = pdmR3CritSectInitDevice(pVM, pDevIns, pCritSect, pszName);
2315
2316 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2317 return rc;
2318}
2319
2320
2321/** @copydoc PDMDEVHLP::pfnUTCNow */
2322static DECLCALLBACK(PRTTIMESPEC) pdmR3DevHlp_UTCNow(PPDMDEVINS pDevIns, PRTTIMESPEC pTime)
2323{
2324 PDMDEV_ASSERT_DEVINS(pDevIns);
2325 LogFlow(("pdmR3DevHlp_UTCNow: caller='%s'/%d: pTime=%p\n",
2326 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pTime));
2327
2328 pTime = TMR3UTCNow(pDevIns->Internal.s.pVMHC, pTime);
2329
2330 LogFlow(("pdmR3DevHlp_UTCNow: caller='%s'/%d: returns %RU64\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, RTTimeSpecGetNano(pTime)));
2331 return pTime;
2332}
2333
2334
2335/** @copydoc PDMDEVHLP::pfnPDMThreadCreate */
2336static DECLCALLBACK(int) pdmR3DevHlp_PDMThreadCreate(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
2337 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName)
2338{
2339 PDMDEV_ASSERT_DEVINS(pDevIns);
2340 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
2341 LogFlow(("pdmR3DevHlp_PDMThreadCreate: caller='%s'/%d: ppThread=%p pvUser=%p pfnThread=%p pfnWakeup=%p cbStack=%#zx enmType=%d pszName=%p:{%s}\n",
2342 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName, pszName));
2343
2344 int rc = pdmR3ThreadCreateDevice(pDevIns->Internal.s.pVMHC, pDevIns, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName);
2345
2346 LogFlow(("pdmR3DevHlp_PDMThreadCreate: caller='%s'/%d: returns %Vrc *ppThread=%RTthrd\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
2347 rc, *ppThread));
2348 return rc;
2349}
2350
2351
2352/** @copydoc PDMDEVHLP::pfnGetVM */
2353static DECLCALLBACK(PVM) pdmR3DevHlp_GetVM(PPDMDEVINS pDevIns)
2354{
2355 PDMDEV_ASSERT_DEVINS(pDevIns);
2356 LogFlow(("pdmR3DevHlp_GetVM: caller='%s'/%d: returns %p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns->Internal.s.pVMHC));
2357 return pDevIns->Internal.s.pVMHC;
2358}
2359
2360
2361/** @copydoc PDMDEVHLP::pfnPCIBusRegister */
2362static DECLCALLBACK(int) pdmR3DevHlp_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3)
2363{
2364 PDMDEV_ASSERT_DEVINS(pDevIns);
2365 PVM pVM = pDevIns->Internal.s.pVMHC;
2366 VM_ASSERT_EMT(pVM);
2367 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: pPciBusReg=%p:{.u32Version=%#x, .pfnRegisterR3=%p, .pfnIORegionRegisterR3=%p, .pfnSetIrqR3=%p, "
2368 ".pfnSaveExecR3=%p, .pfnLoadExecR3=%p, .pfnFakePCIBIOSR3=%p, .pszSetIrqRC=%p:{%s}, .pszSetIrqR0=%p:{%s}} ppPciHlpR3=%p\n",
2369 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pPciBusReg, pPciBusReg->u32Version, pPciBusReg->pfnRegisterR3,
2370 pPciBusReg->pfnIORegionRegisterR3, pPciBusReg->pfnSetIrqR3, pPciBusReg->pfnSaveExecR3, pPciBusReg->pfnLoadExecR3,
2371 pPciBusReg->pfnFakePCIBIOSR3, pPciBusReg->pszSetIrqRC, pPciBusReg->pszSetIrqRC, pPciBusReg->pszSetIrqR0, pPciBusReg->pszSetIrqR0, ppPciHlpR3));
2372
2373 /*
2374 * Validate the structure.
2375 */
2376 if (pPciBusReg->u32Version != PDM_PCIBUSREG_VERSION)
2377 {
2378 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPciBusReg->u32Version, PDM_PCIBUSREG_VERSION));
2379 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc (version)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2380 return VERR_INVALID_PARAMETER;
2381 }
2382 if ( !pPciBusReg->pfnRegisterR3
2383 || !pPciBusReg->pfnIORegionRegisterR3
2384 || !pPciBusReg->pfnSetIrqR3
2385 || !pPciBusReg->pfnSaveExecR3
2386 || !pPciBusReg->pfnLoadExecR3
2387 || !pPciBusReg->pfnFakePCIBIOSR3)
2388 {
2389 Assert(pPciBusReg->pfnRegisterR3);
2390 Assert(pPciBusReg->pfnIORegionRegisterR3);
2391 Assert(pPciBusReg->pfnSetIrqR3);
2392 Assert(pPciBusReg->pfnSaveExecR3);
2393 Assert(pPciBusReg->pfnLoadExecR3);
2394 Assert(pPciBusReg->pfnFakePCIBIOSR3);
2395 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc (R3 callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2396 return VERR_INVALID_PARAMETER;
2397 }
2398 if ( pPciBusReg->pszSetIrqRC
2399 && !VALID_PTR(pPciBusReg->pszSetIrqRC))
2400 {
2401 Assert(VALID_PTR(pPciBusReg->pszSetIrqRC));
2402 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc (GC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2403 return VERR_INVALID_PARAMETER;
2404 }
2405 if ( pPciBusReg->pszSetIrqR0
2406 && !VALID_PTR(pPciBusReg->pszSetIrqR0))
2407 {
2408 Assert(VALID_PTR(pPciBusReg->pszSetIrqR0));
2409 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc (GC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2410 return VERR_INVALID_PARAMETER;
2411 }
2412 if (!ppPciHlpR3)
2413 {
2414 Assert(ppPciHlpR3);
2415 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc (ppPciHlpR3)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2416 return VERR_INVALID_PARAMETER;
2417 }
2418
2419 /*
2420 * Find free PCI bus entry.
2421 */
2422 unsigned iBus = 0;
2423 for (iBus = 0; iBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses); iBus++)
2424 if (!pVM->pdm.s.aPciBuses[iBus].pDevInsR3)
2425 break;
2426 if (iBus >= RT_ELEMENTS(pVM->pdm.s.aPciBuses))
2427 {
2428 AssertMsgFailed(("Too many PCI buses. Max=%u\n", RT_ELEMENTS(pVM->pdm.s.aPciBuses)));
2429 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc (pci bus)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2430 return VERR_INVALID_PARAMETER;
2431 }
2432 PPDMPCIBUS pPciBus = &pVM->pdm.s.aPciBuses[iBus];
2433
2434 /*
2435 * Resolve and init the RC bits.
2436 */
2437 if (pPciBusReg->pszSetIrqRC)
2438 {
2439 int rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pPciBusReg->pszSetIrqRC, &pPciBus->pfnSetIrqRC);
2440 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pPciBusReg->pszSetIrqRC, rc));
2441 if (VBOX_FAILURE(rc))
2442 {
2443 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2444 return rc;
2445 }
2446 pPciBus->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2447 }
2448 else
2449 {
2450 pPciBus->pfnSetIrqRC = 0;
2451 pPciBus->pDevInsRC = 0;
2452 }
2453
2454 /*
2455 * Resolve and init the R0 bits.
2456 */
2457 if (pPciBusReg->pszSetIrqR0)
2458 {
2459 int rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pPciBusReg->pszSetIrqR0, &pPciBus->pfnSetIrqR0);
2460 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pPciBusReg->pszSetIrqR0, rc));
2461 if (VBOX_FAILURE(rc))
2462 {
2463 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2464 return rc;
2465 }
2466 pPciBus->pDevInsR0 = MMHyperR3ToR0(pVM, pDevIns);
2467 }
2468 else
2469 {
2470 pPciBus->pfnSetIrqR0 = 0;
2471 pPciBus->pDevInsR0 = 0;
2472 }
2473
2474 /*
2475 * Init the R3 bits.
2476 */
2477 pPciBus->iBus = iBus;
2478 pPciBus->pDevInsR3 = pDevIns;
2479 pPciBus->pfnRegisterR3 = pPciBusReg->pfnRegisterR3;
2480 pPciBus->pfnIORegionRegisterR3 = pPciBusReg->pfnIORegionRegisterR3;
2481 pPciBus->pfnSetConfigCallbacksR3 = pPciBusReg->pfnSetConfigCallbacksR3;
2482 pPciBus->pfnSetIrqR3 = pPciBusReg->pfnSetIrqR3;
2483 pPciBus->pfnSaveExecR3 = pPciBusReg->pfnSaveExecR3;
2484 pPciBus->pfnLoadExecR3 = pPciBusReg->pfnLoadExecR3;
2485 pPciBus->pfnFakePCIBIOSR3 = pPciBusReg->pfnFakePCIBIOSR3;
2486
2487 Log(("PDM: Registered PCI bus device '%s'/%d pDevIns=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
2488
2489 /* set the helper pointer and return. */
2490 *ppPciHlpR3 = &g_pdmR3DevPciHlp;
2491 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VINF_SUCCESS));
2492 return VINF_SUCCESS;
2493}
2494
2495
2496/** @copydoc PDMDEVHLP::pfnPICRegister */
2497static DECLCALLBACK(int) pdmR3DevHlp_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3)
2498{
2499 PDMDEV_ASSERT_DEVINS(pDevIns);
2500 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
2501 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: pPicReg=%p:{.u32Version=%#x, .pfnSetIrqR3=%p, .pfnGetInterruptR3=%p, .pszGetIrqRC=%p:{%s}, .pszGetInterruptRC=%p:{%s}, .pszGetIrqR0=%p:{%s}, .pszGetInterruptR0=%p:{%s} } ppPicHlpR3=%p\n",
2502 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pPicReg, pPicReg->u32Version, pPicReg->pfnSetIrqR3, pPicReg->pfnGetInterruptR3,
2503 pPicReg->pszSetIrqRC, pPicReg->pszSetIrqRC, pPicReg->pszGetInterruptRC, pPicReg->pszGetInterruptRC,
2504 pPicReg->pszSetIrqR0, pPicReg->pszSetIrqR0, pPicReg->pszGetInterruptR0, pPicReg->pszGetInterruptR0,
2505 ppPicHlpR3));
2506
2507 /*
2508 * Validate input.
2509 */
2510 if (pPicReg->u32Version != PDM_PICREG_VERSION)
2511 {
2512 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPicReg->u32Version, PDM_PICREG_VERSION));
2513 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (version)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2514 return VERR_INVALID_PARAMETER;
2515 }
2516 if ( !pPicReg->pfnSetIrqR3
2517 || !pPicReg->pfnGetInterruptR3)
2518 {
2519 Assert(pPicReg->pfnSetIrqR3);
2520 Assert(pPicReg->pfnGetInterruptR3);
2521 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (R3 callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2522 return VERR_INVALID_PARAMETER;
2523 }
2524 if ( ( pPicReg->pszSetIrqRC
2525 || pPicReg->pszGetInterruptRC)
2526 && ( !VALID_PTR(pPicReg->pszSetIrqRC)
2527 || !VALID_PTR(pPicReg->pszGetInterruptRC))
2528 )
2529 {
2530 Assert(VALID_PTR(pPicReg->pszSetIrqRC));
2531 Assert(VALID_PTR(pPicReg->pszGetInterruptRC));
2532 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (RC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2533 return VERR_INVALID_PARAMETER;
2534 }
2535 if ( pPicReg->pszSetIrqRC
2536 && !(pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC))
2537 {
2538 Assert(pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_GC);
2539 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (RC flag)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2540 return VERR_INVALID_PARAMETER;
2541 }
2542 if ( pPicReg->pszSetIrqR0
2543 && !(pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_R0))
2544 {
2545 Assert(pDevIns->pDevReg->fFlags & PDM_DEVREG_FLAGS_R0);
2546 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (R0 flag)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2547 return VERR_INVALID_PARAMETER;
2548 }
2549 if (!ppPicHlpR3)
2550 {
2551 Assert(ppPicHlpR3);
2552 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc (ppPicHlpR3)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2553 return VERR_INVALID_PARAMETER;
2554 }
2555
2556 /*
2557 * Only one PIC device.
2558 */
2559 PVM pVM = pDevIns->Internal.s.pVMHC;
2560 if (pVM->pdm.s.Pic.pDevInsR3)
2561 {
2562 AssertMsgFailed(("Only one pic device is supported!\n"));
2563 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2564 return VERR_INVALID_PARAMETER;
2565 }
2566
2567 /*
2568 * RC stuff.
2569 */
2570 if (pPicReg->pszSetIrqRC)
2571 {
2572 int rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pPicReg->pszSetIrqRC, &pVM->pdm.s.Pic.pfnSetIrqRC);
2573 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pPicReg->pszSetIrqRC, rc));
2574 if (VBOX_SUCCESS(rc))
2575 {
2576 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pPicReg->pszGetInterruptRC, &pVM->pdm.s.Pic.pfnGetInterruptRC);
2577 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pPicReg->pszGetInterruptRC, rc));
2578 }
2579 if (VBOX_FAILURE(rc))
2580 {
2581 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2582 return rc;
2583 }
2584 pVM->pdm.s.Pic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2585 }
2586 else
2587 {
2588 pVM->pdm.s.Pic.pDevInsRC = 0;
2589 pVM->pdm.s.Pic.pfnSetIrqRC = 0;
2590 pVM->pdm.s.Pic.pfnGetInterruptRC = 0;
2591 }
2592
2593 /*
2594 * R0 stuff.
2595 */
2596 if (pPicReg->pszSetIrqR0)
2597 {
2598 int rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pPicReg->pszSetIrqR0, &pVM->pdm.s.Pic.pfnSetIrqR0);
2599 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pPicReg->pszSetIrqR0, rc));
2600 if (VBOX_SUCCESS(rc))
2601 {
2602 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pPicReg->pszGetInterruptR0, &pVM->pdm.s.Pic.pfnGetInterruptR0);
2603 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pPicReg->pszGetInterruptR0, rc));
2604 }
2605 if (VBOX_FAILURE(rc))
2606 {
2607 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2608 return rc;
2609 }
2610 pVM->pdm.s.Pic.pDevInsR0 = MMHyperR3ToR0(pVM, pDevIns);
2611 Assert(pVM->pdm.s.Pic.pDevInsR0);
2612 }
2613 else
2614 {
2615 pVM->pdm.s.Pic.pfnSetIrqR0 = 0;
2616 pVM->pdm.s.Pic.pfnGetInterruptR0 = 0;
2617 pVM->pdm.s.Pic.pDevInsR0 = 0;
2618 }
2619
2620 /*
2621 * R3 stuff.
2622 */
2623 pVM->pdm.s.Pic.pDevInsR3 = pDevIns;
2624 pVM->pdm.s.Pic.pfnSetIrqR3 = pPicReg->pfnSetIrqR3;
2625 pVM->pdm.s.Pic.pfnGetInterruptR3 = pPicReg->pfnGetInterruptR3;
2626 Log(("PDM: Registered PIC device '%s'/%d pDevIns=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
2627
2628 /* set the helper pointer and return. */
2629 *ppPicHlpR3 = &g_pdmR3DevPicHlp;
2630 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VINF_SUCCESS));
2631 return VINF_SUCCESS;
2632}
2633
2634
2635/** @copydoc PDMDEVHLP::pfnAPICRegister */
2636static DECLCALLBACK(int) pdmR3DevHlp_APICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3)
2637{
2638 PDMDEV_ASSERT_DEVINS(pDevIns);
2639 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
2640 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: pApicReg=%p:{.u32Version=%#x, .pfnGetInterruptR3=%p, .pfnSetBaseR3=%p, .pfnGetBaseR3=%p, "
2641 ".pfnSetTPRR3=%p, .pfnGetTPRR3=%p, .pfnBusDeliverR3=%p, pszGetInterruptRC=%p:{%s}, pszSetBaseRC=%p:{%s}, pszGetBaseRC=%p:{%s}, "
2642 ".pszSetTPRRC=%p:{%s}, .pszGetTPRRC=%p:{%s}, .pszBusDeliverRC=%p:{%s}} ppApicHlpR3=%p\n",
2643 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pApicReg, pApicReg->u32Version, pApicReg->pfnGetInterruptR3, pApicReg->pfnSetBaseR3,
2644 pApicReg->pfnGetBaseR3, pApicReg->pfnSetTPRR3, pApicReg->pfnGetTPRR3, pApicReg->pfnBusDeliverR3, pApicReg->pszGetInterruptRC,
2645 pApicReg->pszGetInterruptRC, pApicReg->pszSetBaseRC, pApicReg->pszSetBaseRC, pApicReg->pszGetBaseRC, pApicReg->pszGetBaseRC,
2646 pApicReg->pszSetTPRRC, pApicReg->pszSetTPRRC, pApicReg->pszGetTPRRC, pApicReg->pszGetTPRRC, pApicReg->pszBusDeliverRC,
2647 pApicReg->pszBusDeliverRC, ppApicHlpR3));
2648
2649 /*
2650 * Validate input.
2651 */
2652 if (pApicReg->u32Version != PDM_APICREG_VERSION)
2653 {
2654 AssertMsgFailed(("u32Version=%#x expected %#x\n", pApicReg->u32Version, PDM_APICREG_VERSION));
2655 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc (version)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2656 return VERR_INVALID_PARAMETER;
2657 }
2658 if ( !pApicReg->pfnGetInterruptR3
2659 || !pApicReg->pfnHasPendingIrqR3
2660 || !pApicReg->pfnSetBaseR3
2661 || !pApicReg->pfnGetBaseR3
2662 || !pApicReg->pfnSetTPRR3
2663 || !pApicReg->pfnGetTPRR3
2664 || !pApicReg->pfnBusDeliverR3)
2665 {
2666 Assert(pApicReg->pfnGetInterruptR3);
2667 Assert(pApicReg->pfnHasPendingIrqR3);
2668 Assert(pApicReg->pfnSetBaseR3);
2669 Assert(pApicReg->pfnGetBaseR3);
2670 Assert(pApicReg->pfnSetTPRR3);
2671 Assert(pApicReg->pfnGetTPRR3);
2672 Assert(pApicReg->pfnBusDeliverR3);
2673 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc (R3 callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2674 return VERR_INVALID_PARAMETER;
2675 }
2676 if ( ( pApicReg->pszGetInterruptRC
2677 || pApicReg->pszHasPendingIrqRC
2678 || pApicReg->pszSetBaseRC
2679 || pApicReg->pszGetBaseRC
2680 || pApicReg->pszSetTPRRC
2681 || pApicReg->pszGetTPRRC
2682 || pApicReg->pszBusDeliverRC)
2683 && ( !VALID_PTR(pApicReg->pszGetInterruptRC)
2684 || !VALID_PTR(pApicReg->pszHasPendingIrqRC)
2685 || !VALID_PTR(pApicReg->pszSetBaseRC)
2686 || !VALID_PTR(pApicReg->pszGetBaseRC)
2687 || !VALID_PTR(pApicReg->pszSetTPRRC)
2688 || !VALID_PTR(pApicReg->pszGetTPRRC)
2689 || !VALID_PTR(pApicReg->pszBusDeliverRC))
2690 )
2691 {
2692 Assert(VALID_PTR(pApicReg->pszGetInterruptRC));
2693 Assert(VALID_PTR(pApicReg->pszHasPendingIrqRC));
2694 Assert(VALID_PTR(pApicReg->pszSetBaseRC));
2695 Assert(VALID_PTR(pApicReg->pszGetBaseRC));
2696 Assert(VALID_PTR(pApicReg->pszSetTPRRC));
2697 Assert(VALID_PTR(pApicReg->pszGetTPRRC));
2698 Assert(VALID_PTR(pApicReg->pszBusDeliverRC));
2699 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc (RC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2700 return VERR_INVALID_PARAMETER;
2701 }
2702 if ( ( pApicReg->pszGetInterruptR0
2703 || pApicReg->pszHasPendingIrqR0
2704 || pApicReg->pszSetBaseR0
2705 || pApicReg->pszGetBaseR0
2706 || pApicReg->pszSetTPRR0
2707 || pApicReg->pszGetTPRR0
2708 || pApicReg->pszBusDeliverR0)
2709 && ( !VALID_PTR(pApicReg->pszGetInterruptR0)
2710 || !VALID_PTR(pApicReg->pszHasPendingIrqR0)
2711 || !VALID_PTR(pApicReg->pszSetBaseR0)
2712 || !VALID_PTR(pApicReg->pszGetBaseR0)
2713 || !VALID_PTR(pApicReg->pszSetTPRR0)
2714 || !VALID_PTR(pApicReg->pszGetTPRR0)
2715 || !VALID_PTR(pApicReg->pszBusDeliverR0))
2716 )
2717 {
2718 Assert(VALID_PTR(pApicReg->pszGetInterruptR0));
2719 Assert(VALID_PTR(pApicReg->pszHasPendingIrqR0));
2720 Assert(VALID_PTR(pApicReg->pszSetBaseR0));
2721 Assert(VALID_PTR(pApicReg->pszGetBaseR0));
2722 Assert(VALID_PTR(pApicReg->pszSetTPRR0));
2723 Assert(VALID_PTR(pApicReg->pszGetTPRR0));
2724 Assert(VALID_PTR(pApicReg->pszBusDeliverR0));
2725 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc (R0 callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2726 return VERR_INVALID_PARAMETER;
2727 }
2728 if (!ppApicHlpR3)
2729 {
2730 Assert(ppApicHlpR3);
2731 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc (ppApicHlpR3)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2732 return VERR_INVALID_PARAMETER;
2733 }
2734
2735 /*
2736 * Only one APIC device. (malc: only in UP case actually)
2737 */
2738 PVM pVM = pDevIns->Internal.s.pVMHC;
2739 if (pVM->pdm.s.Apic.pDevInsR3)
2740 {
2741 AssertMsgFailed(("Only one apic device is supported!\n"));
2742 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2743 return VERR_INVALID_PARAMETER;
2744 }
2745
2746 /*
2747 * Resolve & initialize the RC bits.
2748 */
2749 if (pApicReg->pszGetInterruptRC)
2750 {
2751 int rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszGetInterruptRC, &pVM->pdm.s.Apic.pfnGetInterruptRC);
2752 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszGetInterruptRC, rc));
2753 if (RT_SUCCESS(rc))
2754 {
2755 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszHasPendingIrqRC, &pVM->pdm.s.Apic.pfnHasPendingIrqRC);
2756 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszHasPendingIrqRC, rc));
2757 }
2758 if (RT_SUCCESS(rc))
2759 {
2760 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszSetBaseRC, &pVM->pdm.s.Apic.pfnSetBaseRC);
2761 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszSetBaseRC, rc));
2762 }
2763 if (RT_SUCCESS(rc))
2764 {
2765 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszGetBaseRC, &pVM->pdm.s.Apic.pfnGetBaseRC);
2766 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszGetBaseRC, rc));
2767 }
2768 if (RT_SUCCESS(rc))
2769 {
2770 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszSetTPRRC, &pVM->pdm.s.Apic.pfnSetTPRRC);
2771 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszSetTPRRC, rc));
2772 }
2773 if (RT_SUCCESS(rc))
2774 {
2775 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszGetTPRRC, &pVM->pdm.s.Apic.pfnGetTPRRC);
2776 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszGetTPRRC, rc));
2777 }
2778 if (RT_SUCCESS(rc))
2779 {
2780 rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pApicReg->pszBusDeliverRC, &pVM->pdm.s.Apic.pfnBusDeliverRC);
2781 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pApicReg->pszBusDeliverRC, rc));
2782 }
2783 if (VBOX_FAILURE(rc))
2784 {
2785 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2786 return rc;
2787 }
2788 pVM->pdm.s.Apic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2789 }
2790 else
2791 {
2792 pVM->pdm.s.Apic.pDevInsRC = 0;
2793 pVM->pdm.s.Apic.pfnGetInterruptRC = 0;
2794 pVM->pdm.s.Apic.pfnHasPendingIrqRC = 0;
2795 pVM->pdm.s.Apic.pfnSetBaseRC = 0;
2796 pVM->pdm.s.Apic.pfnGetBaseRC = 0;
2797 pVM->pdm.s.Apic.pfnSetTPRRC = 0;
2798 pVM->pdm.s.Apic.pfnGetTPRRC = 0;
2799 pVM->pdm.s.Apic.pfnBusDeliverRC = 0;
2800 }
2801
2802 /*
2803 * Resolve & initialize the R0 bits.
2804 */
2805 if (pApicReg->pszGetInterruptR0)
2806 {
2807 int rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszGetInterruptR0, &pVM->pdm.s.Apic.pfnGetInterruptR0);
2808 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszGetInterruptR0, rc));
2809 if (RT_SUCCESS(rc))
2810 {
2811 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszHasPendingIrqR0, &pVM->pdm.s.Apic.pfnHasPendingIrqR0);
2812 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszHasPendingIrqR0, rc));
2813 }
2814 if (RT_SUCCESS(rc))
2815 {
2816 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszSetBaseR0, &pVM->pdm.s.Apic.pfnSetBaseR0);
2817 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszSetBaseR0, rc));
2818 }
2819 if (RT_SUCCESS(rc))
2820 {
2821 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszGetBaseR0, &pVM->pdm.s.Apic.pfnGetBaseR0);
2822 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszGetBaseR0, rc));
2823 }
2824 if (RT_SUCCESS(rc))
2825 {
2826 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszSetTPRR0, &pVM->pdm.s.Apic.pfnSetTPRR0);
2827 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszSetTPRR0, rc));
2828 }
2829 if (RT_SUCCESS(rc))
2830 {
2831 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszGetTPRR0, &pVM->pdm.s.Apic.pfnGetTPRR0);
2832 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszGetTPRR0, rc));
2833 }
2834 if (RT_SUCCESS(rc))
2835 {
2836 rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pApicReg->pszBusDeliverR0, &pVM->pdm.s.Apic.pfnBusDeliverR0);
2837 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pApicReg->pszBusDeliverR0, rc));
2838 }
2839 if (VBOX_FAILURE(rc))
2840 {
2841 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2842 return rc;
2843 }
2844 pVM->pdm.s.Apic.pDevInsR0 = MMHyperR3ToR0(pVM, pDevIns);
2845 Assert(pVM->pdm.s.Apic.pDevInsR0);
2846 }
2847 else
2848 {
2849 pVM->pdm.s.Apic.pfnGetInterruptR0 = 0;
2850 pVM->pdm.s.Apic.pfnHasPendingIrqR0 = 0;
2851 pVM->pdm.s.Apic.pfnSetBaseR0 = 0;
2852 pVM->pdm.s.Apic.pfnGetBaseR0 = 0;
2853 pVM->pdm.s.Apic.pfnSetTPRR0 = 0;
2854 pVM->pdm.s.Apic.pfnGetTPRR0 = 0;
2855 pVM->pdm.s.Apic.pfnBusDeliverR0 = 0;
2856 pVM->pdm.s.Apic.pDevInsR0 = 0;
2857 }
2858
2859 /*
2860 * Initialize the HC bits.
2861 */
2862 pVM->pdm.s.Apic.pDevInsR3 = pDevIns;
2863 pVM->pdm.s.Apic.pfnGetInterruptR3 = pApicReg->pfnGetInterruptR3;
2864 pVM->pdm.s.Apic.pfnHasPendingIrqR3 = pApicReg->pfnHasPendingIrqR3;
2865 pVM->pdm.s.Apic.pfnSetBaseR3 = pApicReg->pfnSetBaseR3;
2866 pVM->pdm.s.Apic.pfnGetBaseR3 = pApicReg->pfnGetBaseR3;
2867 pVM->pdm.s.Apic.pfnSetTPRR3 = pApicReg->pfnSetTPRR3;
2868 pVM->pdm.s.Apic.pfnGetTPRR3 = pApicReg->pfnGetTPRR3;
2869 pVM->pdm.s.Apic.pfnBusDeliverR3 = pApicReg->pfnBusDeliverR3;
2870 Log(("PDM: Registered APIC device '%s'/%d pDevIns=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
2871
2872 /* set the helper pointer and return. */
2873 *ppApicHlpR3 = &g_pdmR3DevApicHlp;
2874 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VINF_SUCCESS));
2875 return VINF_SUCCESS;
2876}
2877
2878
2879/** @copydoc PDMDEVHLP::pfnIOAPICRegister */
2880static DECLCALLBACK(int) pdmR3DevHlp_IOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3)
2881{
2882 PDMDEV_ASSERT_DEVINS(pDevIns);
2883 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
2884 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: pIoApicReg=%p:{.u32Version=%#x, .pfnSetIrqR3=%p, .pszSetIrqRC=%p:{%s}, .pszSetIrqR0=%p:{%s}} ppIoApicHlpR3=%p\n",
2885 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pIoApicReg, pIoApicReg->u32Version, pIoApicReg->pfnSetIrqR3,
2886 pIoApicReg->pszSetIrqRC, pIoApicReg->pszSetIrqRC, pIoApicReg->pszSetIrqR0, pIoApicReg->pszSetIrqR0, ppIoApicHlpR3));
2887
2888 /*
2889 * Validate input.
2890 */
2891 if (pIoApicReg->u32Version != PDM_IOAPICREG_VERSION)
2892 {
2893 AssertMsgFailed(("u32Version=%#x expected %#x\n", pIoApicReg->u32Version, PDM_IOAPICREG_VERSION));
2894 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (version)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2895 return VERR_INVALID_PARAMETER;
2896 }
2897 if (!pIoApicReg->pfnSetIrqR3)
2898 {
2899 Assert(pIoApicReg->pfnSetIrqR3);
2900 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (R3 callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2901 return VERR_INVALID_PARAMETER;
2902 }
2903 if ( pIoApicReg->pszSetIrqRC
2904 && !VALID_PTR(pIoApicReg->pszSetIrqRC))
2905 {
2906 Assert(VALID_PTR(pIoApicReg->pszSetIrqRC));
2907 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (GC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2908 return VERR_INVALID_PARAMETER;
2909 }
2910 if ( pIoApicReg->pszSetIrqR0
2911 && !VALID_PTR(pIoApicReg->pszSetIrqR0))
2912 {
2913 Assert(VALID_PTR(pIoApicReg->pszSetIrqR0));
2914 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (GC callbacks)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2915 return VERR_INVALID_PARAMETER;
2916 }
2917 if (!ppIoApicHlpR3)
2918 {
2919 Assert(ppIoApicHlpR3);
2920 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (ppApicHlp)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2921 return VERR_INVALID_PARAMETER;
2922 }
2923
2924 /*
2925 * The I/O APIC requires the APIC to be present (hacks++).
2926 * If the I/O APIC does GC stuff so must the APIC.
2927 */
2928 PVM pVM = pDevIns->Internal.s.pVMHC;
2929 if (!pVM->pdm.s.Apic.pDevInsR3)
2930 {
2931 AssertMsgFailed(("Configuration error / Init order error! No APIC!\n"));
2932 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (no APIC)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2933 return VERR_INVALID_PARAMETER;
2934 }
2935 if ( pIoApicReg->pszSetIrqRC
2936 && !pVM->pdm.s.Apic.pDevInsRC)
2937 {
2938 AssertMsgFailed(("Configuration error! APIC doesn't do GC, I/O APIC does!\n"));
2939 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (no GC APIC)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2940 return VERR_INVALID_PARAMETER;
2941 }
2942
2943 /*
2944 * Only one I/O APIC device.
2945 */
2946 if (pVM->pdm.s.IoApic.pDevInsR3)
2947 {
2948 AssertMsgFailed(("Only one ioapic device is supported!\n"));
2949 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc (only one)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2950 return VERR_INVALID_PARAMETER;
2951 }
2952
2953 /*
2954 * Resolve & initialize the GC bits.
2955 */
2956 if (pIoApicReg->pszSetIrqRC)
2957 {
2958 int rc = PDMR3GetSymbolGCLazy(pVM, pDevIns->pDevReg->szGCMod, pIoApicReg->pszSetIrqRC, &pVM->pdm.s.IoApic.pfnSetIrqRC);
2959 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szGCMod, pIoApicReg->pszSetIrqRC, rc));
2960 if (VBOX_FAILURE(rc))
2961 {
2962 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2963 return rc;
2964 }
2965 pVM->pdm.s.IoApic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2966 }
2967 else
2968 {
2969 pVM->pdm.s.IoApic.pDevInsRC = 0;
2970 pVM->pdm.s.IoApic.pfnSetIrqRC = 0;
2971 }
2972
2973 /*
2974 * Resolve & initialize the R0 bits.
2975 */
2976 if (pIoApicReg->pszSetIrqR0)
2977 {
2978 int rc = PDMR3GetSymbolR0Lazy(pVM, pDevIns->pDevReg->szR0Mod, pIoApicReg->pszSetIrqR0, &pVM->pdm.s.IoApic.pfnSetIrqR0);
2979 AssertMsgRC(rc, ("%s::%s rc=%Vrc\n", pDevIns->pDevReg->szR0Mod, pIoApicReg->pszSetIrqR0, rc));
2980 if (VBOX_FAILURE(rc))
2981 {
2982 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
2983 return rc;
2984 }
2985 pVM->pdm.s.IoApic.pDevInsR0 = MMHyperR3ToR0(pVM, pDevIns);
2986 Assert(pVM->pdm.s.IoApic.pDevInsR0);
2987 }
2988 else
2989 {
2990 pVM->pdm.s.IoApic.pfnSetIrqR0 = 0;
2991 pVM->pdm.s.IoApic.pDevInsR0 = 0;
2992 }
2993
2994 /*
2995 * Initialize the R3 bits.
2996 */
2997 pVM->pdm.s.IoApic.pDevInsR3 = pDevIns;
2998 pVM->pdm.s.IoApic.pfnSetIrqR3 = pIoApicReg->pfnSetIrqR3;
2999 Log(("PDM: Registered I/O APIC device '%s'/%d pDevIns=%p\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
3000
3001 /* set the helper pointer and return. */
3002 *ppIoApicHlpR3 = &g_pdmR3DevIoApicHlp;
3003 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VINF_SUCCESS));
3004 return VINF_SUCCESS;
3005}
3006
3007
3008/** @copydoc PDMDEVHLP::pfnDMACRegister */
3009static DECLCALLBACK(int) pdmR3DevHlp_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp)
3010{
3011 PDMDEV_ASSERT_DEVINS(pDevIns);
3012 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3013 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: pDmacReg=%p:{.u32Version=%#x, .pfnRun=%p, .pfnRegister=%p, .pfnReadMemory=%p, .pfnWriteMemory=%p, .pfnSetDREQ=%p, .pfnGetChannelMode=%p} ppDmacHlp=%p\n",
3014 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDmacReg, pDmacReg->u32Version, pDmacReg->pfnRun, pDmacReg->pfnRegister,
3015 pDmacReg->pfnReadMemory, pDmacReg->pfnWriteMemory, pDmacReg->pfnSetDREQ, pDmacReg->pfnGetChannelMode, ppDmacHlp));
3016
3017 /*
3018 * Validate input.
3019 */
3020 if (pDmacReg->u32Version != PDM_DMACREG_VERSION)
3021 {
3022 AssertMsgFailed(("u32Version=%#x expected %#x\n", pDmacReg->u32Version,
3023 PDM_DMACREG_VERSION));
3024 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Vrc (version)\n",
3025 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3026 return VERR_INVALID_PARAMETER;
3027 }
3028 if ( !pDmacReg->pfnRun
3029 || !pDmacReg->pfnRegister
3030 || !pDmacReg->pfnReadMemory
3031 || !pDmacReg->pfnWriteMemory
3032 || !pDmacReg->pfnSetDREQ
3033 || !pDmacReg->pfnGetChannelMode)
3034 {
3035 Assert(pDmacReg->pfnRun);
3036 Assert(pDmacReg->pfnRegister);
3037 Assert(pDmacReg->pfnReadMemory);
3038 Assert(pDmacReg->pfnWriteMemory);
3039 Assert(pDmacReg->pfnSetDREQ);
3040 Assert(pDmacReg->pfnGetChannelMode);
3041 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Vrc (callbacks)\n",
3042 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3043 return VERR_INVALID_PARAMETER;
3044 }
3045
3046 if (!ppDmacHlp)
3047 {
3048 Assert(ppDmacHlp);
3049 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Vrc (ppDmacHlp)\n",
3050 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3051 return VERR_INVALID_PARAMETER;
3052 }
3053
3054 /*
3055 * Only one DMA device.
3056 */
3057 PVM pVM = pDevIns->Internal.s.pVMHC;
3058 if (pVM->pdm.s.pDmac)
3059 {
3060 AssertMsgFailed(("Only one DMA device is supported!\n"));
3061 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Vrc\n",
3062 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3063 return VERR_INVALID_PARAMETER;
3064 }
3065
3066 /*
3067 * Allocate and initialize pci bus structure.
3068 */
3069 int rc = VINF_SUCCESS;
3070 PPDMDMAC pDmac = (PPDMDMAC)MMR3HeapAlloc(pDevIns->Internal.s.pVMHC, MM_TAG_PDM_DEVICE, sizeof(*pDmac));
3071 if (pDmac)
3072 {
3073 pDmac->pDevIns = pDevIns;
3074 pDmac->Reg = *pDmacReg;
3075 pVM->pdm.s.pDmac = pDmac;
3076
3077 /* set the helper pointer. */
3078 *ppDmacHlp = &g_pdmR3DevDmacHlp;
3079 Log(("PDM: Registered DMAC device '%s'/%d pDevIns=%p\n",
3080 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pDevIns));
3081 }
3082 else
3083 rc = VERR_NO_MEMORY;
3084
3085 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Vrc\n",
3086 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3087 return rc;
3088}
3089
3090
3091/** @copydoc PDMDEVHLP::pfnPhysRead */
3092static DECLCALLBACK(void) pdmR3DevHlp_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
3093{
3094 PDMDEV_ASSERT_DEVINS(pDevIns);
3095 LogFlow(("pdmR3DevHlp_PhysRead: caller='%s'/%d: GCPhys=%VGp pvBuf=%p cbRead=%#x\n",
3096 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhys, pvBuf, cbRead));
3097
3098 /*
3099 * For the convenience of the device we put no thread restriction on this interface.
3100 * That means we'll have to check which thread we're in and choose our path.
3101 */
3102#ifdef PDM_PHYS_READWRITE_FROM_ANY_THREAD
3103 PGMPhysRead(pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbRead);
3104#else
3105 if (VM_IS_EMT(pDevIns->Internal.s.pVMHC) || VMMR3LockIsOwner(pDevIns->Internal.s.pVMHC))
3106 PGMPhysRead(pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbRead);
3107 else
3108 {
3109 Log(("pdmR3DevHlp_PhysRead: caller='%s'/%d: Requesting call in EMT...\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3110 PVMREQ pReq;
3111 AssertCompileSize(RTGCPHYS, 4);
3112 int rc = VMR3ReqCallVoid(pDevIns->Internal.s.pVMHC, &pReq, RT_INDEFINITE_WAIT,
3113 (PFNRT)PGMPhysRead, 4, pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbRead);
3114 while (rc == VERR_TIMEOUT)
3115 rc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
3116 AssertReleaseRC(rc);
3117 VMR3ReqFree(pReq);
3118 }
3119#endif
3120 Log(("pdmR3DevHlp_PhysRead: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3121}
3122
3123
3124/** @copydoc PDMDEVHLP::pfnPhysWrite */
3125static DECLCALLBACK(void) pdmR3DevHlp_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
3126{
3127 PDMDEV_ASSERT_DEVINS(pDevIns);
3128 LogFlow(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: GCPhys=%VGp pvBuf=%p cbWrite=%#x\n",
3129 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhys, pvBuf, cbWrite));
3130
3131 /*
3132 * For the convenience of the device we put no thread restriction on this interface.
3133 * That means we'll have to check which thread we're in and choose our path.
3134 */
3135#ifdef PDM_PHYS_READWRITE_FROM_ANY_THREAD
3136 PGMPhysWrite(pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbWrite);
3137#else
3138 if (VM_IS_EMT(pDevIns->Internal.s.pVMHC) || VMMR3LockIsOwner(pDevIns->Internal.s.pVMHC))
3139 PGMPhysWrite(pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbWrite);
3140 else
3141 {
3142 Log(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: Requesting call in EMT...\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3143 PVMREQ pReq;
3144 AssertCompileSize(RTGCPHYS, 4);
3145 int rc = VMR3ReqCallVoid(pDevIns->Internal.s.pVMHC, &pReq, RT_INDEFINITE_WAIT,
3146 (PFNRT)PGMPhysWrite, 4, pDevIns->Internal.s.pVMHC, GCPhys, pvBuf, cbWrite);
3147 while (rc == VERR_TIMEOUT)
3148 rc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
3149 AssertReleaseRC(rc);
3150 VMR3ReqFree(pReq);
3151 }
3152#endif
3153 Log(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: returns void\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3154}
3155
3156
3157/** @copydoc PDMDEVHLP::pfnPhysReadGCVirt */
3158static DECLCALLBACK(int) pdmR3DevHlp_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb)
3159{
3160 PDMDEV_ASSERT_DEVINS(pDevIns);
3161 PVM pVM = pDevIns->Internal.s.pVMHC;
3162 VM_ASSERT_EMT(pVM);
3163 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: pvDst=%p GCVirt=%VGv cb=%#x\n",
3164 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pvDst, GCVirtSrc, cb));
3165
3166 if (!VM_IS_EMT(pVM))
3167 return VERR_ACCESS_DENIED;
3168
3169 int rc = PGMPhysReadGCPtr(pVM, pvDst, GCVirtSrc, cb);
3170
3171 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3172
3173 return rc;
3174}
3175
3176
3177/** @copydoc PDMDEVHLP::pfnPhysWriteGCVirt */
3178static DECLCALLBACK(int) pdmR3DevHlp_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb)
3179{
3180 PDMDEV_ASSERT_DEVINS(pDevIns);
3181 PVM pVM = pDevIns->Internal.s.pVMHC;
3182 VM_ASSERT_EMT(pVM);
3183 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: GCVirtDst=%VGv pvSrc=%p cb=%#x\n",
3184 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCVirtDst, pvSrc, cb));
3185
3186 if (!VM_IS_EMT(pVM))
3187 return VERR_ACCESS_DENIED;
3188
3189 int rc = PGMPhysWriteGCPtr(pVM, GCVirtDst, pvSrc, cb);
3190
3191 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3192
3193 return rc;
3194}
3195
3196
3197/** @copydoc PDMDEVHLP::pfnPhysReserve */
3198static DECLCALLBACK(int) pdmR3DevHlp_PhysReserve(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, const char *pszDesc)
3199{
3200 PDMDEV_ASSERT_DEVINS(pDevIns);
3201 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3202 LogFlow(("pdmR3DevHlp_PhysReserve: caller='%s'/%d: GCPhys=%VGp cbRange=%#x pszDesc=%p:{%s}\n",
3203 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhys, cbRange, pszDesc, pszDesc));
3204
3205 int rc = MMR3PhysReserve(pDevIns->Internal.s.pVMHC, GCPhys, cbRange, pszDesc);
3206
3207 LogFlow(("pdmR3DevHlp_PhysReserve: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3208
3209 return rc;
3210}
3211
3212
3213/** @copydoc PDMDEVHLP::pfnPhysGCPtr2GCPhys */
3214static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPtr2GCPhys(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys)
3215{
3216 PDMDEV_ASSERT_DEVINS(pDevIns);
3217 PVM pVM = pDevIns->Internal.s.pVMHC;
3218 VM_ASSERT_EMT(pVM);
3219 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: GCPtr=%VGv pGCPhys=%p\n",
3220 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPtr, pGCPhys));
3221
3222 if (!VM_IS_EMT(pVM))
3223 return VERR_ACCESS_DENIED;
3224
3225 int rc = PGMPhysGCPtr2GCPhys(pVM, GCPtr, pGCPhys);
3226
3227 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: returns %Vrc *pGCPhys=%VGp\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc, *pGCPhys));
3228
3229 return rc;
3230}
3231
3232
3233/** @copydoc PDMDEVHLP::pfnVMState */
3234static DECLCALLBACK(VMSTATE) pdmR3DevHlp_VMState(PPDMDEVINS pDevIns)
3235{
3236 PDMDEV_ASSERT_DEVINS(pDevIns);
3237
3238 VMSTATE enmVMState = VMR3GetState(pDevIns->Internal.s.pVMHC);
3239
3240 LogFlow(("pdmR3DevHlp_VMState: caller='%s'/%d: returns %d (%s)\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
3241 enmVMState, VMR3GetStateName(enmVMState)));
3242 return enmVMState;
3243}
3244
3245
3246/** @copydoc PDMDEVHLP::pfnA20IsEnabled */
3247static DECLCALLBACK(bool) pdmR3DevHlp_A20IsEnabled(PPDMDEVINS pDevIns)
3248{
3249 PDMDEV_ASSERT_DEVINS(pDevIns);
3250 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3251
3252 bool fRc = PGMPhysIsA20Enabled(pDevIns->Internal.s.pVMHC);
3253
3254 LogFlow(("pdmR3DevHlp_A20IsEnabled: caller='%s'/%d: returns %d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, fRc));
3255 return fRc;
3256}
3257
3258
3259/** @copydoc PDMDEVHLP::pfnA20Set */
3260static DECLCALLBACK(void) pdmR3DevHlp_A20Set(PPDMDEVINS pDevIns, bool fEnable)
3261{
3262 PDMDEV_ASSERT_DEVINS(pDevIns);
3263 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3264 LogFlow(("pdmR3DevHlp_A20Set: caller='%s'/%d: fEnable=%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, fEnable));
3265 //Assert(*(unsigned *)&fEnable <= 1);
3266 PGMR3PhysSetA20(pDevIns->Internal.s.pVMHC, fEnable);
3267}
3268
3269
3270/** @copydoc PDMDEVHLP::pfnVMReset */
3271static DECLCALLBACK(int) pdmR3DevHlp_VMReset(PPDMDEVINS pDevIns)
3272{
3273 PDMDEV_ASSERT_DEVINS(pDevIns);
3274 PVM pVM = pDevIns->Internal.s.pVMHC;
3275 VM_ASSERT_EMT(pVM);
3276 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: VM_FF_RESET %d -> 1\n",
3277 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VM_FF_ISSET(pVM, VM_FF_RESET)));
3278
3279 /*
3280 * We postpone this operation because we're likely to be inside a I/O instruction
3281 * and the EIP will be updated when we return.
3282 * We still return VINF_EM_RESET to break out of any execution loops and force FF evaluation.
3283 */
3284 bool fHaltOnReset;
3285 int rc = CFGMR3QueryBool(CFGMR3GetChild(CFGMR3GetRoot(pVM), "PDM"), "HaltOnReset", &fHaltOnReset);
3286 if (VBOX_SUCCESS(rc) && fHaltOnReset)
3287 {
3288 Log(("pdmR3DevHlp_VMReset: Halt On Reset!\n"));
3289 rc = VINF_EM_HALT;
3290 }
3291 else
3292 {
3293 VM_FF_SET(pVM, VM_FF_RESET);
3294 rc = VINF_EM_RESET;
3295 }
3296
3297 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3298 return rc;
3299}
3300
3301
3302/** @copydoc PDMDEVHLP::pfnVMSuspend */
3303static DECLCALLBACK(int) pdmR3DevHlp_VMSuspend(PPDMDEVINS pDevIns)
3304{
3305 PDMDEV_ASSERT_DEVINS(pDevIns);
3306 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3307 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d:\n",
3308 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3309
3310 int rc = VMR3Suspend(pDevIns->Internal.s.pVMHC);
3311
3312 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3313 return rc;
3314}
3315
3316
3317/** @copydoc PDMDEVHLP::pfnVMPowerOff */
3318static DECLCALLBACK(int) pdmR3DevHlp_VMPowerOff(PPDMDEVINS pDevIns)
3319{
3320 PDMDEV_ASSERT_DEVINS(pDevIns);
3321 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3322 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d:\n",
3323 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3324
3325 int rc = VMR3PowerOff(pDevIns->Internal.s.pVMHC);
3326
3327 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3328 return rc;
3329}
3330
3331
3332/** @copydoc PDMDEVHLP::pfnLockVM */
3333static DECLCALLBACK(int) pdmR3DevHlp_LockVM(PPDMDEVINS pDevIns)
3334{
3335 return VMMR3Lock(pDevIns->Internal.s.pVMHC);
3336}
3337
3338
3339/** @copydoc PDMDEVHLP::pfnUnlockVM */
3340static DECLCALLBACK(int) pdmR3DevHlp_UnlockVM(PPDMDEVINS pDevIns)
3341{
3342 return VMMR3Unlock(pDevIns->Internal.s.pVMHC);
3343}
3344
3345
3346/** @copydoc PDMDEVHLP::pfnAssertVMLock */
3347static DECLCALLBACK(bool) pdmR3DevHlp_AssertVMLock(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
3348{
3349 PVM pVM = pDevIns->Internal.s.pVMHC;
3350 if (VMMR3LockIsOwner(pVM))
3351 return true;
3352
3353 RTNATIVETHREAD NativeThreadOwner = VMMR3LockGetOwner(pVM);
3354 RTTHREAD ThreadOwner = RTThreadFromNative(NativeThreadOwner);
3355 char szMsg[100];
3356 RTStrPrintf(szMsg, sizeof(szMsg), "AssertVMLocked '%s'/%d ThreadOwner=%RTnthrd/%RTthrd/'%s' Self='%s'\n",
3357 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance,
3358 NativeThreadOwner, ThreadOwner, RTThreadGetName(ThreadOwner), RTThreadSelfName());
3359 AssertMsg1(szMsg, iLine, pszFile, pszFunction);
3360 AssertBreakpoint();
3361 return false;
3362}
3363
3364/** @copydoc PDMDEVHLP::pfnDMARegister */
3365static DECLCALLBACK(int) pdmR3DevHlp_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser)
3366{
3367 PDMDEV_ASSERT_DEVINS(pDevIns);
3368 PVM pVM = pDevIns->Internal.s.pVMHC;
3369 VM_ASSERT_EMT(pVM);
3370 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: uChannel=%d pfnTransferHandler=%p pvUser=%p\n",
3371 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, uChannel, pfnTransferHandler, pvUser));
3372 int rc = VINF_SUCCESS;
3373 if (pVM->pdm.s.pDmac)
3374 pVM->pdm.s.pDmac->Reg.pfnRegister(pVM->pdm.s.pDmac->pDevIns, uChannel, pfnTransferHandler, pvUser);
3375 else
3376 {
3377 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3378 rc = VERR_PDM_NO_DMAC_INSTANCE;
3379 }
3380 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: returns %Vrc\n",
3381 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3382 return rc;
3383}
3384
3385/** @copydoc PDMDEVHLP::pfnDMAReadMemory */
3386static DECLCALLBACK(int) pdmR3DevHlp_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead)
3387{
3388 PDMDEV_ASSERT_DEVINS(pDevIns);
3389 PVM pVM = pDevIns->Internal.s.pVMHC;
3390 VM_ASSERT_EMT(pVM);
3391 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbRead=%p\n",
3392 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbRead));
3393 int rc = VINF_SUCCESS;
3394 if (pVM->pdm.s.pDmac)
3395 {
3396 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnReadMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
3397 if (pcbRead)
3398 *pcbRead = cb;
3399 }
3400 else
3401 {
3402 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3403 rc = VERR_PDM_NO_DMAC_INSTANCE;
3404 }
3405 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: returns %Vrc\n",
3406 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3407 return rc;
3408}
3409
3410/** @copydoc PDMDEVHLP::pfnDMAWriteMemory */
3411static DECLCALLBACK(int) pdmR3DevHlp_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten)
3412{
3413 PDMDEV_ASSERT_DEVINS(pDevIns);
3414 PVM pVM = pDevIns->Internal.s.pVMHC;
3415 VM_ASSERT_EMT(pVM);
3416 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbWritten=%p\n",
3417 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbWritten));
3418 int rc = VINF_SUCCESS;
3419 if (pVM->pdm.s.pDmac)
3420 {
3421 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnWriteMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
3422 if (pcbWritten)
3423 *pcbWritten = cb;
3424 }
3425 else
3426 {
3427 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3428 rc = VERR_PDM_NO_DMAC_INSTANCE;
3429 }
3430 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: returns %Vrc\n",
3431 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3432 return rc;
3433}
3434
3435/** @copydoc PDMDEVHLP::pfnDMASetDREQ */
3436static DECLCALLBACK(int) pdmR3DevHlp_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel)
3437{
3438 PDMDEV_ASSERT_DEVINS(pDevIns);
3439 PVM pVM = pDevIns->Internal.s.pVMHC;
3440 VM_ASSERT_EMT(pVM);
3441 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: uChannel=%d uLevel=%d\n",
3442 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, uChannel, uLevel));
3443 int rc = VINF_SUCCESS;
3444 if (pVM->pdm.s.pDmac)
3445 pVM->pdm.s.pDmac->Reg.pfnSetDREQ(pVM->pdm.s.pDmac->pDevIns, uChannel, uLevel);
3446 else
3447 {
3448 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3449 rc = VERR_PDM_NO_DMAC_INSTANCE;
3450 }
3451 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: returns %Vrc\n",
3452 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3453 return rc;
3454}
3455
3456/** @copydoc PDMDEVHLP::pfnDMAGetChannelMode */
3457static DECLCALLBACK(uint8_t) pdmR3DevHlp_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel)
3458{
3459 PDMDEV_ASSERT_DEVINS(pDevIns);
3460 PVM pVM = pDevIns->Internal.s.pVMHC;
3461 VM_ASSERT_EMT(pVM);
3462 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: uChannel=%d\n",
3463 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, uChannel));
3464 uint8_t u8Mode;
3465 if (pVM->pdm.s.pDmac)
3466 u8Mode = pVM->pdm.s.pDmac->Reg.pfnGetChannelMode(pVM->pdm.s.pDmac->pDevIns, uChannel);
3467 else
3468 {
3469 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3470 u8Mode = 3 << 2 /* illegal mode type */;
3471 }
3472 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: returns %#04x\n",
3473 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, u8Mode));
3474 return u8Mode;
3475}
3476
3477/** @copydoc PDMDEVHLP::pfnDMASchedule */
3478static DECLCALLBACK(void) pdmR3DevHlp_DMASchedule(PPDMDEVINS pDevIns)
3479{
3480 PDMDEV_ASSERT_DEVINS(pDevIns);
3481 PVM pVM = pDevIns->Internal.s.pVMHC;
3482 VM_ASSERT_EMT(pVM);
3483 LogFlow(("pdmR3DevHlp_DMASchedule: caller='%s'/%d: VM_FF_PDM_DMA %d -> 1\n",
3484 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VM_FF_ISSET(pVM, VM_FF_PDM_DMA)));
3485
3486 AssertMsg(pVM->pdm.s.pDmac, ("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3487 VM_FF_SET(pVM, VM_FF_PDM_DMA);
3488 REMR3NotifyDmaPending(pVM);
3489 VMR3NotifyFF(pVM, true);
3490}
3491
3492
3493/** @copydoc PDMDEVHLP::pfnCMOSWrite */
3494static DECLCALLBACK(int) pdmR3DevHlp_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value)
3495{
3496 PDMDEV_ASSERT_DEVINS(pDevIns);
3497 PVM pVM = pDevIns->Internal.s.pVMHC;
3498 VM_ASSERT_EMT(pVM);
3499
3500 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x u8Value=%#04x\n",
3501 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iReg, u8Value));
3502 int rc;
3503 if (pVM->pdm.s.pRtc)
3504 rc = pVM->pdm.s.pRtc->Reg.pfnWrite(pVM->pdm.s.pRtc->pDevIns, iReg, u8Value);
3505 else
3506 rc = VERR_PDM_NO_RTC_INSTANCE;
3507
3508 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Vrc\n",
3509 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3510 return rc;
3511}
3512
3513
3514/** @copydoc PDMDEVHLP::pfnCMOSRead */
3515static DECLCALLBACK(int) pdmR3DevHlp_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value)
3516{
3517 PDMDEV_ASSERT_DEVINS(pDevIns);
3518 PVM pVM = pDevIns->Internal.s.pVMHC;
3519 VM_ASSERT_EMT(pVM);
3520
3521 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x pu8Value=%p\n",
3522 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iReg, pu8Value));
3523 int rc;
3524 if (pVM->pdm.s.pRtc)
3525 rc = pVM->pdm.s.pRtc->Reg.pfnRead(pVM->pdm.s.pRtc->pDevIns, iReg, pu8Value);
3526 else
3527 rc = VERR_PDM_NO_RTC_INSTANCE;
3528
3529 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Vrc\n",
3530 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3531 return rc;
3532}
3533
3534
3535/** @copydoc PDMDEVHLP::pfnGetCpuId */
3536static DECLCALLBACK(void) pdmR3DevHlp_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
3537 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
3538{
3539 PDMDEV_ASSERT_DEVINS(pDevIns);
3540 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: iLeaf=%d pEax=%p pEbx=%p pEcx=%p pEdx=%p\n",
3541 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iLeaf, pEax, pEbx, pEcx, pEdx));
3542 AssertPtr(pEax); AssertPtr(pEbx); AssertPtr(pEcx); AssertPtr(pEdx);
3543
3544 CPUMGetGuestCpuId(pDevIns->Internal.s.pVMHC, iLeaf, pEax, pEbx, pEcx, pEdx);
3545
3546 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: returns void - *pEax=%#x *pEbx=%#x *pEcx=%#x *pEdx=%#x\n",
3547 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, *pEax, *pEbx, *pEcx, *pEdx));
3548}
3549
3550
3551/** @copydoc PDMDEVHLP::pfnROMProtectShadow */
3552static DECLCALLBACK(int) pdmR3DevHlp_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange)
3553{
3554 PDMDEV_ASSERT_DEVINS(pDevIns);
3555 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: GCPhysStart=%VGp cbRange=%#x\n",
3556 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, GCPhysStart, cbRange));
3557
3558 int rc = MMR3PhysRomProtect(pDevIns->Internal.s.pVMHC, GCPhysStart, cbRange);
3559
3560 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: returns %Vrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3561 return rc;
3562}
3563
3564
3565/**
3566 * @copydoc PDMDEVHLP::pfnMMIO2Register
3567 */
3568static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc)
3569{
3570 PDMDEV_ASSERT_DEVINS(pDevIns);
3571 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3572 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: iRegion=#x cb=%#RGp fFlags=%RX32 ppv=%p pszDescp=%p:{%s}\n",
3573 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion, cb, fFlags, ppv, pszDesc, pszDesc));
3574
3575 int rc = PGMR3PhysMMIO2Register(pDevIns->Internal.s.pVMHC, pDevIns, iRegion, cb, fFlags, ppv, pszDesc);
3576
3577 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: returns %Rrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3578 return rc;
3579}
3580
3581
3582/**
3583 * @copydoc PDMDEVHLP::pfnMMIO2Deregister
3584 */
3585static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion)
3586{
3587 PDMDEV_ASSERT_DEVINS(pDevIns);
3588 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3589 LogFlow(("pdmR3DevHlp_MMIO2Deregister: caller='%s'/%d: iRegion=#x\n",
3590 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion));
3591
3592 AssertReturn(iRegion == UINT32_MAX, VERR_INVALID_PARAMETER);
3593
3594 int rc = PGMR3PhysMMIO2Deregister(pDevIns->Internal.s.pVMHC, pDevIns, iRegion);
3595
3596 LogFlow(("pdmR3DevHlp_MMIO2Deregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3597 return rc;
3598}
3599
3600
3601/**
3602 * @copydoc PDMDEVHLP::pfnMMIO2Map
3603 */
3604static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
3605{
3606 PDMDEV_ASSERT_DEVINS(pDevIns);
3607 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3608 LogFlow(("pdmR3DevHlp_MMIO2Map: caller='%s'/%d: iRegion=#x GCPhys=%#RGp\n",
3609 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion, GCPhys));
3610
3611 int rc = PGMR3PhysMMIO2Map(pDevIns->Internal.s.pVMHC, pDevIns, iRegion, GCPhys);
3612
3613 LogFlow(("pdmR3DevHlp_MMIO2Map: caller='%s'/%d: returns %Rrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3614 return rc;
3615}
3616
3617
3618/**
3619 * @copydoc PDMDEVHLP::pfnMMIO2Unmap
3620 */
3621static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
3622{
3623 PDMDEV_ASSERT_DEVINS(pDevIns);
3624 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3625 LogFlow(("pdmR3DevHlp_MMIO2Unmap: caller='%s'/%d: iRegion=#x GCPhys=%#RGp\n",
3626 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion, GCPhys));
3627
3628 int rc = PGMR3PhysMMIO2Unmap(pDevIns->Internal.s.pVMHC, pDevIns, iRegion, GCPhys);
3629
3630 LogFlow(("pdmR3DevHlp_MMIO2Unmap: caller='%s'/%d: returns %Rrc\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc));
3631 return rc;
3632}
3633
3634
3635/**
3636 * @copydoc PDMDEVHLP::pfnMMHyperMapMMIO2
3637 */
3638static DECLCALLBACK(int) pdmR3DevHlp_MMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
3639 const char *pszDesc, PRTRCPTR pRCPtr)
3640{
3641 PDMDEV_ASSERT_DEVINS(pDevIns);
3642 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3643 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: iRegion=#x off=%RGp cb=%RGp pszDesc=%p:{%s} pRCPtr=%p\n",
3644 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, iRegion, off, cb, pszDesc, pszDesc, pRCPtr));
3645
3646 int rc = MMR3HyperMapMMIO2(pDevIns->Internal.s.pVMHC, pDevIns, iRegion, off, cb, pszDesc, pRCPtr);
3647
3648 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: returns %Rrc *pRCPtr=%RRv\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, rc, *pRCPtr));
3649 return rc;
3650}
3651
3652
3653/**
3654 * @copydoc PDMDEVHLP::pfnRegisterVMMDevHeap
3655 */
3656static DECLCALLBACK(int) pdmR3DevHlp_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize)
3657{
3658 PDMDEV_ASSERT_DEVINS(pDevIns);
3659 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3660
3661 int rc = PDMR3RegisterVMMDevHeap(pDevIns->Internal.s.pVMHC, GCPhys, pvHeap, cbSize);
3662 return rc;
3663}
3664
3665
3666/**
3667 * @copydoc PDMDEVHLP::pfnUnregisterVMMDevHeap
3668 */
3669static DECLCALLBACK(int) pdmR3DevHlp_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys)
3670{
3671 PDMDEV_ASSERT_DEVINS(pDevIns);
3672 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
3673
3674 int rc = PDMR3UnregisterVMMDevHeap(pDevIns->Internal.s.pVMHC, GCPhys);
3675 return rc;
3676}
3677
3678
3679
3680
3681
3682/** @copydoc PDMDEVHLP::pfnGetVM */
3683static DECLCALLBACK(PVM) pdmR3DevHlp_Untrusted_GetVM(PPDMDEVINS pDevIns)
3684{
3685 PDMDEV_ASSERT_DEVINS(pDevIns);
3686 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3687 return NULL;
3688}
3689
3690
3691/** @copydoc PDMDEVHLP::pfnPCIBusRegister */
3692static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3)
3693{
3694 PDMDEV_ASSERT_DEVINS(pDevIns);
3695 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3696 NOREF(pPciBusReg);
3697 NOREF(ppPciHlpR3);
3698 return VERR_ACCESS_DENIED;
3699}
3700
3701
3702/** @copydoc PDMDEVHLP::pfnPICRegister */
3703static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3)
3704{
3705 PDMDEV_ASSERT_DEVINS(pDevIns);
3706 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3707 NOREF(pPicReg);
3708 NOREF(ppPicHlpR3);
3709 return VERR_ACCESS_DENIED;
3710}
3711
3712
3713/** @copydoc PDMDEVHLP::pfnAPICRegister */
3714static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_APICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3)
3715{
3716 PDMDEV_ASSERT_DEVINS(pDevIns);
3717 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3718 NOREF(pApicReg);
3719 NOREF(ppApicHlpR3);
3720 return VERR_ACCESS_DENIED;
3721}
3722
3723
3724/** @copydoc PDMDEVHLP::pfnIOAPICRegister */
3725static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_IOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3)
3726{
3727 PDMDEV_ASSERT_DEVINS(pDevIns);
3728 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3729 NOREF(pIoApicReg);
3730 NOREF(ppIoApicHlpR3);
3731 return VERR_ACCESS_DENIED;
3732}
3733
3734
3735/** @copydoc PDMDEVHLP::pfnDMACRegister */
3736static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp)
3737{
3738 PDMDEV_ASSERT_DEVINS(pDevIns);
3739 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3740 NOREF(pDmacReg);
3741 NOREF(ppDmacHlp);
3742 return VERR_ACCESS_DENIED;
3743}
3744
3745
3746/** @copydoc PDMDEVHLP::pfnPhysRead */
3747static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
3748{
3749 PDMDEV_ASSERT_DEVINS(pDevIns);
3750 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3751 NOREF(GCPhys);
3752 NOREF(pvBuf);
3753 NOREF(cbRead);
3754}
3755
3756
3757/** @copydoc PDMDEVHLP::pfnPhysWrite */
3758static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
3759{
3760 PDMDEV_ASSERT_DEVINS(pDevIns);
3761 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3762 NOREF(GCPhys);
3763 NOREF(pvBuf);
3764 NOREF(cbWrite);
3765}
3766
3767
3768/** @copydoc PDMDEVHLP::pfnPhysReadGCVirt */
3769static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb)
3770{
3771 PDMDEV_ASSERT_DEVINS(pDevIns);
3772 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3773 NOREF(pvDst);
3774 NOREF(GCVirtSrc);
3775 NOREF(cb);
3776 return VERR_ACCESS_DENIED;
3777}
3778
3779
3780/** @copydoc PDMDEVHLP::pfnPhysWriteGCVirt */
3781static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb)
3782{
3783 PDMDEV_ASSERT_DEVINS(pDevIns);
3784 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3785 NOREF(GCVirtDst);
3786 NOREF(pvSrc);
3787 NOREF(cb);
3788 return VERR_ACCESS_DENIED;
3789}
3790
3791
3792/** @copydoc PDMDEVHLP::pfnPhysReserve */
3793static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_PhysReserve(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, const char *pszDesc)
3794{
3795 PDMDEV_ASSERT_DEVINS(pDevIns);
3796 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3797 NOREF(GCPhys);
3798 NOREF(cbRange);
3799 return VERR_ACCESS_DENIED;
3800}
3801
3802
3803/** @copydoc PDMDEVHLP::pfnObsoletePhys2HCVirt */
3804static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_Obsolete_Phys2HCVirt(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTUINT cbRange, PRTHCPTR ppvHC)
3805{
3806 PDMDEV_ASSERT_DEVINS(pDevIns);
3807 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3808 NOREF(GCPhys);
3809 NOREF(cbRange);
3810 NOREF(ppvHC);
3811 return VERR_ACCESS_DENIED;
3812}
3813
3814
3815/** @copydoc PDMDEVHLP::pfnObsoletePhysGCPtr2HCPtr */
3816static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_Obsolete_PhysGCPtr2HCPtr(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTHCPTR pHCPtr)
3817{
3818 PDMDEV_ASSERT_DEVINS(pDevIns);
3819 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3820 NOREF(GCPtr);
3821 NOREF(pHCPtr);
3822 return VERR_ACCESS_DENIED;
3823}
3824
3825
3826/** @copydoc PDMDEVHLP::pfnA20IsEnabled */
3827static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_A20IsEnabled(PPDMDEVINS pDevIns)
3828{
3829 PDMDEV_ASSERT_DEVINS(pDevIns);
3830 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3831 return false;
3832}
3833
3834
3835/** @copydoc PDMDEVHLP::pfnA20Set */
3836static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_A20Set(PPDMDEVINS pDevIns, bool fEnable)
3837{
3838 PDMDEV_ASSERT_DEVINS(pDevIns);
3839 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3840 NOREF(fEnable);
3841}
3842
3843
3844/** @copydoc PDMDEVHLP::pfnVMReset */
3845static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMReset(PPDMDEVINS pDevIns)
3846{
3847 PDMDEV_ASSERT_DEVINS(pDevIns);
3848 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3849 return VERR_ACCESS_DENIED;
3850}
3851
3852
3853/** @copydoc PDMDEVHLP::pfnVMSuspend */
3854static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspend(PPDMDEVINS pDevIns)
3855{
3856 PDMDEV_ASSERT_DEVINS(pDevIns);
3857 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3858 return VERR_ACCESS_DENIED;
3859}
3860
3861
3862/** @copydoc PDMDEVHLP::pfnVMPowerOff */
3863static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMPowerOff(PPDMDEVINS pDevIns)
3864{
3865 PDMDEV_ASSERT_DEVINS(pDevIns);
3866 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3867 return VERR_ACCESS_DENIED;
3868}
3869
3870
3871/** @copydoc PDMDEVHLP::pfnLockVM */
3872static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_LockVM(PPDMDEVINS pDevIns)
3873{
3874 PDMDEV_ASSERT_DEVINS(pDevIns);
3875 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3876 return VERR_ACCESS_DENIED;
3877}
3878
3879
3880/** @copydoc PDMDEVHLP::pfnUnlockVM */
3881static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_UnlockVM(PPDMDEVINS pDevIns)
3882{
3883 PDMDEV_ASSERT_DEVINS(pDevIns);
3884 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3885 return VERR_ACCESS_DENIED;
3886}
3887
3888
3889/** @copydoc PDMDEVHLP::pfnAssertVMLock */
3890static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_AssertVMLock(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
3891{
3892 PDMDEV_ASSERT_DEVINS(pDevIns);
3893 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3894 return false;
3895}
3896
3897
3898/** @copydoc PDMDEVHLP::pfnDMARegister */
3899static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser)
3900{
3901 PDMDEV_ASSERT_DEVINS(pDevIns);
3902 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3903 return VERR_ACCESS_DENIED;
3904}
3905
3906
3907/** @copydoc PDMDEVHLP::pfnDMAReadMemory */
3908static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead)
3909{
3910 PDMDEV_ASSERT_DEVINS(pDevIns);
3911 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3912 if (pcbRead)
3913 *pcbRead = 0;
3914 return VERR_ACCESS_DENIED;
3915}
3916
3917
3918/** @copydoc PDMDEVHLP::pfnDMAWriteMemory */
3919static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten)
3920{
3921 PDMDEV_ASSERT_DEVINS(pDevIns);
3922 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3923 if (pcbWritten)
3924 *pcbWritten = 0;
3925 return VERR_ACCESS_DENIED;
3926}
3927
3928
3929/** @copydoc PDMDEVHLP::pfnDMASetDREQ */
3930static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel)
3931{
3932 PDMDEV_ASSERT_DEVINS(pDevIns);
3933 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3934 return VERR_ACCESS_DENIED;
3935}
3936
3937
3938/** @copydoc PDMDEVHLP::pfnDMAGetChannelMode */
3939static DECLCALLBACK(uint8_t) pdmR3DevHlp_Untrusted_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel)
3940{
3941 PDMDEV_ASSERT_DEVINS(pDevIns);
3942 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3943 return 3 << 2 /* illegal mode type */;
3944}
3945
3946
3947/** @copydoc PDMDEVHLP::pfnDMASchedule */
3948static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_DMASchedule(PPDMDEVINS pDevIns)
3949{
3950 PDMDEV_ASSERT_DEVINS(pDevIns);
3951 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3952}
3953
3954
3955/** @copydoc PDMDEVHLP::pfnCMOSWrite */
3956static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value)
3957{
3958 PDMDEV_ASSERT_DEVINS(pDevIns);
3959 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3960 return VERR_ACCESS_DENIED;
3961}
3962
3963
3964/** @copydoc PDMDEVHLP::pfnCMOSRead */
3965static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value)
3966{
3967 PDMDEV_ASSERT_DEVINS(pDevIns);
3968 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3969 return VERR_ACCESS_DENIED;
3970}
3971
3972
3973/** @copydoc PDMDEVHLP::pfnGetCpuId */
3974static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
3975 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
3976{
3977 PDMDEV_ASSERT_DEVINS(pDevIns);
3978 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3979}
3980
3981
3982/** @copydoc PDMDEVHLP::pfnROMProtectShadow */
3983static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange)
3984{
3985 PDMDEV_ASSERT_DEVINS(pDevIns);
3986 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3987 return VERR_ACCESS_DENIED;
3988}
3989
3990
3991/** @copydoc PDMDEVHLP::pfnMMIO2Register */
3992static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc)
3993{
3994 PDMDEV_ASSERT_DEVINS(pDevIns);
3995 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
3996 return VERR_ACCESS_DENIED;
3997}
3998
3999
4000/** @copydoc PDMDEVHLP::pfnMMIO2Deregister */
4001static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion)
4002{
4003 PDMDEV_ASSERT_DEVINS(pDevIns);
4004 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4005 return VERR_ACCESS_DENIED;
4006}
4007
4008
4009/** @copydoc PDMDEVHLP::pfnMMIO2Map */
4010static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
4011{
4012 PDMDEV_ASSERT_DEVINS(pDevIns);
4013 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4014 return VERR_ACCESS_DENIED;
4015}
4016
4017
4018/** @copydoc PDMDEVHLP::pfnMMIO2Unmap */
4019static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
4020{
4021 PDMDEV_ASSERT_DEVINS(pDevIns);
4022 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4023 return VERR_ACCESS_DENIED;
4024}
4025
4026
4027/** @copydoc PDMDEVHLP::pfnMMHyperMapMMIO2 */
4028static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_MMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb, const char *pszDesc, PRTRCPTR pRCPtr)
4029{
4030 PDMDEV_ASSERT_DEVINS(pDevIns);
4031 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4032 return VERR_ACCESS_DENIED;
4033}
4034
4035
4036/** @copydoc PDMDEVHLP::pfnRegisterVMMDevHeap */
4037static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize)
4038{
4039 PDMDEV_ASSERT_DEVINS(pDevIns);
4040 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4041 return VERR_ACCESS_DENIED;
4042}
4043
4044
4045/** @copydoc PDMDEVHLP::pfnUnregisterVMMDevHeap */
4046static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys)
4047{
4048 PDMDEV_ASSERT_DEVINS(pDevIns);
4049 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
4050 return VERR_ACCESS_DENIED;
4051}
4052
4053
4054
4055
4056
4057/** @copydoc PDMPICHLPR3::pfnSetInterruptFF */
4058static DECLCALLBACK(void) pdmR3PicHlp_SetInterruptFF(PPDMDEVINS pDevIns)
4059{
4060 PDMDEV_ASSERT_DEVINS(pDevIns);
4061 PVM pVM = pDevIns->Internal.s.pVMHC;
4062 LogFlow(("pdmR3PicHlp_SetInterruptFF: caller='%s'/%d: VM_FF_INTERRUPT_PIC %d -> 1\n",
4063 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VMCPU_FF_ISSET(pVM, 0, VM_FF_INTERRUPT_PIC)));
4064 /* for PIC we always deliver to CPU 0, MP use APIC */
4065 VMCPU_FF_SET(pVM, 0, VM_FF_INTERRUPT_PIC);
4066 REMR3NotifyInterruptSet(pVM);
4067 VMR3NotifyFF(pVM, true);
4068}
4069
4070
4071/** @copydoc PDMPICHLPR3::pfnClearInterruptFF */
4072static DECLCALLBACK(void) pdmR3PicHlp_ClearInterruptFF(PPDMDEVINS pDevIns)
4073{
4074 PDMDEV_ASSERT_DEVINS(pDevIns);
4075 LogFlow(("pdmR3PicHlp_ClearInterruptFF: caller='%s'/%d: VM_FF_INTERRUPT_PIC %d -> 0\n",
4076 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VMCPU_FF_ISSET(pDevIns->Internal.s.pVMHC, 0, VM_FF_INTERRUPT_PIC)));
4077 /* for PIC we always deliver to CPU 0, MP use APIC */
4078 VMCPU_FF_CLEAR(pDevIns->Internal.s.pVMHC, 0, VM_FF_INTERRUPT_PIC);
4079 REMR3NotifyInterruptClear(pDevIns->Internal.s.pVMHC);
4080}
4081
4082
4083/** @copydoc PDMPICHLPR3::pfnLock */
4084static DECLCALLBACK(int) pdmR3PicHlp_Lock(PPDMDEVINS pDevIns, int rc)
4085{
4086 PDMDEV_ASSERT_DEVINS(pDevIns);
4087 return pdmLockEx(pDevIns->Internal.s.pVMHC, rc);
4088}
4089
4090
4091/** @copydoc PDMPICHLPR3::pfnUnlock */
4092static DECLCALLBACK(void) pdmR3PicHlp_Unlock(PPDMDEVINS pDevIns)
4093{
4094 PDMDEV_ASSERT_DEVINS(pDevIns);
4095 pdmUnlock(pDevIns->Internal.s.pVMHC);
4096}
4097
4098
4099/** @copydoc PDMPICHLPR3::pfnGetRCHelpers */
4100static DECLCALLBACK(PCPDMPICHLPRC) pdmR3PicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
4101{
4102 PDMDEV_ASSERT_DEVINS(pDevIns);
4103 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4104 RTRCPTR pRCHelpers = 0;
4105 int rc = PDMR3GetSymbolGC(pDevIns->Internal.s.pVMHC, NULL, "g_pdmRCPicHlp", &pRCHelpers);
4106 AssertReleaseRC(rc);
4107 AssertRelease(pRCHelpers);
4108 LogFlow(("pdmR3PicHlp_GetRCHelpers: caller='%s'/%d: returns %VGv\n",
4109 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pRCHelpers));
4110 return pRCHelpers;
4111}
4112
4113
4114/** @copydoc PDMPICHLPR3::pfnGetR0Helpers */
4115static DECLCALLBACK(PCPDMPICHLPR0) pdmR3PicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
4116{
4117 PDMDEV_ASSERT_DEVINS(pDevIns);
4118 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4119 PCPDMPICHLPR0 pR0Helpers = 0;
4120 int rc = PDMR3GetSymbolR0(pDevIns->Internal.s.pVMHC, NULL, "g_pdmR0PicHlp", &pR0Helpers);
4121 AssertReleaseRC(rc);
4122 AssertRelease(pR0Helpers);
4123 LogFlow(("pdmR3PicHlp_GetR0Helpers: caller='%s'/%d: returns %VHv\n",
4124 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pR0Helpers));
4125 return pR0Helpers;
4126}
4127
4128
4129/** @copydoc PDMAPICHLPR3::pfnSetInterruptFF */
4130static DECLCALLBACK(void) pdmR3ApicHlp_SetInterruptFF(PPDMDEVINS pDevIns, VMCPUID idCpu)
4131{
4132 PDMDEV_ASSERT_DEVINS(pDevIns);
4133 PVM pVM = pDevIns->Internal.s.pVMHC;
4134 LogFlow(("pdmR3ApicHlp_SetInterruptFF: caller='%s'/%d: VM_FF_INTERRUPT %d -> 1\n",
4135 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VMCPU_FF_ISSET(pVM, idCpu, VM_FF_INTERRUPT_APIC)));
4136 VMCPU_FF_SET(pVM, idCpu, VM_FF_INTERRUPT_APIC);
4137 REMR3NotifyInterruptSet(pVM);
4138 VMR3NotifyFF(pVM, true);
4139}
4140
4141
4142/** @copydoc PDMAPICHLPR3::pfnClearInterruptFF */
4143static DECLCALLBACK(void) pdmR3ApicHlp_ClearInterruptFF(PPDMDEVINS pDevIns, VMCPUID idCpu)
4144{
4145 PDMDEV_ASSERT_DEVINS(pDevIns);
4146 LogFlow(("pdmR3ApicHlp_ClearInterruptFF: caller='%s'/%d: VM_FF_INTERRUPT %d -> 0\n",
4147 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, VMCPU_FF_ISSET(pDevIns->Internal.s.pVMHC, idCpu, VM_FF_INTERRUPT_APIC)));
4148 VMCPU_FF_CLEAR(pDevIns->Internal.s.pVMHC, idCpu, VM_FF_INTERRUPT_APIC);
4149 REMR3NotifyInterruptClear(pDevIns->Internal.s.pVMHC);
4150}
4151
4152
4153/** @copydoc PDMAPICHLPR3::pfnChangeFeature */
4154static DECLCALLBACK(void) pdmR3ApicHlp_ChangeFeature(PPDMDEVINS pDevIns, bool fEnabled)
4155{
4156 PDMDEV_ASSERT_DEVINS(pDevIns);
4157 LogFlow(("pdmR3ApicHlp_ChangeFeature: caller='%s'/%d: fEnabled=%RTbool\n",
4158 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, fEnabled));
4159 if (fEnabled)
4160 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMHC, CPUMCPUIDFEATURE_APIC);
4161 else
4162 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMHC, CPUMCPUIDFEATURE_APIC);
4163}
4164
4165/** @copydoc PDMAPICHLPR3::pfnLock */
4166static DECLCALLBACK(int) pdmR3ApicHlp_Lock(PPDMDEVINS pDevIns, int rc)
4167{
4168 PDMDEV_ASSERT_DEVINS(pDevIns);
4169 return pdmLockEx(pDevIns->Internal.s.pVMHC, rc);
4170}
4171
4172
4173/** @copydoc PDMAPICHLPR3::pfnUnlock */
4174static DECLCALLBACK(void) pdmR3ApicHlp_Unlock(PPDMDEVINS pDevIns)
4175{
4176 PDMDEV_ASSERT_DEVINS(pDevIns);
4177 pdmUnlock(pDevIns->Internal.s.pVMHC);
4178}
4179
4180
4181/** @copydoc PDMAPICHLPR3::pfnGetCpuId */
4182static DECLCALLBACK(VMCPUID) pdmR3ApicHlp_GetCpuId(PPDMDEVINS pDevIns)
4183{
4184 PDMDEV_ASSERT_DEVINS(pDevIns);
4185 return VMMGetCpuId(pDevIns->Internal.s.pVMHC);
4186}
4187
4188
4189/** @copydoc PDMAPICHLPR3::pfnGetRCHelpers */
4190static DECLCALLBACK(PCPDMAPICHLPRC) pdmR3ApicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
4191{
4192 PDMDEV_ASSERT_DEVINS(pDevIns);
4193 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4194 RTRCPTR pRCHelpers = 0;
4195 int rc = PDMR3GetSymbolGC(pDevIns->Internal.s.pVMHC, NULL, "g_pdmRCApicHlp", &pRCHelpers);
4196 AssertReleaseRC(rc);
4197 AssertRelease(pRCHelpers);
4198 LogFlow(("pdmR3ApicHlp_GetRCHelpers: caller='%s'/%d: returns %VGv\n",
4199 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pRCHelpers));
4200 return pRCHelpers;
4201}
4202
4203
4204/** @copydoc PDMAPICHLPR3::pfnGetR0Helpers */
4205static DECLCALLBACK(PCPDMAPICHLPR0) pdmR3ApicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
4206{
4207 PDMDEV_ASSERT_DEVINS(pDevIns);
4208 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4209 PCPDMAPICHLPR0 pR0Helpers = 0;
4210 int rc = PDMR3GetSymbolR0(pDevIns->Internal.s.pVMHC, NULL, "g_pdmR0ApicHlp", &pR0Helpers);
4211 AssertReleaseRC(rc);
4212 AssertRelease(pR0Helpers);
4213 LogFlow(("pdmR3ApicHlp_GetR0Helpers: caller='%s'/%d: returns %VHv\n",
4214 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pR0Helpers));
4215 return pR0Helpers;
4216}
4217
4218
4219/** @copydoc PDMIOAPICHLPR3::pfnApicBusDeliver */
4220static DECLCALLBACK(void) pdmR3IoApicHlp_ApicBusDeliver(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
4221 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode)
4222{
4223 PDMDEV_ASSERT_DEVINS(pDevIns);
4224 PVM pVM = pDevIns->Internal.s.pVMHC;
4225 LogFlow(("pdmR3IoApicHlp_ApicBusDeliver: caller='%s'/%d: u8Dest=%RX8 u8DestMode=%RX8 u8DeliveryMode=%RX8 iVector=%RX8 u8Polarity=%RX8 u8TriggerMode=%RX8\n",
4226 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode));
4227 if (pVM->pdm.s.Apic.pfnBusDeliverR3)
4228 pVM->pdm.s.Apic.pfnBusDeliverR3(pVM->pdm.s.Apic.pDevInsR3, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode);
4229}
4230
4231
4232/** @copydoc PDMIOAPICHLPR3::pfnLock */
4233static DECLCALLBACK(int) pdmR3IoApicHlp_Lock(PPDMDEVINS pDevIns, int rc)
4234{
4235 PDMDEV_ASSERT_DEVINS(pDevIns);
4236 return pdmLockEx(pDevIns->Internal.s.pVMHC, rc);
4237}
4238
4239
4240/** @copydoc PDMIOAPICHLPR3::pfnUnlock */
4241static DECLCALLBACK(void) pdmR3IoApicHlp_Unlock(PPDMDEVINS pDevIns)
4242{
4243 PDMDEV_ASSERT_DEVINS(pDevIns);
4244 pdmUnlock(pDevIns->Internal.s.pVMHC);
4245}
4246
4247
4248/** @copydoc PDMIOAPICHLPR3::pfnGetRCHelpers */
4249static DECLCALLBACK(PCPDMIOAPICHLPRC) pdmR3IoApicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
4250{
4251 PDMDEV_ASSERT_DEVINS(pDevIns);
4252 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4253 RTRCPTR pRCHelpers = 0;
4254 int rc = PDMR3GetSymbolGC(pDevIns->Internal.s.pVMHC, NULL, "g_pdmRCIoApicHlp", &pRCHelpers);
4255 AssertReleaseRC(rc);
4256 AssertRelease(pRCHelpers);
4257 LogFlow(("pdmR3IoApicHlp_GetRCHelpers: caller='%s'/%d: returns %VGv\n",
4258 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pRCHelpers));
4259 return pRCHelpers;
4260}
4261
4262
4263/** @copydoc PDMIOAPICHLPR3::pfnGetR0Helpers */
4264static DECLCALLBACK(PCPDMIOAPICHLPR0) pdmR3IoApicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
4265{
4266 PDMDEV_ASSERT_DEVINS(pDevIns);
4267 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4268 PCPDMIOAPICHLPR0 pR0Helpers = 0;
4269 int rc = PDMR3GetSymbolR0(pDevIns->Internal.s.pVMHC, NULL, "g_pdmR0IoApicHlp", &pR0Helpers);
4270 AssertReleaseRC(rc);
4271 AssertRelease(pR0Helpers);
4272 LogFlow(("pdmR3IoApicHlp_GetR0Helpers: caller='%s'/%d: returns %VHv\n",
4273 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pR0Helpers));
4274 return pR0Helpers;
4275}
4276
4277
4278/** @copydoc PDMPCIHLPR3::pfnIsaSetIrq */
4279static DECLCALLBACK(void) pdmR3PciHlp_IsaSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
4280{
4281 PDMDEV_ASSERT_DEVINS(pDevIns);
4282 Log4(("pdmR3PciHlp_IsaSetIrq: iIrq=%d iLevel=%d\n", iIrq, iLevel));
4283 PDMIsaSetIrq(pDevIns->Internal.s.pVMHC, iIrq, iLevel);
4284}
4285
4286
4287/** @copydoc PDMPCIHLPR3::pfnIoApicSetIrq */
4288static DECLCALLBACK(void) pdmR3PciHlp_IoApicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
4289{
4290 PDMDEV_ASSERT_DEVINS(pDevIns);
4291 Log4(("pdmR3PciHlp_IsaSetIrq: iIrq=%d iLevel=%d\n", iIrq, iLevel));
4292 PDMIoApicSetIrq(pDevIns->Internal.s.pVMHC, iIrq, iLevel);
4293}
4294
4295
4296/** @copydoc PDMPCIHLPR3::pfnIsMMIO2Base */
4297static DECLCALLBACK(bool) pdmR3PciHlp_IsMMIO2Base(PPDMDEVINS pDevIns, PPDMDEVINS pOwner, RTGCPHYS GCPhys)
4298{
4299 PDMDEV_ASSERT_DEVINS(pDevIns);
4300 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4301 bool fRc = PGMR3PhysMMIO2IsBase(pDevIns->Internal.s.pVMHC, pOwner, GCPhys);
4302 Log4(("pdmR3PciHlp_IsMMIO2Base: pOwner=%p GCPhys=%RGp -> %RTbool\n", pOwner, GCPhys, fRc));
4303 return fRc;
4304}
4305
4306
4307/** @copydoc PDMPCIHLPR3::pfnLock */
4308static DECLCALLBACK(int) pdmR3PciHlp_Lock(PPDMDEVINS pDevIns, int rc)
4309{
4310 PDMDEV_ASSERT_DEVINS(pDevIns);
4311 return pdmLockEx(pDevIns->Internal.s.pVMHC, rc);
4312}
4313
4314
4315/** @copydoc PDMPCIHLPR3::pfnUnlock */
4316static DECLCALLBACK(void) pdmR3PciHlp_Unlock(PPDMDEVINS pDevIns)
4317{
4318 PDMDEV_ASSERT_DEVINS(pDevIns);
4319 pdmUnlock(pDevIns->Internal.s.pVMHC);
4320}
4321
4322
4323/** @copydoc PDMPCIHLPR3::pfnGetRCHelpers */
4324static DECLCALLBACK(PCPDMPCIHLPRC) pdmR3PciHlp_GetRCHelpers(PPDMDEVINS pDevIns)
4325{
4326 PDMDEV_ASSERT_DEVINS(pDevIns);
4327 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4328 RTRCPTR pRCHelpers = 0;
4329 int rc = PDMR3GetSymbolGC(pDevIns->Internal.s.pVMHC, NULL, "g_pdmRCPciHlp", &pRCHelpers);
4330 AssertReleaseRC(rc);
4331 AssertRelease(pRCHelpers);
4332 LogFlow(("pdmR3IoApicHlp_GetGCHelpers: caller='%s'/%d: returns %RRv\n",
4333 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pRCHelpers));
4334 return pRCHelpers;
4335}
4336
4337
4338/** @copydoc PDMPCIHLPR3::pfnGetR0Helpers */
4339static DECLCALLBACK(PCPDMPCIHLPR0) pdmR3PciHlp_GetR0Helpers(PPDMDEVINS pDevIns)
4340{
4341 PDMDEV_ASSERT_DEVINS(pDevIns);
4342 VM_ASSERT_EMT(pDevIns->Internal.s.pVMHC);
4343 PCPDMPCIHLPR0 pR0Helpers = 0;
4344 int rc = PDMR3GetSymbolR0(pDevIns->Internal.s.pVMHC, NULL, "g_pdmR0PciHlp", &pR0Helpers);
4345 AssertReleaseRC(rc);
4346 AssertRelease(pR0Helpers);
4347 LogFlow(("pdmR3IoApicHlp_GetR0Helpers: caller='%s'/%d: returns %VHv\n",
4348 pDevIns->pDevReg->szDeviceName, pDevIns->iInstance, pR0Helpers));
4349 return pR0Helpers;
4350}
4351
4352
4353/**
4354 * Locates a LUN.
4355 *
4356 * @returns VBox status code.
4357 * @param pVM VM Handle.
4358 * @param pszDevice Device name.
4359 * @param iInstance Device instance.
4360 * @param iLun The Logical Unit to obtain the interface of.
4361 * @param ppLun Where to store the pointer to the LUN if found.
4362 * @thread Try only do this in EMT...
4363 */
4364int pdmR3DevFindLun(PVM pVM, const char *pszDevice, unsigned iInstance, unsigned iLun, PPDMLUN *ppLun)
4365{
4366 /*
4367 * Iterate registered devices looking for the device.
4368 */
4369 RTUINT cchDevice = strlen(pszDevice);
4370 for (PPDMDEV pDev = pVM->pdm.s.pDevs; pDev; pDev = pDev->pNext)
4371 {
4372 if ( pDev->cchName == cchDevice
4373 && !memcmp(pDev->pDevReg->szDeviceName, pszDevice, cchDevice))
4374 {
4375 /*
4376 * Iterate device instances.
4377 */
4378 for (PPDMDEVINS pDevIns = pDev->pInstances; pDevIns; pDevIns = pDevIns->Internal.s.pPerDeviceNextHC)
4379 {
4380 if (pDevIns->iInstance == iInstance)
4381 {
4382 /*
4383 * Iterate luns.
4384 */
4385 for (PPDMLUN pLun = pDevIns->Internal.s.pLunsHC; pLun; pLun = pLun->pNext)
4386 {
4387 if (pLun->iLun == iLun)
4388 {
4389 *ppLun = pLun;
4390 return VINF_SUCCESS;
4391 }
4392 }
4393 return VERR_PDM_LUN_NOT_FOUND;
4394 }
4395 }
4396 return VERR_PDM_DEVICE_INSTANCE_NOT_FOUND;
4397 }
4398 }
4399 return VERR_PDM_DEVICE_NOT_FOUND;
4400}
4401
4402
4403/**
4404 * Attaches a preconfigured driver to an existing device instance.
4405 *
4406 * This is used to change drivers and suchlike at runtime.
4407 *
4408 * @returns VBox status code.
4409 * @param pVM VM Handle.
4410 * @param pszDevice Device name.
4411 * @param iInstance Device instance.
4412 * @param iLun The Logical Unit to obtain the interface of.
4413 * @param ppBase Where to store the base interface pointer. Optional.
4414 * @thread EMT
4415 */
4416PDMR3DECL(int) PDMR3DeviceAttach(PVM pVM, const char *pszDevice, unsigned iInstance, unsigned iLun, PPDMIBASE *ppBase)
4417{
4418 VM_ASSERT_EMT(pVM);
4419 LogFlow(("PDMR3DeviceAttach: pszDevice=%p:{%s} iInstance=%d iLun=%d ppBase=%p\n",
4420 pszDevice, pszDevice, iInstance, iLun, ppBase));
4421
4422 /*
4423 * Find the LUN in question.
4424 */
4425 PPDMLUN pLun;
4426 int rc = pdmR3DevFindLun(pVM, pszDevice, iInstance, iLun, &pLun);
4427 if (VBOX_SUCCESS(rc))
4428 {
4429 /*
4430 * Can we attach anything at runtime?
4431 */
4432 PPDMDEVINS pDevIns = pLun->pDevIns;
4433 if (pDevIns->pDevReg->pfnAttach)
4434 {
4435 if (!pLun->pTop)
4436 {
4437 rc = pDevIns->pDevReg->pfnAttach(pDevIns, iLun);
4438
4439 }
4440 else
4441 rc = VERR_PDM_DRIVER_ALREADY_ATTACHED;
4442 }
4443 else
4444 rc = VERR_PDM_DEVICE_NO_RT_ATTACH;
4445
4446 if (ppBase)
4447 *ppBase = pLun->pTop ? &pLun->pTop->IBase : NULL;
4448 }
4449 else if (ppBase)
4450 *ppBase = NULL;
4451
4452 if (ppBase)
4453 LogFlow(("PDMR3DeviceAttach: returns %Vrc *ppBase=%p\n", rc, *ppBase));
4454 else
4455 LogFlow(("PDMR3DeviceAttach: returns %Vrc\n", rc));
4456 return rc;
4457}
4458
4459
4460/**
4461 * Detaches a driver chain from an existing device instance.
4462 *
4463 * This is used to change drivers and suchlike at runtime.
4464 *
4465 * @returns VBox status code.
4466 * @param pVM VM Handle.
4467 * @param pszDevice Device name.
4468 * @param iInstance Device instance.
4469 * @param iLun The Logical Unit to obtain the interface of.
4470 * @thread EMT
4471 */
4472PDMR3DECL(int) PDMR3DeviceDetach(PVM pVM, const char *pszDevice, unsigned iInstance, unsigned iLun)
4473{
4474 VM_ASSERT_EMT(pVM);
4475 LogFlow(("PDMR3DeviceDetach: pszDevice=%p:{%s} iInstance=%d iLun=%d\n",
4476 pszDevice, pszDevice, iInstance, iLun));
4477
4478 /*
4479 * Find the LUN in question.
4480 */
4481 PPDMLUN pLun;
4482 int rc = pdmR3DevFindLun(pVM, pszDevice, iInstance, iLun, &pLun);
4483 if (VBOX_SUCCESS(rc))
4484 {
4485 /*
4486 * Can we detach anything at runtime?
4487 */
4488 PPDMDEVINS pDevIns = pLun->pDevIns;
4489 if (pDevIns->pDevReg->pfnDetach)
4490 {
4491 if (pLun->pTop)
4492 rc = pdmR3DrvDetach(pLun->pTop);
4493 else
4494 rc = VINF_PDM_NO_DRIVER_ATTACHED_TO_LUN;
4495 }
4496 else
4497 rc = VERR_PDM_DEVICE_NO_RT_DETACH;
4498 }
4499
4500 LogFlow(("PDMR3DeviceDetach: returns %Vrc\n", rc));
4501 return rc;
4502}
4503
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