VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/PDMDevHlp.cpp@ 82011

Last change on this file since 82011 was 81961, checked in by vboxsync, 5 years ago

DevHPET,PDM: Split structures and refactored registration. bugref:9218

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 204.0 KB
Line 
1/* $Id: PDMDevHlp.cpp 81961 2019-11-18 19:06:25Z vboxsync $ */
2/** @file
3 * PDM - Pluggable Device and Driver Manager, Device Helpers.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_PDM_DEVICE
23#define PDMPCIDEV_INCLUDE_PRIVATE /* Hack to get pdmpcidevint.h included at the right point. */
24#include "PDMInternal.h"
25#include <VBox/vmm/pdm.h>
26#include <VBox/vmm/mm.h>
27#include <VBox/vmm/hm.h>
28#include <VBox/vmm/pgm.h>
29#include <VBox/vmm/iom.h>
30#include <VBox/vmm/dbgf.h>
31#include <VBox/vmm/ssm.h>
32#include <VBox/vmm/vmapi.h>
33#include <VBox/vmm/vmm.h>
34#include <VBox/vmm/vmcc.h>
35
36#include <VBox/version.h>
37#include <VBox/log.h>
38#include <VBox/err.h>
39#include <iprt/asm.h>
40#include <iprt/assert.h>
41#include <iprt/ctype.h>
42#include <iprt/string.h>
43#include <iprt/thread.h>
44
45#include "dtrace/VBoxVMM.h"
46#include "PDMInline.h"
47
48
49/*********************************************************************************************************************************
50* Defined Constants And Macros *
51*********************************************************************************************************************************/
52/** @def PDM_DEVHLP_DEADLOCK_DETECTION
53 * Define this to enable the deadlock detection when accessing physical memory.
54 */
55#if /*defined(DEBUG_bird) ||*/ defined(DOXYGEN_RUNNING)
56# define PDM_DEVHLP_DEADLOCK_DETECTION /**< @todo enable DevHlp deadlock detection! */
57#endif
58
59
60
61/**
62 * Wrapper around PDMR3LdrGetSymbolR0Lazy.
63 */
64DECLINLINE(int) pdmR3DevGetSymbolR0Lazy(PPDMDEVINS pDevIns, const char *pszSymbol, PRTR0PTR ppvValue)
65{
66 return PDMR3LdrGetSymbolR0Lazy(pDevIns->Internal.s.pVMR3,
67 pDevIns->Internal.s.pDevR3->pReg->pszR0Mod,
68 pDevIns->Internal.s.pDevR3->pszR0SearchPath,
69 pszSymbol, ppvValue);
70}
71
72
73/** @name R3 DevHlp
74 * @{
75 */
76
77
78/** @interface_method_impl{PDMDEVHLPR3,pfnIoPortCreateEx} */
79static DECLCALLBACK(int) pdmR3DevHlp_IoPortCreateEx(PPDMDEVINS pDevIns, RTIOPORT cPorts, uint32_t fFlags, PPDMPCIDEV pPciDev,
80 uint32_t iPciRegion, PFNIOMIOPORTNEWOUT pfnOut, PFNIOMIOPORTNEWIN pfnIn,
81 PFNIOMIOPORTNEWOUTSTRING pfnOutStr, PFNIOMIOPORTNEWINSTRING pfnInStr, RTR3PTR pvUser,
82 const char *pszDesc, PCIOMIOPORTDESC paExtDescs, PIOMIOPORTHANDLE phIoPorts)
83{
84 PDMDEV_ASSERT_DEVINS(pDevIns);
85 LogFlow(("pdmR3DevHlp_IoPortCreateEx: caller='%s'/%d: cPorts=%#x fFlags=%#x pPciDev=%p iPciRegion=%#x pfnOut=%p pfnIn=%p pfnOutStr=%p pfnInStr=%p pvUser=%p pszDesc=%p:{%s} paExtDescs=%p phIoPorts=%p\n",
86 pDevIns->pReg->szName, pDevIns->iInstance, cPorts, fFlags, pPciDev, iPciRegion, pfnOut, pfnIn, pfnOutStr, pfnInStr,
87 pvUser, pszDesc, pszDesc, paExtDescs, phIoPorts));
88 PVM pVM = pDevIns->Internal.s.pVMR3;
89 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
90 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_VM_INVALID_VM_STATE);
91
92 int rc = IOMR3IoPortCreate(pVM, pDevIns, cPorts, fFlags, pPciDev, iPciRegion,
93 pfnOut, pfnIn, pfnOutStr, pfnInStr, pvUser, pszDesc, paExtDescs, phIoPorts);
94
95 LogFlow(("pdmR3DevHlp_IoPortCreateEx: caller='%s'/%d: returns %Rrc (*phIoPorts=%#x)\n",
96 pDevIns->pReg->szName, pDevIns->iInstance, rc, *phIoPorts));
97 return rc;
98}
99
100
101/** @interface_method_impl{PDMDEVHLPR3,pfnIoPortMap} */
102static DECLCALLBACK(int) pdmR3DevHlp_IoPortMap(PPDMDEVINS pDevIns, IOMIOPORTHANDLE hIoPorts, RTIOPORT Port)
103{
104 PDMDEV_ASSERT_DEVINS(pDevIns);
105 LogFlow(("pdmR3DevHlp_IoPortMap: caller='%s'/%d: hIoPorts=%#x Port=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hIoPorts, Port));
106 PVM pVM = pDevIns->Internal.s.pVMR3;
107 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
108
109 int rc = IOMR3IoPortMap(pVM, pDevIns, hIoPorts, Port);
110
111 LogFlow(("pdmR3DevHlp_IoPortMap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
112 return rc;
113}
114
115
116/** @interface_method_impl{PDMDEVHLPR3,pfnIoPortUnmap} */
117static DECLCALLBACK(int) pdmR3DevHlp_IoPortUnmap(PPDMDEVINS pDevIns, IOMIOPORTHANDLE hIoPorts)
118{
119 PDMDEV_ASSERT_DEVINS(pDevIns);
120 LogFlow(("pdmR3DevHlp_IoPortMap: caller='%s'/%d: hIoPorts=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hIoPorts));
121 PVM pVM = pDevIns->Internal.s.pVMR3;
122 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
123
124 int rc = IOMR3IoPortUnmap(pVM, pDevIns, hIoPorts);
125
126 LogFlow(("pdmR3DevHlp_IoPortMap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
127 return rc;
128}
129
130
131/** @interface_method_impl{PDMDEVHLPR3,pfnIoPortGetMappingAddress} */
132static DECLCALLBACK(uint32_t) pdmR3DevHlp_IoPortGetMappingAddress(PPDMDEVINS pDevIns, IOMIOPORTHANDLE hIoPorts)
133{
134 PDMDEV_ASSERT_DEVINS(pDevIns);
135 LogFlow(("pdmR3DevHlp_IoPortGetMappingAddress: caller='%s'/%d: hIoPorts=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hIoPorts));
136
137 uint32_t uAddress = IOMR3IoPortGetMappingAddress(pDevIns->Internal.s.pVMR3, pDevIns, hIoPorts);
138
139 LogFlow(("pdmR3DevHlp_IoPortGetMappingAddress: caller='%s'/%d: returns %#RX32\n", pDevIns->pReg->szName, pDevIns->iInstance, uAddress));
140 return uAddress;
141}
142
143
144/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegister} */
145static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegister(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTHCPTR pvUser, PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn,
146 PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc)
147{
148 PDMDEV_ASSERT_DEVINS(pDevIns);
149 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: Port=%#x cPorts=%#x pvUser=%p pfnOut=%p pfnIn=%p pfnOutStr=%p pfnInStr=%p pszDesc=%p:{%s}\n", pDevIns->pReg->szName, pDevIns->iInstance,
150 Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc, pszDesc));
151 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
152
153#if 0 /** @todo needs a real string cache for this */
154 if (pDevIns->iInstance > 0)
155 {
156 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
157 if (pszDesc2)
158 pszDesc = pszDesc2;
159 }
160#endif
161
162 int rc = IOMR3IOPortRegisterR3(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts, pvUser,
163 pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc);
164
165 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
166 return rc;
167}
168
169
170/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegisterRC} */
171static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterRC(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTRCPTR pvUser,
172 const char *pszOut, const char *pszIn,
173 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
174{
175 PDMDEV_ASSERT_DEVINS(pDevIns);
176 Assert(pDevIns->pReg->pszRCMod[0]);
177 Assert(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC);
178 LogFlow(("pdmR3DevHlp_IOPortRegisterRC: 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->pReg->szName, pDevIns->iInstance,
179 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
180
181#if 0
182 /*
183 * Resolve the functions (one of the can be NULL).
184 */
185 PVM pVM = pDevIns->Internal.s.pVMR3;
186 VM_ASSERT_EMT(pVM);
187 int rc = VINF_SUCCESS;
188 if ( pDevIns->pReg->pszRCMod[0]
189 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
190 && VM_IS_RAW_MODE_ENABLED(pVM))
191 {
192 RTRCPTR RCPtrIn = NIL_RTRCPTR;
193 if (pszIn)
194 {
195 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszIn, &RCPtrIn);
196 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pReg->pszRCMod, pszIn));
197 }
198 RTRCPTR RCPtrOut = NIL_RTRCPTR;
199 if (pszOut && RT_SUCCESS(rc))
200 {
201 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszOut, &RCPtrOut);
202 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pReg->pszRCMod, pszOut));
203 }
204 RTRCPTR RCPtrInStr = NIL_RTRCPTR;
205 if (pszInStr && RT_SUCCESS(rc))
206 {
207 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszInStr, &RCPtrInStr);
208 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pReg->pszRCMod, pszInStr));
209 }
210 RTRCPTR RCPtrOutStr = NIL_RTRCPTR;
211 if (pszOutStr && RT_SUCCESS(rc))
212 {
213 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszOutStr, &RCPtrOutStr);
214 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pReg->pszRCMod, pszOutStr));
215 }
216
217 if (RT_SUCCESS(rc))
218 {
219#if 0 /** @todo needs a real string cache for this */
220 if (pDevIns->iInstance > 0)
221 {
222 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
223 if (pszDesc2)
224 pszDesc = pszDesc2;
225 }
226#endif
227
228 rc = IOMR3IOPortRegisterRC(pVM, pDevIns, Port, cPorts, pvUser, RCPtrOut, RCPtrIn, RCPtrOutStr, RCPtrInStr, pszDesc);
229 }
230 }
231 else if (VM_IS_RAW_MODE_ENABLED(pVM))
232 {
233 AssertMsgFailed(("No RC module for this driver!\n"));
234 rc = VERR_INVALID_PARAMETER;
235 }
236#else
237 RT_NOREF(pDevIns, Port, cPorts, pvUser, pszOut, pszIn, pszOutStr, pszInStr, pszDesc);
238 int rc = VINF_SUCCESS;
239#endif
240
241 LogFlow(("pdmR3DevHlp_IOPortRegisterRC: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
242 return rc;
243}
244
245
246/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegisterR0} */
247static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterR0(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTR0PTR pvUser,
248 const char *pszOut, const char *pszIn,
249 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
250{
251 PDMDEV_ASSERT_DEVINS(pDevIns);
252 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
253 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->pReg->szName, pDevIns->iInstance,
254 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
255
256 /*
257 * Resolve the functions (one of the can be NULL).
258 */
259 int rc = VINF_SUCCESS;
260 if ( pDevIns->pReg->pszR0Mod[0]
261 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0))
262 {
263 R0PTRTYPE(PFNIOMIOPORTIN) pfnR0PtrIn = 0;
264 if (pszIn)
265 {
266 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszIn, &pfnR0PtrIn);
267 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pReg->pszR0Mod, pszIn));
268 }
269 R0PTRTYPE(PFNIOMIOPORTOUT) pfnR0PtrOut = 0;
270 if (pszOut && RT_SUCCESS(rc))
271 {
272 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszOut, &pfnR0PtrOut);
273 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pReg->pszR0Mod, pszOut));
274 }
275 R0PTRTYPE(PFNIOMIOPORTINSTRING) pfnR0PtrInStr = 0;
276 if (pszInStr && RT_SUCCESS(rc))
277 {
278 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszInStr, &pfnR0PtrInStr);
279 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pReg->pszR0Mod, pszInStr));
280 }
281 R0PTRTYPE(PFNIOMIOPORTOUTSTRING) pfnR0PtrOutStr = 0;
282 if (pszOutStr && RT_SUCCESS(rc))
283 {
284 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszOutStr, &pfnR0PtrOutStr);
285 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pReg->pszR0Mod, pszOutStr));
286 }
287
288 if (RT_SUCCESS(rc))
289 {
290#if 0 /** @todo needs a real string cache for this */
291 if (pDevIns->iInstance > 0)
292 {
293 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
294 if (pszDesc2)
295 pszDesc = pszDesc2;
296 }
297#endif
298
299 rc = IOMR3IOPortRegisterR0(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts, pvUser, pfnR0PtrOut, pfnR0PtrIn, pfnR0PtrOutStr, pfnR0PtrInStr, pszDesc);
300 }
301 }
302 else
303 {
304 AssertMsgFailed(("No R0 module for this driver!\n"));
305 rc = VERR_INVALID_PARAMETER;
306 }
307
308 LogFlow(("pdmR3DevHlp_IOPortRegisterR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
309 return rc;
310}
311
312
313/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortDeregister} */
314static DECLCALLBACK(int) pdmR3DevHlp_IOPortDeregister(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts)
315{
316 PDMDEV_ASSERT_DEVINS(pDevIns);
317 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
318 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: Port=%#x cPorts=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance,
319 Port, cPorts));
320
321 int rc = IOMR3IOPortDeregister(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts);
322
323 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
324 return rc;
325}
326
327
328/** @interface_method_impl{PDMDEVHLPR3,pfnMmioCreateEx} */
329static DECLCALLBACK(int) pdmR3DevHlp_MmioCreateEx(PPDMDEVINS pDevIns, RTGCPHYS cbRegion,
330 uint32_t fFlags, PPDMPCIDEV pPciDev, uint32_t iPciRegion,
331 PFNIOMMMIONEWWRITE pfnWrite, PFNIOMMMIONEWREAD pfnRead, PFNIOMMMIONEWFILL pfnFill,
332 void *pvUser, const char *pszDesc, PIOMMMIOHANDLE phRegion)
333{
334 PDMDEV_ASSERT_DEVINS(pDevIns);
335 LogFlow(("pdmR3DevHlp_MmioCreateEx: caller='%s'/%d: cbRegion=%#RGp fFlags=%#x pPciDev=%p iPciRegion=%#x pfnWrite=%p pfnRead=%p pfnFill=%p pvUser=%p pszDesc=%p:{%s} phRegion=%p\n",
336 pDevIns->pReg->szName, pDevIns->iInstance, cbRegion, fFlags, pPciDev, iPciRegion, pfnWrite, pfnRead, pfnFill, pvUser, pszDesc, pszDesc, phRegion));
337 PVM pVM = pDevIns->Internal.s.pVMR3;
338 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
339 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_VM_INVALID_VM_STATE);
340
341 /* HACK ALERT! Round the size up to page size. The PCI bus should do something similar before mapping it. */
342 /** @todo It's possible we need to do dummy MMIO fill-in of the PCI bus or
343 * guest adds more alignment to an region. */
344 cbRegion = RT_ALIGN_T(cbRegion, PAGE_SIZE, RTGCPHYS);
345
346 int rc = IOMR3MmioCreate(pVM, pDevIns, cbRegion, fFlags, pPciDev, iPciRegion,
347 pfnWrite, pfnRead, pfnFill, pvUser, pszDesc, phRegion);
348
349 LogFlow(("pdmR3DevHlp_MmioCreateEx: caller='%s'/%d: returns %Rrc (*phRegion=%#x)\n",
350 pDevIns->pReg->szName, pDevIns->iInstance, rc, *phRegion));
351 return rc;
352}
353
354
355/** @interface_method_impl{PDMDEVHLPR3,pfnMmioMap} */
356static DECLCALLBACK(int) pdmR3DevHlp_MmioMap(PPDMDEVINS pDevIns, IOMMMIOHANDLE hRegion, RTGCPHYS GCPhys)
357{
358 PDMDEV_ASSERT_DEVINS(pDevIns);
359 LogFlow(("pdmR3DevHlp_MmioMap: caller='%s'/%d: hRegion=%#x GCPhys=%#RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion, GCPhys));
360 PVM pVM = pDevIns->Internal.s.pVMR3;
361 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
362
363 int rc = IOMR3MmioMap(pVM, pDevIns, hRegion, GCPhys);
364
365 LogFlow(("pdmR3DevHlp_MmioMap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
366 return rc;
367}
368
369
370/** @interface_method_impl{PDMDEVHLPR3,pfnMmioUnmap} */
371static DECLCALLBACK(int) pdmR3DevHlp_MmioUnmap(PPDMDEVINS pDevIns, IOMMMIOHANDLE hRegion)
372{
373 PDMDEV_ASSERT_DEVINS(pDevIns);
374 LogFlow(("pdmR3DevHlp_MmioUnmap: caller='%s'/%d: hRegion=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion));
375 PVM pVM = pDevIns->Internal.s.pVMR3;
376 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
377
378 int rc = IOMR3MmioUnmap(pVM, pDevIns, hRegion);
379
380 LogFlow(("pdmR3DevHlp_MmioUnmap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
381 return rc;
382}
383
384
385/** @interface_method_impl{PDMDEVHLPR3,pfnMmioReduce} */
386static DECLCALLBACK(int) pdmR3DevHlp_MmioReduce(PPDMDEVINS pDevIns, IOMMMIOHANDLE hRegion, RTGCPHYS cbRegion)
387{
388 PDMDEV_ASSERT_DEVINS(pDevIns);
389 LogFlow(("pdmR3DevHlp_MmioReduce: caller='%s'/%d: hRegion=%#x cbRegion=%#RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion, cbRegion));
390 PVM pVM = pDevIns->Internal.s.pVMR3;
391 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
392 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_LOADING, VERR_VM_INVALID_VM_STATE);
393
394 int rc = IOMR3MmioReduce(pVM, pDevIns, hRegion, cbRegion);
395
396 LogFlow(("pdmR3DevHlp_MmioReduce: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
397 return rc;
398}
399
400
401/** @interface_method_impl{PDMDEVHLPR3,pfnMmioGetMappingAddress} */
402static DECLCALLBACK(RTGCPHYS) pdmR3DevHlp_MmioGetMappingAddress(PPDMDEVINS pDevIns, IOMMMIOHANDLE hRegion)
403{
404 PDMDEV_ASSERT_DEVINS(pDevIns);
405 LogFlow(("pdmR3DevHlp_MmioGetMappingAddress: caller='%s'/%d: hRegion=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion));
406
407 RTGCPHYS GCPhys = IOMR3MmioGetMappingAddress(pDevIns->Internal.s.pVMR3, pDevIns, hRegion);
408
409 LogFlow(("pdmR3DevHlp_MmioGetMappingAddress: caller='%s'/%d: returns %RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, GCPhys));
410 return GCPhys;
411}
412
413
414/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegister} */
415static DECLCALLBACK(int) pdmR3DevHlp_MMIORegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTGCPHYS cbRange, RTHCPTR pvUser,
416 PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, PFNIOMMMIOFILL pfnFill,
417 uint32_t fFlags, const char *pszDesc)
418{
419 PDMDEV_ASSERT_DEVINS(pDevIns);
420 PVM pVM = pDevIns->Internal.s.pVMR3;
421 VM_ASSERT_EMT(pVM);
422 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%RGp pvUser=%p pfnWrite=%p pfnRead=%p pfnFill=%p fFlags=%#x pszDesc=%p:{%s}\n",
423 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill, pszDesc, fFlags, pszDesc));
424
425 if (pDevIns->iInstance > 0)
426 {
427 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
428 if (pszDesc2)
429 pszDesc = pszDesc2;
430 }
431
432 int rc = IOMR3MmioRegisterR3(pVM, pDevIns, GCPhysStart, cbRange, pvUser,
433 pfnWrite, pfnRead, pfnFill, fFlags, pszDesc);
434
435 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
436 return rc;
437}
438
439
440/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegisterRC} */
441static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterRC(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTGCPHYS cbRange, RTRCPTR pvUser,
442 const char *pszWrite, const char *pszRead, const char *pszFill)
443{
444 PDMDEV_ASSERT_DEVINS(pDevIns);
445 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
446 Assert(pDevIns->pReg->pszR0Mod[0]);
447 Assert(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0);
448 LogFlow(("pdmR3DevHlp_MMIORegisterRC: caller='%s'/%d: GCPhysStart=%RGp cbRange=%RGp pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
449 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
450
451#if 0
452 /*
453 * Resolve the functions.
454 * Not all function have to present, leave it to IOM to enforce this.
455 */
456 int rc = VINF_SUCCESS;
457 if ( pDevIns->pReg->pszRCMod[0]
458 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
459 && VM_IS_RAW_MODE_ENABLED(pDevIns->Internal.s.pVMR3))
460 {
461 RTRCPTR RCPtrWrite = NIL_RTRCPTR;
462 if (pszWrite)
463 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszWrite, &RCPtrWrite);
464
465 RTRCPTR RCPtrRead = NIL_RTRCPTR;
466 int rc2 = VINF_SUCCESS;
467 if (pszRead)
468 rc2 = pdmR3DevGetSymbolRCLazy(pDevIns, pszRead, &RCPtrRead);
469
470 RTRCPTR RCPtrFill = NIL_RTRCPTR;
471 int rc3 = VINF_SUCCESS;
472 if (pszFill)
473 rc3 = pdmR3DevGetSymbolRCLazy(pDevIns, pszFill, &RCPtrFill);
474
475 if (RT_SUCCESS(rc) && RT_SUCCESS(rc2) && RT_SUCCESS(rc3))
476 rc = IOMR3MmioRegisterRC(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvUser, RCPtrWrite, RCPtrRead, RCPtrFill);
477 else
478 {
479 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pReg->pszRCMod, pszWrite));
480 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pReg->pszRCMod, pszRead));
481 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pReg->pszRCMod, pszFill));
482 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
483 rc = rc2;
484 if (RT_FAILURE(rc3) && RT_SUCCESS(rc))
485 rc = rc3;
486 }
487 }
488 else if (VM_IS_RAW_MODE_ENABLED(pDevIns->Internal.s.pVMR3))
489 {
490 AssertMsgFailed(("No RC module for this driver!\n"));
491 rc = VERR_INVALID_PARAMETER;
492 }
493#else
494 int rc = VINF_SUCCESS;
495 RT_NOREF(pDevIns, GCPhysStart, cbRange, pvUser, pszWrite, pszRead, pszFill);
496#endif
497
498 LogFlow(("pdmR3DevHlp_MMIORegisterRC: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
499 return rc;
500}
501
502/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegisterR0} */
503static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterR0(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTGCPHYS cbRange, RTR0PTR pvUser,
504 const char *pszWrite, const char *pszRead, const char *pszFill)
505{
506 PDMDEV_ASSERT_DEVINS(pDevIns);
507 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
508 Assert(pDevIns->pReg->pszR0Mod[0]);
509 Assert(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0);
510 LogFlow(("pdmR3DevHlp_MMIORegisterHC: caller='%s'/%d: GCPhysStart=%RGp cbRange=%RGp pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
511 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
512
513 /*
514 * Resolve the functions.
515 * Not all function have to present, leave it to IOM to enforce this.
516 */
517 int rc = VINF_SUCCESS;
518 if ( pDevIns->pReg->pszR0Mod[0]
519 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0))
520 {
521 R0PTRTYPE(PFNIOMMMIOWRITE) pfnR0PtrWrite = 0;
522 if (pszWrite)
523 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszWrite, &pfnR0PtrWrite);
524 R0PTRTYPE(PFNIOMMMIOREAD) pfnR0PtrRead = 0;
525 int rc2 = VINF_SUCCESS;
526 if (pszRead)
527 rc2 = pdmR3DevGetSymbolR0Lazy(pDevIns, pszRead, &pfnR0PtrRead);
528 R0PTRTYPE(PFNIOMMMIOFILL) pfnR0PtrFill = 0;
529 int rc3 = VINF_SUCCESS;
530 if (pszFill)
531 rc3 = pdmR3DevGetSymbolR0Lazy(pDevIns, pszFill, &pfnR0PtrFill);
532 if (RT_SUCCESS(rc) && RT_SUCCESS(rc2) && RT_SUCCESS(rc3))
533 rc = IOMR3MmioRegisterR0(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvUser,
534 pfnR0PtrWrite, pfnR0PtrRead, pfnR0PtrFill);
535 else
536 {
537 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pReg->pszR0Mod, pszWrite));
538 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pReg->pszR0Mod, pszRead));
539 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pReg->pszR0Mod, pszFill));
540 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
541 rc = rc2;
542 if (RT_FAILURE(rc3) && RT_SUCCESS(rc))
543 rc = rc3;
544 }
545 }
546 else
547 {
548 AssertMsgFailed(("No R0 module for this driver!\n"));
549 rc = VERR_INVALID_PARAMETER;
550 }
551
552 LogFlow(("pdmR3DevHlp_MMIORegisterR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
553 return rc;
554}
555
556
557/** @interface_method_impl{PDMDEVHLPR3,pfnMMIODeregister} */
558static DECLCALLBACK(int) pdmR3DevHlp_MMIODeregister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTGCPHYS cbRange)
559{
560 PDMDEV_ASSERT_DEVINS(pDevIns);
561 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
562 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%RGp\n",
563 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange));
564
565 int rc = IOMR3MmioDeregister(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange);
566
567 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
568 return rc;
569}
570
571
572/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2Create} */
573static DECLCALLBACK(int) pdmR3DevHlp_Mmio2Create(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iPciRegion, RTGCPHYS cbRegion,
574 uint32_t fFlags, const char *pszDesc, void **ppvMapping, PPGMMMIO2HANDLE phRegion)
575{
576 PDMDEV_ASSERT_DEVINS(pDevIns);
577 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
578 LogFlow(("pdmR3DevHlp_Mmio2Create: caller='%s'/%d: pPciDev=%p (%#x) iPciRegion=%#x cbRegion=%#RGp fFlags=%RX32 pszDesc=%p:{%s} ppvMapping=%p phRegion=%p\n",
579 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iPciRegion, cbRegion,
580 fFlags, pszDesc, pszDesc, ppvMapping, phRegion));
581 *ppvMapping = NULL;
582 *phRegion = NIL_PGMMMIO2HANDLE;
583 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
584
585 PVM pVM = pDevIns->Internal.s.pVMR3;
586 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
587 AssertMsgReturn( pVM->enmVMState == VMSTATE_CREATING
588 || pVM->enmVMState == VMSTATE_LOADING,
589 ("state %s, expected CREATING or LOADING\n", VMGetStateName(pVM->enmVMState)), VERR_VM_INVALID_VM_STATE);
590
591 AssertReturn(!(iPciRegion & UINT16_MAX), VERR_INVALID_PARAMETER); /* not implemented. */
592
593 /** @todo PGMR3PhysMMIO2Register mangles the description, move it here and
594 * use a real string cache. */
595 int rc = PGMR3PhysMMIO2Register(pVM, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iPciRegion >> 16,
596 cbRegion, fFlags, pszDesc, ppvMapping, phRegion);
597
598 LogFlow(("pdmR3DevHlp_Mmio2Create: caller='%s'/%d: returns %Rrc *ppvMapping=%p phRegion=%#RX64\n",
599 pDevIns->pReg->szName, pDevIns->iInstance, rc, *ppvMapping, *phRegion));
600 return rc;
601}
602
603
604/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2Destroy} */
605static DECLCALLBACK(int) pdmR3DevHlp_Mmio2Destroy(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion)
606{
607 PDMDEV_ASSERT_DEVINS(pDevIns);
608 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
609 LogFlow(("pdmR3DevHlp_Mmio2Destroy: caller='%s'/%d: hRegion=%#RX64\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion));
610
611 PVM pVM = pDevIns->Internal.s.pVMR3;
612 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
613 AssertMsgReturn( pVM->enmVMState == VMSTATE_DESTROYING
614 || pVM->enmVMState == VMSTATE_LOADING,
615 ("state %s, expected DESTROYING or LOADING\n", VMGetStateName(pVM->enmVMState)), VERR_VM_INVALID_VM_STATE);
616
617 int rc = PGMR3PhysMMIOExDeregister(pDevIns->Internal.s.pVMR3, pDevIns, UINT32_MAX, UINT32_MAX, hRegion);
618
619 LogFlow(("pdmR3DevHlp_Mmio2Destroy: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
620 return rc;
621}
622
623
624/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2Map} */
625static DECLCALLBACK(int) pdmR3DevHlp_Mmio2Map(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion, RTGCPHYS GCPhys)
626{
627 PDMDEV_ASSERT_DEVINS(pDevIns);
628 LogFlow(("pdmR3DevHlp_Mmio2Map: caller='%s'/%d: hRegion=%#RX64 GCPhys=%RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion, GCPhys));
629
630 PVM pVM = pDevIns->Internal.s.pVMR3;
631 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
632
633 int rc = PGMR3PhysMMIOExMap(pDevIns->Internal.s.pVMR3, pDevIns, UINT32_MAX, UINT32_MAX, hRegion, GCPhys);
634
635 LogFlow(("pdmR3DevHlp_Mmio2Map: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
636 return rc;
637}
638
639
640/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2Unmap} */
641static DECLCALLBACK(int) pdmR3DevHlp_Mmio2Unmap(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion)
642{
643 PDMDEV_ASSERT_DEVINS(pDevIns);
644 LogFlow(("pdmR3DevHlp_Mmio2Unmap: caller='%s'/%d: hRegion=%#RX64\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion));
645
646 PVM pVM = pDevIns->Internal.s.pVMR3;
647 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
648
649 int rc = PGMR3PhysMMIOExUnmap(pDevIns->Internal.s.pVMR3, pDevIns, UINT32_MAX, UINT32_MAX, hRegion, NIL_RTGCPHYS);
650
651 LogFlow(("pdmR3DevHlp_Mmio2Unmap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
652 return rc;
653}
654
655
656/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2Reduce} */
657static DECLCALLBACK(int) pdmR3DevHlp_Mmio2Reduce(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion, RTGCPHYS cbRegion)
658{
659 PDMDEV_ASSERT_DEVINS(pDevIns);
660 LogFlow(("pdmR3DevHlp_Mmio2Reduce: caller='%s'/%d: hRegion=%#RX64 cbRegion=%RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion, cbRegion));
661 PVM pVM = pDevIns->Internal.s.pVMR3;
662 VM_ASSERT_EMT_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
663 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_LOADING, VERR_VM_INVALID_VM_STATE);
664
665 int rc = PGMR3PhysMMIOExReduce(pDevIns->Internal.s.pVMR3, pDevIns, UINT32_MAX, UINT32_MAX, hRegion, cbRegion);
666
667 LogFlow(("pdmR3DevHlp_Mmio2Reduce: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
668 return rc;
669}
670
671
672/** @interface_method_impl{PDMDEVHLPR3,pfnMmio2GetMappingAddress} */
673static DECLCALLBACK(RTGCPHYS) pdmR3DevHlp_Mmio2GetMappingAddress(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion)
674{
675 PDMDEV_ASSERT_DEVINS(pDevIns);
676 PVM pVM = pDevIns->Internal.s.pVMR3;
677 LogFlow(("pdmR3DevHlp_Mmio2GetMappingAddress: caller='%s'/%d: hRegion=%#RX6r\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion));
678 VM_ASSERT_EMT0_RETURN(pVM, NIL_RTGCPHYS);
679
680 RTGCPHYS GCPhys = PGMR3PhysMmio2GetMappingAddress(pVM, pDevIns, hRegion);
681
682 LogFlow(("pdmR3DevHlp_Mmio2GetMappingAddress: caller='%s'/%d: returns %RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, GCPhys));
683 return GCPhys;
684}
685
686/**
687 * @copydoc PDMDEVHLPR3::pfnMmio2ChangeRegionNo
688 */
689static DECLCALLBACK(int) pdmR3DevHlp_Mmio2ChangeRegionNo(PPDMDEVINS pDevIns, PGMMMIO2HANDLE hRegion, uint32_t iNewRegion)
690{
691 PDMDEV_ASSERT_DEVINS(pDevIns);
692 PVM pVM = pDevIns->Internal.s.pVMR3;
693 LogFlow(("pdmR3DevHlp_Mmio2ChangeRegionNo: caller='%s'/%d: hRegion=%#RX6r iNewRegion=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, hRegion, iNewRegion));
694 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
695
696 int rc = PGMR3PhysMMIOExChangeRegionNo(pVM, pDevIns, UINT32_MAX, UINT32_MAX, hRegion, iNewRegion);
697
698 LogFlow(("pdmR3DevHlp_Mmio2ChangeRegionNo: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
699 return rc;
700}
701
702
703/**
704 * @copydoc PDMDEVHLPR3::pfnMMIO2Register
705 */
706static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Register(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS cb,
707 uint32_t fFlags, void **ppv, const char *pszDesc)
708{
709 PDMDEV_ASSERT_DEVINS(pDevIns);
710 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
711 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: pPciDev=%p (%#x) iRegion=%#x cb=%#RGp fFlags=%RX32 ppv=%p pszDescp=%p:{%s}\n",
712 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion,
713 cb, fFlags, ppv, pszDesc, pszDesc));
714 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
715
716/** @todo PGMR3PhysMMIO2Register mangles the description, move it here and
717 * use a real string cache. */
718 int rc = PGMR3PhysMMIO2Register(pDevIns->Internal.s.pVMR3, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion,
719 cb, fFlags, pszDesc, ppv, NULL);
720
721 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
722 return rc;
723}
724
725
726/**
727 * @copydoc PDMDEVHLPR3::pfnMMIOExDeregister
728 */
729static DECLCALLBACK(int) pdmR3DevHlp_MMIOExDeregister(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion)
730{
731 PDMDEV_ASSERT_DEVINS(pDevIns);
732 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
733 LogFlow(("pdmR3DevHlp_MMIOExDeregister: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x\n",
734 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion));
735
736 AssertReturn(iRegion <= UINT8_MAX || iRegion == UINT32_MAX, VERR_INVALID_PARAMETER);
737 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
738
739 int rc = PGMR3PhysMMIOExDeregister(pDevIns->Internal.s.pVMR3, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254,
740 iRegion, NIL_PGMMMIO2HANDLE);
741
742 LogFlow(("pdmR3DevHlp_MMIOExDeregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
743 return rc;
744}
745
746
747/**
748 * @copydoc PDMDEVHLPR3::pfnMMIOExMap
749 */
750static DECLCALLBACK(int) pdmR3DevHlp_MMIOExMap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS GCPhys)
751{
752 PDMDEV_ASSERT_DEVINS(pDevIns);
753 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
754 LogFlow(("pdmR3DevHlp_MMIOExMap: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x GCPhys=%#RGp\n",
755 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, GCPhys));
756 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 != NULL, VERR_INVALID_PARAMETER);
757
758 int rc = PGMR3PhysMMIOExMap(pDevIns->Internal.s.pVMR3, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion,
759 NIL_PGMMMIO2HANDLE, GCPhys);
760
761 LogFlow(("pdmR3DevHlp_MMIOExMap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
762 return rc;
763}
764
765
766/**
767 * @copydoc PDMDEVHLPR3::pfnMMIOExUnmap
768 */
769static DECLCALLBACK(int) pdmR3DevHlp_MMIOExUnmap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS GCPhys)
770{
771 PDMDEV_ASSERT_DEVINS(pDevIns);
772 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
773 LogFlow(("pdmR3DevHlp_MMIOExUnmap: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x GCPhys=%#RGp\n",
774 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, GCPhys));
775 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 != NULL, VERR_INVALID_PARAMETER);
776
777 int rc = PGMR3PhysMMIOExUnmap(pDevIns->Internal.s.pVMR3, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion,
778 NIL_PGMMMIO2HANDLE, GCPhys);
779
780 LogFlow(("pdmR3DevHlp_MMIOExUnmap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
781 return rc;
782}
783
784
785/**
786 * @copydoc PDMDEVHLPR3::pfnMMIOExReduce
787 */
788static DECLCALLBACK(int) pdmR3DevHlp_MMIOExReduce(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS cbRegion)
789{
790 PDMDEV_ASSERT_DEVINS(pDevIns);
791 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
792 LogFlow(("pdmR3DevHlp_MMIOExReduce: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x cbRegion=%RGp\n",
793 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, cbRegion));
794 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 != NULL, VERR_INVALID_PARAMETER);
795
796 int rc = PGMR3PhysMMIOExReduce(pDevIns->Internal.s.pVMR3, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion,
797 NIL_PGMMMIO2HANDLE, cbRegion);
798
799 LogFlow(("pdmR3DevHlp_MMIOExReduce: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
800 return rc;
801}
802
803
804/**
805 * @copydoc PDMDEVHLPR3::pfnMMHyperMapMMIO2
806 */
807static DECLCALLBACK(int) pdmR3DevHlp_MMHyperMapMMIO2(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS off,
808 RTGCPHYS cb, const char *pszDesc, PRTRCPTR pRCPtr)
809{
810 PDMDEV_ASSERT_DEVINS(pDevIns);
811#ifndef PGM_WITHOUT_MAPPINGS
812 PVM pVM = pDevIns->Internal.s.pVMR3;
813 VM_ASSERT_EMT(pVM);
814 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x off=%RGp cb=%RGp pszDesc=%p:{%s} pRCPtr=%p\n",
815 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, off, cb, pszDesc, pszDesc, pRCPtr));
816 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
817
818 if (pDevIns->iInstance > 0)
819 {
820 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
821 if (pszDesc2)
822 pszDesc = pszDesc2;
823 }
824
825 int rc = MMR3HyperMapMMIO2(pVM, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion, off, cb, pszDesc, pRCPtr);
826
827#else
828 RT_NOREF(pDevIns, pPciDev, iRegion, off, cb, pszDesc, pRCPtr);
829 AssertFailed();
830 int rc = VERR_RAW_MODE_NOT_SUPPORTED;
831#endif
832 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: returns %Rrc *pRCPtr=%RRv\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pRCPtr));
833 return rc;
834}
835
836
837/**
838 * @copydoc PDMDEVHLPR3::pfnMMIO2MapKernel
839 */
840static DECLCALLBACK(int) pdmR3DevHlp_MMIO2MapKernel(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion, RTGCPHYS off,
841 RTGCPHYS cb,const char *pszDesc, PRTR0PTR pR0Ptr)
842{
843 PDMDEV_ASSERT_DEVINS(pDevIns);
844 PVM pVM = pDevIns->Internal.s.pVMR3;
845 VM_ASSERT_EMT(pVM);
846 LogFlow(("pdmR3DevHlp_MMIO2MapKernel: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x off=%RGp cb=%RGp pszDesc=%p:{%s} pR0Ptr=%p\n",
847 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, off, cb, pszDesc, pszDesc, pR0Ptr));
848 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
849
850 if (pDevIns->iInstance > 0)
851 {
852 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
853 if (pszDesc2)
854 pszDesc = pszDesc2;
855 }
856
857 int rc = PGMR3PhysMMIO2MapKernel(pVM, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion, off, cb, pszDesc, pR0Ptr);
858
859 LogFlow(("pdmR3DevHlp_MMIO2MapKernel: caller='%s'/%d: returns %Rrc *pR0Ptr=%RHv\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pR0Ptr));
860 return rc;
861}
862
863
864/**
865 * @copydoc PDMDEVHLPR3::pfnMMIOExChangeRegionNo
866 */
867static DECLCALLBACK(int) pdmR3DevHlp_MMIOExChangeRegionNo(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
868 uint32_t iNewRegion)
869{
870 PDMDEV_ASSERT_DEVINS(pDevIns);
871 PVM pVM = pDevIns->Internal.s.pVMR3;
872 VM_ASSERT_EMT(pVM);
873 LogFlow(("pdmR3DevHlp_MMIOExChangeRegionNo: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%#x iNewRegion=%#x\n",
874 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev ? pPciDev->uDevFn : UINT32_MAX, iRegion, iNewRegion));
875 AssertReturn(!pPciDev || pPciDev->Int.s.pDevInsR3 == pDevIns, VERR_INVALID_PARAMETER);
876
877 int rc = PGMR3PhysMMIOExChangeRegionNo(pVM, pDevIns, pPciDev ? pPciDev->Int.s.idxDevCfg : 254, iRegion,
878 NIL_PGMMMIO2HANDLE, iNewRegion);
879
880 LogFlow(("pdmR3DevHlp_MMIOExChangeRegionNo: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
881 return rc;
882}
883
884
885/** @interface_method_impl{PDMDEVHLPR3,pfnROMRegister} */
886static DECLCALLBACK(int) pdmR3DevHlp_ROMRegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange,
887 const void *pvBinary, uint32_t cbBinary, uint32_t fFlags, const char *pszDesc)
888{
889 PDMDEV_ASSERT_DEVINS(pDevIns);
890 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
891 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x pvBinary=%p cbBinary=%#x fFlags=%#RX32 pszDesc=%p:{%s}\n",
892 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvBinary, cbBinary, fFlags, pszDesc, pszDesc));
893
894/** @todo can we mangle pszDesc? */
895 int rc = PGMR3PhysRomRegister(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvBinary, cbBinary, fFlags, pszDesc);
896
897 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
898 return rc;
899}
900
901
902/** @interface_method_impl{PDMDEVHLPR3,pfnROMProtectShadow} */
903static DECLCALLBACK(int) pdmR3DevHlp_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, PGMROMPROT enmProt)
904{
905 PDMDEV_ASSERT_DEVINS(pDevIns);
906 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x enmProt=%d\n",
907 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, enmProt));
908
909 int rc = PGMR3PhysRomProtect(pDevIns->Internal.s.pVMR3, GCPhysStart, cbRange, enmProt);
910
911 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
912 return rc;
913}
914
915
916/** @interface_method_impl{PDMDEVHLPR3,pfnSSMRegister} */
917static DECLCALLBACK(int) pdmR3DevHlp_SSMRegister(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess, const char *pszBefore,
918 PFNSSMDEVLIVEPREP pfnLivePrep, PFNSSMDEVLIVEEXEC pfnLiveExec, PFNSSMDEVLIVEVOTE pfnLiveVote,
919 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
920 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone)
921{
922 PDMDEV_ASSERT_DEVINS(pDevIns);
923 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
924 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: uVersion=%#x cbGuess=%#x pszBefore=%p:{%s}\n"
925 " pfnLivePrep=%p pfnLiveExec=%p pfnLiveVote=%p pfnSavePrep=%p pfnSaveExec=%p pfnSaveDone=%p pszLoadPrep=%p pfnLoadExec=%p pfnLoadDone=%p\n",
926 pDevIns->pReg->szName, pDevIns->iInstance, uVersion, cbGuess, pszBefore, pszBefore,
927 pfnLivePrep, pfnLiveExec, pfnLiveVote,
928 pfnSavePrep, pfnSaveExec, pfnSaveDone,
929 pfnLoadPrep, pfnLoadExec, pfnLoadDone));
930
931 int rc = SSMR3RegisterDevice(pDevIns->Internal.s.pVMR3, pDevIns, pDevIns->pReg->szName, pDevIns->iInstance,
932 uVersion, cbGuess, pszBefore,
933 pfnLivePrep, pfnLiveExec, pfnLiveVote,
934 pfnSavePrep, pfnSaveExec, pfnSaveDone,
935 pfnLoadPrep, pfnLoadExec, pfnLoadDone);
936
937 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
938 return rc;
939}
940
941
942/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimerCreate} */
943static DECLCALLBACK(int) pdmR3DevHlp_TMTimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback, void *pvUser, uint32_t fFlags, const char *pszDesc, PPTMTIMERR3 ppTimer)
944{
945 PDMDEV_ASSERT_DEVINS(pDevIns);
946 PVM pVM = pDevIns->Internal.s.pVMR3;
947 VM_ASSERT_EMT(pVM);
948 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: enmClock=%d pfnCallback=%p pvUser=%p fFlags=%#x pszDesc=%p:{%s} ppTimer=%p\n",
949 pDevIns->pReg->szName, pDevIns->iInstance, enmClock, pfnCallback, pvUser, fFlags, pszDesc, pszDesc, ppTimer));
950
951 if (pDevIns->iInstance > 0) /** @todo use a string cache here later. */
952 {
953 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
954 if (pszDesc2)
955 pszDesc = pszDesc2;
956 }
957
958 int rc = TMR3TimerCreateDevice(pVM, pDevIns, enmClock, pfnCallback, pvUser, fFlags, pszDesc, ppTimer);
959
960 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
961 return rc;
962}
963
964
965
966/** @interface_method_impl{PDMDEVHLPR3,pfnTimerCreate} */
967static DECLCALLBACK(int) pdmR3DevHlp_TimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback,
968 void *pvUser, uint32_t fFlags, const char *pszDesc, PTMTIMERHANDLE phTimer)
969{
970 PDMDEV_ASSERT_DEVINS(pDevIns);
971 PVM pVM = pDevIns->Internal.s.pVMR3;
972 VM_ASSERT_EMT(pVM);
973 LogFlow(("pdmR3DevHlp_TimerCreate: caller='%s'/%d: enmClock=%d pfnCallback=%p pvUser=%p fFlags=%#x pszDesc=%p:{%s} phTimer=%p\n",
974 pDevIns->pReg->szName, pDevIns->iInstance, enmClock, pfnCallback, pvUser, fFlags, pszDesc, pszDesc, phTimer));
975
976 if (pDevIns->iInstance > 0) /** @todo use a string cache here later. */
977 {
978 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
979 if (pszDesc2)
980 pszDesc = pszDesc2;
981 }
982
983 PTMTIMER pTimer = NULL;
984 int rc = TMR3TimerCreateDevice(pVM, pDevIns, enmClock, pfnCallback, pvUser, fFlags, pszDesc, &pTimer);
985 *phTimer = (uintptr_t)pTimer;
986
987 LogFlow(("pdmR3DevHlp_TimerCreate: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
988 return rc;
989}
990
991
992/** @interface_method_impl{PDMDEVHLPR3,pfnTimerToPtr} */
993static DECLCALLBACK(PTMTIMERR3) pdmR3DevHlp_TimerToPtr(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
994{
995 PDMDEV_ASSERT_DEVINS(pDevIns);
996 RT_NOREF(pDevIns);
997 return (PTMTIMERR3)hTimer;
998}
999
1000
1001/** @interface_method_impl{PDMDEVHLPR3,pfnTimerFromMicro} */
1002static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerFromMicro(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cMicroSecs)
1003{
1004 return TMTimerFromMicro(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cMicroSecs);
1005}
1006
1007
1008/** @interface_method_impl{PDMDEVHLPR3,pfnTimerFromMilli} */
1009static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerFromMilli(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cMilliSecs)
1010{
1011 return TMTimerFromMilli(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cMilliSecs);
1012}
1013
1014
1015/** @interface_method_impl{PDMDEVHLPR3,pfnTimerFromNano} */
1016static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerFromNano(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cNanoSecs)
1017{
1018 return TMTimerFromNano(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cNanoSecs);
1019}
1020
1021/** @interface_method_impl{PDMDEVHLPR3,pfnTimerGet} */
1022static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerGet(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1023{
1024 return TMTimerGet(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1025}
1026
1027
1028/** @interface_method_impl{PDMDEVHLPR3,pfnTimerGetFreq} */
1029static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerGetFreq(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1030{
1031 return TMTimerGetFreq(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1032}
1033
1034
1035/** @interface_method_impl{PDMDEVHLPR3,pfnTimerGetNano} */
1036static DECLCALLBACK(uint64_t) pdmR3DevHlp_TimerGetNano(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1037{
1038 return TMTimerGetNano(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1039}
1040
1041
1042/** @interface_method_impl{PDMDEVHLPR3,pfnTimerIsActive} */
1043static DECLCALLBACK(bool) pdmR3DevHlp_TimerIsActive(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1044{
1045 return TMTimerIsActive(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1046}
1047
1048
1049/** @interface_method_impl{PDMDEVHLPR3,pfnTimerIsLockOwner} */
1050static DECLCALLBACK(bool) pdmR3DevHlp_TimerIsLockOwner(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1051{
1052 return TMTimerIsLockOwner(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1053}
1054
1055
1056/** @interface_method_impl{PDMDEVHLPR3,pfnTimerLock} */
1057static DECLCALLBACK(int) pdmR3DevHlp_TimerLock(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, int rcBusy)
1058{
1059 return TMTimerLock(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), rcBusy);
1060}
1061
1062
1063/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSet} */
1064static DECLCALLBACK(int) pdmR3DevHlp_TimerSet(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t uExpire)
1065{
1066 return TMTimerSet(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), uExpire);
1067}
1068
1069
1070/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetFrequencyHint} */
1071static DECLCALLBACK(int) pdmR3DevHlp_TimerSetFrequencyHint(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint32_t uHz)
1072{
1073 return TMTimerSetFrequencyHint(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), uHz);
1074}
1075
1076
1077/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetMicro} */
1078static DECLCALLBACK(int) pdmR3DevHlp_TimerSetMicro(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cMicrosToNext)
1079{
1080 return TMTimerSetMicro(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cMicrosToNext);
1081}
1082
1083
1084/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetMillies} */
1085static DECLCALLBACK(int) pdmR3DevHlp_TimerSetMillies(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cMilliesToNext)
1086{
1087 return TMTimerSetMillies(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cMilliesToNext);
1088}
1089
1090
1091/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetNano} */
1092static DECLCALLBACK(int) pdmR3DevHlp_TimerSetNano(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cNanosToNext)
1093{
1094 return TMTimerSetNano(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cNanosToNext);
1095}
1096
1097
1098/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetRelative} */
1099static DECLCALLBACK(int) pdmR3DevHlp_TimerSetRelative(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, uint64_t cTicksToNext, uint64_t *pu64Now)
1100{
1101 return TMTimerSetRelative(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), cTicksToNext, pu64Now);
1102}
1103
1104
1105/** @interface_method_impl{PDMDEVHLPR3,pfnTimerStop} */
1106static DECLCALLBACK(int) pdmR3DevHlp_TimerStop(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1107{
1108 return TMTimerStop(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1109}
1110
1111
1112/** @interface_method_impl{PDMDEVHLPR3,pfnTimerUnlock} */
1113static DECLCALLBACK(void) pdmR3DevHlp_TimerUnlock(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1114{
1115 TMTimerUnlock(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1116}
1117
1118
1119/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSetCritSect} */
1120static DECLCALLBACK(int) pdmR3DevHlp_TimerSetCritSect(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, PPDMCRITSECT pCritSect)
1121{
1122 return TMR3TimerSetCritSect(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), pCritSect);
1123}
1124
1125
1126/** @interface_method_impl{PDMDEVHLPR3,pfnTimerSave} */
1127static DECLCALLBACK(int) pdmR3DevHlp_TimerSave(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, PSSMHANDLE pSSM)
1128{
1129 return TMR3TimerSave(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), pSSM);
1130}
1131
1132
1133/** @interface_method_impl{PDMDEVHLPR3,pfnTimerLoad} */
1134static DECLCALLBACK(int) pdmR3DevHlp_TimerLoad(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, PSSMHANDLE pSSM)
1135{
1136 return TMR3TimerLoad(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer), pSSM);
1137}
1138
1139
1140/** @interface_method_impl{PDMDEVHLPR3,pfnTimerDestroy} */
1141static DECLCALLBACK(int) pdmR3DevHlp_TimerDestroy(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer)
1142{
1143 return TMR3TimerDestroy(pdmR3DevHlp_TimerToPtr(pDevIns, hTimer));
1144}
1145
1146
1147/** @interface_method_impl{PDMDEVHLPR3,pfnTMUtcNow} */
1148static DECLCALLBACK(PRTTIMESPEC) pdmR3DevHlp_TMUtcNow(PPDMDEVINS pDevIns, PRTTIMESPEC pTime)
1149{
1150 PDMDEV_ASSERT_DEVINS(pDevIns);
1151 LogFlow(("pdmR3DevHlp_TMUtcNow: caller='%s'/%d: pTime=%p\n",
1152 pDevIns->pReg->szName, pDevIns->iInstance, pTime));
1153
1154 pTime = TMR3UtcNow(pDevIns->Internal.s.pVMR3, pTime);
1155
1156 LogFlow(("pdmR3DevHlp_TMUtcNow: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, RTTimeSpecGetNano(pTime)));
1157 return pTime;
1158}
1159
1160
1161/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGet} */
1162static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGet(PPDMDEVINS pDevIns)
1163{
1164 PDMDEV_ASSERT_DEVINS(pDevIns);
1165 LogFlow(("pdmR3DevHlp_TMTimeVirtGet: caller='%s'/%d\n",
1166 pDevIns->pReg->szName, pDevIns->iInstance));
1167
1168 uint64_t u64Time = TMVirtualSyncGet(pDevIns->Internal.s.pVMR3);
1169
1170 LogFlow(("pdmR3DevHlp_TMTimeVirtGet: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Time));
1171 return u64Time;
1172}
1173
1174
1175/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGetFreq} */
1176static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGetFreq(PPDMDEVINS pDevIns)
1177{
1178 PDMDEV_ASSERT_DEVINS(pDevIns);
1179 LogFlow(("pdmR3DevHlp_TMTimeVirtGetFreq: caller='%s'/%d\n",
1180 pDevIns->pReg->szName, pDevIns->iInstance));
1181
1182 uint64_t u64Freq = TMVirtualGetFreq(pDevIns->Internal.s.pVMR3);
1183
1184 LogFlow(("pdmR3DevHlp_TMTimeVirtGetFreq: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Freq));
1185 return u64Freq;
1186}
1187
1188
1189/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGetNano} */
1190static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGetNano(PPDMDEVINS pDevIns)
1191{
1192 PDMDEV_ASSERT_DEVINS(pDevIns);
1193 LogFlow(("pdmR3DevHlp_TMTimeVirtGetNano: caller='%s'/%d\n",
1194 pDevIns->pReg->szName, pDevIns->iInstance));
1195
1196 uint64_t u64Time = TMVirtualGet(pDevIns->Internal.s.pVMR3);
1197 uint64_t u64Nano = TMVirtualToNano(pDevIns->Internal.s.pVMR3, u64Time);
1198
1199 LogFlow(("pdmR3DevHlp_TMTimeVirtGetNano: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Nano));
1200 return u64Nano;
1201}
1202
1203
1204/** @interface_method_impl{PDMDEVHLPR3,pfnGetSupDrvSession} */
1205static DECLCALLBACK(PSUPDRVSESSION) pdmR3DevHlp_GetSupDrvSession(PPDMDEVINS pDevIns)
1206{
1207 PDMDEV_ASSERT_DEVINS(pDevIns);
1208 LogFlow(("pdmR3DevHlp_GetSupDrvSession: caller='%s'/%d\n",
1209 pDevIns->pReg->szName, pDevIns->iInstance));
1210
1211 PSUPDRVSESSION pSession = pDevIns->Internal.s.pVMR3->pSession;
1212
1213 LogFlow(("pdmR3DevHlp_GetSupDrvSession: caller='%s'/%d: returns %#p\n", pDevIns->pReg->szName, pDevIns->iInstance, pSession));
1214 return pSession;
1215}
1216
1217
1218/** @interface_method_impl{PDMDEVHLPR3,pfnQueryGenericUserObject} */
1219static DECLCALLBACK(void *) pdmR3DevHlp_QueryGenericUserObject(PPDMDEVINS pDevIns, PCRTUUID pUuid)
1220{
1221 PDMDEV_ASSERT_DEVINS(pDevIns);
1222 LogFlow(("pdmR3DevHlp_QueryGenericUserObject: caller='%s'/%d: pUuid=%p:%RTuuid\n",
1223 pDevIns->pReg->szName, pDevIns->iInstance, pUuid, pUuid));
1224
1225#if defined(DEBUG_bird) || defined(DEBUG_ramshankar) || defined(DEBUG_sunlover) || defined(DEBUG_michael) || defined(DEBUG_andy)
1226 AssertMsgFailed(("'%s' wants %RTuuid - external only interface!\n", pDevIns->pReg->szName, pUuid));
1227#endif
1228
1229 void *pvRet;
1230 PUVM pUVM = pDevIns->Internal.s.pVMR3->pUVM;
1231 if (pUVM->pVmm2UserMethods->pfnQueryGenericObject)
1232 pvRet = pUVM->pVmm2UserMethods->pfnQueryGenericObject(pUVM->pVmm2UserMethods, pUVM, pUuid);
1233 else
1234 pvRet = NULL;
1235
1236 LogRel(("pdmR3DevHlp_QueryGenericUserObject: caller='%s'/%d: returns %#p for %RTuuid\n",
1237 pDevIns->pReg->szName, pDevIns->iInstance, pvRet, pUuid));
1238 return pvRet;
1239}
1240
1241
1242/** @interface_method_impl{PDMDEVHLPR3,pfnPhysRead} */
1243static DECLCALLBACK(int) pdmR3DevHlp_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
1244{
1245 PDMDEV_ASSERT_DEVINS(pDevIns);
1246 PVM pVM = pDevIns->Internal.s.pVMR3;
1247 LogFlow(("pdmR3DevHlp_PhysRead: caller='%s'/%d: GCPhys=%RGp pvBuf=%p cbRead=%#x\n",
1248 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, pvBuf, cbRead));
1249
1250#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1251 if (!VM_IS_EMT(pVM))
1252 {
1253 char szNames[128];
1254 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1255 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1256 }
1257#endif
1258
1259 VBOXSTRICTRC rcStrict;
1260 if (VM_IS_EMT(pVM))
1261 rcStrict = PGMPhysRead(pVM, GCPhys, pvBuf, cbRead, PGMACCESSORIGIN_DEVICE);
1262 else
1263 rcStrict = PGMR3PhysReadExternal(pVM, GCPhys, pvBuf, cbRead, PGMACCESSORIGIN_DEVICE);
1264 AssertMsg(rcStrict == VINF_SUCCESS, ("%Rrc\n", VBOXSTRICTRC_VAL(rcStrict))); /** @todo track down the users for this bugger. */
1265
1266 Log(("pdmR3DevHlp_PhysRead: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VBOXSTRICTRC_VAL(rcStrict) ));
1267 return VBOXSTRICTRC_VAL(rcStrict);
1268}
1269
1270
1271/** @interface_method_impl{PDMDEVHLPR3,pfnPhysWrite} */
1272static DECLCALLBACK(int) pdmR3DevHlp_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
1273{
1274 PDMDEV_ASSERT_DEVINS(pDevIns);
1275 PVM pVM = pDevIns->Internal.s.pVMR3;
1276 LogFlow(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: GCPhys=%RGp pvBuf=%p cbWrite=%#x\n",
1277 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, pvBuf, cbWrite));
1278
1279#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1280 if (!VM_IS_EMT(pVM))
1281 {
1282 char szNames[128];
1283 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1284 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1285 }
1286#endif
1287
1288 VBOXSTRICTRC rcStrict;
1289 if (VM_IS_EMT(pVM))
1290 rcStrict = PGMPhysWrite(pVM, GCPhys, pvBuf, cbWrite, PGMACCESSORIGIN_DEVICE);
1291 else
1292 rcStrict = PGMR3PhysWriteExternal(pVM, GCPhys, pvBuf, cbWrite, PGMACCESSORIGIN_DEVICE);
1293 AssertMsg(rcStrict == VINF_SUCCESS, ("%Rrc\n", VBOXSTRICTRC_VAL(rcStrict))); /** @todo track down the users for this bugger. */
1294
1295 Log(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VBOXSTRICTRC_VAL(rcStrict) ));
1296 return VBOXSTRICTRC_VAL(rcStrict);
1297}
1298
1299
1300/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPhys2CCPtr} */
1301static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPhys2CCPtr(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, void **ppv, PPGMPAGEMAPLOCK pLock)
1302{
1303 PDMDEV_ASSERT_DEVINS(pDevIns);
1304 PVM pVM = pDevIns->Internal.s.pVMR3;
1305 LogFlow(("pdmR3DevHlp_PhysGCPhys2CCPtr: caller='%s'/%d: GCPhys=%RGp fFlags=%#x ppv=%p pLock=%p\n",
1306 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, fFlags, ppv, pLock));
1307 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
1308
1309#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1310 if (!VM_IS_EMT(pVM))
1311 {
1312 char szNames[128];
1313 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1314 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1315 }
1316#endif
1317
1318 int rc = PGMR3PhysGCPhys2CCPtrExternal(pVM, GCPhys, ppv, pLock);
1319
1320 Log(("pdmR3DevHlp_PhysGCPhys2CCPtr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1321 return rc;
1322}
1323
1324
1325/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPhys2CCPtrReadOnly} */
1326static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, const void **ppv, PPGMPAGEMAPLOCK pLock)
1327{
1328 PDMDEV_ASSERT_DEVINS(pDevIns);
1329 PVM pVM = pDevIns->Internal.s.pVMR3;
1330 LogFlow(("pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly: caller='%s'/%d: GCPhys=%RGp fFlags=%#x ppv=%p pLock=%p\n",
1331 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, fFlags, ppv, pLock));
1332 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
1333
1334#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1335 if (!VM_IS_EMT(pVM))
1336 {
1337 char szNames[128];
1338 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1339 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1340 }
1341#endif
1342
1343 int rc = PGMR3PhysGCPhys2CCPtrReadOnlyExternal(pVM, GCPhys, ppv, pLock);
1344
1345 Log(("pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1346 return rc;
1347}
1348
1349
1350/** @interface_method_impl{PDMDEVHLPR3,pfnPhysReleasePageMappingLock} */
1351static DECLCALLBACK(void) pdmR3DevHlp_PhysReleasePageMappingLock(PPDMDEVINS pDevIns, PPGMPAGEMAPLOCK pLock)
1352{
1353 PDMDEV_ASSERT_DEVINS(pDevIns);
1354 PVM pVM = pDevIns->Internal.s.pVMR3;
1355 LogFlow(("pdmR3DevHlp_PhysReleasePageMappingLock: caller='%s'/%d: pLock=%p\n",
1356 pDevIns->pReg->szName, pDevIns->iInstance, pLock));
1357
1358 PGMPhysReleasePageMappingLock(pVM, pLock);
1359
1360 Log(("pdmR3DevHlp_PhysReleasePageMappingLock: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1361}
1362
1363
1364/** @interface_method_impl{PDMDEVHLPR3,pfnPhysBulkGCPhys2CCPtr} */
1365static DECLCALLBACK(int) pdmR3DevHlp_PhysBulkGCPhys2CCPtr(PPDMDEVINS pDevIns, uint32_t cPages, PCRTGCPHYS paGCPhysPages,
1366 uint32_t fFlags, void **papvPages, PPGMPAGEMAPLOCK paLocks)
1367{
1368 PDMDEV_ASSERT_DEVINS(pDevIns);
1369 PVM pVM = pDevIns->Internal.s.pVMR3;
1370 LogFlow(("pdmR3DevHlp_PhysBulkGCPhys2CCPtr: caller='%s'/%d: cPages=%#x paGCPhysPages=%p (%RGp,..) fFlags=%#x papvPages=%p paLocks=%p\n",
1371 pDevIns->pReg->szName, pDevIns->iInstance, cPages, paGCPhysPages, paGCPhysPages[0], fFlags, papvPages, paLocks));
1372 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
1373 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
1374
1375#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1376 if (!VM_IS_EMT(pVM))
1377 {
1378 char szNames[128];
1379 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1380 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1381 }
1382#endif
1383
1384 int rc = PGMR3PhysBulkGCPhys2CCPtrExternal(pVM, cPages, paGCPhysPages, papvPages, paLocks);
1385
1386 Log(("pdmR3DevHlp_PhysBulkGCPhys2CCPtr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1387 return rc;
1388}
1389
1390
1391/** @interface_method_impl{PDMDEVHLPR3,pfnPhysBulkGCPhys2CCPtrReadOnly} */
1392static DECLCALLBACK(int) pdmR3DevHlp_PhysBulkGCPhys2CCPtrReadOnly(PPDMDEVINS pDevIns, uint32_t cPages, PCRTGCPHYS paGCPhysPages,
1393 uint32_t fFlags, const void **papvPages, PPGMPAGEMAPLOCK paLocks)
1394{
1395 PDMDEV_ASSERT_DEVINS(pDevIns);
1396 PVM pVM = pDevIns->Internal.s.pVMR3;
1397 LogFlow(("pdmR3DevHlp_PhysBulkGCPhys2CCPtrReadOnly: caller='%s'/%d: cPages=%#x paGCPhysPages=%p (%RGp,...) fFlags=%#x papvPages=%p paLocks=%p\n",
1398 pDevIns->pReg->szName, pDevIns->iInstance, cPages, paGCPhysPages, paGCPhysPages[0], fFlags, papvPages, paLocks));
1399 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
1400 AssertReturn(cPages > 0, VERR_INVALID_PARAMETER);
1401
1402#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1403 if (!VM_IS_EMT(pVM))
1404 {
1405 char szNames[128];
1406 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
1407 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
1408 }
1409#endif
1410
1411 int rc = PGMR3PhysBulkGCPhys2CCPtrReadOnlyExternal(pVM, cPages, paGCPhysPages, papvPages, paLocks);
1412
1413 Log(("pdmR3DevHlp_PhysBulkGCPhys2CCPtrReadOnly: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1414 return rc;
1415}
1416
1417
1418/** @interface_method_impl{PDMDEVHLPR3,pfnPhysBulkReleasePageMappingLocks} */
1419static DECLCALLBACK(void) pdmR3DevHlp_PhysBulkReleasePageMappingLocks(PPDMDEVINS pDevIns, uint32_t cPages, PPGMPAGEMAPLOCK paLocks)
1420{
1421 PDMDEV_ASSERT_DEVINS(pDevIns);
1422 PVM pVM = pDevIns->Internal.s.pVMR3;
1423 LogFlow(("pdmR3DevHlp_PhysBulkReleasePageMappingLocks: caller='%s'/%d: cPages=%#x paLocks=%p\n",
1424 pDevIns->pReg->szName, pDevIns->iInstance, cPages, paLocks));
1425 Assert(cPages > 0);
1426
1427 PGMPhysBulkReleasePageMappingLocks(pVM, cPages, paLocks);
1428
1429 Log(("pdmR3DevHlp_PhysBulkReleasePageMappingLocks: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1430}
1431
1432
1433/** @interface_method_impl{PDMDEVHLPR3,pfnPhysReadGCVirt} */
1434static DECLCALLBACK(int) pdmR3DevHlp_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb)
1435{
1436 PDMDEV_ASSERT_DEVINS(pDevIns);
1437 PVM pVM = pDevIns->Internal.s.pVMR3;
1438 VM_ASSERT_EMT(pVM);
1439 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: pvDst=%p GCVirt=%RGv cb=%#x\n",
1440 pDevIns->pReg->szName, pDevIns->iInstance, pvDst, GCVirtSrc, cb));
1441
1442 PVMCPU pVCpu = VMMGetCpu(pVM);
1443 if (!pVCpu)
1444 return VERR_ACCESS_DENIED;
1445#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1446 /** @todo SMP. */
1447#endif
1448
1449 int rc = PGMPhysSimpleReadGCPtr(pVCpu, pvDst, GCVirtSrc, cb);
1450
1451 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1452
1453 return rc;
1454}
1455
1456
1457/** @interface_method_impl{PDMDEVHLPR3,pfnPhysWriteGCVirt} */
1458static DECLCALLBACK(int) pdmR3DevHlp_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb)
1459{
1460 PDMDEV_ASSERT_DEVINS(pDevIns);
1461 PVM pVM = pDevIns->Internal.s.pVMR3;
1462 VM_ASSERT_EMT(pVM);
1463 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: GCVirtDst=%RGv pvSrc=%p cb=%#x\n",
1464 pDevIns->pReg->szName, pDevIns->iInstance, GCVirtDst, pvSrc, cb));
1465
1466 PVMCPU pVCpu = VMMGetCpu(pVM);
1467 if (!pVCpu)
1468 return VERR_ACCESS_DENIED;
1469#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1470 /** @todo SMP. */
1471#endif
1472
1473 int rc = PGMPhysSimpleWriteGCPtr(pVCpu, GCVirtDst, pvSrc, cb);
1474
1475 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1476
1477 return rc;
1478}
1479
1480
1481/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPtr2GCPhys} */
1482static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPtr2GCPhys(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys)
1483{
1484 PDMDEV_ASSERT_DEVINS(pDevIns);
1485 PVM pVM = pDevIns->Internal.s.pVMR3;
1486 VM_ASSERT_EMT(pVM);
1487 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: GCPtr=%RGv pGCPhys=%p\n",
1488 pDevIns->pReg->szName, pDevIns->iInstance, GCPtr, pGCPhys));
1489
1490 PVMCPU pVCpu = VMMGetCpu(pVM);
1491 if (!pVCpu)
1492 return VERR_ACCESS_DENIED;
1493#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
1494 /** @todo SMP. */
1495#endif
1496
1497 int rc = PGMPhysGCPtr2GCPhys(pVCpu, GCPtr, pGCPhys);
1498
1499 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: returns %Rrc *pGCPhys=%RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pGCPhys));
1500
1501 return rc;
1502}
1503
1504
1505/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapAlloc} */
1506static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAlloc(PPDMDEVINS pDevIns, size_t cb)
1507{
1508 PDMDEV_ASSERT_DEVINS(pDevIns);
1509 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: cb=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cb));
1510
1511 void *pv = MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE_USER, cb);
1512
1513 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
1514 return pv;
1515}
1516
1517
1518/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapAllocZ} */
1519static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAllocZ(PPDMDEVINS pDevIns, size_t cb)
1520{
1521 PDMDEV_ASSERT_DEVINS(pDevIns);
1522 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: cb=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cb));
1523
1524 void *pv = MMR3HeapAllocZ(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE_USER, cb);
1525
1526 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
1527 return pv;
1528}
1529
1530
1531/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapFree} */
1532static DECLCALLBACK(void) pdmR3DevHlp_MMHeapFree(PPDMDEVINS pDevIns, void *pv)
1533{
1534 PDMDEV_ASSERT_DEVINS(pDevIns); RT_NOREF_PV(pDevIns);
1535 LogFlow(("pdmR3DevHlp_MMHeapFree: caller='%s'/%d: pv=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
1536
1537 MMR3HeapFree(pv);
1538
1539 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1540}
1541
1542
1543/** @interface_method_impl{PDMDEVHLPR3,pfnVMState} */
1544static DECLCALLBACK(VMSTATE) pdmR3DevHlp_VMState(PPDMDEVINS pDevIns)
1545{
1546 PDMDEV_ASSERT_DEVINS(pDevIns);
1547
1548 VMSTATE enmVMState = VMR3GetState(pDevIns->Internal.s.pVMR3);
1549
1550 LogFlow(("pdmR3DevHlp_VMState: caller='%s'/%d: returns %d (%s)\n", pDevIns->pReg->szName, pDevIns->iInstance,
1551 enmVMState, VMR3GetStateName(enmVMState)));
1552 return enmVMState;
1553}
1554
1555
1556/** @interface_method_impl{PDMDEVHLPR3,pfnVMTeleportedAndNotFullyResumedYet} */
1557static DECLCALLBACK(bool) pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet(PPDMDEVINS pDevIns)
1558{
1559 PDMDEV_ASSERT_DEVINS(pDevIns);
1560
1561 bool fRc = VMR3TeleportedAndNotFullyResumedYet(pDevIns->Internal.s.pVMR3);
1562
1563 LogFlow(("pdmR3DevHlp_VMState: caller='%s'/%d: returns %RTbool\n", pDevIns->pReg->szName, pDevIns->iInstance,
1564 fRc));
1565 return fRc;
1566}
1567
1568
1569/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetError} */
1570static DECLCALLBACK(int) pdmR3DevHlp_VMSetError(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
1571{
1572 PDMDEV_ASSERT_DEVINS(pDevIns);
1573 va_list args;
1574 va_start(args, pszFormat);
1575 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMR3, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc2 == rc); NOREF(rc2);
1576 va_end(args);
1577 return rc;
1578}
1579
1580
1581/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetErrorV} */
1582static DECLCALLBACK(int) pdmR3DevHlp_VMSetErrorV(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
1583{
1584 PDMDEV_ASSERT_DEVINS(pDevIns);
1585 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMR3, rc, RT_SRC_POS_ARGS, pszFormat, va); Assert(rc2 == rc); NOREF(rc2);
1586 return rc;
1587}
1588
1589
1590/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetRuntimeError} */
1591static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeError(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
1592{
1593 PDMDEV_ASSERT_DEVINS(pDevIns);
1594 va_list args;
1595 va_start(args, pszFormat);
1596 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMR3, fFlags, pszErrorId, pszFormat, args);
1597 va_end(args);
1598 return rc;
1599}
1600
1601
1602/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetRuntimeErrorV} */
1603static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeErrorV(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
1604{
1605 PDMDEV_ASSERT_DEVINS(pDevIns);
1606 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMR3, fFlags, pszErrorId, pszFormat, va);
1607 return rc;
1608}
1609
1610
1611/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFStopV} */
1612static DECLCALLBACK(int) pdmR3DevHlp_DBGFStopV(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction, const char *pszFormat, va_list args)
1613{
1614 PDMDEV_ASSERT_DEVINS(pDevIns);
1615#ifdef LOG_ENABLED
1616 va_list va2;
1617 va_copy(va2, args);
1618 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: pszFile=%p:{%s} iLine=%d pszFunction=%p:{%s} pszFormat=%p:{%s} (%N)\n",
1619 pDevIns->pReg->szName, pDevIns->iInstance, pszFile, pszFile, iLine, pszFunction, pszFunction, pszFormat, pszFormat, pszFormat, &va2));
1620 va_end(va2);
1621#endif
1622
1623 PVM pVM = pDevIns->Internal.s.pVMR3;
1624 VM_ASSERT_EMT(pVM);
1625 int rc = DBGFR3EventSrcV(pVM, DBGFEVENT_DEV_STOP, pszFile, iLine, pszFunction, pszFormat, args);
1626 if (rc == VERR_DBGF_NOT_ATTACHED)
1627 rc = VINF_SUCCESS;
1628
1629 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1630 return rc;
1631}
1632
1633
1634/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFInfoRegister} */
1635static DECLCALLBACK(int) pdmR3DevHlp_DBGFInfoRegister(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler)
1636{
1637 PDMDEV_ASSERT_DEVINS(pDevIns);
1638 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: pszName=%p:{%s} pszDesc=%p:{%s} pfnHandler=%p\n",
1639 pDevIns->pReg->szName, pDevIns->iInstance, pszName, pszName, pszDesc, pszDesc, pfnHandler));
1640
1641 PVM pVM = pDevIns->Internal.s.pVMR3;
1642 VM_ASSERT_EMT(pVM);
1643 int rc = DBGFR3InfoRegisterDevice(pVM, pszName, pszDesc, pfnHandler, pDevIns);
1644
1645 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1646 return rc;
1647}
1648
1649
1650/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFInfoRegisterArgv} */
1651static DECLCALLBACK(int) pdmR3DevHlp_DBGFInfoRegisterArgv(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFINFOARGVDEV pfnHandler)
1652{
1653 PDMDEV_ASSERT_DEVINS(pDevIns);
1654 LogFlow(("pdmR3DevHlp_DBGFInfoRegisterArgv: caller='%s'/%d: pszName=%p:{%s} pszDesc=%p:{%s} pfnHandler=%p\n",
1655 pDevIns->pReg->szName, pDevIns->iInstance, pszName, pszName, pszDesc, pszDesc, pfnHandler));
1656
1657 PVM pVM = pDevIns->Internal.s.pVMR3;
1658 VM_ASSERT_EMT(pVM);
1659 int rc = DBGFR3InfoRegisterDeviceArgv(pVM, pszName, pszDesc, pfnHandler, pDevIns);
1660
1661 LogFlow(("pdmR3DevHlp_DBGFInfoRegisterArgv: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1662 return rc;
1663}
1664
1665
1666/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFRegRegister} */
1667static DECLCALLBACK(int) pdmR3DevHlp_DBGFRegRegister(PPDMDEVINS pDevIns, PCDBGFREGDESC paRegisters)
1668{
1669 PDMDEV_ASSERT_DEVINS(pDevIns);
1670 LogFlow(("pdmR3DevHlp_DBGFRegRegister: caller='%s'/%d: paRegisters=%p\n",
1671 pDevIns->pReg->szName, pDevIns->iInstance, paRegisters));
1672
1673 PVM pVM = pDevIns->Internal.s.pVMR3;
1674 VM_ASSERT_EMT(pVM);
1675 int rc = DBGFR3RegRegisterDevice(pVM, paRegisters, pDevIns, pDevIns->pReg->szName, pDevIns->iInstance);
1676
1677 LogFlow(("pdmR3DevHlp_DBGFRegRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1678 return rc;
1679}
1680
1681
1682/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFTraceBuf} */
1683static DECLCALLBACK(RTTRACEBUF) pdmR3DevHlp_DBGFTraceBuf(PPDMDEVINS pDevIns)
1684{
1685 PDMDEV_ASSERT_DEVINS(pDevIns);
1686 RTTRACEBUF hTraceBuf = pDevIns->Internal.s.pVMR3->hTraceBufR3;
1687 LogFlow(("pdmR3DevHlp_DBGFTraceBuf: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, hTraceBuf));
1688 return hTraceBuf;
1689}
1690
1691
1692/** @interface_method_impl{PDMDEVHLPR3,pfnSTAMRegister} */
1693static DECLCALLBACK(void) pdmR3DevHlp_STAMRegister(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName,
1694 STAMUNIT enmUnit, const char *pszDesc)
1695{
1696 PDMDEV_ASSERT_DEVINS(pDevIns);
1697 PVM pVM = pDevIns->Internal.s.pVMR3;
1698 VM_ASSERT_EMT(pVM);
1699
1700 int rc;
1701 if (*pszName == '/')
1702 rc = STAMR3Register(pVM, pvSample, enmType, STAMVISIBILITY_ALWAYS, pszName, enmUnit, pszDesc);
1703 else /* Provide default device statistics prefix: */
1704 rc = STAMR3RegisterF(pVM, pvSample, enmType, STAMVISIBILITY_ALWAYS, enmUnit, pszDesc,
1705 "/Devices/%s#%u/%s", pDevIns->pReg->szName, pDevIns->iInstance, pszName);
1706 AssertRC(rc);
1707}
1708
1709
1710/** @interface_method_impl{PDMDEVHLPR3,pfnSTAMRegisterV} */
1711static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterV(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility,
1712 STAMUNIT enmUnit, const char *pszDesc, const char *pszName, va_list args)
1713{
1714 PDMDEV_ASSERT_DEVINS(pDevIns);
1715 PVM pVM = pDevIns->Internal.s.pVMR3;
1716 VM_ASSERT_EMT(pVM);
1717
1718 int rc;
1719 if (*pszName == '/')
1720 rc = STAMR3RegisterV(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
1721 else
1722 {
1723 /* Provide default device statistics prefix: */
1724 va_list vaCopy;
1725 va_copy(vaCopy, args);
1726 rc = STAMR3RegisterF(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc,
1727 "/Devices/%s#%u/%N", pDevIns->pReg->szName, pDevIns->iInstance, pszName, &vaCopy);
1728 va_end(vaCopy);
1729 }
1730 AssertRC(rc);
1731}
1732
1733
1734/**
1735 * @interface_method_impl{PDMDEVHLPR3,pfnPCIRegister}
1736 */
1737static DECLCALLBACK(int) pdmR3DevHlp_PCIRegister(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t fFlags,
1738 uint8_t uPciDevNo, uint8_t uPciFunNo, const char *pszName)
1739{
1740 PDMDEV_ASSERT_DEVINS(pDevIns);
1741 PVM pVM = pDevIns->Internal.s.pVMR3;
1742 VM_ASSERT_EMT(pVM);
1743 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: pPciDev=%p:{.config={%#.256Rhxs} fFlags=%#x uPciDevNo=%#x uPciFunNo=%#x pszName=%p:{%s}\n",
1744 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev->abConfig, fFlags, uPciDevNo, uPciFunNo, pszName, pszName ? pszName : ""));
1745
1746 /*
1747 * Validate input.
1748 */
1749 AssertLogRelMsgReturn(pDevIns->pReg->cMaxPciDevices > 0,
1750 ("'%s'/%d: cMaxPciDevices is 0\n", pDevIns->pReg->szName, pDevIns->iInstance),
1751 VERR_WRONG_ORDER);
1752 AssertLogRelMsgReturn(RT_VALID_PTR(pPciDev),
1753 ("'%s'/%d: Invalid pPciDev value: %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pPciDev),
1754 VERR_INVALID_POINTER);
1755 AssertLogRelMsgReturn(PDMPciDevGetVendorId(pPciDev),
1756 ("'%s'/%d: Vendor ID is not set!\n", pDevIns->pReg->szName, pDevIns->iInstance),
1757 VERR_INVALID_POINTER);
1758 AssertLogRelMsgReturn( uPciDevNo < 32
1759 || uPciDevNo == PDMPCIDEVREG_DEV_NO_FIRST_UNUSED
1760 || uPciDevNo == PDMPCIDEVREG_DEV_NO_SAME_AS_PREV,
1761 ("'%s'/%d: Invalid PCI device number: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, uPciDevNo),
1762 VERR_INVALID_PARAMETER);
1763 AssertLogRelMsgReturn( uPciFunNo < 8
1764 || uPciFunNo == PDMPCIDEVREG_FUN_NO_FIRST_UNUSED,
1765 ("'%s'/%d: Invalid PCI funcion number: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, uPciFunNo),
1766 VERR_INVALID_PARAMETER);
1767 AssertLogRelMsgReturn(!(fFlags & ~PDMPCIDEVREG_F_VALID_MASK),
1768 ("'%s'/%d: Invalid flags: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, fFlags),
1769 VERR_INVALID_FLAGS);
1770 if (!pszName)
1771 pszName = pDevIns->pReg->szName;
1772 AssertLogRelReturn(RT_VALID_PTR(pszName), VERR_INVALID_POINTER);
1773 AssertLogRelReturn(!pPciDev->Int.s.fRegistered, VERR_PDM_NOT_PCI_DEVICE);
1774 AssertLogRelReturn(pPciDev == PDMDEV_GET_PPCIDEV(pDevIns, pPciDev->Int.s.idxSubDev), VERR_PDM_NOT_PCI_DEVICE);
1775 AssertLogRelReturn(pPciDev == PDMDEV_CALC_PPCIDEV(pDevIns, pPciDev->Int.s.idxSubDev), VERR_PDM_NOT_PCI_DEVICE);
1776 AssertMsgReturn(pPciDev->u32Magic == PDMPCIDEV_MAGIC, ("%#x\n", pPciDev->u32Magic), VERR_PDM_NOT_PCI_DEVICE);
1777
1778 /*
1779 * Check the registration order - must be following PDMDEVINSR3::apPciDevs.
1780 */
1781 PPDMPCIDEV const pPrevPciDev = pPciDev->Int.s.idxSubDev == 0 ? NULL
1782 : PDMDEV_GET_PPCIDEV(pDevIns, pPciDev->Int.s.idxSubDev - 1);
1783 if (pPrevPciDev)
1784 {
1785 AssertLogRelReturn(pPrevPciDev->u32Magic == PDMPCIDEV_MAGIC, VERR_INVALID_MAGIC);
1786 AssertLogRelReturn(pPrevPciDev->Int.s.fRegistered, VERR_WRONG_ORDER);
1787 }
1788
1789 /*
1790 * Resolve the PCI configuration node for the device. The default (zero'th)
1791 * is the same as the PDM device, the rest are "PciCfg1..255" CFGM sub-nodes.
1792 */
1793 PCFGMNODE pCfg = pDevIns->Internal.s.pCfgHandle;
1794 if (pPciDev->Int.s.idxSubDev > 0)
1795 pCfg = CFGMR3GetChildF(pDevIns->Internal.s.pCfgHandle, "PciCfg%u", pPciDev->Int.s.idxSubDev);
1796
1797 /*
1798 * We resolve PDMPCIDEVREG_DEV_NO_SAME_AS_PREV, the PCI bus handles
1799 * PDMPCIDEVREG_DEV_NO_FIRST_UNUSED and PDMPCIDEVREG_FUN_NO_FIRST_UNUSED.
1800 */
1801 uint8_t const uPciDevNoRaw = uPciDevNo;
1802 uint32_t uDefPciBusNo = 0;
1803 if (uPciDevNo == PDMPCIDEVREG_DEV_NO_SAME_AS_PREV)
1804 {
1805 if (pPrevPciDev)
1806 {
1807 uPciDevNo = pPrevPciDev->uDevFn >> 3;
1808 uDefPciBusNo = pPrevPciDev->Int.s.idxPdmBus;
1809 }
1810 else
1811 {
1812 /* Look for PCI device registered with an earlier device instance so we can more
1813 easily have multiple functions spanning multiple PDM device instances. */
1814 PPDMDEVINS pPrevIns = pDevIns->Internal.s.pDevR3->pInstances;
1815 for (;;)
1816 {
1817 AssertLogRelMsgReturn(pPrevIns && pPrevIns != pDevIns,
1818 ("'%s'/%d: Can't use PDMPCIDEVREG_DEV_NO_SAME_AS_PREV without a previously registered PCI device by the same or earlier PDM device instance!\n",
1819 pDevIns->pReg->szName, pDevIns->iInstance), VERR_WRONG_ORDER);
1820 if (pPrevIns->Internal.s.pNextR3 == pDevIns)
1821 break;
1822 pPrevIns = pPrevIns->Internal.s.pNextR3;
1823 }
1824
1825 PPDMPCIDEV pOtherPciDev = PDMDEV_GET_PPCIDEV(pPrevIns, 0);
1826 AssertLogRelMsgReturn(pOtherPciDev && pOtherPciDev->Int.s.fRegistered,
1827 ("'%s'/%d: Can't use PDMPCIDEVREG_DEV_NO_SAME_AS_PREV without a previously registered PCI device by the same or earlier PDM device instance!\n",
1828 pDevIns->pReg->szName, pDevIns->iInstance),
1829 VERR_WRONG_ORDER);
1830 for (uint32_t iPrevPciDev = 1; iPrevPciDev < pDevIns->cPciDevs; iPrevPciDev++)
1831 {
1832 PPDMPCIDEV pCur = PDMDEV_GET_PPCIDEV(pPrevIns, iPrevPciDev);
1833 AssertBreak(pCur);
1834 if (!pCur->Int.s.fRegistered)
1835 break;
1836 pOtherPciDev = pCur;
1837 }
1838
1839 uPciDevNo = pOtherPciDev->uDevFn >> 3;
1840 uDefPciBusNo = pOtherPciDev->Int.s.idxPdmBus;
1841 }
1842 }
1843
1844 /*
1845 * Choose the PCI bus for the device.
1846 *
1847 * This is simple. If the device was configured for a particular bus, the PCIBusNo
1848 * configuration value will be set. If not the default bus is 0.
1849 */
1850 /** @cfgm{/Devices/NAME/XX/[PciCfgYY/]PCIBusNo, uint8_t, 0, 7, 0}
1851 * Selects the PCI bus number of a device. The default value isn't necessarily
1852 * zero if the device is registered using PDMPCIDEVREG_DEV_NO_SAME_AS_PREV, it
1853 * will then also inherit the bus number from the previously registered device.
1854 */
1855 uint8_t u8Bus;
1856 int rc = CFGMR3QueryU8Def(pCfg, "PCIBusNo", &u8Bus, (uint8_t)uDefPciBusNo);
1857 AssertLogRelMsgRCReturn(rc, ("Configuration error: PCIBusNo query failed with rc=%Rrc (%s/%d)\n",
1858 rc, pDevIns->pReg->szName, pDevIns->iInstance), rc);
1859 AssertLogRelMsgReturn(u8Bus < RT_ELEMENTS(pVM->pdm.s.aPciBuses),
1860 ("Configuration error: PCIBusNo=%d, max is %d. (%s/%d)\n", u8Bus,
1861 RT_ELEMENTS(pVM->pdm.s.aPciBuses), pDevIns->pReg->szName, pDevIns->iInstance),
1862 VERR_PDM_NO_PCI_BUS);
1863 pPciDev->Int.s.idxPdmBus = u8Bus;
1864 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[u8Bus];
1865 if (pBus->pDevInsR3)
1866 {
1867 /*
1868 * Check the configuration for PCI device and function assignment.
1869 */
1870 /** @cfgm{/Devices/NAME/XX/[PciCfgYY/]PCIDeviceNo, uint8_t, 0, 31}
1871 * Overrides the default PCI device number of a device.
1872 */
1873 uint8_t uCfgDevice;
1874 rc = CFGMR3QueryU8(pCfg, "PCIDeviceNo", &uCfgDevice);
1875 if (RT_SUCCESS(rc))
1876 {
1877 AssertMsgReturn(uCfgDevice <= 31,
1878 ("Configuration error: PCIDeviceNo=%d, max is 31. (%s/%d/%d)\n",
1879 uCfgDevice, pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->Int.s.idxSubDev),
1880 VERR_PDM_BAD_PCI_CONFIG);
1881 uPciDevNo = uCfgDevice;
1882 }
1883 else
1884 AssertMsgReturn(rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT,
1885 ("Configuration error: PCIDeviceNo query failed with rc=%Rrc (%s/%d/%d)\n",
1886 rc, pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->Int.s.idxSubDev),
1887 rc);
1888
1889 /** @cfgm{/Devices/NAME/XX/[PciCfgYY/]PCIFunctionNo, uint8_t, 0, 7}
1890 * Overrides the default PCI function number of a device.
1891 */
1892 uint8_t uCfgFunction;
1893 rc = CFGMR3QueryU8(pCfg, "PCIFunctionNo", &uCfgFunction);
1894 if (RT_SUCCESS(rc))
1895 {
1896 AssertMsgReturn(uCfgFunction <= 7,
1897 ("Configuration error: PCIFunctionNo=%#x, max is 7. (%s/%d/%d)\n",
1898 uCfgFunction, pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->Int.s.idxSubDev),
1899 VERR_PDM_BAD_PCI_CONFIG);
1900 uPciFunNo = uCfgFunction;
1901 }
1902 else
1903 AssertMsgReturn(rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT,
1904 ("Configuration error: PCIFunctionNo query failed with rc=%Rrc (%s/%d/%d)\n",
1905 rc, pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->Int.s.idxSubDev),
1906 rc);
1907
1908
1909 /*
1910 * Initialize the internal data. We only do the wipe and the members
1911 * owned by PDM, the PCI bus does the rest in the registration call.
1912 */
1913 RT_ZERO(pPciDev->Int);
1914
1915 pPciDev->Int.s.idxDevCfg = pPciDev->Int.s.idxSubDev;
1916 pPciDev->Int.s.fReassignableDevNo = uPciDevNoRaw >= VBOX_PCI_MAX_DEVICES;
1917 pPciDev->Int.s.fReassignableFunNo = uPciFunNo >= VBOX_PCI_MAX_FUNCTIONS;
1918 pPciDev->Int.s.pDevInsR3 = pDevIns;
1919 pPciDev->Int.s.idxPdmBus = u8Bus;
1920 pPciDev->Int.s.fRegistered = true;
1921
1922 /* Set some of the public members too. */
1923 pPciDev->pszNameR3 = pszName;
1924
1925 /*
1926 * Call the pci bus device to do the actual registration.
1927 */
1928 pdmLock(pVM);
1929 rc = pBus->pfnRegister(pBus->pDevInsR3, pPciDev, fFlags, uPciDevNo, uPciFunNo, pszName);
1930 pdmUnlock(pVM);
1931 if (RT_SUCCESS(rc))
1932 Log(("PDM: Registered device '%s'/%d as PCI device %d on bus %d\n",
1933 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->uDevFn, pBus->iBus));
1934 else
1935 pPciDev->Int.s.fRegistered = false;
1936 }
1937 else
1938 {
1939 AssertLogRelMsgFailed(("Configuration error: No PCI bus available. This could be related to init order too!\n"));
1940 rc = VERR_PDM_NO_PCI_BUS;
1941 }
1942
1943 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1944 return rc;
1945}
1946
1947
1948/** @interface_method_impl{PDMDEVHLPR3,pfnPCIRegisterMsi} */
1949static DECLCALLBACK(int) pdmR3DevHlp_PCIRegisterMsi(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, PPDMMSIREG pMsiReg)
1950{
1951 PDMDEV_ASSERT_DEVINS(pDevIns);
1952 if (!pPciDev) /* NULL is an alias for the default PCI device. */
1953 pPciDev = pDevIns->apPciDevs[0];
1954 AssertReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
1955 LogFlow(("pdmR3DevHlp_PCIRegisterMsi: caller='%s'/%d: pPciDev=%p:{%#x} pMsgReg=%p:{cMsiVectors=%d, cMsixVectors=%d}\n",
1956 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev->uDevFn, pMsiReg, pMsiReg->cMsiVectors, pMsiReg->cMsixVectors));
1957 PDMPCIDEV_ASSERT_VALID_RET(pDevIns, pPciDev);
1958
1959 AssertLogRelMsgReturn(pDevIns->pReg->cMaxPciDevices > 0,
1960 ("'%s'/%d: cMaxPciDevices is 0\n", pDevIns->pReg->szName, pDevIns->iInstance),
1961 VERR_WRONG_ORDER);
1962 AssertLogRelMsgReturn(pMsiReg->cMsixVectors <= pDevIns->pReg->cMaxMsixVectors,
1963 ("'%s'/%d: cMsixVectors=%u cMaxMsixVectors=%u\n",
1964 pDevIns->pReg->szName, pDevIns->iInstance, pMsiReg->cMsixVectors, pDevIns->pReg->cMaxMsixVectors),
1965 VERR_INVALID_FLAGS);
1966
1967 PVM pVM = pDevIns->Internal.s.pVMR3;
1968 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
1969 AssertReturn(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses), VERR_WRONG_ORDER);
1970 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
1971
1972 pdmLock(pVM);
1973 int rc;
1974 if (pBus->pfnRegisterMsi)
1975 rc = pBus->pfnRegisterMsi(pBus->pDevInsR3, pPciDev, pMsiReg);
1976 else
1977 rc = VERR_NOT_IMPLEMENTED;
1978 pdmUnlock(pVM);
1979
1980 LogFlow(("pdmR3DevHlp_PCIRegisterMsi: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1981 return rc;
1982}
1983
1984
1985/** @interface_method_impl{PDMDEVHLPR3,pfnPCIIORegionRegister} */
1986static DECLCALLBACK(int) pdmR3DevHlp_PCIIORegionRegister(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
1987 RTGCPHYS cbRegion, PCIADDRESSSPACE enmType, uint32_t fFlags,
1988 uint64_t hHandle, PFNPCIIOREGIONMAP pfnMapUnmap)
1989{
1990 PDMDEV_ASSERT_DEVINS(pDevIns);
1991 PVM pVM = pDevIns->Internal.s.pVMR3;
1992 VM_ASSERT_EMT(pVM);
1993 if (!pPciDev) /* NULL is an alias for the default PCI device. */
1994 pPciDev = pDevIns->apPciDevs[0];
1995 AssertReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
1996 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: pPciDev=%p:{%#x} iRegion=%d cbRegion=%RGp enmType=%d fFlags=%#x, hHandle=%#RX64 pfnMapUnmap=%p\n",
1997 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev->uDevFn, iRegion, cbRegion, enmType, fFlags, hHandle, pfnMapUnmap));
1998 PDMPCIDEV_ASSERT_VALID_RET(pDevIns, pPciDev);
1999
2000 /*
2001 * Validate input.
2002 */
2003 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
2004 AssertLogRelMsgReturn(VMR3GetState(pVM) == VMSTATE_CREATING,
2005 ("caller='%s'/%d: %s\n", pDevIns->pReg->szName, pDevIns->iInstance, VMR3GetStateName(VMR3GetState(pVM))),
2006 VERR_WRONG_ORDER);
2007
2008 if (iRegion >= VBOX_PCI_NUM_REGIONS)
2009 {
2010 Assert(iRegion < VBOX_PCI_NUM_REGIONS);
2011 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc (iRegion)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2012 return VERR_INVALID_PARAMETER;
2013 }
2014
2015 switch ((int)enmType)
2016 {
2017 case PCI_ADDRESS_SPACE_IO:
2018 /*
2019 * Sanity check: don't allow to register more than 32K of the PCI I/O space.
2020 */
2021 AssertLogRelMsgReturn(cbRegion <= _32K,
2022 ("caller='%s'/%d: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cbRegion),
2023 VERR_INVALID_PARAMETER);
2024 break;
2025
2026 case PCI_ADDRESS_SPACE_MEM:
2027 case PCI_ADDRESS_SPACE_MEM_PREFETCH:
2028 /*
2029 * Sanity check: Don't allow to register more than 2GB of the PCI MMIO space.
2030 */
2031 AssertLogRelMsgReturn(cbRegion <= MM_MMIO_32_MAX,
2032 ("caller='%s'/%d: %RGp (max %RGp)\n",
2033 pDevIns->pReg->szName, pDevIns->iInstance, cbRegion, (RTGCPHYS)MM_MMIO_32_MAX),
2034 VERR_OUT_OF_RANGE);
2035 break;
2036
2037 case PCI_ADDRESS_SPACE_BAR64 | PCI_ADDRESS_SPACE_MEM:
2038 case PCI_ADDRESS_SPACE_BAR64 | PCI_ADDRESS_SPACE_MEM_PREFETCH:
2039 /*
2040 * Sanity check: Don't allow to register more than 64GB of the 64-bit PCI MMIO space.
2041 */
2042 AssertLogRelMsgReturn(cbRegion <= MM_MMIO_64_MAX,
2043 ("caller='%s'/%d: %RGp (max %RGp)\n",
2044 pDevIns->pReg->szName, pDevIns->iInstance, cbRegion, MM_MMIO_64_MAX),
2045 VERR_OUT_OF_RANGE);
2046 break;
2047
2048 default:
2049 AssertMsgFailed(("enmType=%#x is unknown\n", enmType));
2050 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc (enmType)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2051 return VERR_INVALID_PARAMETER;
2052 }
2053
2054 AssertMsgReturn( pfnMapUnmap
2055 || ( hHandle != UINT64_MAX
2056 && (fFlags & PDMPCIDEV_IORGN_F_HANDLE_MASK) != PDMPCIDEV_IORGN_F_NO_HANDLE),
2057 ("caller='%s'/%d: fFlags=%#x hHandle=%#RX64\n", pDevIns->pReg->szName, pDevIns->iInstance, fFlags, hHandle),
2058 VERR_INVALID_PARAMETER);
2059
2060 AssertMsgReturn(!(fFlags & ~PDMPCIDEV_IORGN_F_VALID_MASK), ("fFlags=%#x\n", fFlags), VERR_INVALID_FLAGS);
2061 int rc;
2062 switch (fFlags & PDMPCIDEV_IORGN_F_HANDLE_MASK)
2063 {
2064 case PDMPCIDEV_IORGN_F_NO_HANDLE:
2065 break;
2066 case PDMPCIDEV_IORGN_F_IOPORT_HANDLE:
2067 AssertReturn(enmType == PCI_ADDRESS_SPACE_IO, VERR_INVALID_FLAGS);
2068 rc = IOMR3IoPortValidateHandle(pVM, pDevIns, (IOMIOPORTHANDLE)hHandle);
2069 AssertRCReturn(rc, rc);
2070 break;
2071 case PDMPCIDEV_IORGN_F_MMIO_HANDLE:
2072 AssertReturn( (enmType & ~PCI_ADDRESS_SPACE_BAR64) == PCI_ADDRESS_SPACE_MEM
2073 || (enmType & ~PCI_ADDRESS_SPACE_BAR64) == PCI_ADDRESS_SPACE_MEM_PREFETCH,
2074 VERR_INVALID_FLAGS);
2075 rc = IOMR3MmioValidateHandle(pVM, pDevIns, (IOMMMIOHANDLE)hHandle);
2076 AssertRCReturn(rc, rc);
2077 break;
2078 case PDMPCIDEV_IORGN_F_MMIO2_HANDLE:
2079 AssertReturn( (enmType & ~PCI_ADDRESS_SPACE_BAR64) == PCI_ADDRESS_SPACE_MEM
2080 || (enmType & ~PCI_ADDRESS_SPACE_BAR64) == PCI_ADDRESS_SPACE_MEM_PREFETCH,
2081 VERR_INVALID_FLAGS);
2082 rc = PGMR3PhysMmio2ValidateHandle(pVM, pDevIns, (PGMMMIO2HANDLE)hHandle);
2083 AssertRCReturn(rc, rc);
2084 break;
2085 default:
2086 AssertFailedReturn(VERR_IPE_NOT_REACHED_DEFAULT_CASE);
2087 break;
2088 }
2089
2090 /*
2091 * We're currently restricted to page aligned MMIO regions.
2092 */
2093 if ( ((enmType & ~(PCI_ADDRESS_SPACE_BAR64 | PCI_ADDRESS_SPACE_MEM_PREFETCH)) == PCI_ADDRESS_SPACE_MEM)
2094 && cbRegion != RT_ALIGN_64(cbRegion, PAGE_SIZE))
2095 {
2096 Log(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: aligning cbRegion %RGp -> %RGp\n",
2097 pDevIns->pReg->szName, pDevIns->iInstance, cbRegion, RT_ALIGN_64(cbRegion, PAGE_SIZE)));
2098 cbRegion = RT_ALIGN_64(cbRegion, PAGE_SIZE);
2099 }
2100
2101 /*
2102 * For registering PCI MMIO memory or PCI I/O memory, the size of the region must be a power of 2!
2103 */
2104 int iLastSet = ASMBitLastSetU64(cbRegion);
2105 Assert(iLastSet > 0);
2106 uint64_t cbRegionAligned = RT_BIT_64(iLastSet - 1);
2107 if (cbRegion > cbRegionAligned)
2108 cbRegion = cbRegionAligned * 2; /* round up */
2109
2110 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
2111 AssertReturn(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses), VERR_WRONG_ORDER);
2112 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
2113
2114 pdmLock(pVM);
2115 rc = pBus->pfnIORegionRegister(pBus->pDevInsR3, pPciDev, iRegion, cbRegion, enmType, fFlags, hHandle, pfnMapUnmap);
2116 pdmUnlock(pVM);
2117
2118 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2119 return rc;
2120}
2121
2122
2123/** @interface_method_impl{PDMDEVHLPR3,pfnPCIInterceptConfigAccesses} */
2124static DECLCALLBACK(int) pdmR3DevHlp_PCIInterceptConfigAccesses(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev,
2125 PFNPCICONFIGREAD pfnRead, PFNPCICONFIGWRITE pfnWrite)
2126{
2127 PDMDEV_ASSERT_DEVINS(pDevIns);
2128 PVM pVM = pDevIns->Internal.s.pVMR3;
2129 VM_ASSERT_EMT(pVM);
2130 if (!pPciDev) /* NULL is an alias for the default PCI device. */
2131 pPciDev = pDevIns->apPciDevs[0];
2132 AssertReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
2133 LogFlow(("pdmR3DevHlp_PCIInterceptConfigAccesses: caller='%s'/%d: pPciDev=%p pfnRead=%p pfnWrite=%p\n",
2134 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pfnRead, pfnWrite));
2135 PDMPCIDEV_ASSERT_VALID_RET(pDevIns, pPciDev);
2136
2137 /*
2138 * Validate input.
2139 */
2140 AssertPtr(pfnRead);
2141 AssertPtr(pfnWrite);
2142 AssertPtr(pPciDev);
2143
2144 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
2145 AssertReturn(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses), VERR_INTERNAL_ERROR_2);
2146 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
2147 AssertRelease(VMR3GetState(pVM) != VMSTATE_RUNNING);
2148
2149 /*
2150 * Do the job.
2151 */
2152 pdmLock(pVM);
2153 pBus->pfnInterceptConfigAccesses(pBus->pDevInsR3, pPciDev, pfnRead, pfnWrite);
2154 pdmUnlock(pVM);
2155
2156 LogFlow(("pdmR3DevHlp_PCIInterceptConfigAccesses: caller='%s'/%d: returns VINF_SUCCESS\n",
2157 pDevIns->pReg->szName, pDevIns->iInstance));
2158 return VINF_SUCCESS;
2159}
2160
2161
2162/** @interface_method_impl{PDMDEVHLPR3,pfnPCIConfigWrite} */
2163static DECLCALLBACK(VBOXSTRICTRC)
2164pdmR3DevHlp_PCIConfigWrite(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t uAddress, unsigned cb, uint32_t u32Value)
2165{
2166 PDMDEV_ASSERT_DEVINS(pDevIns);
2167 PVM pVM = pDevIns->Internal.s.pVMR3;
2168 AssertPtrReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
2169 LogFlow(("pdmR3DevHlp_PCIConfigWrite: caller='%s'/%d: pPciDev=%p uAddress=%#x cd=%d u32Value=%#x\n",
2170 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, uAddress, cb, u32Value));
2171
2172 /*
2173 * Resolve the bus.
2174 */
2175 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
2176 AssertReturn(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses), VERR_INTERNAL_ERROR_2);
2177 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
2178
2179 /*
2180 * Do the job.
2181 */
2182 VBOXSTRICTRC rcStrict = pBus->pfnConfigWrite(pBus->pDevInsR3, pPciDev, uAddress, cb, u32Value);
2183
2184 LogFlow(("pdmR3DevHlp_PCIConfigWrite: caller='%s'/%d: returns %Rrc\n",
2185 pDevIns->pReg->szName, pDevIns->iInstance, VBOXSTRICTRC_VAL(rcStrict)));
2186 return rcStrict;
2187}
2188
2189
2190/** @interface_method_impl{PDMDEVHLPR3,pfnPCIConfigRead} */
2191static DECLCALLBACK(VBOXSTRICTRC)
2192pdmR3DevHlp_PCIConfigRead(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t uAddress, unsigned cb, uint32_t *pu32Value)
2193{
2194 PDMDEV_ASSERT_DEVINS(pDevIns);
2195 PVM pVM = pDevIns->Internal.s.pVMR3;
2196 AssertPtrReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
2197 LogFlow(("pdmR3DevHlp_PCIConfigRead: caller='%s'/%d: pPciDev=%p uAddress=%#x cd=%d pu32Value=%p:{%#x}\n",
2198 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, uAddress, cb, pu32Value, *pu32Value));
2199
2200 /*
2201 * Resolve the bus.
2202 */
2203 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
2204 AssertReturn(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses), VERR_INTERNAL_ERROR_2);
2205 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
2206
2207 /*
2208 * Do the job.
2209 */
2210 VBOXSTRICTRC rcStrict = pBus->pfnConfigRead(pBus->pDevInsR3, pPciDev, uAddress, cb, pu32Value);
2211
2212 LogFlow(("pdmR3DevHlp_PCIConfigRead: caller='%s'/%d: returns %Rrc (*pu32Value=%#x)\n",
2213 pDevIns->pReg->szName, pDevIns->iInstance, VBOXSTRICTRC_VAL(rcStrict), *pu32Value));
2214 return rcStrict;
2215}
2216
2217
2218/** @interface_method_impl{PDMDEVHLPR3,pfnPCIPhysRead} */
2219static DECLCALLBACK(int)
2220pdmR3DevHlp_PCIPhysRead(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
2221{
2222 PDMDEV_ASSERT_DEVINS(pDevIns);
2223 if (!pPciDev) /* NULL is an alias for the default PCI device. */
2224 pPciDev = pDevIns->apPciDevs[0];
2225 AssertReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
2226 PDMPCIDEV_ASSERT_VALID_AND_REGISTERED(pDevIns, pPciDev);
2227
2228#ifndef PDM_DO_NOT_RESPECT_PCI_BM_BIT
2229 /*
2230 * Just check the busmaster setting here and forward the request to the generic read helper.
2231 */
2232 if (PCIDevIsBusmaster(pPciDev))
2233 { /* likely */ }
2234 else
2235 {
2236 Log(("pdmR3DevHlp_PCIPhysRead: caller='%s'/%d: returns %Rrc - Not bus master! GCPhys=%RGp cbRead=%#zx\n",
2237 pDevIns->pReg->szName, pDevIns->iInstance, VERR_PDM_NOT_PCI_BUS_MASTER, GCPhys, cbRead));
2238 memset(pvBuf, 0xff, cbRead);
2239 return VERR_PDM_NOT_PCI_BUS_MASTER;
2240 }
2241#endif
2242
2243 return pDevIns->pHlpR3->pfnPhysRead(pDevIns, GCPhys, pvBuf, cbRead);
2244}
2245
2246
2247/** @interface_method_impl{PDMDEVHLPR3,pfnPCIPhysWrite} */
2248static DECLCALLBACK(int)
2249pdmR3DevHlp_PCIPhysWrite(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
2250{
2251 PDMDEV_ASSERT_DEVINS(pDevIns);
2252 if (!pPciDev) /* NULL is an alias for the default PCI device. */
2253 pPciDev = pDevIns->apPciDevs[0];
2254 AssertReturn(pPciDev, VERR_PDM_NOT_PCI_DEVICE);
2255 PDMPCIDEV_ASSERT_VALID_AND_REGISTERED(pDevIns, pPciDev);
2256
2257#ifndef PDM_DO_NOT_RESPECT_PCI_BM_BIT
2258 /*
2259 * Just check the busmaster setting here and forward the request to the generic read helper.
2260 */
2261 if (PCIDevIsBusmaster(pPciDev))
2262 { /* likely */ }
2263 else
2264 {
2265 Log(("pdmR3DevHlp_PCIPhysWrite: caller='%s'/%d: returns %Rrc - Not bus master! GCPhys=%RGp cbWrite=%#zx\n",
2266 pDevIns->pReg->szName, pDevIns->iInstance, VERR_PDM_NOT_PCI_BUS_MASTER, GCPhys, cbWrite));
2267 return VERR_PDM_NOT_PCI_BUS_MASTER;
2268 }
2269#endif
2270
2271 return pDevIns->pHlpR3->pfnPhysWrite(pDevIns, GCPhys, pvBuf, cbWrite);
2272}
2273
2274
2275/** @interface_method_impl{PDMDEVHLPR3,pfnPCISetIrq} */
2276static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrq(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, int iIrq, int iLevel)
2277{
2278 PDMDEV_ASSERT_DEVINS(pDevIns);
2279 if (!pPciDev) /* NULL is an alias for the default PCI device. */
2280 pPciDev = pDevIns->apPciDevs[0];
2281 AssertReturnVoid(pPciDev);
2282 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: pPciDev=%p:{%#x} iIrq=%d iLevel=%d\n",
2283 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev->uDevFn, iIrq, iLevel));
2284 PDMPCIDEV_ASSERT_VALID_AND_REGISTERED(pDevIns, pPciDev);
2285
2286 /*
2287 * Validate input.
2288 */
2289 Assert(iIrq == 0);
2290 Assert((uint32_t)iLevel <= PDM_IRQ_LEVEL_FLIP_FLOP);
2291
2292 /*
2293 * Must have a PCI device registered!
2294 */
2295 PVM pVM = pDevIns->Internal.s.pVMR3;
2296 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
2297 AssertReturnVoid(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses));
2298 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
2299
2300 pdmLock(pVM);
2301 uint32_t uTagSrc;
2302 if (iLevel & PDM_IRQ_LEVEL_HIGH)
2303 {
2304 pDevIns->Internal.s.uLastIrqTag = uTagSrc = pdmCalcIrqTag(pVM, pDevIns->idTracing);
2305 if (iLevel == PDM_IRQ_LEVEL_HIGH)
2306 VBOXVMM_PDM_IRQ_HIGH(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2307 else
2308 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2309 }
2310 else
2311 uTagSrc = pDevIns->Internal.s.uLastIrqTag;
2312
2313 pBus->pfnSetIrqR3(pBus->pDevInsR3, pPciDev, iIrq, iLevel, uTagSrc);
2314
2315 if (iLevel == PDM_IRQ_LEVEL_LOW)
2316 VBOXVMM_PDM_IRQ_LOW(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2317 pdmUnlock(pVM);
2318
2319 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
2320}
2321
2322
2323/** @interface_method_impl{PDMDEVHLPR3,pfnPCISetIrqNoWait} */
2324static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrqNoWait(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, int iIrq, int iLevel)
2325{
2326 pdmR3DevHlp_PCISetIrq(pDevIns, pPciDev, iIrq, iLevel);
2327}
2328
2329
2330/** @interface_method_impl{PDMDEVHLPR3,pfnISASetIrq} */
2331static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
2332{
2333 PDMDEV_ASSERT_DEVINS(pDevIns);
2334 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, iIrq, iLevel));
2335
2336 /*
2337 * Validate input.
2338 */
2339 Assert(iIrq < 16);
2340 Assert((uint32_t)iLevel <= PDM_IRQ_LEVEL_FLIP_FLOP);
2341
2342 PVM pVM = pDevIns->Internal.s.pVMR3;
2343
2344 /*
2345 * Do the job.
2346 */
2347 pdmLock(pVM);
2348 uint32_t uTagSrc;
2349 if (iLevel & PDM_IRQ_LEVEL_HIGH)
2350 {
2351 pDevIns->Internal.s.uLastIrqTag = uTagSrc = pdmCalcIrqTag(pVM, pDevIns->idTracing);
2352 if (iLevel == PDM_IRQ_LEVEL_HIGH)
2353 VBOXVMM_PDM_IRQ_HIGH(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2354 else
2355 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2356 }
2357 else
2358 uTagSrc = pDevIns->Internal.s.uLastIrqTag;
2359
2360 PDMIsaSetIrq(pVM, iIrq, iLevel, uTagSrc); /* (The API takes the lock recursively.) */
2361
2362 if (iLevel == PDM_IRQ_LEVEL_LOW)
2363 VBOXVMM_PDM_IRQ_LOW(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2364 pdmUnlock(pVM);
2365
2366 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
2367}
2368
2369
2370/** @interface_method_impl{PDMDEVHLPR3,pfnISASetIrqNoWait} */
2371static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
2372{
2373 pdmR3DevHlp_ISASetIrq(pDevIns, iIrq, iLevel);
2374}
2375
2376
2377/** @interface_method_impl{PDMDEVHLPR3,pfnIoApicSendMsi} */
2378static DECLCALLBACK(void) pdmR3DevHlp_IoApicSendMsi(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue)
2379{
2380 PDMDEV_ASSERT_DEVINS(pDevIns);
2381 LogFlow(("pdmR3DevHlp_IoApicSendMsi: caller='%s'/%d: GCPhys=%RGp uValue=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, uValue));
2382
2383 /*
2384 * Validate input.
2385 */
2386 Assert(GCPhys != 0);
2387 Assert(uValue != 0);
2388
2389 PVM pVM = pDevIns->Internal.s.pVMR3;
2390
2391 /*
2392 * Do the job.
2393 */
2394 pdmLock(pVM);
2395 uint32_t uTagSrc;
2396 pDevIns->Internal.s.uLastIrqTag = uTagSrc = pdmCalcIrqTag(pVM, pDevIns->idTracing);
2397 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
2398
2399 PDMIoApicSendMsi(pVM, GCPhys, uValue, uTagSrc); /* (The API takes the lock recursively.) */
2400
2401 pdmUnlock(pVM);
2402
2403 LogFlow(("pdmR3DevHlp_IoApicSendMsi: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
2404}
2405
2406
2407/** @interface_method_impl{PDMDEVHLPR3,pfnDriverAttach} */
2408static DECLCALLBACK(int) pdmR3DevHlp_DriverAttach(PPDMDEVINS pDevIns, uint32_t iLun, PPDMIBASE pBaseInterface, PPDMIBASE *ppBaseInterface, const char *pszDesc)
2409{
2410 PDMDEV_ASSERT_DEVINS(pDevIns);
2411 PVM pVM = pDevIns->Internal.s.pVMR3;
2412 VM_ASSERT_EMT(pVM);
2413 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: iLun=%d pBaseInterface=%p ppBaseInterface=%p pszDesc=%p:{%s}\n",
2414 pDevIns->pReg->szName, pDevIns->iInstance, iLun, pBaseInterface, ppBaseInterface, pszDesc, pszDesc));
2415
2416 /*
2417 * Lookup the LUN, it might already be registered.
2418 */
2419 PPDMLUN pLunPrev = NULL;
2420 PPDMLUN pLun = pDevIns->Internal.s.pLunsR3;
2421 for (; pLun; pLunPrev = pLun, pLun = pLun->pNext)
2422 if (pLun->iLun == iLun)
2423 break;
2424
2425 /*
2426 * Create the LUN if if wasn't found, else check if driver is already attached to it.
2427 */
2428 if (!pLun)
2429 {
2430 if ( !pBaseInterface
2431 || !pszDesc
2432 || !*pszDesc)
2433 {
2434 Assert(pBaseInterface);
2435 Assert(pszDesc || *pszDesc);
2436 return VERR_INVALID_PARAMETER;
2437 }
2438
2439 pLun = (PPDMLUN)MMR3HeapAlloc(pVM, MM_TAG_PDM_LUN, sizeof(*pLun));
2440 if (!pLun)
2441 return VERR_NO_MEMORY;
2442
2443 pLun->iLun = iLun;
2444 pLun->pNext = pLunPrev ? pLunPrev->pNext : NULL;
2445 pLun->pTop = NULL;
2446 pLun->pBottom = NULL;
2447 pLun->pDevIns = pDevIns;
2448 pLun->pUsbIns = NULL;
2449 pLun->pszDesc = pszDesc;
2450 pLun->pBase = pBaseInterface;
2451 if (!pLunPrev)
2452 pDevIns->Internal.s.pLunsR3 = pLun;
2453 else
2454 pLunPrev->pNext = pLun;
2455 Log(("pdmR3DevHlp_DriverAttach: Registered LUN#%d '%s' with device '%s'/%d.\n",
2456 iLun, pszDesc, pDevIns->pReg->szName, pDevIns->iInstance));
2457 }
2458 else if (pLun->pTop)
2459 {
2460 AssertMsgFailed(("Already attached! The device should keep track of such things!\n"));
2461 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_PDM_DRIVER_ALREADY_ATTACHED));
2462 return VERR_PDM_DRIVER_ALREADY_ATTACHED;
2463 }
2464 Assert(pLun->pBase == pBaseInterface);
2465
2466
2467 /*
2468 * Get the attached driver configuration.
2469 */
2470 int rc;
2471 PCFGMNODE pNode = CFGMR3GetChildF(pDevIns->Internal.s.pCfgHandle, "LUN#%u", iLun);
2472 if (pNode)
2473 rc = pdmR3DrvInstantiate(pVM, pNode, pBaseInterface, NULL /*pDrvAbove*/, pLun, ppBaseInterface);
2474 else
2475 rc = VERR_PDM_NO_ATTACHED_DRIVER;
2476
2477 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2478 return rc;
2479}
2480
2481
2482/** @interface_method_impl{PDMDEVHLPR3,pfnDriverDetach} */
2483static DECLCALLBACK(int) pdmR3DevHlp_DriverDetach(PPDMDEVINS pDevIns, PPDMDRVINS pDrvIns, uint32_t fFlags)
2484{
2485 PDMDEV_ASSERT_DEVINS(pDevIns); RT_NOREF_PV(pDevIns);
2486 LogFlow(("pdmR3DevHlp_DriverDetach: caller='%s'/%d: pDrvIns=%p\n",
2487 pDevIns->pReg->szName, pDevIns->iInstance, pDrvIns));
2488
2489#ifdef VBOX_STRICT
2490 PVM pVM = pDevIns->Internal.s.pVMR3;
2491 VM_ASSERT_EMT(pVM);
2492#endif
2493
2494 int rc = pdmR3DrvDetach(pDrvIns, fFlags);
2495
2496 LogFlow(("pdmR3DevHlp_DriverDetach: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2497 return rc;
2498}
2499
2500
2501/** @interface_method_impl{PDMDEVHLPR3,pfnQueueCreatePtr} */
2502static DECLCALLBACK(int) pdmR3DevHlp_QueueCreatePtr(PPDMDEVINS pDevIns, size_t cbItem, uint32_t cItems, uint32_t cMilliesInterval,
2503 PFNPDMQUEUEDEV pfnCallback, bool fRZEnabled, const char *pszName, PPDMQUEUE *ppQueue)
2504{
2505 PDMDEV_ASSERT_DEVINS(pDevIns);
2506 LogFlow(("pdmR3DevHlp_QueueCreatePtr: caller='%s'/%d: cbItem=%#x cItems=%#x cMilliesInterval=%u pfnCallback=%p fRZEnabled=%RTbool pszName=%p:{%s} ppQueue=%p\n",
2507 pDevIns->pReg->szName, pDevIns->iInstance, cbItem, cItems, cMilliesInterval, pfnCallback, fRZEnabled, pszName, pszName, ppQueue));
2508
2509 PVM pVM = pDevIns->Internal.s.pVMR3;
2510 VM_ASSERT_EMT(pVM);
2511
2512 if (pDevIns->iInstance > 0)
2513 {
2514 pszName = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s_%u", pszName, pDevIns->iInstance);
2515 AssertLogRelReturn(pszName, VERR_NO_MEMORY);
2516 }
2517
2518 int rc = PDMR3QueueCreateDevice(pVM, pDevIns, cbItem, cItems, cMilliesInterval, pfnCallback, fRZEnabled, pszName, ppQueue);
2519
2520 LogFlow(("pdmR3DevHlp_QueueCreatePtr: caller='%s'/%d: returns %Rrc *ppQueue=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *ppQueue));
2521 return rc;
2522}
2523
2524
2525/** @interface_method_impl{PDMDEVHLPR3,pfnQueueCreate} */
2526static DECLCALLBACK(int) pdmR3DevHlp_QueueCreate(PPDMDEVINS pDevIns, size_t cbItem, uint32_t cItems, uint32_t cMilliesInterval,
2527 PFNPDMQUEUEDEV pfnCallback, bool fRZEnabled, const char *pszName,
2528 PDMQUEUEHANDLE *phQueue)
2529{
2530 PDMDEV_ASSERT_DEVINS(pDevIns);
2531 LogFlow(("pdmR3DevHlp_QueueCreate: caller='%s'/%d: cbItem=%#x cItems=%#x cMilliesInterval=%u pfnCallback=%p fRZEnabled=%RTbool pszName=%p:{%s} phQueue=%p\n",
2532 pDevIns->pReg->szName, pDevIns->iInstance, cbItem, cItems, cMilliesInterval, pfnCallback, fRZEnabled, pszName, pszName, phQueue));
2533
2534 PVM pVM = pDevIns->Internal.s.pVMR3;
2535 VM_ASSERT_EMT(pVM);
2536
2537 if (pDevIns->iInstance > 0)
2538 {
2539 pszName = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s_%u", pszName, pDevIns->iInstance);
2540 AssertLogRelReturn(pszName, VERR_NO_MEMORY);
2541 }
2542
2543 PPDMQUEUE pQueue = NULL;
2544 int rc = PDMR3QueueCreateDevice(pVM, pDevIns, cbItem, cItems, cMilliesInterval, pfnCallback, fRZEnabled, pszName, &pQueue);
2545 *phQueue = (uintptr_t)pQueue;
2546
2547 LogFlow(("pdmR3DevHlp_QueueCreate: caller='%s'/%d: returns %Rrc *ppQueue=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *phQueue));
2548 return rc;
2549}
2550
2551
2552/** @interface_method_impl{PDMDEVHLPR3,pfnQueueToPtr} */
2553static DECLCALLBACK(PPDMQUEUE) pdmR3DevHlp_QueueToPtr(PPDMDEVINS pDevIns, PDMQUEUEHANDLE hQueue)
2554{
2555 PDMDEV_ASSERT_DEVINS(pDevIns);
2556 RT_NOREF(pDevIns);
2557 return (PPDMQUEUE)hQueue;
2558}
2559
2560
2561/** @interface_method_impl{PDMDEVHLPR3,pfnQueueAlloc} */
2562static DECLCALLBACK(PPDMQUEUEITEMCORE) pdmR3DevHlp_QueueAlloc(PPDMDEVINS pDevIns, PDMQUEUEHANDLE hQueue)
2563{
2564 return PDMQueueAlloc(pdmR3DevHlp_QueueToPtr(pDevIns, hQueue));
2565}
2566
2567
2568/** @interface_method_impl{PDMDEVHLPR3,pfnQueueInsert} */
2569static DECLCALLBACK(void) pdmR3DevHlp_QueueInsert(PPDMDEVINS pDevIns, PDMQUEUEHANDLE hQueue, PPDMQUEUEITEMCORE pItem)
2570{
2571 return PDMQueueInsert(pdmR3DevHlp_QueueToPtr(pDevIns, hQueue), pItem);
2572}
2573
2574
2575/** @interface_method_impl{PDMDEVHLPR3,pfnQueueInsertEx} */
2576static DECLCALLBACK(void) pdmR3DevHlp_QueueInsertEx(PPDMDEVINS pDevIns, PDMQUEUEHANDLE hQueue, PPDMQUEUEITEMCORE pItem,
2577 uint64_t cNanoMaxDelay)
2578{
2579 return PDMQueueInsertEx(pdmR3DevHlp_QueueToPtr(pDevIns, hQueue), pItem, cNanoMaxDelay);
2580}
2581
2582
2583/** @interface_method_impl{PDMDEVHLPR3,pfnQueueFlushIfNecessary} */
2584static DECLCALLBACK(bool) pdmR3DevHlp_QueueFlushIfNecessary(PPDMDEVINS pDevIns, PDMQUEUEHANDLE hQueue)
2585{
2586 return PDMQueueFlushIfNecessary(pdmR3DevHlp_QueueToPtr(pDevIns, hQueue));
2587}
2588
2589
2590/** @interface_method_impl{PDMDEVHLPR3,pfnTaskCreate} */
2591static DECLCALLBACK(int) pdmR3DevHlp_TaskCreate(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszName,
2592 PFNPDMTASKDEV pfnCallback, void *pvUser, PDMTASKHANDLE *phTask)
2593{
2594 PDMDEV_ASSERT_DEVINS(pDevIns);
2595 LogFlow(("pdmR3DevHlp_TaskTrigger: caller='%s'/%d: pfnCallback=%p fFlags=%#x pszName=%p:{%s} phTask=%p\n",
2596 pDevIns->pReg->szName, pDevIns->iInstance, pfnCallback, fFlags, pszName, pszName, phTask));
2597 PVM pVM = pDevIns->Internal.s.pVMR3;
2598 VM_ASSERT_EMT(pVM);
2599
2600 int rc = PDMR3TaskCreate(pVM, fFlags, pszName, PDMTASKTYPE_DEV, pDevIns, (PFNRT)pfnCallback, pvUser, phTask);
2601
2602 LogFlow(("pdmR3DevHlp_TaskTrigger: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2603 return rc;
2604}
2605
2606
2607/** @interface_method_impl{PDMDEVHLPR3,pfnTaskTrigger} */
2608static DECLCALLBACK(int) pdmR3DevHlp_TaskTrigger(PPDMDEVINS pDevIns, PDMTASKHANDLE hTask)
2609{
2610 PDMDEV_ASSERT_DEVINS(pDevIns);
2611 LogFlow(("pdmR3DevHlp_TaskTrigger: caller='%s'/%d: hTask=%RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, hTask));
2612
2613 int rc = PDMTaskTrigger(pDevIns->Internal.s.pVMR3, PDMTASKTYPE_DEV, pDevIns, hTask);
2614
2615 LogFlow(("pdmR3DevHlp_TaskTrigger: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2616 return rc;
2617}
2618
2619
2620/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventCreate} */
2621static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventCreate(PPDMDEVINS pDevIns, PSUPSEMEVENT phEvent)
2622{
2623 PDMDEV_ASSERT_DEVINS(pDevIns);
2624 LogFlow(("pdmR3DevHlp_SUPSemEventCreate: caller='%s'/%d: phEvent=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, phEvent));
2625 PVM pVM = pDevIns->Internal.s.pVMR3;
2626 VM_ASSERT_EMT(pVM);
2627
2628 int rc = SUPSemEventCreate(pVM->pSession, phEvent);
2629
2630 LogFlow(("pdmR3DevHlp_SUPSemEventCreate: caller='%s'/%d: returns %Rrc *phEvent=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *phEvent));
2631 return rc;
2632}
2633
2634
2635/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventClose} */
2636static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventClose(PPDMDEVINS pDevIns, SUPSEMEVENT hEvent)
2637{
2638 PDMDEV_ASSERT_DEVINS(pDevIns);
2639 LogFlow(("pdmR3DevHlp_SUPSemEventClose: caller='%s'/%d: hEvent=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, hEvent));
2640
2641 int rc = SUPSemEventClose(pDevIns->Internal.s.pVMR3->pSession, hEvent);
2642
2643 LogFlow(("pdmR3DevHlp_SUPSemEventClose: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2644 return rc;
2645}
2646
2647
2648/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventSignal} */
2649static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventSignal(PPDMDEVINS pDevIns, SUPSEMEVENT hEvent)
2650{
2651 PDMDEV_ASSERT_DEVINS(pDevIns);
2652 LogFlow(("pdmR3DevHlp_SUPSemEventSignal: caller='%s'/%d: hEvent=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, hEvent));
2653
2654 int rc = SUPSemEventSignal(pDevIns->Internal.s.pVMR3->pSession, hEvent);
2655
2656 LogFlow(("pdmR3DevHlp_SUPSemEventSignal: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2657 return rc;
2658}
2659
2660
2661/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventWaitNoResume} */
2662static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventWaitNoResume(PPDMDEVINS pDevIns, SUPSEMEVENT hEvent, uint32_t cMillies)
2663{
2664 PDMDEV_ASSERT_DEVINS(pDevIns);
2665 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNoResume: caller='%s'/%d: hEvent=%p cNsTimeout=%RU32\n",
2666 pDevIns->pReg->szName, pDevIns->iInstance, hEvent, cMillies));
2667
2668 int rc = SUPSemEventWaitNoResume(pDevIns->Internal.s.pVMR3->pSession, hEvent, cMillies);
2669
2670 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNoResume: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2671 return rc;
2672}
2673
2674
2675/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventWaitNsAbsIntr} */
2676static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventWaitNsAbsIntr(PPDMDEVINS pDevIns, SUPSEMEVENT hEvent, uint64_t uNsTimeout)
2677{
2678 PDMDEV_ASSERT_DEVINS(pDevIns);
2679 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNsAbsIntr: caller='%s'/%d: hEvent=%p uNsTimeout=%RU64\n",
2680 pDevIns->pReg->szName, pDevIns->iInstance, hEvent, uNsTimeout));
2681
2682 int rc = SUPSemEventWaitNsAbsIntr(pDevIns->Internal.s.pVMR3->pSession, hEvent, uNsTimeout);
2683
2684 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNsAbsIntr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2685 return rc;
2686}
2687
2688
2689/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventWaitNsRelIntr} */
2690static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventWaitNsRelIntr(PPDMDEVINS pDevIns, SUPSEMEVENT hEvent, uint64_t cNsTimeout)
2691{
2692 PDMDEV_ASSERT_DEVINS(pDevIns);
2693 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNsRelIntr: caller='%s'/%d: hEvent=%p cNsTimeout=%RU64\n",
2694 pDevIns->pReg->szName, pDevIns->iInstance, hEvent, cNsTimeout));
2695
2696 int rc = SUPSemEventWaitNsRelIntr(pDevIns->Internal.s.pVMR3->pSession, hEvent, cNsTimeout);
2697
2698 LogFlow(("pdmR3DevHlp_SUPSemEventWaitNsRelIntr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2699 return rc;
2700}
2701
2702
2703/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventGetResolution} */
2704static DECLCALLBACK(uint32_t) pdmR3DevHlp_SUPSemEventGetResolution(PPDMDEVINS pDevIns)
2705{
2706 PDMDEV_ASSERT_DEVINS(pDevIns);
2707 LogFlow(("pdmR3DevHlp_SUPSemEventGetResolution: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
2708
2709 uint32_t cNsResolution = SUPSemEventGetResolution(pDevIns->Internal.s.pVMR3->pSession);
2710
2711 LogFlow(("pdmR3DevHlp_SUPSemEventGetResolution: caller='%s'/%d: returns %u\n", pDevIns->pReg->szName, pDevIns->iInstance, cNsResolution));
2712 return cNsResolution;
2713}
2714
2715
2716/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiCreate} */
2717static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiCreate(PPDMDEVINS pDevIns, PSUPSEMEVENTMULTI phEventMulti)
2718{
2719 PDMDEV_ASSERT_DEVINS(pDevIns);
2720 LogFlow(("pdmR3DevHlp_SUPSemEventMultiCreate: caller='%s'/%d: phEventMulti=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, phEventMulti));
2721 PVM pVM = pDevIns->Internal.s.pVMR3;
2722 VM_ASSERT_EMT(pVM);
2723
2724 int rc = SUPSemEventMultiCreate(pVM->pSession, phEventMulti);
2725
2726 LogFlow(("pdmR3DevHlp_SUPSemEventMultiCreate: caller='%s'/%d: returns %Rrc *phEventMulti=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *phEventMulti));
2727 return rc;
2728}
2729
2730
2731/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiClose} */
2732static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiClose(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti)
2733{
2734 PDMDEV_ASSERT_DEVINS(pDevIns);
2735 LogFlow(("pdmR3DevHlp_SUPSemEventMultiClose: caller='%s'/%d: hEventMulti=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti));
2736
2737 int rc = SUPSemEventMultiClose(pDevIns->Internal.s.pVMR3->pSession, hEventMulti);
2738
2739 LogFlow(("pdmR3DevHlp_SUPSemEventMultiClose: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2740 return rc;
2741}
2742
2743
2744/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiSignal} */
2745static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiSignal(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti)
2746{
2747 PDMDEV_ASSERT_DEVINS(pDevIns);
2748 LogFlow(("pdmR3DevHlp_SUPSemEventMultiSignal: caller='%s'/%d: hEventMulti=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti));
2749
2750 int rc = SUPSemEventMultiSignal(pDevIns->Internal.s.pVMR3->pSession, hEventMulti);
2751
2752 LogFlow(("pdmR3DevHlp_SUPSemEventMultiSignal: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2753 return rc;
2754}
2755
2756
2757/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiReset} */
2758static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiReset(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti)
2759{
2760 PDMDEV_ASSERT_DEVINS(pDevIns);
2761 LogFlow(("pdmR3DevHlp_SUPSemEventMultiReset: caller='%s'/%d: hEventMulti=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti));
2762
2763 int rc = SUPSemEventMultiReset(pDevIns->Internal.s.pVMR3->pSession, hEventMulti);
2764
2765 LogFlow(("pdmR3DevHlp_SUPSemEventMultiReset: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2766 return rc;
2767}
2768
2769
2770/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiWaitNoResume} */
2771static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiWaitNoResume(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti,
2772 uint32_t cMillies)
2773{
2774 PDMDEV_ASSERT_DEVINS(pDevIns);
2775 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNoResume: caller='%s'/%d: hEventMulti=%p cMillies=%RU32\n",
2776 pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti, cMillies));
2777
2778 int rc = SUPSemEventMultiWaitNoResume(pDevIns->Internal.s.pVMR3->pSession, hEventMulti, cMillies);
2779
2780 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNoResume: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2781 return rc;
2782}
2783
2784
2785/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiWaitNsAbsIntr} */
2786static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiWaitNsAbsIntr(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti,
2787 uint64_t uNsTimeout)
2788{
2789 PDMDEV_ASSERT_DEVINS(pDevIns);
2790 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNsAbsIntr: caller='%s'/%d: hEventMulti=%p uNsTimeout=%RU64\n",
2791 pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti, uNsTimeout));
2792
2793 int rc = SUPSemEventMultiWaitNsAbsIntr(pDevIns->Internal.s.pVMR3->pSession, hEventMulti, uNsTimeout);
2794
2795 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNsAbsIntr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2796 return rc;
2797}
2798
2799
2800/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiWaitNsRelIntr} */
2801static DECLCALLBACK(int) pdmR3DevHlp_SUPSemEventMultiWaitNsRelIntr(PPDMDEVINS pDevIns, SUPSEMEVENTMULTI hEventMulti,
2802 uint64_t cNsTimeout)
2803{
2804 PDMDEV_ASSERT_DEVINS(pDevIns);
2805 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNsRelIntr: caller='%s'/%d: hEventMulti=%p cNsTimeout=%RU64\n",
2806 pDevIns->pReg->szName, pDevIns->iInstance, hEventMulti, cNsTimeout));
2807
2808 int rc = SUPSemEventMultiWaitNsRelIntr(pDevIns->Internal.s.pVMR3->pSession, hEventMulti, cNsTimeout);
2809
2810 LogFlow(("pdmR3DevHlp_SUPSemEventMultiWaitNsRelIntr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2811 return rc;
2812}
2813
2814
2815/** @interface_method_impl{PDMDEVHLPR3,pfnSUPSemEventMultiGetResolution} */
2816static DECLCALLBACK(uint32_t) pdmR3DevHlp_SUPSemEventMultiGetResolution(PPDMDEVINS pDevIns)
2817{
2818 PDMDEV_ASSERT_DEVINS(pDevIns);
2819 LogFlow(("pdmR3DevHlp_SUPSemEventMultiGetResolution: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
2820
2821 uint32_t cNsResolution = SUPSemEventMultiGetResolution(pDevIns->Internal.s.pVMR3->pSession);
2822
2823 LogFlow(("pdmR3DevHlp_SUPSemEventMultiGetResolution: caller='%s'/%d: returns %u\n", pDevIns->pReg->szName, pDevIns->iInstance, cNsResolution));
2824 return cNsResolution;
2825}
2826
2827
2828/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectInit} */
2829static DECLCALLBACK(int) pdmR3DevHlp_CritSectInit(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, RT_SRC_POS_DECL,
2830 const char *pszNameFmt, va_list va)
2831{
2832 PDMDEV_ASSERT_DEVINS(pDevIns);
2833 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: pCritSect=%p pszNameFmt=%p:{%s}\n",
2834 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect, pszNameFmt, pszNameFmt));
2835
2836 PVM pVM = pDevIns->Internal.s.pVMR3;
2837 VM_ASSERT_EMT(pVM);
2838 int rc = pdmR3CritSectInitDevice(pVM, pDevIns, pCritSect, RT_SRC_POS_ARGS, pszNameFmt, va);
2839
2840 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2841 return rc;
2842}
2843
2844
2845/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectGetNop} */
2846static DECLCALLBACK(PPDMCRITSECT) pdmR3DevHlp_CritSectGetNop(PPDMDEVINS pDevIns)
2847{
2848 PDMDEV_ASSERT_DEVINS(pDevIns);
2849 PVM pVM = pDevIns->Internal.s.pVMR3;
2850 VM_ASSERT_EMT(pVM);
2851
2852 PPDMCRITSECT pCritSect = PDMR3CritSectGetNop(pVM);
2853 LogFlow(("pdmR3DevHlp_CritSectGetNop: caller='%s'/%d: return %p\n",
2854 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect));
2855 return pCritSect;
2856}
2857
2858
2859/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectGetNopR0} */
2860static DECLCALLBACK(R0PTRTYPE(PPDMCRITSECT)) pdmR3DevHlp_CritSectGetNopR0(PPDMDEVINS pDevIns)
2861{
2862 PDMDEV_ASSERT_DEVINS(pDevIns);
2863 PVM pVM = pDevIns->Internal.s.pVMR3;
2864 VM_ASSERT_EMT(pVM);
2865
2866 R0PTRTYPE(PPDMCRITSECT) pCritSect = PDMR3CritSectGetNopR0(pVM);
2867 LogFlow(("pdmR3DevHlp_CritSectGetNopR0: caller='%s'/%d: return %RHv\n",
2868 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect));
2869 return pCritSect;
2870}
2871
2872
2873/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectGetNopRC} */
2874static DECLCALLBACK(RCPTRTYPE(PPDMCRITSECT)) pdmR3DevHlp_CritSectGetNopRC(PPDMDEVINS pDevIns)
2875{
2876 PDMDEV_ASSERT_DEVINS(pDevIns);
2877 PVM pVM = pDevIns->Internal.s.pVMR3;
2878 VM_ASSERT_EMT(pVM);
2879
2880 RCPTRTYPE(PPDMCRITSECT) pCritSect = PDMR3CritSectGetNopRC(pVM);
2881 LogFlow(("pdmR3DevHlp_CritSectGetNopRC: caller='%s'/%d: return %RRv\n",
2882 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect));
2883 return pCritSect;
2884}
2885
2886
2887/** @interface_method_impl{PDMDEVHLPR3,pfnSetDeviceCritSect} */
2888static DECLCALLBACK(int) pdmR3DevHlp_SetDeviceCritSect(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
2889{
2890 /*
2891 * Validate input.
2892 *
2893 * Note! We only allow the automatically created default critical section
2894 * to be replaced by this API.
2895 */
2896 PDMDEV_ASSERT_DEVINS(pDevIns);
2897 AssertPtrReturn(pCritSect, VERR_INVALID_POINTER);
2898 LogFlow(("pdmR3DevHlp_SetDeviceCritSect: caller='%s'/%d: pCritSect=%p (%s)\n",
2899 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect, pCritSect->s.pszName));
2900 AssertReturn(PDMCritSectIsInitialized(pCritSect), VERR_INVALID_PARAMETER);
2901 PVM pVM = pDevIns->Internal.s.pVMR3;
2902 AssertReturn(pCritSect->s.pVMR3 == pVM, VERR_INVALID_PARAMETER);
2903
2904 VM_ASSERT_EMT(pVM);
2905 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_WRONG_ORDER);
2906
2907 AssertReturn(pDevIns->pCritSectRoR3, VERR_PDM_DEV_IPE_1);
2908 AssertReturn(pDevIns->pCritSectRoR3->s.fAutomaticDefaultCritsect, VERR_WRONG_ORDER);
2909 AssertReturn(!pDevIns->pCritSectRoR3->s.fUsedByTimerOrSimilar, VERR_WRONG_ORDER);
2910 AssertReturn(pDevIns->pCritSectRoR3 != pCritSect, VERR_INVALID_PARAMETER);
2911
2912 /*
2913 * Replace the critical section and destroy the automatic default section.
2914 */
2915 PPDMCRITSECT pOldCritSect = pDevIns->pCritSectRoR3;
2916 pDevIns->pCritSectRoR3 = pCritSect;
2917 pDevIns->Internal.s.fIntFlags |= PDMDEVINSINT_FLAGS_CHANGED_CRITSECT;
2918
2919 Assert(RT_BOOL(pDevIns->Internal.s.fIntFlags & PDMDEVINSINT_FLAGS_R0_ENABLED) == pDevIns->fR0Enabled);
2920 if ( (pDevIns->Internal.s.fIntFlags & PDMDEVINSINT_FLAGS_R0_ENABLED)
2921 && !(pDevIns->Internal.s.pDevR3->pReg->fFlags & PDM_DEVREG_FLAGS_NEW_STYLE))
2922 {
2923 PDMDEVICECOMPATSETCRITSECTREQ Req;
2924 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
2925 Req.Hdr.cbReq = sizeof(Req);
2926 Req.idxR0Device = pDevIns->Internal.s.idxR0Device;
2927 Req.pDevInsR3 = pDevIns;
2928 Req.pCritSectR3 = pCritSect;
2929 int rc = VMMR3CallR0(pVM, VMMR0_DO_PDM_DEVICE_COMPAT_SET_CRITSECT, 0, &Req.Hdr);
2930 AssertLogRelRCReturn(rc, rc);
2931 }
2932
2933 PDMR3CritSectDelete(pOldCritSect);
2934 Assert((uintptr_t)pOldCritSect - (uintptr_t)pDevIns < pDevIns->cbRing3);
2935
2936 LogFlow(("pdmR3DevHlp_SetDeviceCritSect: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2937 return VINF_SUCCESS;
2938}
2939
2940
2941/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectYield} */
2942static DECLCALLBACK(bool) pdmR3DevHlp_CritSectYield(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
2943{
2944 PDMDEV_ASSERT_DEVINS(pDevIns);
2945 return PDMR3CritSectYield(pDevIns->Internal.s.pVMR3, pCritSect);
2946}
2947
2948
2949/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectEnter} */
2950static DECLCALLBACK(int) pdmR3DevHlp_CritSectEnter(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, int rcBusy)
2951{
2952 PDMDEV_ASSERT_DEVINS(pDevIns);
2953 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2954 return PDMCritSectEnter(pCritSect, rcBusy);
2955}
2956
2957
2958/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectEnterDebug} */
2959static DECLCALLBACK(int) pdmR3DevHlp_CritSectEnterDebug(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, int rcBusy, RTHCUINTPTR uId, RT_SRC_POS_DECL)
2960{
2961 PDMDEV_ASSERT_DEVINS(pDevIns);
2962 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2963 return PDMCritSectEnterDebug(pCritSect, rcBusy, uId, RT_SRC_POS_ARGS);
2964}
2965
2966
2967/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectTryEnter} */
2968static DECLCALLBACK(int) pdmR3DevHlp_CritSectTryEnter(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
2969{
2970 PDMDEV_ASSERT_DEVINS(pDevIns);
2971 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2972 return PDMCritSectTryEnter(pCritSect);
2973}
2974
2975
2976/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectTryEnterDebug} */
2977static DECLCALLBACK(int) pdmR3DevHlp_CritSectTryEnterDebug(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, RTHCUINTPTR uId, RT_SRC_POS_DECL)
2978{
2979 PDMDEV_ASSERT_DEVINS(pDevIns);
2980 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2981 return PDMCritSectTryEnterDebug(pCritSect, uId, RT_SRC_POS_ARGS);
2982}
2983
2984
2985/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectLeave} */
2986static DECLCALLBACK(int) pdmR3DevHlp_CritSectLeave(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
2987{
2988 PDMDEV_ASSERT_DEVINS(pDevIns);
2989 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2990 return PDMCritSectLeave(pCritSect);
2991}
2992
2993
2994/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectIsOwner} */
2995static DECLCALLBACK(bool) pdmR3DevHlp_CritSectIsOwner(PPDMDEVINS pDevIns, PCPDMCRITSECT pCritSect)
2996{
2997 PDMDEV_ASSERT_DEVINS(pDevIns);
2998 RT_NOREF(pDevIns); /** @todo pass pDevIns->Internal.s.pVMR3 to the crit sect code. */
2999 return PDMCritSectIsOwner(pCritSect);
3000}
3001
3002
3003/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectIsInitialized} */
3004static DECLCALLBACK(bool) pdmR3DevHlp_CritSectIsInitialized(PPDMDEVINS pDevIns, PCPDMCRITSECT pCritSect)
3005{
3006 PDMDEV_ASSERT_DEVINS(pDevIns);
3007 RT_NOREF(pDevIns);
3008 return PDMCritSectIsInitialized(pCritSect);
3009}
3010
3011
3012/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectHasWaiters} */
3013static DECLCALLBACK(bool) pdmR3DevHlp_CritSectHasWaiters(PPDMDEVINS pDevIns, PCPDMCRITSECT pCritSect)
3014{
3015 PDMDEV_ASSERT_DEVINS(pDevIns);
3016 RT_NOREF(pDevIns);
3017 return PDMCritSectHasWaiters(pCritSect);
3018}
3019
3020
3021/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectGetRecursion} */
3022static DECLCALLBACK(uint32_t) pdmR3DevHlp_CritSectGetRecursion(PPDMDEVINS pDevIns, PCPDMCRITSECT pCritSect)
3023{
3024 PDMDEV_ASSERT_DEVINS(pDevIns);
3025 RT_NOREF(pDevIns);
3026 return PDMCritSectGetRecursion(pCritSect);
3027}
3028
3029
3030/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectScheduleExitEvent} */
3031static DECLCALLBACK(int) pdmR3DevHlp_CritSectScheduleExitEvent(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect,
3032 SUPSEMEVENT hEventToSignal)
3033{
3034 PDMDEV_ASSERT_DEVINS(pDevIns);
3035 RT_NOREF(pDevIns);
3036 return PDMHCCritSectScheduleExitEvent(pCritSect, hEventToSignal);
3037}
3038
3039
3040/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectDelete} */
3041static DECLCALLBACK(int) pdmR3DevHlp_CritSectDelete(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
3042{
3043 PDMDEV_ASSERT_DEVINS(pDevIns);
3044 RT_NOREF(pDevIns);
3045 return PDMR3CritSectDelete(pCritSect);
3046}
3047
3048
3049/** @interface_method_impl{PDMDEVHLPR3,pfnThreadCreate} */
3050static DECLCALLBACK(int) pdmR3DevHlp_ThreadCreate(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
3051 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName)
3052{
3053 PDMDEV_ASSERT_DEVINS(pDevIns);
3054 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3055 LogFlow(("pdmR3DevHlp_ThreadCreate: caller='%s'/%d: ppThread=%p pvUser=%p pfnThread=%p pfnWakeup=%p cbStack=%#zx enmType=%d pszName=%p:{%s}\n",
3056 pDevIns->pReg->szName, pDevIns->iInstance, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName, pszName));
3057
3058 int rc = pdmR3ThreadCreateDevice(pDevIns->Internal.s.pVMR3, pDevIns, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName);
3059
3060 LogFlow(("pdmR3DevHlp_ThreadCreate: caller='%s'/%d: returns %Rrc *ppThread=%RTthrd\n", pDevIns->pReg->szName, pDevIns->iInstance,
3061 rc, *ppThread));
3062 return rc;
3063}
3064
3065
3066/** @interface_method_impl{PDMDEVHLPR3,pfnSetAsyncNotification} */
3067static DECLCALLBACK(int) pdmR3DevHlp_SetAsyncNotification(PPDMDEVINS pDevIns, PFNPDMDEVASYNCNOTIFY pfnAsyncNotify)
3068{
3069 PDMDEV_ASSERT_DEVINS(pDevIns);
3070 VM_ASSERT_EMT0(pDevIns->Internal.s.pVMR3);
3071 LogFlow(("pdmR3DevHlp_SetAsyncNotification: caller='%s'/%d: pfnAsyncNotify=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pfnAsyncNotify));
3072
3073 int rc = VINF_SUCCESS;
3074 AssertStmt(pfnAsyncNotify, rc = VERR_INVALID_PARAMETER);
3075 AssertStmt(!pDevIns->Internal.s.pfnAsyncNotify, rc = VERR_WRONG_ORDER);
3076 AssertStmt(pDevIns->Internal.s.fIntFlags & (PDMDEVINSINT_FLAGS_SUSPENDED | PDMDEVINSINT_FLAGS_RESET), rc = VERR_WRONG_ORDER);
3077 VMSTATE enmVMState = VMR3GetState(pDevIns->Internal.s.pVMR3);
3078 AssertStmt( enmVMState == VMSTATE_SUSPENDING
3079 || enmVMState == VMSTATE_SUSPENDING_EXT_LS
3080 || enmVMState == VMSTATE_SUSPENDING_LS
3081 || enmVMState == VMSTATE_RESETTING
3082 || enmVMState == VMSTATE_RESETTING_LS
3083 || enmVMState == VMSTATE_POWERING_OFF
3084 || enmVMState == VMSTATE_POWERING_OFF_LS,
3085 rc = VERR_INVALID_STATE);
3086
3087 if (RT_SUCCESS(rc))
3088 pDevIns->Internal.s.pfnAsyncNotify = pfnAsyncNotify;
3089
3090 LogFlow(("pdmR3DevHlp_SetAsyncNotification: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
3091 return rc;
3092}
3093
3094
3095/** @interface_method_impl{PDMDEVHLPR3,pfnAsyncNotificationCompleted} */
3096static DECLCALLBACK(void) pdmR3DevHlp_AsyncNotificationCompleted(PPDMDEVINS pDevIns)
3097{
3098 PDMDEV_ASSERT_DEVINS(pDevIns);
3099 PVM pVM = pDevIns->Internal.s.pVMR3;
3100
3101 VMSTATE enmVMState = VMR3GetState(pVM);
3102 if ( enmVMState == VMSTATE_SUSPENDING
3103 || enmVMState == VMSTATE_SUSPENDING_EXT_LS
3104 || enmVMState == VMSTATE_SUSPENDING_LS
3105 || enmVMState == VMSTATE_RESETTING
3106 || enmVMState == VMSTATE_RESETTING_LS
3107 || enmVMState == VMSTATE_POWERING_OFF
3108 || enmVMState == VMSTATE_POWERING_OFF_LS)
3109 {
3110 LogFlow(("pdmR3DevHlp_AsyncNotificationCompleted: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
3111 VMR3AsyncPdmNotificationWakeupU(pVM->pUVM);
3112 }
3113 else
3114 LogFlow(("pdmR3DevHlp_AsyncNotificationCompleted: caller='%s'/%d: enmVMState=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, enmVMState));
3115}
3116
3117
3118/** @interface_method_impl{PDMDEVHLPR3,pfnRTCRegister} */
3119static DECLCALLBACK(int) pdmR3DevHlp_RTCRegister(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp)
3120{
3121 PDMDEV_ASSERT_DEVINS(pDevIns);
3122 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3123 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: pRtcReg=%p:{.u32Version=%#x, .pfnWrite=%p, .pfnRead=%p} ppRtcHlp=%p\n",
3124 pDevIns->pReg->szName, pDevIns->iInstance, pRtcReg, pRtcReg->u32Version, pRtcReg->pfnWrite,
3125 pRtcReg->pfnWrite, ppRtcHlp));
3126
3127 /*
3128 * Validate input.
3129 */
3130 if (pRtcReg->u32Version != PDM_RTCREG_VERSION)
3131 {
3132 AssertMsgFailed(("u32Version=%#x expected %#x\n", pRtcReg->u32Version,
3133 PDM_RTCREG_VERSION));
3134 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (version)\n",
3135 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3136 return VERR_INVALID_PARAMETER;
3137 }
3138 if ( !pRtcReg->pfnWrite
3139 || !pRtcReg->pfnRead)
3140 {
3141 Assert(pRtcReg->pfnWrite);
3142 Assert(pRtcReg->pfnRead);
3143 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (callbacks)\n",
3144 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3145 return VERR_INVALID_PARAMETER;
3146 }
3147
3148 if (!ppRtcHlp)
3149 {
3150 Assert(ppRtcHlp);
3151 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (ppRtcHlp)\n",
3152 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3153 return VERR_INVALID_PARAMETER;
3154 }
3155
3156 /*
3157 * Only one DMA device.
3158 */
3159 PVM pVM = pDevIns->Internal.s.pVMR3;
3160 if (pVM->pdm.s.pRtc)
3161 {
3162 AssertMsgFailed(("Only one RTC device is supported!\n"));
3163 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc\n",
3164 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3165 return VERR_INVALID_PARAMETER;
3166 }
3167
3168 /*
3169 * Allocate and initialize pci bus structure.
3170 */
3171 int rc = VINF_SUCCESS;
3172 PPDMRTC pRtc = (PPDMRTC)MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE, sizeof(*pRtc));
3173 if (pRtc)
3174 {
3175 pRtc->pDevIns = pDevIns;
3176 pRtc->Reg = *pRtcReg;
3177 pVM->pdm.s.pRtc = pRtc;
3178
3179 /* set the helper pointer. */
3180 *ppRtcHlp = &g_pdmR3DevRtcHlp;
3181 Log(("PDM: Registered RTC device '%s'/%d pDevIns=%p\n",
3182 pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
3183 }
3184 else
3185 rc = VERR_NO_MEMORY;
3186
3187 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc\n",
3188 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3189 return rc;
3190}
3191
3192
3193/** @interface_method_impl{PDMDEVHLPR3,pfnDMARegister} */
3194static DECLCALLBACK(int) pdmR3DevHlp_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser)
3195{
3196 PDMDEV_ASSERT_DEVINS(pDevIns);
3197 PVM pVM = pDevIns->Internal.s.pVMR3;
3198 VM_ASSERT_EMT(pVM);
3199 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: uChannel=%d pfnTransferHandler=%p pvUser=%p\n",
3200 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pfnTransferHandler, pvUser));
3201 int rc = VINF_SUCCESS;
3202 if (pVM->pdm.s.pDmac)
3203 pVM->pdm.s.pDmac->Reg.pfnRegister(pVM->pdm.s.pDmac->pDevIns, uChannel, pfnTransferHandler, pvUser);
3204 else
3205 {
3206 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3207 rc = VERR_PDM_NO_DMAC_INSTANCE;
3208 }
3209 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: returns %Rrc\n",
3210 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3211 return rc;
3212}
3213
3214
3215/** @interface_method_impl{PDMDEVHLPR3,pfnDMAReadMemory} */
3216static DECLCALLBACK(int) pdmR3DevHlp_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead)
3217{
3218 PDMDEV_ASSERT_DEVINS(pDevIns);
3219 PVM pVM = pDevIns->Internal.s.pVMR3;
3220 VM_ASSERT_EMT(pVM);
3221 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbRead=%p\n",
3222 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbRead));
3223 int rc = VINF_SUCCESS;
3224 if (pVM->pdm.s.pDmac)
3225 {
3226 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnReadMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
3227 if (pcbRead)
3228 *pcbRead = cb;
3229 }
3230 else
3231 {
3232 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3233 rc = VERR_PDM_NO_DMAC_INSTANCE;
3234 }
3235 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: returns %Rrc\n",
3236 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3237 return rc;
3238}
3239
3240
3241/** @interface_method_impl{PDMDEVHLPR3,pfnDMAWriteMemory} */
3242static DECLCALLBACK(int) pdmR3DevHlp_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten)
3243{
3244 PDMDEV_ASSERT_DEVINS(pDevIns);
3245 PVM pVM = pDevIns->Internal.s.pVMR3;
3246 VM_ASSERT_EMT(pVM);
3247 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbWritten=%p\n",
3248 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbWritten));
3249 int rc = VINF_SUCCESS;
3250 if (pVM->pdm.s.pDmac)
3251 {
3252 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnWriteMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
3253 if (pcbWritten)
3254 *pcbWritten = cb;
3255 }
3256 else
3257 {
3258 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3259 rc = VERR_PDM_NO_DMAC_INSTANCE;
3260 }
3261 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: returns %Rrc\n",
3262 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3263 return rc;
3264}
3265
3266
3267/** @interface_method_impl{PDMDEVHLPR3,pfnDMASetDREQ} */
3268static DECLCALLBACK(int) pdmR3DevHlp_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel)
3269{
3270 PDMDEV_ASSERT_DEVINS(pDevIns);
3271 PVM pVM = pDevIns->Internal.s.pVMR3;
3272 VM_ASSERT_EMT(pVM);
3273 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: uChannel=%d uLevel=%d\n",
3274 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, uLevel));
3275 int rc = VINF_SUCCESS;
3276 if (pVM->pdm.s.pDmac)
3277 pVM->pdm.s.pDmac->Reg.pfnSetDREQ(pVM->pdm.s.pDmac->pDevIns, uChannel, uLevel);
3278 else
3279 {
3280 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3281 rc = VERR_PDM_NO_DMAC_INSTANCE;
3282 }
3283 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: returns %Rrc\n",
3284 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3285 return rc;
3286}
3287
3288/** @interface_method_impl{PDMDEVHLPR3,pfnDMAGetChannelMode} */
3289static DECLCALLBACK(uint8_t) pdmR3DevHlp_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel)
3290{
3291 PDMDEV_ASSERT_DEVINS(pDevIns);
3292 PVM pVM = pDevIns->Internal.s.pVMR3;
3293 VM_ASSERT_EMT(pVM);
3294 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: uChannel=%d\n",
3295 pDevIns->pReg->szName, pDevIns->iInstance, uChannel));
3296 uint8_t u8Mode;
3297 if (pVM->pdm.s.pDmac)
3298 u8Mode = pVM->pdm.s.pDmac->Reg.pfnGetChannelMode(pVM->pdm.s.pDmac->pDevIns, uChannel);
3299 else
3300 {
3301 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3302 u8Mode = 3 << 2 /* illegal mode type */;
3303 }
3304 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: returns %#04x\n",
3305 pDevIns->pReg->szName, pDevIns->iInstance, u8Mode));
3306 return u8Mode;
3307}
3308
3309/** @interface_method_impl{PDMDEVHLPR3,pfnDMASchedule} */
3310static DECLCALLBACK(void) pdmR3DevHlp_DMASchedule(PPDMDEVINS pDevIns)
3311{
3312 PDMDEV_ASSERT_DEVINS(pDevIns);
3313 PVM pVM = pDevIns->Internal.s.pVMR3;
3314 VM_ASSERT_EMT(pVM);
3315 LogFlow(("pdmR3DevHlp_DMASchedule: caller='%s'/%d: VM_FF_PDM_DMA %d -> 1\n",
3316 pDevIns->pReg->szName, pDevIns->iInstance, VM_FF_IS_SET(pVM, VM_FF_PDM_DMA)));
3317
3318 AssertMsg(pVM->pdm.s.pDmac, ("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
3319 VM_FF_SET(pVM, VM_FF_PDM_DMA);
3320 VMR3NotifyGlobalFFU(pVM->pUVM, VMNOTIFYFF_FLAGS_DONE_REM);
3321}
3322
3323
3324/** @interface_method_impl{PDMDEVHLPR3,pfnCMOSWrite} */
3325static DECLCALLBACK(int) pdmR3DevHlp_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value)
3326{
3327 PDMDEV_ASSERT_DEVINS(pDevIns);
3328 PVM pVM = pDevIns->Internal.s.pVMR3;
3329 VM_ASSERT_EMT(pVM);
3330
3331 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x u8Value=%#04x\n",
3332 pDevIns->pReg->szName, pDevIns->iInstance, iReg, u8Value));
3333 int rc;
3334 if (pVM->pdm.s.pRtc)
3335 {
3336 PPDMDEVINS pDevInsRtc = pVM->pdm.s.pRtc->pDevIns;
3337 rc = PDMCritSectEnter(pDevInsRtc->pCritSectRoR3, VERR_IGNORED);
3338 if (RT_SUCCESS(rc))
3339 {
3340 rc = pVM->pdm.s.pRtc->Reg.pfnWrite(pDevInsRtc, iReg, u8Value);
3341 PDMCritSectLeave(pDevInsRtc->pCritSectRoR3);
3342 }
3343 }
3344 else
3345 rc = VERR_PDM_NO_RTC_INSTANCE;
3346
3347 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Rrc\n",
3348 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3349 return rc;
3350}
3351
3352
3353/** @interface_method_impl{PDMDEVHLPR3,pfnCMOSRead} */
3354static DECLCALLBACK(int) pdmR3DevHlp_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value)
3355{
3356 PDMDEV_ASSERT_DEVINS(pDevIns);
3357 PVM pVM = pDevIns->Internal.s.pVMR3;
3358 VM_ASSERT_EMT(pVM);
3359
3360 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x pu8Value=%p\n",
3361 pDevIns->pReg->szName, pDevIns->iInstance, iReg, pu8Value));
3362 int rc;
3363 if (pVM->pdm.s.pRtc)
3364 {
3365 PPDMDEVINS pDevInsRtc = pVM->pdm.s.pRtc->pDevIns;
3366 rc = PDMCritSectEnter(pDevInsRtc->pCritSectRoR3, VERR_IGNORED);
3367 if (RT_SUCCESS(rc))
3368 {
3369 rc = pVM->pdm.s.pRtc->Reg.pfnRead(pDevInsRtc, iReg, pu8Value);
3370 PDMCritSectLeave(pDevInsRtc->pCritSectRoR3);
3371 }
3372 }
3373 else
3374 rc = VERR_PDM_NO_RTC_INSTANCE;
3375
3376 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Rrc\n",
3377 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3378 return rc;
3379}
3380
3381
3382/** @interface_method_impl{PDMDEVHLPR3,pfnAssertEMT} */
3383static DECLCALLBACK(bool) pdmR3DevHlp_AssertEMT(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
3384{
3385 PDMDEV_ASSERT_DEVINS(pDevIns);
3386 if (VM_IS_EMT(pDevIns->Internal.s.pVMR3))
3387 return true;
3388
3389 char szMsg[100];
3390 RTStrPrintf(szMsg, sizeof(szMsg), "AssertEMT '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance);
3391 RTAssertMsg1Weak(szMsg, iLine, pszFile, pszFunction);
3392 AssertBreakpoint();
3393 return false;
3394}
3395
3396
3397/** @interface_method_impl{PDMDEVHLPR3,pfnAssertOther} */
3398static DECLCALLBACK(bool) pdmR3DevHlp_AssertOther(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
3399{
3400 PDMDEV_ASSERT_DEVINS(pDevIns);
3401 if (!VM_IS_EMT(pDevIns->Internal.s.pVMR3))
3402 return true;
3403
3404 char szMsg[100];
3405 RTStrPrintf(szMsg, sizeof(szMsg), "AssertOther '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance);
3406 RTAssertMsg1Weak(szMsg, iLine, pszFile, pszFunction);
3407 AssertBreakpoint();
3408 return false;
3409}
3410
3411
3412/** @interface_method_impl{PDMDEVHLPR3,pfnLdrGetRCInterfaceSymbols} */
3413static DECLCALLBACK(int) pdmR3DevHlp_LdrGetRCInterfaceSymbols(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
3414 const char *pszSymPrefix, const char *pszSymList)
3415{
3416 PDMDEV_ASSERT_DEVINS(pDevIns);
3417 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3418 LogFlow(("pdmR3DevHlp_PDMLdrGetRCInterfaceSymbols: caller='%s'/%d: pvInterface=%p cbInterface=%zu pszSymPrefix=%p:{%s} pszSymList=%p:{%s}\n",
3419 pDevIns->pReg->szName, pDevIns->iInstance, pvInterface, cbInterface, pszSymPrefix, pszSymPrefix, pszSymList, pszSymList));
3420
3421 int rc;
3422 if ( strncmp(pszSymPrefix, "dev", 3) == 0
3423 && RTStrIStr(pszSymPrefix + 3, pDevIns->pReg->szName) != NULL)
3424 {
3425 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
3426 rc = PDMR3LdrGetInterfaceSymbols(pDevIns->Internal.s.pVMR3,
3427 pvInterface, cbInterface,
3428 pDevIns->pReg->pszRCMod, pDevIns->Internal.s.pDevR3->pszRCSearchPath,
3429 pszSymPrefix, pszSymList,
3430 false /*fRing0OrRC*/);
3431 else
3432 {
3433 AssertMsgFailed(("Not a raw-mode enabled driver\n"));
3434 rc = VERR_PERMISSION_DENIED;
3435 }
3436 }
3437 else
3438 {
3439 AssertMsgFailed(("Invalid prefix '%s' for '%s'; must start with 'dev' and contain the driver name!\n",
3440 pszSymPrefix, pDevIns->pReg->szName));
3441 rc = VERR_INVALID_NAME;
3442 }
3443
3444 LogFlow(("pdmR3DevHlp_PDMLdrGetRCInterfaceSymbols: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
3445 pDevIns->iInstance, rc));
3446 return rc;
3447}
3448
3449
3450/** @interface_method_impl{PDMDEVHLPR3,pfnLdrGetR0InterfaceSymbols} */
3451static DECLCALLBACK(int) pdmR3DevHlp_LdrGetR0InterfaceSymbols(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
3452 const char *pszSymPrefix, const char *pszSymList)
3453{
3454 PDMDEV_ASSERT_DEVINS(pDevIns);
3455 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3456 LogFlow(("pdmR3DevHlp_PDMLdrGetR0InterfaceSymbols: caller='%s'/%d: pvInterface=%p cbInterface=%zu pszSymPrefix=%p:{%s} pszSymList=%p:{%s}\n",
3457 pDevIns->pReg->szName, pDevIns->iInstance, pvInterface, cbInterface, pszSymPrefix, pszSymPrefix, pszSymList, pszSymList));
3458
3459 int rc;
3460 if ( strncmp(pszSymPrefix, "dev", 3) == 0
3461 && RTStrIStr(pszSymPrefix + 3, pDevIns->pReg->szName) != NULL)
3462 {
3463 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
3464 rc = PDMR3LdrGetInterfaceSymbols(pDevIns->Internal.s.pVMR3,
3465 pvInterface, cbInterface,
3466 pDevIns->pReg->pszR0Mod, pDevIns->Internal.s.pDevR3->pszR0SearchPath,
3467 pszSymPrefix, pszSymList,
3468 true /*fRing0OrRC*/);
3469 else
3470 {
3471 AssertMsgFailed(("Not a ring-0 enabled driver\n"));
3472 rc = VERR_PERMISSION_DENIED;
3473 }
3474 }
3475 else
3476 {
3477 AssertMsgFailed(("Invalid prefix '%s' for '%s'; must start with 'dev' and contain the driver name!\n",
3478 pszSymPrefix, pDevIns->pReg->szName));
3479 rc = VERR_INVALID_NAME;
3480 }
3481
3482 LogFlow(("pdmR3DevHlp_PDMLdrGetR0InterfaceSymbols: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
3483 pDevIns->iInstance, rc));
3484 return rc;
3485}
3486
3487
3488/** @interface_method_impl{PDMDEVHLPR3,pfnCallR0} */
3489static DECLCALLBACK(int) pdmR3DevHlp_CallR0(PPDMDEVINS pDevIns, uint32_t uOperation, uint64_t u64Arg)
3490{
3491 PDMDEV_ASSERT_DEVINS(pDevIns);
3492 PVM pVM = pDevIns->Internal.s.pVMR3;
3493 PVMCPU pVCpu = VMMGetCpu(pVM);
3494 AssertReturn(pVCpu, VERR_VM_THREAD_IS_EMT);
3495 LogFlow(("pdmR3DevHlp_CallR0: caller='%s'/%d: uOperation=%#x u64Arg=%#RX64\n",
3496 pDevIns->pReg->szName, pDevIns->iInstance, uOperation, u64Arg));
3497
3498 /*
3499 * Resolve the ring-0 entry point. There is not need to remember this like
3500 * we do for drivers since this is mainly for construction time hacks and
3501 * other things that aren't performance critical.
3502 */
3503 int rc;
3504 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
3505 {
3506 /*
3507 * Make the ring-0 call.
3508 */
3509 PDMDEVICEGENCALLREQ Req;
3510 RT_ZERO(Req.Params);
3511 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
3512 Req.Hdr.cbReq = sizeof(Req);
3513 Req.pDevInsR3 = pDevIns;
3514 Req.idxR0Device = pDevIns->Internal.s.idxR0Device;
3515 Req.enmCall = PDMDEVICEGENCALL_REQUEST;
3516 Req.Params.Req.uReq = uOperation;
3517 Req.Params.Req.uArg = u64Arg;
3518 rc = VMMR3CallR0Emt(pVM, pVCpu, VMMR0_DO_PDM_DEVICE_GEN_CALL, 0, &Req.Hdr);
3519 }
3520 else
3521 rc = VERR_ACCESS_DENIED;
3522 LogFlow(("pdmR3DevHlp_CallR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
3523 pDevIns->iInstance, rc));
3524 return rc;
3525}
3526
3527
3528/** @interface_method_impl{PDMDEVHLPR3,pfnVMGetSuspendReason} */
3529static DECLCALLBACK(VMSUSPENDREASON) pdmR3DevHlp_VMGetSuspendReason(PPDMDEVINS pDevIns)
3530{
3531 PDMDEV_ASSERT_DEVINS(pDevIns);
3532 PVM pVM = pDevIns->Internal.s.pVMR3;
3533 VM_ASSERT_EMT(pVM);
3534 VMSUSPENDREASON enmReason = VMR3GetSuspendReason(pVM->pUVM);
3535 LogFlow(("pdmR3DevHlp_VMGetSuspendReason: caller='%s'/%d: returns %d\n",
3536 pDevIns->pReg->szName, pDevIns->iInstance, enmReason));
3537 return enmReason;
3538}
3539
3540
3541/** @interface_method_impl{PDMDEVHLPR3,pfnVMGetResumeReason} */
3542static DECLCALLBACK(VMRESUMEREASON) pdmR3DevHlp_VMGetResumeReason(PPDMDEVINS pDevIns)
3543{
3544 PDMDEV_ASSERT_DEVINS(pDevIns);
3545 PVM pVM = pDevIns->Internal.s.pVMR3;
3546 VM_ASSERT_EMT(pVM);
3547 VMRESUMEREASON enmReason = VMR3GetResumeReason(pVM->pUVM);
3548 LogFlow(("pdmR3DevHlp_VMGetResumeReason: caller='%s'/%d: returns %d\n",
3549 pDevIns->pReg->szName, pDevIns->iInstance, enmReason));
3550 return enmReason;
3551}
3552
3553
3554/** @interface_method_impl{PDMDEVHLPR3,pfnGetUVM} */
3555static DECLCALLBACK(PUVM) pdmR3DevHlp_GetUVM(PPDMDEVINS pDevIns)
3556{
3557 PDMDEV_ASSERT_DEVINS(pDevIns);
3558 LogFlow(("pdmR3DevHlp_GetUVM: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns->Internal.s.pVMR3));
3559 return pDevIns->Internal.s.pVMR3->pUVM;
3560}
3561
3562
3563/** @interface_method_impl{PDMDEVHLPR3,pfnGetVM} */
3564static DECLCALLBACK(PVM) pdmR3DevHlp_GetVM(PPDMDEVINS pDevIns)
3565{
3566 PDMDEV_ASSERT_DEVINS(pDevIns);
3567 LogFlow(("pdmR3DevHlp_GetVM: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns->Internal.s.pVMR3));
3568 return pDevIns->Internal.s.pVMR3;
3569}
3570
3571
3572/** @interface_method_impl{PDMDEVHLPR3,pfnGetVMCPU} */
3573static DECLCALLBACK(PVMCPU) pdmR3DevHlp_GetVMCPU(PPDMDEVINS pDevIns)
3574{
3575 PDMDEV_ASSERT_DEVINS(pDevIns);
3576 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3577 LogFlow(("pdmR3DevHlp_GetVMCPU: caller='%s'/%d for CPU %u\n", pDevIns->pReg->szName, pDevIns->iInstance, VMMGetCpuId(pDevIns->Internal.s.pVMR3)));
3578 return VMMGetCpu(pDevIns->Internal.s.pVMR3);
3579}
3580
3581
3582/** @interface_method_impl{PDMDEVHLPR3,pfnGetCurrentCpuId} */
3583static DECLCALLBACK(VMCPUID) pdmR3DevHlp_GetCurrentCpuId(PPDMDEVINS pDevIns)
3584{
3585 PDMDEV_ASSERT_DEVINS(pDevIns);
3586 VMCPUID idCpu = VMMGetCpuId(pDevIns->Internal.s.pVMR3);
3587 LogFlow(("pdmR3DevHlp_GetCurrentCpuId: caller='%s'/%d for CPU %u\n", pDevIns->pReg->szName, pDevIns->iInstance, idCpu));
3588 return idCpu;
3589}
3590
3591
3592/** @interface_method_impl{PDMDEVHLPR3,pfnPCIBusRegister} */
3593static DECLCALLBACK(int) pdmR3DevHlp_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREGR3 pPciBusReg,
3594 PCPDMPCIHLPR3 *ppPciHlp, uint32_t *piBus)
3595{
3596 PDMDEV_ASSERT_DEVINS(pDevIns);
3597 PVM pVM = pDevIns->Internal.s.pVMR3;
3598 VM_ASSERT_EMT(pVM);
3599 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: pPciBusReg=%p:{.u32Version=%#x, .pfnRegisterR3=%p, .pfnIORegionRegisterR3=%p, "
3600 ".pfnInterceptConfigAccesses=%p, pfnConfigRead=%p, pfnConfigWrite=%p, .pfnSetIrqR3=%p, .u32EndVersion=%#x} ppPciHlpR3=%p piBus=%p\n",
3601 pDevIns->pReg->szName, pDevIns->iInstance, pPciBusReg, pPciBusReg->u32Version, pPciBusReg->pfnRegisterR3,
3602 pPciBusReg->pfnIORegionRegisterR3, pPciBusReg->pfnInterceptConfigAccesses, pPciBusReg->pfnConfigRead,
3603 pPciBusReg->pfnConfigWrite, pPciBusReg->pfnSetIrqR3, pPciBusReg->u32EndVersion, ppPciHlp, piBus));
3604
3605 /*
3606 * Validate the structure and output parameters.
3607 */
3608 AssertLogRelMsgReturn(pPciBusReg->u32Version == PDM_PCIBUSREGR3_VERSION,
3609 ("u32Version=%#x expected %#x\n", pPciBusReg->u32Version, PDM_PCIBUSREGR3_VERSION),
3610 VERR_INVALID_PARAMETER);
3611 AssertPtrReturn(pPciBusReg->pfnRegisterR3, VERR_INVALID_PARAMETER);
3612 AssertPtrNullReturn(pPciBusReg->pfnRegisterMsiR3, VERR_INVALID_POINTER);
3613 AssertPtrReturn(pPciBusReg->pfnIORegionRegisterR3, VERR_INVALID_POINTER);
3614 AssertPtrReturn(pPciBusReg->pfnInterceptConfigAccesses, VERR_INVALID_POINTER);
3615 AssertPtrReturn(pPciBusReg->pfnConfigWrite, VERR_INVALID_POINTER);
3616 AssertPtrReturn(pPciBusReg->pfnConfigRead, VERR_INVALID_POINTER);
3617 AssertPtrReturn(pPciBusReg->pfnSetIrqR3, VERR_INVALID_POINTER);
3618 AssertLogRelMsgReturn(pPciBusReg->u32EndVersion == PDM_PCIBUSREGR3_VERSION,
3619 ("u32Version=%#x expected %#x\n", pPciBusReg->u32Version, PDM_PCIBUSREGR3_VERSION),
3620 VERR_INVALID_PARAMETER);
3621 AssertPtrReturn(ppPciHlp, VERR_INVALID_POINTER);
3622 AssertPtrNullReturn(piBus, VERR_INVALID_POINTER);
3623 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_WRONG_ORDER);
3624
3625 /*
3626 * Find free PCI bus entry.
3627 */
3628 unsigned iBus = 0;
3629 for (iBus = 0; iBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses); iBus++)
3630 if (!pVM->pdm.s.aPciBuses[iBus].pDevInsR3)
3631 break;
3632 AssertLogRelMsgReturn(iBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses),
3633 ("Too many PCI buses. Max=%u\n", RT_ELEMENTS(pVM->pdm.s.aPciBuses)),
3634 VERR_OUT_OF_RESOURCES);
3635 PPDMPCIBUS pPciBus = &pVM->pdm.s.aPciBuses[iBus];
3636
3637 /*
3638 * Init the R3 bits.
3639 */
3640 pPciBus->iBus = iBus;
3641 pPciBus->pDevInsR3 = pDevIns;
3642 pPciBus->pfnRegister = pPciBusReg->pfnRegisterR3;
3643 pPciBus->pfnRegisterMsi = pPciBusReg->pfnRegisterMsiR3;
3644 pPciBus->pfnIORegionRegister = pPciBusReg->pfnIORegionRegisterR3;
3645 pPciBus->pfnInterceptConfigAccesses = pPciBusReg->pfnInterceptConfigAccesses;
3646 pPciBus->pfnConfigRead = pPciBusReg->pfnConfigRead;
3647 pPciBus->pfnConfigWrite = pPciBusReg->pfnConfigWrite;
3648 pPciBus->pfnSetIrqR3 = pPciBusReg->pfnSetIrqR3;
3649
3650 Log(("PDM: Registered PCI bus device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
3651
3652 /* set the helper pointer and return. */
3653 *ppPciHlp = &g_pdmR3DevPciHlp;
3654 if (piBus)
3655 *piBus = iBus;
3656 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc *piBus=%u\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS, iBus));
3657 return VINF_SUCCESS;
3658}
3659
3660
3661/** @interface_method_impl{PDMDEVHLPR3,pfnPICRegister} */
3662static DECLCALLBACK(int) pdmR3DevHlp_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLP *ppPicHlp)
3663{
3664 PDMDEV_ASSERT_DEVINS(pDevIns);
3665 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3666 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: pPicReg=%p:{.u32Version=%#x, .pfnSetIrq=%p, .pfnGetInterrupt=%p, .u32TheEnd=%#x } ppPicHlp=%p\n",
3667 pDevIns->pReg->szName, pDevIns->iInstance, pPicReg, pPicReg->u32Version, pPicReg->pfnSetIrq, pPicReg->pfnGetInterrupt, pPicReg->u32TheEnd, ppPicHlp));
3668 PVM pVM = pDevIns->Internal.s.pVMR3;
3669
3670 /*
3671 * Validate input.
3672 */
3673 AssertMsgReturn(pPicReg->u32Version == PDM_PICREG_VERSION,
3674 ("%s/%d: u32Version=%#x expected %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, pPicReg->u32Version, PDM_PICREG_VERSION),
3675 VERR_INVALID_PARAMETER);
3676 AssertPtrReturn(pPicReg->pfnSetIrq, VERR_INVALID_POINTER);
3677 AssertPtrReturn(pPicReg->pfnGetInterrupt, VERR_INVALID_POINTER);
3678 AssertMsgReturn(pPicReg->u32TheEnd == PDM_PICREG_VERSION,
3679 ("%s/%d: u32TheEnd=%#x expected %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, pPicReg->u32TheEnd, PDM_PICREG_VERSION),
3680 VERR_INVALID_PARAMETER);
3681 AssertPtrReturn(ppPicHlp, VERR_INVALID_POINTER);
3682
3683 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_WRONG_ORDER);
3684 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
3685
3686 /*
3687 * Only one PIC device.
3688 */
3689 AssertMsgReturn(pVM->pdm.s.Pic.pDevInsR3 == NULL, ("%s/%d: Only one PIC!\n", pDevIns->pReg->szName, pDevIns->iInstance),
3690 VERR_ALREADY_EXISTS);
3691
3692 /*
3693 * Take down the callbacks and instance.
3694 */
3695 pVM->pdm.s.Pic.pDevInsR3 = pDevIns;
3696 pVM->pdm.s.Pic.pfnSetIrqR3 = pPicReg->pfnSetIrq;
3697 pVM->pdm.s.Pic.pfnGetInterruptR3 = pPicReg->pfnGetInterrupt;
3698 Log(("PDM: Registered PIC device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
3699
3700 /* set the helper pointer and return. */
3701 *ppPicHlp = &g_pdmR3DevPicHlp;
3702 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3703 return VINF_SUCCESS;
3704}
3705
3706
3707/** @interface_method_impl{PDMDEVHLPR3,pfnAPICRegister} */
3708static DECLCALLBACK(int) pdmR3DevHlp_APICRegister(PPDMDEVINS pDevIns)
3709{
3710 PDMDEV_ASSERT_DEVINS(pDevIns);
3711 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3712
3713 /*
3714 * Only one APIC device. On SMP we have single logical device covering all LAPICs,
3715 * as they need to communicate and share state easily.
3716 */
3717 PVM pVM = pDevIns->Internal.s.pVMR3;
3718 if (pVM->pdm.s.Apic.pDevInsR3)
3719 {
3720 AssertMsgFailed(("Only one APIC device is supported!\n"));
3721 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3722 return VERR_INVALID_PARAMETER;
3723 }
3724
3725 /*
3726 * Initialize the RC, R0 and HC bits.
3727 */
3728 pVM->pdm.s.Apic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
3729 Assert(pVM->pdm.s.Apic.pDevInsRC || !VM_IS_RAW_MODE_ENABLED(pVM));
3730
3731 pVM->pdm.s.Apic.pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
3732 Assert(pVM->pdm.s.Apic.pDevInsR0);
3733
3734 pVM->pdm.s.Apic.pDevInsR3 = pDevIns;
3735 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3736 return VINF_SUCCESS;
3737}
3738
3739
3740/** @interface_method_impl{PDMDEVHLPR3,pfnIoApicRegister} */
3741static DECLCALLBACK(int) pdmR3DevHlp_IoApicRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLP *ppIoApicHlp)
3742{
3743 PDMDEV_ASSERT_DEVINS(pDevIns);
3744 LogFlow(("pdmR3DevHlp_IoApicRegister: caller='%s'/%d: pIoApicReg=%p:{.u32Version=%#x, .pfnSetIrq=%p, .pfnSendMsi=%p, .pfnSetEoi=%p, .u32TheEnd=%#x } ppIoApicHlp=%p\n",
3745 pDevIns->pReg->szName, pDevIns->iInstance, pIoApicReg, pIoApicReg->u32Version, pIoApicReg->pfnSetIrq, pIoApicReg->pfnSendMsi, pIoApicReg->pfnSetEoi, pIoApicReg->u32TheEnd, ppIoApicHlp));
3746 PVM pVM = pDevIns->Internal.s.pVMR3;
3747
3748 /*
3749 * Validate input.
3750 */
3751 AssertMsgReturn(pIoApicReg->u32Version == PDM_IOAPICREG_VERSION,
3752 ("%s/%d: u32Version=%#x expected %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, pIoApicReg->u32Version, PDM_IOAPICREG_VERSION),
3753 VERR_VERSION_MISMATCH);
3754 AssertPtrReturn(pIoApicReg->pfnSetIrq, VERR_INVALID_POINTER);
3755 AssertPtrReturn(pIoApicReg->pfnSendMsi, VERR_INVALID_POINTER);
3756 AssertPtrReturn(pIoApicReg->pfnSetEoi, VERR_INVALID_POINTER);
3757 AssertMsgReturn(pIoApicReg->u32TheEnd == PDM_IOAPICREG_VERSION,
3758 ("%s/%d: u32TheEnd=%#x expected %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, pIoApicReg->u32TheEnd, PDM_IOAPICREG_VERSION),
3759 VERR_VERSION_MISMATCH);
3760 AssertPtrReturn(ppIoApicHlp, VERR_INVALID_POINTER);
3761 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_WRONG_ORDER);
3762 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
3763
3764 /*
3765 * The I/O APIC requires the APIC to be present (hacks++).
3766 * If the I/O APIC does GC stuff so must the APIC.
3767 */
3768 AssertMsgReturn(pVM->pdm.s.Apic.pDevInsR3 != NULL, ("Configuration error / Init order error! No APIC!\n"), VERR_WRONG_ORDER);
3769
3770 /*
3771 * Only one I/O APIC device.
3772 */
3773 AssertMsgReturn(pVM->pdm.s.IoApic.pDevInsR3 == NULL,
3774 ("Only one IOAPIC device is supported! (caller %s/%d)\n", pDevIns->pReg->szName, pDevIns->iInstance),
3775 VERR_ALREADY_EXISTS);
3776
3777 /*
3778 * Initialize the R3 bits.
3779 */
3780 pVM->pdm.s.IoApic.pDevInsR3 = pDevIns;
3781 pVM->pdm.s.IoApic.pfnSetIrqR3 = pIoApicReg->pfnSetIrq;
3782 pVM->pdm.s.IoApic.pfnSendMsiR3 = pIoApicReg->pfnSendMsi;
3783 pVM->pdm.s.IoApic.pfnSetEoiR3 = pIoApicReg->pfnSetEoi;
3784 Log(("PDM: Registered I/O APIC device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
3785
3786 /* set the helper pointer and return. */
3787 *ppIoApicHlp = &g_pdmR3DevIoApicHlp;
3788 LogFlow(("pdmR3DevHlp_IoApicRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3789 return VINF_SUCCESS;
3790}
3791
3792
3793/** @interface_method_impl{PDMDEVHLPR3,pfnHpetRegister} */
3794static DECLCALLBACK(int) pdmR3DevHlp_HpetRegister(PPDMDEVINS pDevIns, PPDMHPETREG pHpetReg, PCPDMHPETHLPR3 *ppHpetHlpR3)
3795{
3796 PDMDEV_ASSERT_DEVINS(pDevIns);
3797 LogFlow(("pdmR3DevHlp_HpetRegister: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
3798 PVM pVM = pDevIns->Internal.s.pVMR3;
3799
3800 /*
3801 * Validate input.
3802 */
3803 AssertMsgReturn(pHpetReg->u32Version == PDM_HPETREG_VERSION,
3804 ("%s/%u: u32Version=%#x expected %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, pHpetReg->u32Version, PDM_HPETREG_VERSION),
3805 VERR_VERSION_MISMATCH);
3806 AssertPtrReturn(ppHpetHlpR3, VERR_INVALID_POINTER);
3807 VM_ASSERT_STATE_RETURN(pVM, VMSTATE_CREATING, VERR_WRONG_ORDER);
3808 VM_ASSERT_EMT0_RETURN(pVM, VERR_VM_THREAD_NOT_EMT);
3809
3810 /*
3811 * Only one HPET device.
3812 */
3813 AssertMsgReturn(pVM->pdm.s.pHpet == NULL,
3814 ("Only one HPET device is supported! (caller %s/%d)\n", pDevIns->pReg->szName, pDevIns->iInstance),
3815 VERR_ALREADY_EXISTS);
3816
3817 /*
3818 * Do the job (what there is of it).
3819 */
3820 pVM->pdm.s.pHpet = pDevIns;
3821 *ppHpetHlpR3 = &g_pdmR3DevHpetHlp;
3822
3823 LogFlow(("pdmR3DevHlp_HpetRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3824 return VINF_SUCCESS;
3825}
3826
3827
3828/** @interface_method_impl{PDMDEVHLPR3,pfnPciRawRegister} */
3829static DECLCALLBACK(int) pdmR3DevHlp_PciRawRegister(PPDMDEVINS pDevIns, PPDMPCIRAWREG pPciRawReg, PCPDMPCIRAWHLPR3 *ppPciRawHlpR3)
3830{
3831 PDMDEV_ASSERT_DEVINS(pDevIns); RT_NOREF_PV(pDevIns);
3832 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3833 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
3834
3835 /*
3836 * Validate input.
3837 */
3838 if (pPciRawReg->u32Version != PDM_PCIRAWREG_VERSION)
3839 {
3840 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPciRawReg->u32Version, PDM_PCIRAWREG_VERSION));
3841 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3842 return VERR_INVALID_PARAMETER;
3843 }
3844
3845 if (!ppPciRawHlpR3)
3846 {
3847 Assert(ppPciRawHlpR3);
3848 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc (ppPciRawHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3849 return VERR_INVALID_PARAMETER;
3850 }
3851
3852 /* set the helper pointer and return. */
3853 *ppPciRawHlpR3 = &g_pdmR3DevPciRawHlp;
3854 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3855 return VINF_SUCCESS;
3856}
3857
3858
3859/** @interface_method_impl{PDMDEVHLPR3,pfnDMACRegister} */
3860static DECLCALLBACK(int) pdmR3DevHlp_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp)
3861{
3862 PDMDEV_ASSERT_DEVINS(pDevIns);
3863 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3864 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",
3865 pDevIns->pReg->szName, pDevIns->iInstance, pDmacReg, pDmacReg->u32Version, pDmacReg->pfnRun, pDmacReg->pfnRegister,
3866 pDmacReg->pfnReadMemory, pDmacReg->pfnWriteMemory, pDmacReg->pfnSetDREQ, pDmacReg->pfnGetChannelMode, ppDmacHlp));
3867
3868 /*
3869 * Validate input.
3870 */
3871 if (pDmacReg->u32Version != PDM_DMACREG_VERSION)
3872 {
3873 AssertMsgFailed(("u32Version=%#x expected %#x\n", pDmacReg->u32Version,
3874 PDM_DMACREG_VERSION));
3875 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (version)\n",
3876 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3877 return VERR_INVALID_PARAMETER;
3878 }
3879 if ( !pDmacReg->pfnRun
3880 || !pDmacReg->pfnRegister
3881 || !pDmacReg->pfnReadMemory
3882 || !pDmacReg->pfnWriteMemory
3883 || !pDmacReg->pfnSetDREQ
3884 || !pDmacReg->pfnGetChannelMode)
3885 {
3886 Assert(pDmacReg->pfnRun);
3887 Assert(pDmacReg->pfnRegister);
3888 Assert(pDmacReg->pfnReadMemory);
3889 Assert(pDmacReg->pfnWriteMemory);
3890 Assert(pDmacReg->pfnSetDREQ);
3891 Assert(pDmacReg->pfnGetChannelMode);
3892 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (callbacks)\n",
3893 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3894 return VERR_INVALID_PARAMETER;
3895 }
3896
3897 if (!ppDmacHlp)
3898 {
3899 Assert(ppDmacHlp);
3900 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (ppDmacHlp)\n",
3901 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3902 return VERR_INVALID_PARAMETER;
3903 }
3904
3905 /*
3906 * Only one DMA device.
3907 */
3908 PVM pVM = pDevIns->Internal.s.pVMR3;
3909 if (pVM->pdm.s.pDmac)
3910 {
3911 AssertMsgFailed(("Only one DMA device is supported!\n"));
3912 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc\n",
3913 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3914 return VERR_INVALID_PARAMETER;
3915 }
3916
3917 /*
3918 * Allocate and initialize pci bus structure.
3919 */
3920 int rc = VINF_SUCCESS;
3921 PPDMDMAC pDmac = (PPDMDMAC)MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE, sizeof(*pDmac));
3922 if (pDmac)
3923 {
3924 pDmac->pDevIns = pDevIns;
3925 pDmac->Reg = *pDmacReg;
3926 pVM->pdm.s.pDmac = pDmac;
3927
3928 /* set the helper pointer. */
3929 *ppDmacHlp = &g_pdmR3DevDmacHlp;
3930 Log(("PDM: Registered DMAC device '%s'/%d pDevIns=%p\n",
3931 pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
3932 }
3933 else
3934 rc = VERR_NO_MEMORY;
3935
3936 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc\n",
3937 pDevIns->pReg->szName, pDevIns->iInstance, rc));
3938 return rc;
3939}
3940
3941
3942/**
3943 * @copydoc PDMDEVHLPR3::pfnRegisterVMMDevHeap
3944 */
3945static DECLCALLBACK(int) pdmR3DevHlp_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbHeap)
3946{
3947 PDMDEV_ASSERT_DEVINS(pDevIns);
3948 PVM pVM = pDevIns->Internal.s.pVMR3;
3949 VM_ASSERT_EMT(pVM);
3950 LogFlow(("pdmR3DevHlp_RegisterVMMDevHeap: caller='%s'/%d: GCPhys=%RGp pvHeap=%p cbHeap=%#x\n",
3951 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, pvHeap, cbHeap));
3952
3953 if (pVM->pdm.s.pvVMMDevHeap == NULL)
3954 {
3955 pVM->pdm.s.pvVMMDevHeap = pvHeap;
3956 pVM->pdm.s.GCPhysVMMDevHeap = GCPhys;
3957 pVM->pdm.s.cbVMMDevHeap = cbHeap;
3958 pVM->pdm.s.cbVMMDevHeapLeft = cbHeap;
3959 }
3960 else
3961 {
3962 Assert(pVM->pdm.s.pvVMMDevHeap == pvHeap);
3963 Assert(pVM->pdm.s.cbVMMDevHeap == cbHeap);
3964 Assert(pVM->pdm.s.GCPhysVMMDevHeap != GCPhys || GCPhys == NIL_RTGCPHYS);
3965 if (pVM->pdm.s.GCPhysVMMDevHeap != GCPhys)
3966 {
3967 pVM->pdm.s.GCPhysVMMDevHeap = GCPhys;
3968 if (pVM->pdm.s.pfnVMMDevHeapNotify)
3969 pVM->pdm.s.pfnVMMDevHeapNotify(pVM, pvHeap, GCPhys);
3970 }
3971 }
3972
3973 LogFlow(("pdmR3DevHlp_RegisterVMMDevHeap: caller='%s'/%d: returns %Rrc\n",
3974 pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
3975 return VINF_SUCCESS;
3976}
3977
3978
3979/**
3980 * @interface_method_impl{PDMDEVHLPR3,pfnFirmwareRegister}
3981 */
3982static DECLCALLBACK(int) pdmR3DevHlp_FirmwareRegister(PPDMDEVINS pDevIns, PCPDMFWREG pFwReg, PCPDMFWHLPR3 *ppFwHlp)
3983{
3984 PDMDEV_ASSERT_DEVINS(pDevIns);
3985 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3986 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: pFWReg=%p:{.u32Version=%#x, .pfnIsHardReset=%p, .u32TheEnd=%#x} ppFwHlp=%p\n",
3987 pDevIns->pReg->szName, pDevIns->iInstance, pFwReg, pFwReg->u32Version, pFwReg->pfnIsHardReset, pFwReg->u32TheEnd, ppFwHlp));
3988
3989 /*
3990 * Validate input.
3991 */
3992 if (pFwReg->u32Version != PDM_FWREG_VERSION)
3993 {
3994 AssertMsgFailed(("u32Version=%#x expected %#x\n", pFwReg->u32Version, PDM_FWREG_VERSION));
3995 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: returns %Rrc (version)\n",
3996 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
3997 return VERR_INVALID_PARAMETER;
3998 }
3999 if (!pFwReg->pfnIsHardReset)
4000 {
4001 Assert(pFwReg->pfnIsHardReset);
4002 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: returns %Rrc (callbacks)\n",
4003 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
4004 return VERR_INVALID_PARAMETER;
4005 }
4006
4007 if (!ppFwHlp)
4008 {
4009 Assert(ppFwHlp);
4010 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: returns %Rrc (ppFwHlp)\n",
4011 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
4012 return VERR_INVALID_PARAMETER;
4013 }
4014
4015 /*
4016 * Only one DMA device.
4017 */
4018 PVM pVM = pDevIns->Internal.s.pVMR3;
4019 if (pVM->pdm.s.pFirmware)
4020 {
4021 AssertMsgFailed(("Only one firmware device is supported!\n"));
4022 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: returns %Rrc\n",
4023 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
4024 return VERR_INVALID_PARAMETER;
4025 }
4026
4027 /*
4028 * Allocate and initialize pci bus structure.
4029 */
4030 int rc = VINF_SUCCESS;
4031 PPDMFW pFirmware = (PPDMFW)MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE, sizeof(*pFirmware));
4032 if (pFirmware)
4033 {
4034 pFirmware->pDevIns = pDevIns;
4035 pFirmware->Reg = *pFwReg;
4036 pVM->pdm.s.pFirmware = pFirmware;
4037
4038 /* set the helper pointer. */
4039 *ppFwHlp = &g_pdmR3DevFirmwareHlp;
4040 Log(("PDM: Registered firmware device '%s'/%d pDevIns=%p\n",
4041 pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
4042 }
4043 else
4044 rc = VERR_NO_MEMORY;
4045
4046 LogFlow(("pdmR3DevHlp_FirmwareRegister: caller='%s'/%d: returns %Rrc\n",
4047 pDevIns->pReg->szName, pDevIns->iInstance, rc));
4048 return rc;
4049}
4050
4051
4052/** @interface_method_impl{PDMDEVHLPR3,pfnVMReset} */
4053static DECLCALLBACK(int) pdmR3DevHlp_VMReset(PPDMDEVINS pDevIns, uint32_t fFlags)
4054{
4055 PDMDEV_ASSERT_DEVINS(pDevIns);
4056 PVM pVM = pDevIns->Internal.s.pVMR3;
4057 VM_ASSERT_EMT(pVM);
4058 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: fFlags=%#x VM_FF_RESET %d -> 1\n",
4059 pDevIns->pReg->szName, pDevIns->iInstance, fFlags, VM_FF_IS_SET(pVM, VM_FF_RESET)));
4060
4061 /*
4062 * We postpone this operation because we're likely to be inside a I/O instruction
4063 * and the EIP will be updated when we return.
4064 * We still return VINF_EM_RESET to break out of any execution loops and force FF evaluation.
4065 */
4066 bool fHaltOnReset;
4067 int rc = CFGMR3QueryBool(CFGMR3GetChild(CFGMR3GetRoot(pVM), "PDM"), "HaltOnReset", &fHaltOnReset);
4068 if (RT_SUCCESS(rc) && fHaltOnReset)
4069 {
4070 Log(("pdmR3DevHlp_VMReset: Halt On Reset!\n"));
4071 rc = VINF_EM_HALT;
4072 }
4073 else
4074 {
4075 pVM->pdm.s.fResetFlags = fFlags;
4076 VM_FF_SET(pVM, VM_FF_RESET);
4077 rc = VINF_EM_RESET;
4078 }
4079
4080 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
4081 return rc;
4082}
4083
4084
4085/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspend} */
4086static DECLCALLBACK(int) pdmR3DevHlp_VMSuspend(PPDMDEVINS pDevIns)
4087{
4088 int rc;
4089 PDMDEV_ASSERT_DEVINS(pDevIns);
4090 PVM pVM = pDevIns->Internal.s.pVMR3;
4091 VM_ASSERT_EMT(pVM);
4092 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d:\n",
4093 pDevIns->pReg->szName, pDevIns->iInstance));
4094
4095 /** @todo Always take the SMP path - fewer code paths. */
4096 if (pVM->cCpus > 1)
4097 {
4098 /* We own the IOM lock here and could cause a deadlock by waiting for a VCPU that is blocking on the IOM lock. */
4099 rc = VMR3ReqCallNoWait(pVM, VMCPUID_ANY_QUEUE, (PFNRT)VMR3Suspend, 2, pVM->pUVM, VMSUSPENDREASON_VM);
4100 AssertRC(rc);
4101 rc = VINF_EM_SUSPEND;
4102 }
4103 else
4104 rc = VMR3Suspend(pVM->pUVM, VMSUSPENDREASON_VM);
4105
4106 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
4107 return rc;
4108}
4109
4110
4111/**
4112 * Worker for pdmR3DevHlp_VMSuspendSaveAndPowerOff that is invoked via a queued
4113 * EMT request to avoid deadlocks.
4114 *
4115 * @returns VBox status code fit for scheduling.
4116 * @param pVM The cross context VM structure.
4117 * @param pDevIns The device that triggered this action.
4118 */
4119static DECLCALLBACK(int) pdmR3DevHlp_VMSuspendSaveAndPowerOffWorker(PVM pVM, PPDMDEVINS pDevIns)
4120{
4121 /*
4122 * Suspend the VM first then do the saving.
4123 */
4124 int rc = VMR3Suspend(pVM->pUVM, VMSUSPENDREASON_VM);
4125 if (RT_SUCCESS(rc))
4126 {
4127 PUVM pUVM = pVM->pUVM;
4128 rc = pUVM->pVmm2UserMethods->pfnSaveState(pVM->pUVM->pVmm2UserMethods, pUVM);
4129
4130 /*
4131 * On success, power off the VM, on failure we'll leave it suspended.
4132 */
4133 if (RT_SUCCESS(rc))
4134 {
4135 rc = VMR3PowerOff(pVM->pUVM);
4136 if (RT_FAILURE(rc))
4137 LogRel(("%s/SSP: VMR3PowerOff failed: %Rrc\n", pDevIns->pReg->szName, rc));
4138 }
4139 else
4140 LogRel(("%s/SSP: pfnSaveState failed: %Rrc\n", pDevIns->pReg->szName, rc));
4141 }
4142 else
4143 LogRel(("%s/SSP: Suspend failed: %Rrc\n", pDevIns->pReg->szName, rc));
4144 return rc;
4145}
4146
4147
4148/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspendSaveAndPowerOff} */
4149static DECLCALLBACK(int) pdmR3DevHlp_VMSuspendSaveAndPowerOff(PPDMDEVINS pDevIns)
4150{
4151 PDMDEV_ASSERT_DEVINS(pDevIns);
4152 PVM pVM = pDevIns->Internal.s.pVMR3;
4153 VM_ASSERT_EMT(pVM);
4154 LogFlow(("pdmR3DevHlp_VMSuspendSaveAndPowerOff: caller='%s'/%d:\n",
4155 pDevIns->pReg->szName, pDevIns->iInstance));
4156
4157 int rc;
4158 if ( pVM->pUVM->pVmm2UserMethods
4159 && pVM->pUVM->pVmm2UserMethods->pfnSaveState)
4160 {
4161 rc = VMR3ReqCallNoWait(pVM, VMCPUID_ANY_QUEUE, (PFNRT)pdmR3DevHlp_VMSuspendSaveAndPowerOffWorker, 2, pVM, pDevIns);
4162 if (RT_SUCCESS(rc))
4163 {
4164 LogRel(("%s: Suspending, Saving and Powering Off the VM\n", pDevIns->pReg->szName));
4165 rc = VINF_EM_SUSPEND;
4166 }
4167 }
4168 else
4169 rc = VERR_NOT_SUPPORTED;
4170
4171 LogFlow(("pdmR3DevHlp_VMSuspendSaveAndPowerOff: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
4172 return rc;
4173}
4174
4175
4176/** @interface_method_impl{PDMDEVHLPR3,pfnVMPowerOff} */
4177static DECLCALLBACK(int) pdmR3DevHlp_VMPowerOff(PPDMDEVINS pDevIns)
4178{
4179 int rc;
4180 PDMDEV_ASSERT_DEVINS(pDevIns);
4181 PVM pVM = pDevIns->Internal.s.pVMR3;
4182 VM_ASSERT_EMT(pVM);
4183 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d:\n",
4184 pDevIns->pReg->szName, pDevIns->iInstance));
4185
4186 /** @todo Always take the SMP path - fewer code paths. */
4187 if (pVM->cCpus > 1)
4188 {
4189 /* We might be holding locks here and could cause a deadlock since
4190 VMR3PowerOff rendezvous with the other CPUs. */
4191 rc = VMR3ReqCallNoWait(pVM, VMCPUID_ANY_QUEUE, (PFNRT)VMR3PowerOff, 1, pVM->pUVM);
4192 AssertRC(rc);
4193 /* Set the VCPU state to stopped here as well to make sure no
4194 inconsistency with the EM state occurs. */
4195 VMCPU_SET_STATE(VMMGetCpu(pVM), VMCPUSTATE_STOPPED);
4196 rc = VINF_EM_OFF;
4197 }
4198 else
4199 rc = VMR3PowerOff(pVM->pUVM);
4200
4201 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
4202 return rc;
4203}
4204
4205
4206/** @interface_method_impl{PDMDEVHLPR3,pfnA20IsEnabled} */
4207static DECLCALLBACK(bool) pdmR3DevHlp_A20IsEnabled(PPDMDEVINS pDevIns)
4208{
4209 PDMDEV_ASSERT_DEVINS(pDevIns);
4210 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
4211
4212 bool fRc = PGMPhysIsA20Enabled(VMMGetCpu(pDevIns->Internal.s.pVMR3));
4213
4214 LogFlow(("pdmR3DevHlp_A20IsEnabled: caller='%s'/%d: returns %d\n", pDevIns->pReg->szName, pDevIns->iInstance, fRc));
4215 return fRc;
4216}
4217
4218
4219/** @interface_method_impl{PDMDEVHLPR3,pfnA20Set} */
4220static DECLCALLBACK(void) pdmR3DevHlp_A20Set(PPDMDEVINS pDevIns, bool fEnable)
4221{
4222 PDMDEV_ASSERT_DEVINS(pDevIns);
4223 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
4224 LogFlow(("pdmR3DevHlp_A20Set: caller='%s'/%d: fEnable=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, fEnable));
4225 PGMR3PhysSetA20(VMMGetCpu(pDevIns->Internal.s.pVMR3), fEnable);
4226}
4227
4228
4229/** @interface_method_impl{PDMDEVHLPR3,pfnGetCpuId} */
4230static DECLCALLBACK(void) pdmR3DevHlp_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
4231 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
4232{
4233 PDMDEV_ASSERT_DEVINS(pDevIns);
4234 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
4235
4236 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: iLeaf=%d pEax=%p pEbx=%p pEcx=%p pEdx=%p\n",
4237 pDevIns->pReg->szName, pDevIns->iInstance, iLeaf, pEax, pEbx, pEcx, pEdx));
4238 AssertPtr(pEax); AssertPtr(pEbx); AssertPtr(pEcx); AssertPtr(pEdx);
4239
4240 CPUMGetGuestCpuId(VMMGetCpu(pDevIns->Internal.s.pVMR3), iLeaf, 0 /*iSubLeaf*/, pEax, pEbx, pEcx, pEdx);
4241
4242 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: returns void - *pEax=%#x *pEbx=%#x *pEcx=%#x *pEdx=%#x\n",
4243 pDevIns->pReg->szName, pDevIns->iInstance, *pEax, *pEbx, *pEcx, *pEdx));
4244}
4245
4246
4247/**
4248 * The device helper structure for trusted devices.
4249 */
4250const PDMDEVHLPR3 g_pdmR3DevHlpTrusted =
4251{
4252 PDM_DEVHLPR3_VERSION,
4253 pdmR3DevHlp_IoPortCreateEx,
4254 pdmR3DevHlp_IoPortMap,
4255 pdmR3DevHlp_IoPortUnmap,
4256 pdmR3DevHlp_IoPortGetMappingAddress,
4257 pdmR3DevHlp_IOPortRegister,
4258 pdmR3DevHlp_IOPortRegisterRC,
4259 pdmR3DevHlp_IOPortRegisterR0,
4260 pdmR3DevHlp_IOPortDeregister,
4261 pdmR3DevHlp_MmioCreateEx,
4262 pdmR3DevHlp_MmioMap,
4263 pdmR3DevHlp_MmioUnmap,
4264 pdmR3DevHlp_MmioReduce,
4265 pdmR3DevHlp_MmioGetMappingAddress,
4266 pdmR3DevHlp_MMIORegister,
4267 pdmR3DevHlp_MMIORegisterRC,
4268 pdmR3DevHlp_MMIORegisterR0,
4269 pdmR3DevHlp_MMIODeregister,
4270 pdmR3DevHlp_Mmio2Create,
4271 pdmR3DevHlp_Mmio2Destroy,
4272 pdmR3DevHlp_Mmio2Map,
4273 pdmR3DevHlp_Mmio2Unmap,
4274 pdmR3DevHlp_Mmio2Reduce,
4275 pdmR3DevHlp_Mmio2GetMappingAddress,
4276 pdmR3DevHlp_Mmio2ChangeRegionNo,
4277 pdmR3DevHlp_MMIO2Register,
4278 pdmR3DevHlp_MMIOExDeregister,
4279 pdmR3DevHlp_MMIOExMap,
4280 pdmR3DevHlp_MMIOExUnmap,
4281 pdmR3DevHlp_MMIOExReduce,
4282 pdmR3DevHlp_MMHyperMapMMIO2,
4283 pdmR3DevHlp_MMIO2MapKernel,
4284 pdmR3DevHlp_ROMRegister,
4285 pdmR3DevHlp_ROMProtectShadow,
4286 pdmR3DevHlp_SSMRegister,
4287 SSMR3PutStruct,
4288 SSMR3PutStructEx,
4289 SSMR3PutBool,
4290 SSMR3PutU8,
4291 SSMR3PutS8,
4292 SSMR3PutU16,
4293 SSMR3PutS16,
4294 SSMR3PutU32,
4295 SSMR3PutS32,
4296 SSMR3PutU64,
4297 SSMR3PutS64,
4298 SSMR3PutU128,
4299 SSMR3PutS128,
4300 SSMR3PutUInt,
4301 SSMR3PutSInt,
4302 SSMR3PutGCUInt,
4303 SSMR3PutGCUIntReg,
4304 SSMR3PutGCPhys32,
4305 SSMR3PutGCPhys64,
4306 SSMR3PutGCPhys,
4307 SSMR3PutGCPtr,
4308 SSMR3PutGCUIntPtr,
4309 SSMR3PutRCPtr,
4310 SSMR3PutIOPort,
4311 SSMR3PutSel,
4312 SSMR3PutMem,
4313 SSMR3PutStrZ,
4314 SSMR3GetStruct,
4315 SSMR3GetStructEx,
4316 SSMR3GetBool,
4317 SSMR3GetBoolV,
4318 SSMR3GetU8,
4319 SSMR3GetU8V,
4320 SSMR3GetS8,
4321 SSMR3GetS8V,
4322 SSMR3GetU16,
4323 SSMR3GetU16V,
4324 SSMR3GetS16,
4325 SSMR3GetS16V,
4326 SSMR3GetU32,
4327 SSMR3GetU32V,
4328 SSMR3GetS32,
4329 SSMR3GetS32V,
4330 SSMR3GetU64,
4331 SSMR3GetU64V,
4332 SSMR3GetS64,
4333 SSMR3GetS64V,
4334 SSMR3GetU128,
4335 SSMR3GetU128V,
4336 SSMR3GetS128,
4337 SSMR3GetS128V,
4338 SSMR3GetGCPhys32,
4339 SSMR3GetGCPhys32V,
4340 SSMR3GetGCPhys64,
4341 SSMR3GetGCPhys64V,
4342 SSMR3GetGCPhys,
4343 SSMR3GetGCPhysV,
4344 SSMR3GetUInt,
4345 SSMR3GetSInt,
4346 SSMR3GetGCUInt,
4347 SSMR3GetGCUIntReg,
4348 SSMR3GetGCPtr,
4349 SSMR3GetGCUIntPtr,
4350 SSMR3GetRCPtr,
4351 SSMR3GetIOPort,
4352 SSMR3GetSel,
4353 SSMR3GetMem,
4354 SSMR3GetStrZ,
4355 SSMR3GetStrZEx,
4356 SSMR3Skip,
4357 SSMR3SkipToEndOfUnit,
4358 SSMR3SetLoadError,
4359 SSMR3SetLoadErrorV,
4360 SSMR3SetCfgError,
4361 SSMR3SetCfgErrorV,
4362 SSMR3HandleGetStatus,
4363 SSMR3HandleGetAfter,
4364 SSMR3HandleIsLiveSave,
4365 SSMR3HandleMaxDowntime,
4366 SSMR3HandleHostBits,
4367 SSMR3HandleRevision,
4368 SSMR3HandleVersion,
4369 pdmR3DevHlp_TMTimerCreate,
4370 pdmR3DevHlp_TimerCreate,
4371 pdmR3DevHlp_TimerToPtr,
4372 pdmR3DevHlp_TimerFromMicro,
4373 pdmR3DevHlp_TimerFromMilli,
4374 pdmR3DevHlp_TimerFromNano,
4375 pdmR3DevHlp_TimerGet,
4376 pdmR3DevHlp_TimerGetFreq,
4377 pdmR3DevHlp_TimerGetNano,
4378 pdmR3DevHlp_TimerIsActive,
4379 pdmR3DevHlp_TimerIsLockOwner,
4380 pdmR3DevHlp_TimerLock,
4381 pdmR3DevHlp_TimerSet,
4382 pdmR3DevHlp_TimerSetFrequencyHint,
4383 pdmR3DevHlp_TimerSetMicro,
4384 pdmR3DevHlp_TimerSetMillies,
4385 pdmR3DevHlp_TimerSetNano,
4386 pdmR3DevHlp_TimerSetRelative,
4387 pdmR3DevHlp_TimerStop,
4388 pdmR3DevHlp_TimerUnlock,
4389 pdmR3DevHlp_TimerSetCritSect,
4390 pdmR3DevHlp_TimerSave,
4391 pdmR3DevHlp_TimerLoad,
4392 pdmR3DevHlp_TimerDestroy,
4393 TMR3TimerSkip,
4394 pdmR3DevHlp_TMUtcNow,
4395 CFGMR3Exists,
4396 CFGMR3QueryType,
4397 CFGMR3QuerySize,
4398 CFGMR3QueryInteger,
4399 CFGMR3QueryIntegerDef,
4400 CFGMR3QueryString,
4401 CFGMR3QueryStringDef,
4402 CFGMR3QueryBytes,
4403 CFGMR3QueryU64,
4404 CFGMR3QueryU64Def,
4405 CFGMR3QueryS64,
4406 CFGMR3QueryS64Def,
4407 CFGMR3QueryU32,
4408 CFGMR3QueryU32Def,
4409 CFGMR3QueryS32,
4410 CFGMR3QueryS32Def,
4411 CFGMR3QueryU16,
4412 CFGMR3QueryU16Def,
4413 CFGMR3QueryS16,
4414 CFGMR3QueryS16Def,
4415 CFGMR3QueryU8,
4416 CFGMR3QueryU8Def,
4417 CFGMR3QueryS8,
4418 CFGMR3QueryS8Def,
4419 CFGMR3QueryBool,
4420 CFGMR3QueryBoolDef,
4421 CFGMR3QueryPort,
4422 CFGMR3QueryPortDef,
4423 CFGMR3QueryUInt,
4424 CFGMR3QueryUIntDef,
4425 CFGMR3QuerySInt,
4426 CFGMR3QuerySIntDef,
4427 CFGMR3QueryPtr,
4428 CFGMR3QueryPtrDef,
4429 CFGMR3QueryGCPtr,
4430 CFGMR3QueryGCPtrDef,
4431 CFGMR3QueryGCPtrU,
4432 CFGMR3QueryGCPtrUDef,
4433 CFGMR3QueryGCPtrS,
4434 CFGMR3QueryGCPtrSDef,
4435 CFGMR3QueryStringAlloc,
4436 CFGMR3QueryStringAllocDef,
4437 CFGMR3GetParent,
4438 CFGMR3GetChild,
4439 CFGMR3GetChildF,
4440 CFGMR3GetChildFV,
4441 CFGMR3GetFirstChild,
4442 CFGMR3GetNextChild,
4443 CFGMR3GetName,
4444 CFGMR3GetNameLen,
4445 CFGMR3AreChildrenValid,
4446 CFGMR3GetFirstValue,
4447 CFGMR3GetNextValue,
4448 CFGMR3GetValueName,
4449 CFGMR3GetValueNameLen,
4450 CFGMR3GetValueType,
4451 CFGMR3AreValuesValid,
4452 CFGMR3ValidateConfig,
4453 pdmR3DevHlp_PhysRead,
4454 pdmR3DevHlp_PhysWrite,
4455 pdmR3DevHlp_PhysGCPhys2CCPtr,
4456 pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly,
4457 pdmR3DevHlp_PhysReleasePageMappingLock,
4458 pdmR3DevHlp_PhysReadGCVirt,
4459 pdmR3DevHlp_PhysWriteGCVirt,
4460 pdmR3DevHlp_PhysGCPtr2GCPhys,
4461 pdmR3DevHlp_MMHeapAlloc,
4462 pdmR3DevHlp_MMHeapAllocZ,
4463 pdmR3DevHlp_MMHeapFree,
4464 pdmR3DevHlp_VMState,
4465 pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet,
4466 pdmR3DevHlp_VMSetError,
4467 pdmR3DevHlp_VMSetErrorV,
4468 pdmR3DevHlp_VMSetRuntimeError,
4469 pdmR3DevHlp_VMSetRuntimeErrorV,
4470 pdmR3DevHlp_DBGFStopV,
4471 pdmR3DevHlp_DBGFInfoRegister,
4472 pdmR3DevHlp_DBGFInfoRegisterArgv,
4473 pdmR3DevHlp_DBGFRegRegister,
4474 pdmR3DevHlp_DBGFTraceBuf,
4475 pdmR3DevHlp_STAMRegister,
4476 pdmR3DevHlp_STAMRegisterV,
4477 pdmR3DevHlp_PCIRegister,
4478 pdmR3DevHlp_PCIRegisterMsi,
4479 pdmR3DevHlp_PCIIORegionRegister,
4480 pdmR3DevHlp_PCIInterceptConfigAccesses,
4481 pdmR3DevHlp_PCIConfigWrite,
4482 pdmR3DevHlp_PCIConfigRead,
4483 pdmR3DevHlp_PCIPhysRead,
4484 pdmR3DevHlp_PCIPhysWrite,
4485 pdmR3DevHlp_PCISetIrq,
4486 pdmR3DevHlp_PCISetIrqNoWait,
4487 pdmR3DevHlp_ISASetIrq,
4488 pdmR3DevHlp_ISASetIrqNoWait,
4489 pdmR3DevHlp_IoApicSendMsi,
4490 pdmR3DevHlp_DriverAttach,
4491 pdmR3DevHlp_DriverDetach,
4492 pdmR3DevHlp_QueueCreatePtr,
4493 pdmR3DevHlp_QueueCreate,
4494 pdmR3DevHlp_QueueToPtr,
4495 pdmR3DevHlp_QueueAlloc,
4496 pdmR3DevHlp_QueueInsert,
4497 pdmR3DevHlp_QueueInsertEx,
4498 pdmR3DevHlp_QueueFlushIfNecessary,
4499 pdmR3DevHlp_TaskCreate,
4500 pdmR3DevHlp_TaskTrigger,
4501 pdmR3DevHlp_SUPSemEventCreate,
4502 pdmR3DevHlp_SUPSemEventClose,
4503 pdmR3DevHlp_SUPSemEventSignal,
4504 pdmR3DevHlp_SUPSemEventWaitNoResume,
4505 pdmR3DevHlp_SUPSemEventWaitNsAbsIntr,
4506 pdmR3DevHlp_SUPSemEventWaitNsRelIntr,
4507 pdmR3DevHlp_SUPSemEventGetResolution,
4508 pdmR3DevHlp_SUPSemEventMultiCreate,
4509 pdmR3DevHlp_SUPSemEventMultiClose,
4510 pdmR3DevHlp_SUPSemEventMultiSignal,
4511 pdmR3DevHlp_SUPSemEventMultiReset,
4512 pdmR3DevHlp_SUPSemEventMultiWaitNoResume,
4513 pdmR3DevHlp_SUPSemEventMultiWaitNsAbsIntr,
4514 pdmR3DevHlp_SUPSemEventMultiWaitNsRelIntr,
4515 pdmR3DevHlp_SUPSemEventMultiGetResolution,
4516 pdmR3DevHlp_CritSectInit,
4517 pdmR3DevHlp_CritSectGetNop,
4518 pdmR3DevHlp_CritSectGetNopR0,
4519 pdmR3DevHlp_CritSectGetNopRC,
4520 pdmR3DevHlp_SetDeviceCritSect,
4521 pdmR3DevHlp_CritSectYield,
4522 pdmR3DevHlp_CritSectEnter,
4523 pdmR3DevHlp_CritSectEnterDebug,
4524 pdmR3DevHlp_CritSectTryEnter,
4525 pdmR3DevHlp_CritSectTryEnterDebug,
4526 pdmR3DevHlp_CritSectLeave,
4527 pdmR3DevHlp_CritSectIsOwner,
4528 pdmR3DevHlp_CritSectIsInitialized,
4529 pdmR3DevHlp_CritSectHasWaiters,
4530 pdmR3DevHlp_CritSectGetRecursion,
4531 pdmR3DevHlp_CritSectScheduleExitEvent,
4532 pdmR3DevHlp_CritSectDelete,
4533 pdmR3DevHlp_ThreadCreate,
4534 PDMR3ThreadDestroy,
4535 PDMR3ThreadIAmSuspending,
4536 PDMR3ThreadIAmRunning,
4537 PDMR3ThreadSleep,
4538 PDMR3ThreadSuspend,
4539 PDMR3ThreadResume,
4540 pdmR3DevHlp_SetAsyncNotification,
4541 pdmR3DevHlp_AsyncNotificationCompleted,
4542 pdmR3DevHlp_RTCRegister,
4543 pdmR3DevHlp_PCIBusRegister,
4544 pdmR3DevHlp_PICRegister,
4545 pdmR3DevHlp_APICRegister,
4546 pdmR3DevHlp_IoApicRegister,
4547 pdmR3DevHlp_HpetRegister,
4548 pdmR3DevHlp_PciRawRegister,
4549 pdmR3DevHlp_DMACRegister,
4550 pdmR3DevHlp_DMARegister,
4551 pdmR3DevHlp_DMAReadMemory,
4552 pdmR3DevHlp_DMAWriteMemory,
4553 pdmR3DevHlp_DMASetDREQ,
4554 pdmR3DevHlp_DMAGetChannelMode,
4555 pdmR3DevHlp_DMASchedule,
4556 pdmR3DevHlp_CMOSWrite,
4557 pdmR3DevHlp_CMOSRead,
4558 pdmR3DevHlp_AssertEMT,
4559 pdmR3DevHlp_AssertOther,
4560 pdmR3DevHlp_LdrGetRCInterfaceSymbols,
4561 pdmR3DevHlp_LdrGetR0InterfaceSymbols,
4562 pdmR3DevHlp_CallR0,
4563 pdmR3DevHlp_VMGetSuspendReason,
4564 pdmR3DevHlp_VMGetResumeReason,
4565 pdmR3DevHlp_PhysBulkGCPhys2CCPtr,
4566 pdmR3DevHlp_PhysBulkGCPhys2CCPtrReadOnly,
4567 pdmR3DevHlp_PhysBulkReleasePageMappingLocks,
4568 pdmR3DevHlp_MMIOExChangeRegionNo,
4569 0,
4570 0,
4571 0,
4572 0,
4573 0,
4574 0,
4575 0,
4576 0,
4577 0,
4578 0,
4579 pdmR3DevHlp_GetUVM,
4580 pdmR3DevHlp_GetVM,
4581 pdmR3DevHlp_GetVMCPU,
4582 pdmR3DevHlp_GetCurrentCpuId,
4583 pdmR3DevHlp_RegisterVMMDevHeap,
4584 pdmR3DevHlp_FirmwareRegister,
4585 pdmR3DevHlp_VMReset,
4586 pdmR3DevHlp_VMSuspend,
4587 pdmR3DevHlp_VMSuspendSaveAndPowerOff,
4588 pdmR3DevHlp_VMPowerOff,
4589 pdmR3DevHlp_A20IsEnabled,
4590 pdmR3DevHlp_A20Set,
4591 pdmR3DevHlp_GetCpuId,
4592 pdmR3DevHlp_TMTimeVirtGet,
4593 pdmR3DevHlp_TMTimeVirtGetFreq,
4594 pdmR3DevHlp_TMTimeVirtGetNano,
4595 pdmR3DevHlp_GetSupDrvSession,
4596 pdmR3DevHlp_QueryGenericUserObject,
4597 PDM_DEVHLPR3_VERSION /* the end */
4598};
4599
4600
4601
4602
4603/** @interface_method_impl{PDMDEVHLPR3,pfnGetUVM} */
4604static DECLCALLBACK(PUVM) pdmR3DevHlp_Untrusted_GetUVM(PPDMDEVINS pDevIns)
4605{
4606 PDMDEV_ASSERT_DEVINS(pDevIns);
4607 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4608 return NULL;
4609}
4610
4611
4612/** @interface_method_impl{PDMDEVHLPR3,pfnGetVM} */
4613static DECLCALLBACK(PVM) pdmR3DevHlp_Untrusted_GetVM(PPDMDEVINS pDevIns)
4614{
4615 PDMDEV_ASSERT_DEVINS(pDevIns);
4616 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4617 return NULL;
4618}
4619
4620
4621/** @interface_method_impl{PDMDEVHLPR3,pfnGetVMCPU} */
4622static DECLCALLBACK(PVMCPU) pdmR3DevHlp_Untrusted_GetVMCPU(PPDMDEVINS pDevIns)
4623{
4624 PDMDEV_ASSERT_DEVINS(pDevIns);
4625 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4626 return NULL;
4627}
4628
4629
4630/** @interface_method_impl{PDMDEVHLPR3,pfnGetCurrentCpuId} */
4631static DECLCALLBACK(VMCPUID) pdmR3DevHlp_Untrusted_GetCurrentCpuId(PPDMDEVINS pDevIns)
4632{
4633 PDMDEV_ASSERT_DEVINS(pDevIns);
4634 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4635 return NIL_VMCPUID;
4636}
4637
4638
4639/** @interface_method_impl{PDMDEVHLPR3,pfnRegisterVMMDevHeap} */
4640static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys,
4641 RTR3PTR pvHeap, unsigned cbHeap)
4642{
4643 PDMDEV_ASSERT_DEVINS(pDevIns);
4644 NOREF(GCPhys); NOREF(pvHeap); NOREF(cbHeap);
4645 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4646 return VERR_ACCESS_DENIED;
4647}
4648
4649
4650/** @interface_method_impl{PDMDEVHLPR3,pfnFirmwareRegister} */
4651static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_FirmwareRegister(PPDMDEVINS pDevIns, PCPDMFWREG pFwReg, PCPDMFWHLPR3 *ppFwHlp)
4652{
4653 PDMDEV_ASSERT_DEVINS(pDevIns);
4654 NOREF(pFwReg); NOREF(ppFwHlp);
4655 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4656 return VERR_ACCESS_DENIED;
4657}
4658
4659
4660/** @interface_method_impl{PDMDEVHLPR3,pfnVMReset} */
4661static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMReset(PPDMDEVINS pDevIns, uint32_t fFlags)
4662{
4663 PDMDEV_ASSERT_DEVINS(pDevIns); NOREF(fFlags);
4664 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4665 return VERR_ACCESS_DENIED;
4666}
4667
4668
4669/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspend} */
4670static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspend(PPDMDEVINS pDevIns)
4671{
4672 PDMDEV_ASSERT_DEVINS(pDevIns);
4673 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4674 return VERR_ACCESS_DENIED;
4675}
4676
4677
4678/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspendSaveAndPowerOff} */
4679static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspendSaveAndPowerOff(PPDMDEVINS pDevIns)
4680{
4681 PDMDEV_ASSERT_DEVINS(pDevIns);
4682 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4683 return VERR_ACCESS_DENIED;
4684}
4685
4686
4687/** @interface_method_impl{PDMDEVHLPR3,pfnVMPowerOff} */
4688static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMPowerOff(PPDMDEVINS pDevIns)
4689{
4690 PDMDEV_ASSERT_DEVINS(pDevIns);
4691 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4692 return VERR_ACCESS_DENIED;
4693}
4694
4695
4696/** @interface_method_impl{PDMDEVHLPR3,pfnA20IsEnabled} */
4697static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_A20IsEnabled(PPDMDEVINS pDevIns)
4698{
4699 PDMDEV_ASSERT_DEVINS(pDevIns);
4700 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4701 return false;
4702}
4703
4704
4705/** @interface_method_impl{PDMDEVHLPR3,pfnA20Set} */
4706static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_A20Set(PPDMDEVINS pDevIns, bool fEnable)
4707{
4708 PDMDEV_ASSERT_DEVINS(pDevIns);
4709 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4710 NOREF(fEnable);
4711}
4712
4713
4714/** @interface_method_impl{PDMDEVHLPR3,pfnGetCpuId} */
4715static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
4716 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
4717{
4718 PDMDEV_ASSERT_DEVINS(pDevIns);
4719 NOREF(iLeaf); NOREF(pEax); NOREF(pEbx); NOREF(pEcx); NOREF(pEdx);
4720 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4721}
4722
4723
4724/** @interface_method_impl{PDMDEVHLPR3,pfnGetSupDrvSession} */
4725static DECLCALLBACK(PSUPDRVSESSION) pdmR3DevHlp_Untrusted_GetSupDrvSession(PPDMDEVINS pDevIns)
4726{
4727 PDMDEV_ASSERT_DEVINS(pDevIns);
4728 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
4729 return (PSUPDRVSESSION)0;
4730}
4731
4732
4733/** @interface_method_impl{PDMDEVHLPR3,pfnQueryGenericUserObject} */
4734static DECLCALLBACK(void *) pdmR3DevHlp_Untrusted_QueryGenericUserObject(PPDMDEVINS pDevIns, PCRTUUID pUuid)
4735{
4736 PDMDEV_ASSERT_DEVINS(pDevIns);
4737 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d %RTuuid\n",
4738 pDevIns->pReg->szName, pDevIns->iInstance, pUuid));
4739 return NULL;
4740}
4741
4742
4743/**
4744 * The device helper structure for non-trusted devices.
4745 */
4746const PDMDEVHLPR3 g_pdmR3DevHlpUnTrusted =
4747{
4748 PDM_DEVHLPR3_VERSION,
4749 pdmR3DevHlp_IoPortCreateEx,
4750 pdmR3DevHlp_IoPortMap,
4751 pdmR3DevHlp_IoPortUnmap,
4752 pdmR3DevHlp_IoPortGetMappingAddress,
4753 pdmR3DevHlp_IOPortRegister,
4754 pdmR3DevHlp_IOPortRegisterRC,
4755 pdmR3DevHlp_IOPortRegisterR0,
4756 pdmR3DevHlp_IOPortDeregister,
4757 pdmR3DevHlp_MmioCreateEx,
4758 pdmR3DevHlp_MmioMap,
4759 pdmR3DevHlp_MmioUnmap,
4760 pdmR3DevHlp_MmioReduce,
4761 pdmR3DevHlp_MmioGetMappingAddress,
4762 pdmR3DevHlp_MMIORegister,
4763 pdmR3DevHlp_MMIORegisterRC,
4764 pdmR3DevHlp_MMIORegisterR0,
4765 pdmR3DevHlp_MMIODeregister,
4766 pdmR3DevHlp_Mmio2Create,
4767 pdmR3DevHlp_Mmio2Destroy,
4768 pdmR3DevHlp_Mmio2Map,
4769 pdmR3DevHlp_Mmio2Unmap,
4770 pdmR3DevHlp_Mmio2Reduce,
4771 pdmR3DevHlp_Mmio2GetMappingAddress,
4772 pdmR3DevHlp_Mmio2ChangeRegionNo,
4773 pdmR3DevHlp_MMIO2Register,
4774 pdmR3DevHlp_MMIOExDeregister,
4775 pdmR3DevHlp_MMIOExMap,
4776 pdmR3DevHlp_MMIOExUnmap,
4777 pdmR3DevHlp_MMIOExReduce,
4778 pdmR3DevHlp_MMHyperMapMMIO2,
4779 pdmR3DevHlp_MMIO2MapKernel,
4780 pdmR3DevHlp_ROMRegister,
4781 pdmR3DevHlp_ROMProtectShadow,
4782 pdmR3DevHlp_SSMRegister,
4783 SSMR3PutStruct,
4784 SSMR3PutStructEx,
4785 SSMR3PutBool,
4786 SSMR3PutU8,
4787 SSMR3PutS8,
4788 SSMR3PutU16,
4789 SSMR3PutS16,
4790 SSMR3PutU32,
4791 SSMR3PutS32,
4792 SSMR3PutU64,
4793 SSMR3PutS64,
4794 SSMR3PutU128,
4795 SSMR3PutS128,
4796 SSMR3PutUInt,
4797 SSMR3PutSInt,
4798 SSMR3PutGCUInt,
4799 SSMR3PutGCUIntReg,
4800 SSMR3PutGCPhys32,
4801 SSMR3PutGCPhys64,
4802 SSMR3PutGCPhys,
4803 SSMR3PutGCPtr,
4804 SSMR3PutGCUIntPtr,
4805 SSMR3PutRCPtr,
4806 SSMR3PutIOPort,
4807 SSMR3PutSel,
4808 SSMR3PutMem,
4809 SSMR3PutStrZ,
4810 SSMR3GetStruct,
4811 SSMR3GetStructEx,
4812 SSMR3GetBool,
4813 SSMR3GetBoolV,
4814 SSMR3GetU8,
4815 SSMR3GetU8V,
4816 SSMR3GetS8,
4817 SSMR3GetS8V,
4818 SSMR3GetU16,
4819 SSMR3GetU16V,
4820 SSMR3GetS16,
4821 SSMR3GetS16V,
4822 SSMR3GetU32,
4823 SSMR3GetU32V,
4824 SSMR3GetS32,
4825 SSMR3GetS32V,
4826 SSMR3GetU64,
4827 SSMR3GetU64V,
4828 SSMR3GetS64,
4829 SSMR3GetS64V,
4830 SSMR3GetU128,
4831 SSMR3GetU128V,
4832 SSMR3GetS128,
4833 SSMR3GetS128V,
4834 SSMR3GetGCPhys32,
4835 SSMR3GetGCPhys32V,
4836 SSMR3GetGCPhys64,
4837 SSMR3GetGCPhys64V,
4838 SSMR3GetGCPhys,
4839 SSMR3GetGCPhysV,
4840 SSMR3GetUInt,
4841 SSMR3GetSInt,
4842 SSMR3GetGCUInt,
4843 SSMR3GetGCUIntReg,
4844 SSMR3GetGCPtr,
4845 SSMR3GetGCUIntPtr,
4846 SSMR3GetRCPtr,
4847 SSMR3GetIOPort,
4848 SSMR3GetSel,
4849 SSMR3GetMem,
4850 SSMR3GetStrZ,
4851 SSMR3GetStrZEx,
4852 SSMR3Skip,
4853 SSMR3SkipToEndOfUnit,
4854 SSMR3SetLoadError,
4855 SSMR3SetLoadErrorV,
4856 SSMR3SetCfgError,
4857 SSMR3SetCfgErrorV,
4858 SSMR3HandleGetStatus,
4859 SSMR3HandleGetAfter,
4860 SSMR3HandleIsLiveSave,
4861 SSMR3HandleMaxDowntime,
4862 SSMR3HandleHostBits,
4863 SSMR3HandleRevision,
4864 SSMR3HandleVersion,
4865 pdmR3DevHlp_TMTimerCreate,
4866 pdmR3DevHlp_TimerCreate,
4867 pdmR3DevHlp_TimerToPtr,
4868 pdmR3DevHlp_TimerFromMicro,
4869 pdmR3DevHlp_TimerFromMilli,
4870 pdmR3DevHlp_TimerFromNano,
4871 pdmR3DevHlp_TimerGet,
4872 pdmR3DevHlp_TimerGetFreq,
4873 pdmR3DevHlp_TimerGetNano,
4874 pdmR3DevHlp_TimerIsActive,
4875 pdmR3DevHlp_TimerIsLockOwner,
4876 pdmR3DevHlp_TimerLock,
4877 pdmR3DevHlp_TimerSet,
4878 pdmR3DevHlp_TimerSetFrequencyHint,
4879 pdmR3DevHlp_TimerSetMicro,
4880 pdmR3DevHlp_TimerSetMillies,
4881 pdmR3DevHlp_TimerSetNano,
4882 pdmR3DevHlp_TimerSetRelative,
4883 pdmR3DevHlp_TimerStop,
4884 pdmR3DevHlp_TimerUnlock,
4885 pdmR3DevHlp_TimerSetCritSect,
4886 pdmR3DevHlp_TimerSave,
4887 pdmR3DevHlp_TimerLoad,
4888 pdmR3DevHlp_TimerDestroy,
4889 TMR3TimerSkip,
4890 pdmR3DevHlp_TMUtcNow,
4891 CFGMR3Exists,
4892 CFGMR3QueryType,
4893 CFGMR3QuerySize,
4894 CFGMR3QueryInteger,
4895 CFGMR3QueryIntegerDef,
4896 CFGMR3QueryString,
4897 CFGMR3QueryStringDef,
4898 CFGMR3QueryBytes,
4899 CFGMR3QueryU64,
4900 CFGMR3QueryU64Def,
4901 CFGMR3QueryS64,
4902 CFGMR3QueryS64Def,
4903 CFGMR3QueryU32,
4904 CFGMR3QueryU32Def,
4905 CFGMR3QueryS32,
4906 CFGMR3QueryS32Def,
4907 CFGMR3QueryU16,
4908 CFGMR3QueryU16Def,
4909 CFGMR3QueryS16,
4910 CFGMR3QueryS16Def,
4911 CFGMR3QueryU8,
4912 CFGMR3QueryU8Def,
4913 CFGMR3QueryS8,
4914 CFGMR3QueryS8Def,
4915 CFGMR3QueryBool,
4916 CFGMR3QueryBoolDef,
4917 CFGMR3QueryPort,
4918 CFGMR3QueryPortDef,
4919 CFGMR3QueryUInt,
4920 CFGMR3QueryUIntDef,
4921 CFGMR3QuerySInt,
4922 CFGMR3QuerySIntDef,
4923 CFGMR3QueryPtr,
4924 CFGMR3QueryPtrDef,
4925 CFGMR3QueryGCPtr,
4926 CFGMR3QueryGCPtrDef,
4927 CFGMR3QueryGCPtrU,
4928 CFGMR3QueryGCPtrUDef,
4929 CFGMR3QueryGCPtrS,
4930 CFGMR3QueryGCPtrSDef,
4931 CFGMR3QueryStringAlloc,
4932 CFGMR3QueryStringAllocDef,
4933 CFGMR3GetParent,
4934 CFGMR3GetChild,
4935 CFGMR3GetChildF,
4936 CFGMR3GetChildFV,
4937 CFGMR3GetFirstChild,
4938 CFGMR3GetNextChild,
4939 CFGMR3GetName,
4940 CFGMR3GetNameLen,
4941 CFGMR3AreChildrenValid,
4942 CFGMR3GetFirstValue,
4943 CFGMR3GetNextValue,
4944 CFGMR3GetValueName,
4945 CFGMR3GetValueNameLen,
4946 CFGMR3GetValueType,
4947 CFGMR3AreValuesValid,
4948 CFGMR3ValidateConfig,
4949 pdmR3DevHlp_PhysRead,
4950 pdmR3DevHlp_PhysWrite,
4951 pdmR3DevHlp_PhysGCPhys2CCPtr,
4952 pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly,
4953 pdmR3DevHlp_PhysReleasePageMappingLock,
4954 pdmR3DevHlp_PhysReadGCVirt,
4955 pdmR3DevHlp_PhysWriteGCVirt,
4956 pdmR3DevHlp_PhysGCPtr2GCPhys,
4957 pdmR3DevHlp_MMHeapAlloc,
4958 pdmR3DevHlp_MMHeapAllocZ,
4959 pdmR3DevHlp_MMHeapFree,
4960 pdmR3DevHlp_VMState,
4961 pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet,
4962 pdmR3DevHlp_VMSetError,
4963 pdmR3DevHlp_VMSetErrorV,
4964 pdmR3DevHlp_VMSetRuntimeError,
4965 pdmR3DevHlp_VMSetRuntimeErrorV,
4966 pdmR3DevHlp_DBGFStopV,
4967 pdmR3DevHlp_DBGFInfoRegister,
4968 pdmR3DevHlp_DBGFInfoRegisterArgv,
4969 pdmR3DevHlp_DBGFRegRegister,
4970 pdmR3DevHlp_DBGFTraceBuf,
4971 pdmR3DevHlp_STAMRegister,
4972 pdmR3DevHlp_STAMRegisterV,
4973 pdmR3DevHlp_PCIRegister,
4974 pdmR3DevHlp_PCIRegisterMsi,
4975 pdmR3DevHlp_PCIIORegionRegister,
4976 pdmR3DevHlp_PCIInterceptConfigAccesses,
4977 pdmR3DevHlp_PCIConfigWrite,
4978 pdmR3DevHlp_PCIConfigRead,
4979 pdmR3DevHlp_PCIPhysRead,
4980 pdmR3DevHlp_PCIPhysWrite,
4981 pdmR3DevHlp_PCISetIrq,
4982 pdmR3DevHlp_PCISetIrqNoWait,
4983 pdmR3DevHlp_ISASetIrq,
4984 pdmR3DevHlp_ISASetIrqNoWait,
4985 pdmR3DevHlp_IoApicSendMsi,
4986 pdmR3DevHlp_DriverAttach,
4987 pdmR3DevHlp_DriverDetach,
4988 pdmR3DevHlp_QueueCreatePtr,
4989 pdmR3DevHlp_QueueCreate,
4990 pdmR3DevHlp_QueueToPtr,
4991 pdmR3DevHlp_QueueAlloc,
4992 pdmR3DevHlp_QueueInsert,
4993 pdmR3DevHlp_QueueInsertEx,
4994 pdmR3DevHlp_QueueFlushIfNecessary,
4995 pdmR3DevHlp_TaskCreate,
4996 pdmR3DevHlp_TaskTrigger,
4997 pdmR3DevHlp_SUPSemEventCreate,
4998 pdmR3DevHlp_SUPSemEventClose,
4999 pdmR3DevHlp_SUPSemEventSignal,
5000 pdmR3DevHlp_SUPSemEventWaitNoResume,
5001 pdmR3DevHlp_SUPSemEventWaitNsAbsIntr,
5002 pdmR3DevHlp_SUPSemEventWaitNsRelIntr,
5003 pdmR3DevHlp_SUPSemEventGetResolution,
5004 pdmR3DevHlp_SUPSemEventMultiCreate,
5005 pdmR3DevHlp_SUPSemEventMultiClose,
5006 pdmR3DevHlp_SUPSemEventMultiSignal,
5007 pdmR3DevHlp_SUPSemEventMultiReset,
5008 pdmR3DevHlp_SUPSemEventMultiWaitNoResume,
5009 pdmR3DevHlp_SUPSemEventMultiWaitNsAbsIntr,
5010 pdmR3DevHlp_SUPSemEventMultiWaitNsRelIntr,
5011 pdmR3DevHlp_SUPSemEventMultiGetResolution,
5012 pdmR3DevHlp_CritSectInit,
5013 pdmR3DevHlp_CritSectGetNop,
5014 pdmR3DevHlp_CritSectGetNopR0,
5015 pdmR3DevHlp_CritSectGetNopRC,
5016 pdmR3DevHlp_SetDeviceCritSect,
5017 pdmR3DevHlp_CritSectYield,
5018 pdmR3DevHlp_CritSectEnter,
5019 pdmR3DevHlp_CritSectEnterDebug,
5020 pdmR3DevHlp_CritSectTryEnter,
5021 pdmR3DevHlp_CritSectTryEnterDebug,
5022 pdmR3DevHlp_CritSectLeave,
5023 pdmR3DevHlp_CritSectIsOwner,
5024 pdmR3DevHlp_CritSectIsInitialized,
5025 pdmR3DevHlp_CritSectHasWaiters,
5026 pdmR3DevHlp_CritSectGetRecursion,
5027 pdmR3DevHlp_CritSectScheduleExitEvent,
5028 pdmR3DevHlp_CritSectDelete,
5029 pdmR3DevHlp_ThreadCreate,
5030 PDMR3ThreadDestroy,
5031 PDMR3ThreadIAmSuspending,
5032 PDMR3ThreadIAmRunning,
5033 PDMR3ThreadSleep,
5034 PDMR3ThreadSuspend,
5035 PDMR3ThreadResume,
5036 pdmR3DevHlp_SetAsyncNotification,
5037 pdmR3DevHlp_AsyncNotificationCompleted,
5038 pdmR3DevHlp_RTCRegister,
5039 pdmR3DevHlp_PCIBusRegister,
5040 pdmR3DevHlp_PICRegister,
5041 pdmR3DevHlp_APICRegister,
5042 pdmR3DevHlp_IoApicRegister,
5043 pdmR3DevHlp_HpetRegister,
5044 pdmR3DevHlp_PciRawRegister,
5045 pdmR3DevHlp_DMACRegister,
5046 pdmR3DevHlp_DMARegister,
5047 pdmR3DevHlp_DMAReadMemory,
5048 pdmR3DevHlp_DMAWriteMemory,
5049 pdmR3DevHlp_DMASetDREQ,
5050 pdmR3DevHlp_DMAGetChannelMode,
5051 pdmR3DevHlp_DMASchedule,
5052 pdmR3DevHlp_CMOSWrite,
5053 pdmR3DevHlp_CMOSRead,
5054 pdmR3DevHlp_AssertEMT,
5055 pdmR3DevHlp_AssertOther,
5056 pdmR3DevHlp_LdrGetRCInterfaceSymbols,
5057 pdmR3DevHlp_LdrGetR0InterfaceSymbols,
5058 pdmR3DevHlp_CallR0,
5059 pdmR3DevHlp_VMGetSuspendReason,
5060 pdmR3DevHlp_VMGetResumeReason,
5061 pdmR3DevHlp_PhysBulkGCPhys2CCPtr,
5062 pdmR3DevHlp_PhysBulkGCPhys2CCPtrReadOnly,
5063 pdmR3DevHlp_PhysBulkReleasePageMappingLocks,
5064 pdmR3DevHlp_MMIOExChangeRegionNo,
5065 0,
5066 0,
5067 0,
5068 0,
5069 0,
5070 0,
5071 0,
5072 0,
5073 0,
5074 0,
5075 pdmR3DevHlp_Untrusted_GetUVM,
5076 pdmR3DevHlp_Untrusted_GetVM,
5077 pdmR3DevHlp_Untrusted_GetVMCPU,
5078 pdmR3DevHlp_Untrusted_GetCurrentCpuId,
5079 pdmR3DevHlp_Untrusted_RegisterVMMDevHeap,
5080 pdmR3DevHlp_Untrusted_FirmwareRegister,
5081 pdmR3DevHlp_Untrusted_VMReset,
5082 pdmR3DevHlp_Untrusted_VMSuspend,
5083 pdmR3DevHlp_Untrusted_VMSuspendSaveAndPowerOff,
5084 pdmR3DevHlp_Untrusted_VMPowerOff,
5085 pdmR3DevHlp_Untrusted_A20IsEnabled,
5086 pdmR3DevHlp_Untrusted_A20Set,
5087 pdmR3DevHlp_Untrusted_GetCpuId,
5088 pdmR3DevHlp_TMTimeVirtGet,
5089 pdmR3DevHlp_TMTimeVirtGetFreq,
5090 pdmR3DevHlp_TMTimeVirtGetNano,
5091 pdmR3DevHlp_Untrusted_GetSupDrvSession,
5092 pdmR3DevHlp_Untrusted_QueryGenericUserObject,
5093 PDM_DEVHLPR3_VERSION /* the end */
5094};
5095
5096
5097
5098/**
5099 * Queue consumer callback for internal component.
5100 *
5101 * @returns Success indicator.
5102 * If false the item will not be removed and the flushing will stop.
5103 * @param pVM The cross context VM structure.
5104 * @param pItem The item to consume. Upon return this item will be freed.
5105 */
5106DECLCALLBACK(bool) pdmR3DevHlpQueueConsumer(PVM pVM, PPDMQUEUEITEMCORE pItem)
5107{
5108 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)pItem;
5109 LogFlow(("pdmR3DevHlpQueueConsumer: enmOp=%d pDevIns=%p\n", pTask->enmOp, pTask->pDevInsR3));
5110 switch (pTask->enmOp)
5111 {
5112 case PDMDEVHLPTASKOP_ISA_SET_IRQ:
5113 PDMIsaSetIrq(pVM, pTask->u.IsaSetIRQ.iIrq, pTask->u.IsaSetIRQ.iLevel, pTask->u.IsaSetIRQ.uTagSrc);
5114 break;
5115
5116 case PDMDEVHLPTASKOP_PCI_SET_IRQ:
5117 {
5118 /* Same as pdmR3DevHlp_PCISetIrq, except we've got a tag already. */
5119 PPDMPCIDEV pPciDev = pTask->u.PciSetIRQ.pPciDevR3;
5120 if (pPciDev)
5121 {
5122 size_t const idxBus = pPciDev->Int.s.idxPdmBus;
5123 AssertBreak(idxBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses));
5124 PPDMPCIBUS pBus = &pVM->pdm.s.aPciBuses[idxBus];
5125
5126 pdmLock(pVM);
5127 pBus->pfnSetIrqR3(pBus->pDevInsR3, pPciDev, pTask->u.PciSetIRQ.iIrq,
5128 pTask->u.PciSetIRQ.iLevel, pTask->u.PciSetIRQ.uTagSrc);
5129 pdmUnlock(pVM);
5130 }
5131 else
5132 AssertReleaseMsgFailed(("No PCI device registered!\n"));
5133 break;
5134 }
5135
5136 case PDMDEVHLPTASKOP_IOAPIC_SET_IRQ:
5137 PDMIoApicSetIrq(pVM, pTask->u.IoApicSetIRQ.iIrq, pTask->u.IoApicSetIRQ.iLevel, pTask->u.IoApicSetIRQ.uTagSrc);
5138 break;
5139
5140 default:
5141 AssertReleaseMsgFailed(("Invalid operation %d\n", pTask->enmOp));
5142 break;
5143 }
5144 return true;
5145}
5146
5147/** @} */
5148
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