VirtualBox

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

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

Main, VMM: PCI passthrough work

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 140.0 KB
Line 
1/* $Id: PDMDevHlp.cpp 35676 2011-01-24 14:24:34Z vboxsync $ */
2/** @file
3 * PDM - Pluggable Device and Driver Manager, Device Helpers.
4 */
5
6/*
7 * Copyright (C) 2006-2010 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#include "PDMInternal.h"
24#include <VBox/vmm/pdm.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/pgm.h>
27#include <VBox/vmm/iom.h>
28#include <VBox/vmm/rem.h>
29#include <VBox/vmm/dbgf.h>
30#include <VBox/vmm/vmapi.h>
31#include <VBox/vmm/vm.h>
32#include <VBox/vmm/uvm.h>
33#include <VBox/vmm/vmm.h>
34
35#include <VBox/version.h>
36#include <VBox/log.h>
37#include <VBox/err.h>
38#include <iprt/asm.h>
39#include <iprt/assert.h>
40#include <iprt/ctype.h>
41#include <iprt/string.h>
42#include <iprt/thread.h>
43
44
45/*******************************************************************************
46* Defined Constants And Macros *
47*******************************************************************************/
48/** @def PDM_DEVHLP_DEADLOCK_DETECTION
49 * Define this to enable the deadlock detection when accessing physical memory.
50 */
51#if /*defined(DEBUG_bird) ||*/ defined(DOXYGEN_RUNNING)
52# define PDM_DEVHLP_DEADLOCK_DETECTION /**< @todo enable DevHlp deadlock detection! */
53#endif
54
55
56
57/**
58 * Wrapper around PDMR3LdrGetSymbolRCLazy.
59 */
60DECLINLINE(int) pdmR3DevGetSymbolRCLazy(PPDMDEVINS pDevIns, const char *pszSymbol, PRTRCPTR ppvValue)
61{
62 return PDMR3LdrGetSymbolRCLazy(pDevIns->Internal.s.pVMR3,
63 pDevIns->Internal.s.pDevR3->pReg->szRCMod,
64 pDevIns->Internal.s.pDevR3->pszRCSearchPath,
65 pszSymbol, ppvValue);
66}
67
68
69/**
70 * Wrapper around PDMR3LdrGetSymbolR0Lazy.
71 */
72DECLINLINE(int) pdmR3DevGetSymbolR0Lazy(PPDMDEVINS pDevIns, const char *pszSymbol, PRTR0PTR ppvValue)
73{
74 return PDMR3LdrGetSymbolR0Lazy(pDevIns->Internal.s.pVMR3,
75 pDevIns->Internal.s.pDevR3->pReg->szR0Mod,
76 pDevIns->Internal.s.pDevR3->pszR0SearchPath,
77 pszSymbol, ppvValue);
78}
79
80
81/** @name R3 DevHlp
82 * @{
83 */
84
85
86/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegister} */
87static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTHCPTR pvUser, PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn,
88 PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc)
89{
90 PDMDEV_ASSERT_DEVINS(pDevIns);
91 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: Port=%#x cPorts=%#x pvUser=%p pfnOut=%p pfnIn=%p pfnOutStr=%p pfnInStr=%p p32_tszDesc=%p:{%s}\n", pDevIns->pReg->szName, pDevIns->iInstance,
92 Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc, pszDesc));
93 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
94
95#if 0 /** @todo needs a real string cache for this */
96 if (pDevIns->iInstance > 0)
97 {
98 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
99 if (pszDesc2)
100 pszDesc = pszDesc2;
101 }
102#endif
103
104 int rc = IOMR3IOPortRegisterR3(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc);
105
106 LogFlow(("pdmR3DevHlp_IOPortRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
107 return rc;
108}
109
110
111/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegisterRC} */
112static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterRC(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTRCPTR pvUser,
113 const char *pszOut, const char *pszIn,
114 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
115{
116 PDMDEV_ASSERT_DEVINS(pDevIns);
117 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
118 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,
119 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
120
121 /*
122 * Resolve the functions (one of the can be NULL).
123 */
124 int rc = VINF_SUCCESS;
125 if ( pDevIns->pReg->szRCMod[0]
126 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC))
127 {
128 RTRCPTR RCPtrIn = NIL_RTRCPTR;
129 if (pszIn)
130 {
131 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszIn, &RCPtrIn);
132 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pReg->szRCMod, pszIn));
133 }
134 RTRCPTR RCPtrOut = NIL_RTRCPTR;
135 if (pszOut && RT_SUCCESS(rc))
136 {
137 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszOut, &RCPtrOut);
138 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pReg->szRCMod, pszOut));
139 }
140 RTRCPTR RCPtrInStr = NIL_RTRCPTR;
141 if (pszInStr && RT_SUCCESS(rc))
142 {
143 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszInStr, &RCPtrInStr);
144 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pReg->szRCMod, pszInStr));
145 }
146 RTRCPTR RCPtrOutStr = NIL_RTRCPTR;
147 if (pszOutStr && RT_SUCCESS(rc))
148 {
149 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszOutStr, &RCPtrOutStr);
150 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pReg->szRCMod, pszOutStr));
151 }
152
153 if (RT_SUCCESS(rc))
154 {
155#if 0 /** @todo needs a real string cache for this */
156 if (pDevIns->iInstance > 0)
157 {
158 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
159 if (pszDesc2)
160 pszDesc = pszDesc2;
161 }
162#endif
163
164 rc = IOMR3IOPortRegisterRC(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts, pvUser, RCPtrOut, RCPtrIn, RCPtrOutStr, RCPtrInStr, pszDesc);
165 }
166 }
167 else
168 {
169 AssertMsgFailed(("No GC module for this driver!\n"));
170 rc = VERR_INVALID_PARAMETER;
171 }
172
173 LogFlow(("pdmR3DevHlp_IOPortRegisterRC: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
174 return rc;
175}
176
177
178/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortRegisterR0} */
179static DECLCALLBACK(int) pdmR3DevHlp_IOPortRegisterR0(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts, RTR0PTR pvUser,
180 const char *pszOut, const char *pszIn,
181 const char *pszOutStr, const char *pszInStr, const char *pszDesc)
182{
183 PDMDEV_ASSERT_DEVINS(pDevIns);
184 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
185 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,
186 Port, cPorts, pvUser, pszOut, pszOut, pszIn, pszIn, pszOutStr, pszOutStr, pszInStr, pszInStr, pszDesc, pszDesc));
187
188 /*
189 * Resolve the functions (one of the can be NULL).
190 */
191 int rc = VINF_SUCCESS;
192 if ( pDevIns->pReg->szR0Mod[0]
193 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0))
194 {
195 R0PTRTYPE(PFNIOMIOPORTIN) pfnR0PtrIn = 0;
196 if (pszIn)
197 {
198 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszIn, &pfnR0PtrIn);
199 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszIn)\n", pDevIns->pReg->szR0Mod, pszIn));
200 }
201 R0PTRTYPE(PFNIOMIOPORTOUT) pfnR0PtrOut = 0;
202 if (pszOut && RT_SUCCESS(rc))
203 {
204 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszOut, &pfnR0PtrOut);
205 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOut)\n", pDevIns->pReg->szR0Mod, pszOut));
206 }
207 R0PTRTYPE(PFNIOMIOPORTINSTRING) pfnR0PtrInStr = 0;
208 if (pszInStr && RT_SUCCESS(rc))
209 {
210 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszInStr, &pfnR0PtrInStr);
211 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszInStr)\n", pDevIns->pReg->szR0Mod, pszInStr));
212 }
213 R0PTRTYPE(PFNIOMIOPORTOUTSTRING) pfnR0PtrOutStr = 0;
214 if (pszOutStr && RT_SUCCESS(rc))
215 {
216 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszOutStr, &pfnR0PtrOutStr);
217 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszOutStr)\n", pDevIns->pReg->szR0Mod, pszOutStr));
218 }
219
220 if (RT_SUCCESS(rc))
221 {
222#if 0 /** @todo needs a real string cache for this */
223 if (pDevIns->iInstance > 0)
224 {
225 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
226 if (pszDesc2)
227 pszDesc = pszDesc2;
228 }
229#endif
230
231 rc = IOMR3IOPortRegisterR0(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts, pvUser, pfnR0PtrOut, pfnR0PtrIn, pfnR0PtrOutStr, pfnR0PtrInStr, pszDesc);
232 }
233 }
234 else
235 {
236 AssertMsgFailed(("No R0 module for this driver!\n"));
237 rc = VERR_INVALID_PARAMETER;
238 }
239
240 LogFlow(("pdmR3DevHlp_IOPortRegisterR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
241 return rc;
242}
243
244
245/** @interface_method_impl{PDMDEVHLPR3,pfnIOPortDeregister} */
246static DECLCALLBACK(int) pdmR3DevHlp_IOPortDeregister(PPDMDEVINS pDevIns, RTIOPORT Port, RTUINT cPorts)
247{
248 PDMDEV_ASSERT_DEVINS(pDevIns);
249 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
250 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: Port=%#x cPorts=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance,
251 Port, cPorts));
252
253 int rc = IOMR3IOPortDeregister(pDevIns->Internal.s.pVMR3, pDevIns, Port, cPorts);
254
255 LogFlow(("pdmR3DevHlp_IOPortDeregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
256 return rc;
257}
258
259
260/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegister} */
261static DECLCALLBACK(int) pdmR3DevHlp_MMIORegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTHCPTR pvUser,
262 PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, PFNIOMMMIOFILL pfnFill,
263 const char *pszDesc)
264{
265 PDMDEV_ASSERT_DEVINS(pDevIns);
266 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
267 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x pvUser=%p pfnWrite=%p pfnRead=%p pfnFill=%p pszDesc=%p:{%s}\n",
268 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill, pszDesc, pszDesc));
269
270/** @todo IOMR3MMIORegisterR3 mangles the description, move it here. */
271 int rc = IOMR3MMIORegisterR3(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill, pszDesc);
272
273 LogFlow(("pdmR3DevHlp_MMIORegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
274 return rc;
275}
276
277
278/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegisterRC} */
279static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterRC(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTGCPTR pvUser,
280 const char *pszWrite, const char *pszRead, const char *pszFill,
281 const char *pszDesc)
282{
283 PDMDEV_ASSERT_DEVINS(pDevIns);
284 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
285 LogFlow(("pdmR3DevHlp_MMIORegisterRC: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
286 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
287
288/** @todo pszDesc is unused here, drop it. */
289
290 /*
291 * Resolve the functions.
292 * Not all function have to present, leave it to IOM to enforce this.
293 */
294 int rc = VINF_SUCCESS;
295 if ( pDevIns->pReg->szRCMod[0]
296 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC))
297 {
298 RTRCPTR RCPtrWrite = NIL_RTRCPTR;
299 if (pszWrite)
300 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pszWrite, &RCPtrWrite);
301
302 RTRCPTR RCPtrRead = NIL_RTRCPTR;
303 int rc2 = VINF_SUCCESS;
304 if (pszRead)
305 rc2 = pdmR3DevGetSymbolRCLazy(pDevIns, pszRead, &RCPtrRead);
306
307 RTRCPTR RCPtrFill = NIL_RTRCPTR;
308 int rc3 = VINF_SUCCESS;
309 if (pszFill)
310 rc3 = pdmR3DevGetSymbolRCLazy(pDevIns, pszFill, &RCPtrFill);
311
312 if (RT_SUCCESS(rc) && RT_SUCCESS(rc2) && RT_SUCCESS(rc3))
313 rc = IOMR3MMIORegisterRC(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvUser, RCPtrWrite, RCPtrRead, RCPtrFill);
314 else
315 {
316 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pReg->szRCMod, pszWrite));
317 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pReg->szRCMod, pszRead));
318 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pReg->szRCMod, pszFill));
319 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
320 rc = rc2;
321 if (RT_FAILURE(rc3) && RT_SUCCESS(rc))
322 rc = rc3;
323 }
324 }
325 else
326 {
327 AssertMsgFailed(("No GC module for this driver!\n"));
328 rc = VERR_INVALID_PARAMETER;
329 }
330
331 LogFlow(("pdmR3DevHlp_MMIORegisterRC: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
332 return rc;
333}
334
335/** @interface_method_impl{PDMDEVHLPR3,pfnMMIORegisterR0} */
336static DECLCALLBACK(int) pdmR3DevHlp_MMIORegisterR0(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, RTR0PTR pvUser,
337 const char *pszWrite, const char *pszRead, const char *pszFill,
338 const char *pszDesc)
339{
340 PDMDEV_ASSERT_DEVINS(pDevIns);
341 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
342 LogFlow(("pdmR3DevHlp_MMIORegisterHC: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x pvUser=%p pszWrite=%p:{%s} pszRead=%p:{%s} pszFill=%p:{%s}\n",
343 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvUser, pszWrite, pszWrite, pszRead, pszRead, pszFill, pszFill));
344
345/** @todo pszDesc is unused here, remove it. */
346
347 /*
348 * Resolve the functions.
349 * Not all function have to present, leave it to IOM to enforce this.
350 */
351 int rc = VINF_SUCCESS;
352 if ( pDevIns->pReg->szR0Mod[0]
353 && (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0))
354 {
355 R0PTRTYPE(PFNIOMMMIOWRITE) pfnR0PtrWrite = 0;
356 if (pszWrite)
357 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pszWrite, &pfnR0PtrWrite);
358 R0PTRTYPE(PFNIOMMMIOREAD) pfnR0PtrRead = 0;
359 int rc2 = VINF_SUCCESS;
360 if (pszRead)
361 rc2 = pdmR3DevGetSymbolR0Lazy(pDevIns, pszRead, &pfnR0PtrRead);
362 R0PTRTYPE(PFNIOMMMIOFILL) pfnR0PtrFill = 0;
363 int rc3 = VINF_SUCCESS;
364 if (pszFill)
365 rc3 = pdmR3DevGetSymbolR0Lazy(pDevIns, pszFill, &pfnR0PtrFill);
366 if (RT_SUCCESS(rc) && RT_SUCCESS(rc2) && RT_SUCCESS(rc3))
367 rc = IOMR3MMIORegisterR0(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvUser, pfnR0PtrWrite, pfnR0PtrRead, pfnR0PtrFill);
368 else
369 {
370 AssertMsgRC(rc, ("Failed to resolve %s.%s (pszWrite)\n", pDevIns->pReg->szR0Mod, pszWrite));
371 AssertMsgRC(rc2, ("Failed to resolve %s.%s (pszRead)\n", pDevIns->pReg->szR0Mod, pszRead));
372 AssertMsgRC(rc3, ("Failed to resolve %s.%s (pszFill)\n", pDevIns->pReg->szR0Mod, pszFill));
373 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
374 rc = rc2;
375 if (RT_FAILURE(rc3) && RT_SUCCESS(rc))
376 rc = rc3;
377 }
378 }
379 else
380 {
381 AssertMsgFailed(("No R0 module for this driver!\n"));
382 rc = VERR_INVALID_PARAMETER;
383 }
384
385 LogFlow(("pdmR3DevHlp_MMIORegisterR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
386 return rc;
387}
388
389
390/** @interface_method_impl{PDMDEVHLPR3,pfnMMIODeregister} */
391static DECLCALLBACK(int) pdmR3DevHlp_MMIODeregister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange)
392{
393 PDMDEV_ASSERT_DEVINS(pDevIns);
394 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
395 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x\n",
396 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange));
397
398 int rc = IOMR3MMIODeregister(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange);
399
400 LogFlow(("pdmR3DevHlp_MMIODeregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
401 return rc;
402}
403
404
405/**
406 * @copydoc PDMDEVHLPR3::pfnMMIO2Register
407 */
408static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc)
409{
410 PDMDEV_ASSERT_DEVINS(pDevIns);
411 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
412 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: iRegion=%#x cb=%#RGp fFlags=%RX32 ppv=%p pszDescp=%p:{%s}\n",
413 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, cb, fFlags, ppv, pszDesc, pszDesc));
414
415/** @todo PGMR3PhysMMIO2Register mangles the description, move it here and
416 * use a real string cache. */
417 int rc = PGMR3PhysMMIO2Register(pDevIns->Internal.s.pVMR3, pDevIns, iRegion, cb, fFlags, ppv, pszDesc);
418
419 LogFlow(("pdmR3DevHlp_MMIO2Register: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
420 return rc;
421}
422
423
424/**
425 * @copydoc PDMDEVHLPR3::pfnMMIO2Deregister
426 */
427static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion)
428{
429 PDMDEV_ASSERT_DEVINS(pDevIns);
430 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
431 LogFlow(("pdmR3DevHlp_MMIO2Deregister: caller='%s'/%d: iRegion=%#x\n",
432 pDevIns->pReg->szName, pDevIns->iInstance, iRegion));
433
434 AssertReturn(iRegion == UINT32_MAX, VERR_INVALID_PARAMETER);
435
436 int rc = PGMR3PhysMMIO2Deregister(pDevIns->Internal.s.pVMR3, pDevIns, iRegion);
437
438 LogFlow(("pdmR3DevHlp_MMIO2Deregister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
439 return rc;
440}
441
442
443/**
444 * @copydoc PDMDEVHLPR3::pfnMMIO2Map
445 */
446static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
447{
448 PDMDEV_ASSERT_DEVINS(pDevIns);
449 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
450 LogFlow(("pdmR3DevHlp_MMIO2Map: caller='%s'/%d: iRegion=%#x GCPhys=%#RGp\n",
451 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, GCPhys));
452
453 int rc = PGMR3PhysMMIO2Map(pDevIns->Internal.s.pVMR3, pDevIns, iRegion, GCPhys);
454
455 LogFlow(("pdmR3DevHlp_MMIO2Map: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
456 return rc;
457}
458
459
460/**
461 * @copydoc PDMDEVHLPR3::pfnMMIO2Unmap
462 */
463static DECLCALLBACK(int) pdmR3DevHlp_MMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
464{
465 PDMDEV_ASSERT_DEVINS(pDevIns);
466 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
467 LogFlow(("pdmR3DevHlp_MMIO2Unmap: caller='%s'/%d: iRegion=%#x GCPhys=%#RGp\n",
468 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, GCPhys));
469
470 int rc = PGMR3PhysMMIO2Unmap(pDevIns->Internal.s.pVMR3, pDevIns, iRegion, GCPhys);
471
472 LogFlow(("pdmR3DevHlp_MMIO2Unmap: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
473 return rc;
474}
475
476
477/**
478 * @copydoc PDMDEVHLPR3::pfnMMHyperMapMMIO2
479 */
480static DECLCALLBACK(int) pdmR3DevHlp_MMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
481 const char *pszDesc, PRTRCPTR pRCPtr)
482{
483 PDMDEV_ASSERT_DEVINS(pDevIns);
484 PVM pVM = pDevIns->Internal.s.pVMR3;
485 VM_ASSERT_EMT(pVM);
486 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: iRegion=%#x off=%RGp cb=%RGp pszDesc=%p:{%s} pRCPtr=%p\n",
487 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, off, cb, pszDesc, pszDesc, pRCPtr));
488
489 if (pDevIns->iInstance > 0)
490 {
491 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
492 if (pszDesc2)
493 pszDesc = pszDesc2;
494 }
495
496 int rc = MMR3HyperMapMMIO2(pVM, pDevIns, iRegion, off, cb, pszDesc, pRCPtr);
497
498 LogFlow(("pdmR3DevHlp_MMHyperMapMMIO2: caller='%s'/%d: returns %Rrc *pRCPtr=%RRv\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pRCPtr));
499 return rc;
500}
501
502
503/**
504 * @copydoc PDMDEVHLPR3::pfnMMIO2MapKernel
505 */
506static DECLCALLBACK(int) pdmR3DevHlp_MMIO2MapKernel(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
507 const char *pszDesc, PRTR0PTR pR0Ptr)
508{
509 PDMDEV_ASSERT_DEVINS(pDevIns);
510 PVM pVM = pDevIns->Internal.s.pVMR3;
511 VM_ASSERT_EMT(pVM);
512 LogFlow(("pdmR3DevHlp_MMIO2MapKernel: caller='%s'/%d: iRegion=%#x off=%RGp cb=%RGp pszDesc=%p:{%s} pR0Ptr=%p\n",
513 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, off, cb, pszDesc, pszDesc, pR0Ptr));
514
515 if (pDevIns->iInstance > 0)
516 {
517 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
518 if (pszDesc2)
519 pszDesc = pszDesc2;
520 }
521
522 int rc = PGMR3PhysMMIO2MapKernel(pVM, pDevIns, iRegion, off, cb, pszDesc, pR0Ptr);
523
524 LogFlow(("pdmR3DevHlp_MMIO2MapKernel: caller='%s'/%d: returns %Rrc *pR0Ptr=%RHv\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pR0Ptr));
525 return rc;
526}
527
528
529/** @interface_method_impl{PDMDEVHLPR3,pfnROMRegister} */
530static DECLCALLBACK(int) pdmR3DevHlp_ROMRegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange,
531 const void *pvBinary, uint32_t cbBinary, uint32_t fFlags, const char *pszDesc)
532{
533 PDMDEV_ASSERT_DEVINS(pDevIns);
534 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
535 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x pvBinary=%p cbBinary=%#x fFlags=%#RX32 pszDesc=%p:{%s}\n",
536 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, pvBinary, cbBinary, fFlags, pszDesc, pszDesc));
537
538/** @todo can we mangle pszDesc? */
539 int rc = PGMR3PhysRomRegister(pDevIns->Internal.s.pVMR3, pDevIns, GCPhysStart, cbRange, pvBinary, cbBinary, fFlags, pszDesc);
540
541 LogFlow(("pdmR3DevHlp_ROMRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
542 return rc;
543}
544
545
546/** @interface_method_impl{PDMDEVHLPR3,pfnROMProtectShadow} */
547static DECLCALLBACK(int) pdmR3DevHlp_ROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, RTUINT cbRange, PGMROMPROT enmProt)
548{
549 PDMDEV_ASSERT_DEVINS(pDevIns);
550 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: GCPhysStart=%RGp cbRange=%#x enmProt=%d\n",
551 pDevIns->pReg->szName, pDevIns->iInstance, GCPhysStart, cbRange, enmProt));
552
553 int rc = PGMR3PhysRomProtect(pDevIns->Internal.s.pVMR3, GCPhysStart, cbRange, enmProt);
554
555 LogFlow(("pdmR3DevHlp_ROMProtectShadow: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
556 return rc;
557}
558
559
560/** @interface_method_impl{PDMDEVHLPR3,pfnSSMRegister} */
561static DECLCALLBACK(int) pdmR3DevHlp_SSMRegister(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess, const char *pszBefore,
562 PFNSSMDEVLIVEPREP pfnLivePrep, PFNSSMDEVLIVEEXEC pfnLiveExec, PFNSSMDEVLIVEVOTE pfnLiveVote,
563 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
564 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone)
565{
566 PDMDEV_ASSERT_DEVINS(pDevIns);
567 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
568 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: uVersion=#x cbGuess=%#x pszBefore=%p:{%s}\n"
569 " pfnLivePrep=%p pfnLiveExec=%p pfnLiveVote=%p pfnSavePrep=%p pfnSaveExec=%p pfnSaveDone=%p pszLoadPrep=%p pfnLoadExec=%p pfnLoadDone=%p\n",
570 pDevIns->pReg->szName, pDevIns->iInstance, uVersion, cbGuess, pszBefore, pszBefore,
571 pfnLivePrep, pfnLiveExec, pfnLiveVote,
572 pfnSavePrep, pfnSaveExec, pfnSaveDone,
573 pfnLoadPrep, pfnLoadExec, pfnLoadDone));
574
575 int rc = SSMR3RegisterDevice(pDevIns->Internal.s.pVMR3, pDevIns, pDevIns->pReg->szName, pDevIns->iInstance,
576 uVersion, cbGuess, pszBefore,
577 pfnLivePrep, pfnLiveExec, pfnLiveVote,
578 pfnSavePrep, pfnSaveExec, pfnSaveDone,
579 pfnLoadPrep, pfnLoadExec, pfnLoadDone);
580
581 LogFlow(("pdmR3DevHlp_SSMRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
582 return rc;
583}
584
585
586/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimerCreate} */
587static DECLCALLBACK(int) pdmR3DevHlp_TMTimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback, void *pvUser, uint32_t fFlags, const char *pszDesc, PPTMTIMERR3 ppTimer)
588{
589 PDMDEV_ASSERT_DEVINS(pDevIns);
590 PVM pVM = pDevIns->Internal.s.pVMR3;
591 VM_ASSERT_EMT(pVM);
592 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: enmClock=%d pfnCallback=%p pvUser=%p fFlags=%#x pszDesc=%p:{%s} ppTimer=%p\n",
593 pDevIns->pReg->szName, pDevIns->iInstance, enmClock, pfnCallback, pvUser, fFlags, pszDesc, pszDesc, ppTimer));
594
595 if (pDevIns->iInstance > 0) /** @todo use a string cache here later. */
596 {
597 char *pszDesc2 = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s [%u]", pszDesc, pDevIns->iInstance);
598 if (pszDesc2)
599 pszDesc = pszDesc2;
600 }
601
602 int rc = TMR3TimerCreateDevice(pVM, pDevIns, enmClock, pfnCallback, pvUser, fFlags, pszDesc, ppTimer);
603
604 LogFlow(("pdmR3DevHlp_TMTimerCreate: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
605 return rc;
606}
607
608
609/** @interface_method_impl{PDMDEVHLPR3,pfnTMUtcNow} */
610static DECLCALLBACK(PRTTIMESPEC) pdmR3DevHlp_TMUtcNow(PPDMDEVINS pDevIns, PRTTIMESPEC pTime)
611{
612 PDMDEV_ASSERT_DEVINS(pDevIns);
613 LogFlow(("pdmR3DevHlp_TMUtcNow: caller='%s'/%d: pTime=%p\n",
614 pDevIns->pReg->szName, pDevIns->iInstance, pTime));
615
616 pTime = TMR3UtcNow(pDevIns->Internal.s.pVMR3, pTime);
617
618 LogFlow(("pdmR3DevHlp_TMUtcNow: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, RTTimeSpecGetNano(pTime)));
619 return pTime;
620}
621
622
623/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGet} */
624static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGet(PPDMDEVINS pDevIns)
625{
626 PDMDEV_ASSERT_DEVINS(pDevIns);
627 LogFlow(("pdmR3DevHlp_TMTimeVirtGet: caller='%s'\n",
628 pDevIns->pReg->szName, pDevIns->iInstance));
629
630 uint64_t u64Time = TMVirtualSyncGet(pDevIns->Internal.s.pVMR3);
631
632 LogFlow(("pdmR3DevHlp_TMTimeVirtGet: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Time));
633 return u64Time;
634}
635
636
637/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGetFreq} */
638static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGetFreq(PPDMDEVINS pDevIns)
639{
640 PDMDEV_ASSERT_DEVINS(pDevIns);
641 LogFlow(("pdmR3DevHlp_TMTimeVirtGetFreq: caller='%s'\n",
642 pDevIns->pReg->szName, pDevIns->iInstance));
643
644 uint64_t u64Freq = TMVirtualGetFreq(pDevIns->Internal.s.pVMR3);
645
646 LogFlow(("pdmR3DevHlp_TMTimeVirtGetFreq: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Freq));
647 return u64Freq;
648}
649
650
651/** @interface_method_impl{PDMDEVHLPR3,pfnTMTimeVirtGetNano} */
652static DECLCALLBACK(uint64_t) pdmR3DevHlp_TMTimeVirtGetNano(PPDMDEVINS pDevIns)
653{
654 PDMDEV_ASSERT_DEVINS(pDevIns);
655 LogFlow(("pdmR3DevHlp_TMTimeVirtGetNano: caller='%s'\n",
656 pDevIns->pReg->szName, pDevIns->iInstance));
657
658 uint64_t u64Time = TMVirtualSyncGet(pDevIns->Internal.s.pVMR3);
659 uint64_t u64Nano = TMVirtualToNano(pDevIns->Internal.s.pVMR3, u64Time);
660
661 LogFlow(("pdmR3DevHlp_TMTimeVirtGetNano: caller='%s'/%d: returns %RU64\n", pDevIns->pReg->szName, pDevIns->iInstance, u64Nano));
662 return u64Nano;
663}
664
665
666/** @interface_method_impl{PDMDEVHLPR3,pfnPhysRead} */
667static DECLCALLBACK(int) pdmR3DevHlp_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
668{
669 PDMDEV_ASSERT_DEVINS(pDevIns);
670 PVM pVM = pDevIns->Internal.s.pVMR3;
671 LogFlow(("pdmR3DevHlp_PhysRead: caller='%s'/%d: GCPhys=%RGp pvBuf=%p cbRead=%#x\n",
672 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, pvBuf, cbRead));
673
674#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
675 if (!VM_IS_EMT(pVM))
676 {
677 char szNames[128];
678 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
679 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
680 }
681#endif
682
683 int rc;
684 if (VM_IS_EMT(pVM))
685 rc = PGMPhysRead(pVM, GCPhys, pvBuf, cbRead);
686 else
687 rc = PGMR3PhysReadExternal(pVM, GCPhys, pvBuf, cbRead);
688
689 Log(("pdmR3DevHlp_PhysRead: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
690 return rc;
691}
692
693
694/** @interface_method_impl{PDMDEVHLPR3,pfnPhysWrite} */
695static DECLCALLBACK(int) pdmR3DevHlp_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
696{
697 PDMDEV_ASSERT_DEVINS(pDevIns);
698 PVM pVM = pDevIns->Internal.s.pVMR3;
699 LogFlow(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: GCPhys=%RGp pvBuf=%p cbWrite=%#x\n",
700 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, pvBuf, cbWrite));
701
702#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
703 if (!VM_IS_EMT(pVM))
704 {
705 char szNames[128];
706 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
707 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
708 }
709#endif
710
711 int rc;
712 if (VM_IS_EMT(pVM))
713 rc = PGMPhysWrite(pVM, GCPhys, pvBuf, cbWrite);
714 else
715 rc = PGMR3PhysWriteExternal(pVM, GCPhys, pvBuf, cbWrite, pDevIns->pReg->szName);
716
717 Log(("pdmR3DevHlp_PhysWrite: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
718 return rc;
719}
720
721
722/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPhys2CCPtr} */
723static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPhys2CCPtr(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, void **ppv, PPGMPAGEMAPLOCK pLock)
724{
725 PDMDEV_ASSERT_DEVINS(pDevIns);
726 PVM pVM = pDevIns->Internal.s.pVMR3;
727 LogFlow(("pdmR3DevHlp_PhysGCPhys2CCPtr: caller='%s'/%d: GCPhys=%RGp fFlags=%#x ppv=%p pLock=%p\n",
728 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, fFlags, ppv, pLock));
729 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
730
731#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
732 if (!VM_IS_EMT(pVM))
733 {
734 char szNames[128];
735 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
736 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
737 }
738#endif
739
740 int rc = PGMR3PhysGCPhys2CCPtrExternal(pVM, GCPhys, ppv, pLock);
741
742 Log(("pdmR3DevHlp_PhysGCPhys2CCPtr: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
743 return rc;
744}
745
746
747/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPhys2CCPtrReadOnly} */
748static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, const void **ppv, PPGMPAGEMAPLOCK pLock)
749{
750 PDMDEV_ASSERT_DEVINS(pDevIns);
751 PVM pVM = pDevIns->Internal.s.pVMR3;
752 LogFlow(("pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly: caller='%s'/%d: GCPhys=%RGp fFlags=%#x ppv=%p pLock=%p\n",
753 pDevIns->pReg->szName, pDevIns->iInstance, GCPhys, fFlags, ppv, pLock));
754 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
755
756#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
757 if (!VM_IS_EMT(pVM))
758 {
759 char szNames[128];
760 uint32_t cLocks = PDMR3CritSectCountOwned(pVM, szNames, sizeof(szNames));
761 AssertMsg(cLocks == 0, ("cLocks=%u %s\n", cLocks, szNames));
762 }
763#endif
764
765 int rc = PGMR3PhysGCPhys2CCPtrReadOnlyExternal(pVM, GCPhys, ppv, pLock);
766
767 Log(("pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
768 return rc;
769}
770
771
772/** @interface_method_impl{PDMDEVHLPR3,pfnPhysReleasePageMappingLock} */
773static DECLCALLBACK(void) pdmR3DevHlp_PhysReleasePageMappingLock(PPDMDEVINS pDevIns, PPGMPAGEMAPLOCK pLock)
774{
775 PDMDEV_ASSERT_DEVINS(pDevIns);
776 PVM pVM = pDevIns->Internal.s.pVMR3;
777 LogFlow(("pdmR3DevHlp_PhysReleasePageMappingLock: caller='%s'/%d: pLock=%p\n",
778 pDevIns->pReg->szName, pDevIns->iInstance, pLock));
779
780 PGMPhysReleasePageMappingLock(pVM, pLock);
781
782 Log(("pdmR3DevHlp_PhysReleasePageMappingLock: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
783}
784
785
786/** @interface_method_impl{PDMDEVHLPR3,pfnPhysReadGCVirt} */
787static DECLCALLBACK(int) pdmR3DevHlp_PhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb)
788{
789 PDMDEV_ASSERT_DEVINS(pDevIns);
790 PVM pVM = pDevIns->Internal.s.pVMR3;
791 VM_ASSERT_EMT(pVM);
792 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: pvDst=%p GCVirt=%RGv cb=%#x\n",
793 pDevIns->pReg->szName, pDevIns->iInstance, pvDst, GCVirtSrc, cb));
794
795 PVMCPU pVCpu = VMMGetCpu(pVM);
796 if (!pVCpu)
797 return VERR_ACCESS_DENIED;
798#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
799 /** @todo SMP. */
800#endif
801
802 int rc = PGMPhysSimpleReadGCPtr(pVCpu, pvDst, GCVirtSrc, cb);
803
804 LogFlow(("pdmR3DevHlp_PhysReadGCVirt: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
805
806 return rc;
807}
808
809
810/** @interface_method_impl{PDMDEVHLPR3,pfnPhysWriteGCVirt} */
811static DECLCALLBACK(int) pdmR3DevHlp_PhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb)
812{
813 PDMDEV_ASSERT_DEVINS(pDevIns);
814 PVM pVM = pDevIns->Internal.s.pVMR3;
815 VM_ASSERT_EMT(pVM);
816 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: GCVirtDst=%RGv pvSrc=%p cb=%#x\n",
817 pDevIns->pReg->szName, pDevIns->iInstance, GCVirtDst, pvSrc, cb));
818
819 PVMCPU pVCpu = VMMGetCpu(pVM);
820 if (!pVCpu)
821 return VERR_ACCESS_DENIED;
822#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
823 /** @todo SMP. */
824#endif
825
826 int rc = PGMPhysSimpleWriteGCPtr(pVCpu, GCVirtDst, pvSrc, cb);
827
828 LogFlow(("pdmR3DevHlp_PhysWriteGCVirt: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
829
830 return rc;
831}
832
833
834/** @interface_method_impl{PDMDEVHLPR3,pfnPhysGCPtr2GCPhys} */
835static DECLCALLBACK(int) pdmR3DevHlp_PhysGCPtr2GCPhys(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys)
836{
837 PDMDEV_ASSERT_DEVINS(pDevIns);
838 PVM pVM = pDevIns->Internal.s.pVMR3;
839 VM_ASSERT_EMT(pVM);
840 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: GCPtr=%RGv pGCPhys=%p\n",
841 pDevIns->pReg->szName, pDevIns->iInstance, GCPtr, pGCPhys));
842
843 PVMCPU pVCpu = VMMGetCpu(pVM);
844 if (!pVCpu)
845 return VERR_ACCESS_DENIED;
846#if defined(VBOX_STRICT) && defined(PDM_DEVHLP_DEADLOCK_DETECTION)
847 /** @todo SMP. */
848#endif
849
850 int rc = PGMPhysGCPtr2GCPhys(pVCpu, GCPtr, pGCPhys);
851
852 LogFlow(("pdmR3DevHlp_PhysGCPtr2GCPhys: caller='%s'/%d: returns %Rrc *pGCPhys=%RGp\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *pGCPhys));
853
854 return rc;
855}
856
857
858/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapAlloc} */
859static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAlloc(PPDMDEVINS pDevIns, size_t cb)
860{
861 PDMDEV_ASSERT_DEVINS(pDevIns);
862 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: cb=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cb));
863
864 void *pv = MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE_USER, cb);
865
866 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
867 return pv;
868}
869
870
871/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapAllocZ} */
872static DECLCALLBACK(void *) pdmR3DevHlp_MMHeapAllocZ(PPDMDEVINS pDevIns, size_t cb)
873{
874 PDMDEV_ASSERT_DEVINS(pDevIns);
875 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: cb=%#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cb));
876
877 void *pv = MMR3HeapAllocZ(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE_USER, cb);
878
879 LogFlow(("pdmR3DevHlp_MMHeapAllocZ: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
880 return pv;
881}
882
883
884/** @interface_method_impl{PDMDEVHLPR3,pfnMMHeapFree} */
885static DECLCALLBACK(void) pdmR3DevHlp_MMHeapFree(PPDMDEVINS pDevIns, void *pv)
886{
887 PDMDEV_ASSERT_DEVINS(pDevIns);
888 LogFlow(("pdmR3DevHlp_MMHeapFree: caller='%s'/%d: pv=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pv));
889
890 MMR3HeapFree(pv);
891
892 LogFlow(("pdmR3DevHlp_MMHeapAlloc: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
893}
894
895
896/** @interface_method_impl{PDMDEVHLPR3,pfnVMState} */
897static DECLCALLBACK(VMSTATE) pdmR3DevHlp_VMState(PPDMDEVINS pDevIns)
898{
899 PDMDEV_ASSERT_DEVINS(pDevIns);
900
901 VMSTATE enmVMState = VMR3GetState(pDevIns->Internal.s.pVMR3);
902
903 LogFlow(("pdmR3DevHlp_VMState: caller='%s'/%d: returns %d (%s)\n", pDevIns->pReg->szName, pDevIns->iInstance,
904 enmVMState, VMR3GetStateName(enmVMState)));
905 return enmVMState;
906}
907
908
909/** @interface_method_impl{PDMDEVHLPR3,pfnVMTeleportedAndNotFullyResumedYet} */
910static DECLCALLBACK(bool) pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet(PPDMDEVINS pDevIns)
911{
912 PDMDEV_ASSERT_DEVINS(pDevIns);
913
914 bool fRc = VMR3TeleportedAndNotFullyResumedYet(pDevIns->Internal.s.pVMR3);
915
916 LogFlow(("pdmR3DevHlp_VMState: caller='%s'/%d: returns %RTbool\n", pDevIns->pReg->szName, pDevIns->iInstance,
917 fRc));
918 return fRc;
919}
920
921
922/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetError} */
923static DECLCALLBACK(int) pdmR3DevHlp_VMSetError(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
924{
925 PDMDEV_ASSERT_DEVINS(pDevIns);
926 va_list args;
927 va_start(args, pszFormat);
928 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMR3, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc2 == rc); NOREF(rc2);
929 va_end(args);
930 return rc;
931}
932
933
934/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetErrorV} */
935static DECLCALLBACK(int) pdmR3DevHlp_VMSetErrorV(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
936{
937 PDMDEV_ASSERT_DEVINS(pDevIns);
938 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMR3, rc, RT_SRC_POS_ARGS, pszFormat, va); Assert(rc2 == rc); NOREF(rc2);
939 return rc;
940}
941
942
943/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetRuntimeError} */
944static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeError(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
945{
946 PDMDEV_ASSERT_DEVINS(pDevIns);
947 va_list args;
948 va_start(args, pszFormat);
949 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMR3, fFlags, pszErrorId, pszFormat, args);
950 va_end(args);
951 return rc;
952}
953
954
955/** @interface_method_impl{PDMDEVHLPR3,pfnVMSetRuntimeErrorV} */
956static DECLCALLBACK(int) pdmR3DevHlp_VMSetRuntimeErrorV(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
957{
958 PDMDEV_ASSERT_DEVINS(pDevIns);
959 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMR3, fFlags, pszErrorId, pszFormat, va);
960 return rc;
961}
962
963
964/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFStopV} */
965static DECLCALLBACK(int) pdmR3DevHlp_DBGFStopV(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction, const char *pszFormat, va_list args)
966{
967 PDMDEV_ASSERT_DEVINS(pDevIns);
968#ifdef LOG_ENABLED
969 va_list va2;
970 va_copy(va2, args);
971 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: pszFile=%p:{%s} iLine=%d pszFunction=%p:{%s} pszFormat=%p:{%s} (%N)\n",
972 pDevIns->pReg->szName, pDevIns->iInstance, pszFile, pszFile, iLine, pszFunction, pszFunction, pszFormat, pszFormat, pszFormat, &va2));
973 va_end(va2);
974#endif
975
976 PVM pVM = pDevIns->Internal.s.pVMR3;
977 VM_ASSERT_EMT(pVM);
978 int rc = DBGFR3EventSrcV(pVM, DBGFEVENT_DEV_STOP, pszFile, iLine, pszFunction, pszFormat, args);
979 if (rc == VERR_DBGF_NOT_ATTACHED)
980 rc = VINF_SUCCESS;
981
982 LogFlow(("pdmR3DevHlp_DBGFStopV: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
983 return rc;
984}
985
986
987/** @interface_method_impl{PDMDEVHLPR3,pfnDBGFInfoRegister} */
988static DECLCALLBACK(int) pdmR3DevHlp_DBGFInfoRegister(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler)
989{
990 PDMDEV_ASSERT_DEVINS(pDevIns);
991 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: pszName=%p:{%s} pszDesc=%p:{%s} pfnHandler=%p\n",
992 pDevIns->pReg->szName, pDevIns->iInstance, pszName, pszName, pszDesc, pszDesc, pfnHandler));
993
994 PVM pVM = pDevIns->Internal.s.pVMR3;
995 VM_ASSERT_EMT(pVM);
996 int rc = DBGFR3InfoRegisterDevice(pVM, pszName, pszDesc, pfnHandler, pDevIns);
997
998 LogFlow(("pdmR3DevHlp_DBGFInfoRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
999 return rc;
1000}
1001
1002
1003/** @interface_method_impl{PDMDEVHLPR3,pfnSTAMRegister} */
1004static DECLCALLBACK(void) pdmR3DevHlp_STAMRegister(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName, STAMUNIT enmUnit, const char *pszDesc)
1005{
1006 PDMDEV_ASSERT_DEVINS(pDevIns);
1007 PVM pVM = pDevIns->Internal.s.pVMR3;
1008 VM_ASSERT_EMT(pVM);
1009
1010 STAM_REG(pVM, pvSample, enmType, pszName, enmUnit, pszDesc);
1011 NOREF(pVM);
1012}
1013
1014
1015
1016/** @interface_method_impl{PDMDEVHLPR3,pfnSTAMRegisterF} */
1017static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterF(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility,
1018 STAMUNIT enmUnit, const char *pszDesc, const char *pszName, ...)
1019{
1020 PDMDEV_ASSERT_DEVINS(pDevIns);
1021 PVM pVM = pDevIns->Internal.s.pVMR3;
1022 VM_ASSERT_EMT(pVM);
1023
1024 va_list args;
1025 va_start(args, pszName);
1026 int rc = STAMR3RegisterV(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
1027 va_end(args);
1028 AssertRC(rc);
1029
1030 NOREF(pVM);
1031}
1032
1033
1034/** @interface_method_impl{PDMDEVHLPR3,pfnSTAMRegisterV} */
1035static DECLCALLBACK(void) pdmR3DevHlp_STAMRegisterV(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, STAMVISIBILITY enmVisibility,
1036 STAMUNIT enmUnit, const char *pszDesc, const char *pszName, va_list args)
1037{
1038 PDMDEV_ASSERT_DEVINS(pDevIns);
1039 PVM pVM = pDevIns->Internal.s.pVMR3;
1040 VM_ASSERT_EMT(pVM);
1041
1042 int rc = STAMR3RegisterV(pVM, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
1043 AssertRC(rc);
1044
1045 NOREF(pVM);
1046}
1047
1048
1049/** @interface_method_impl{PDMDEVHLPR3,pfnPCIRegister} */
1050static DECLCALLBACK(int) pdmR3DevHlp_PCIRegister(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev)
1051{
1052 PDMDEV_ASSERT_DEVINS(pDevIns);
1053 PVM pVM = pDevIns->Internal.s.pVMR3;
1054 VM_ASSERT_EMT(pVM);
1055 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: pPciDev=%p:{.config={%#.256Rhxs}\n",
1056 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pPciDev->config));
1057
1058 /*
1059 * Validate input.
1060 */
1061 if (!pPciDev)
1062 {
1063 Assert(pPciDev);
1064 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc (pPciDev)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1065 return VERR_INVALID_PARAMETER;
1066 }
1067 if (!pPciDev->config[0] && !pPciDev->config[1])
1068 {
1069 Assert(pPciDev->config[0] || pPciDev->config[1]);
1070 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc (vendor)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1071 return VERR_INVALID_PARAMETER;
1072 }
1073 if (pDevIns->Internal.s.pPciDeviceR3)
1074 {
1075 /** @todo the PCI device vs. PDM device designed is a bit flawed if we have to
1076 * support a PDM device with multiple PCI devices. This might become a problem
1077 * when upgrading the chipset for instance because of multiple functions in some
1078 * devices...
1079 */
1080 AssertMsgFailed(("Only one PCI device per device is currently implemented!\n"));
1081 return VERR_INTERNAL_ERROR;
1082 }
1083
1084 /*
1085 * Choose the PCI bus for the device.
1086 *
1087 * This is simple. If the device was configured for a particular bus, the PCIBusNo
1088 * configuration value will be set. If not the default bus is 0.
1089 */
1090 int rc;
1091 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusR3;
1092 if (!pBus)
1093 {
1094 uint8_t u8Bus;
1095 rc = CFGMR3QueryU8Def(pDevIns->Internal.s.pCfgHandle, "PCIBusNo", &u8Bus, 0);
1096 AssertLogRelMsgRCReturn(rc, ("Configuration error: PCIBusNo query failed with rc=%Rrc (%s/%d)\n",
1097 rc, pDevIns->pReg->szName, pDevIns->iInstance), rc);
1098 AssertLogRelMsgReturn(u8Bus < RT_ELEMENTS(pVM->pdm.s.aPciBuses),
1099 ("Configuration error: PCIBusNo=%d, max is %d. (%s/%d)\n", u8Bus,
1100 RT_ELEMENTS(pVM->pdm.s.aPciBuses), pDevIns->pReg->szName, pDevIns->iInstance),
1101 VERR_PDM_NO_PCI_BUS);
1102 pBus = pDevIns->Internal.s.pPciBusR3 = &pVM->pdm.s.aPciBuses[u8Bus];
1103 }
1104 if (pBus->pDevInsR3)
1105 {
1106 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
1107 pDevIns->Internal.s.pPciBusR0 = MMHyperR3ToR0(pVM, pDevIns->Internal.s.pPciBusR3);
1108 else
1109 pDevIns->Internal.s.pPciBusR0 = NIL_RTR0PTR;
1110
1111 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
1112 pDevIns->Internal.s.pPciBusRC = MMHyperR3ToRC(pVM, pDevIns->Internal.s.pPciBusR3);
1113 else
1114 pDevIns->Internal.s.pPciBusRC = NIL_RTRCPTR;
1115
1116 /*
1117 * Check the configuration for PCI device and function assignment.
1118 */
1119 int iDev = -1;
1120 uint8_t u8Device;
1121 rc = CFGMR3QueryU8(pDevIns->Internal.s.pCfgHandle, "PCIDeviceNo", &u8Device);
1122 if (RT_SUCCESS(rc))
1123 {
1124 if (u8Device > 31)
1125 {
1126 AssertMsgFailed(("Configuration error: PCIDeviceNo=%d, max is 31. (%s/%d)\n",
1127 u8Device, pDevIns->pReg->szName, pDevIns->iInstance));
1128 return VERR_INTERNAL_ERROR;
1129 }
1130
1131 uint8_t u8Function;
1132 rc = CFGMR3QueryU8(pDevIns->Internal.s.pCfgHandle, "PCIFunctionNo", &u8Function);
1133 if (RT_FAILURE(rc))
1134 {
1135 AssertMsgFailed(("Configuration error: PCIDeviceNo, but PCIFunctionNo query failed with rc=%Rrc (%s/%d)\n",
1136 rc, pDevIns->pReg->szName, pDevIns->iInstance));
1137 return rc;
1138 }
1139 if (u8Function > 7)
1140 {
1141 AssertMsgFailed(("Configuration error: PCIFunctionNo=%d, max is 7. (%s/%d)\n",
1142 u8Function, pDevIns->pReg->szName, pDevIns->iInstance));
1143 return VERR_INTERNAL_ERROR;
1144 }
1145 iDev = (u8Device << 3) | u8Function;
1146 }
1147 else if (rc != VERR_CFGM_VALUE_NOT_FOUND)
1148 {
1149 AssertMsgFailed(("Configuration error: PCIDeviceNo query failed with rc=%Rrc (%s/%d)\n",
1150 rc, pDevIns->pReg->szName, pDevIns->iInstance));
1151 return rc;
1152 }
1153
1154 /*
1155 * Call the pci bus device to do the actual registration.
1156 */
1157 pdmLock(pVM);
1158 rc = pBus->pfnRegisterR3(pBus->pDevInsR3, pPciDev, pDevIns->pReg->szName, iDev);
1159 pdmUnlock(pVM);
1160 if (RT_SUCCESS(rc))
1161 {
1162 pPciDev->pDevIns = pDevIns;
1163
1164 pDevIns->Internal.s.pPciDeviceR3 = pPciDev;
1165 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
1166 pDevIns->Internal.s.pPciDeviceR0 = MMHyperR3ToR0(pVM, pPciDev);
1167 else
1168 pDevIns->Internal.s.pPciDeviceR0 = NIL_RTR0PTR;
1169
1170 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
1171 pDevIns->Internal.s.pPciDeviceRC = MMHyperR3ToRC(pVM, pPciDev);
1172 else
1173 pDevIns->Internal.s.pPciDeviceRC = NIL_RTRCPTR;
1174
1175 Log(("PDM: Registered device '%s'/%d as PCI device %d on bus %d\n",
1176 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev->devfn, pDevIns->Internal.s.pPciBusR3->iBus));
1177 }
1178 }
1179 else
1180 {
1181 AssertLogRelMsgFailed(("Configuration error: No PCI bus available. This could be related to init order too!\n"));
1182 rc = VERR_PDM_NO_PCI_BUS;
1183 }
1184
1185 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1186 return rc;
1187}
1188
1189
1190/** @interface_method_impl{PDMDEVHLPR3,pfnPCIIORegionRegister} */
1191static DECLCALLBACK(int) pdmR3DevHlp_PCIIORegionRegister(PPDMDEVINS pDevIns, int iRegion, uint32_t cbRegion, PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback)
1192{
1193 PDMDEV_ASSERT_DEVINS(pDevIns);
1194 PVM pVM = pDevIns->Internal.s.pVMR3;
1195 VM_ASSERT_EMT(pVM);
1196 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: iRegion=%d cbRegion=%#x enmType=%d pfnCallback=%p\n",
1197 pDevIns->pReg->szName, pDevIns->iInstance, iRegion, cbRegion, enmType, pfnCallback));
1198
1199 /*
1200 * Validate input.
1201 */
1202 if (iRegion < 0 || iRegion >= PCI_NUM_REGIONS)
1203 {
1204 Assert(iRegion >= 0 && iRegion < PCI_NUM_REGIONS);
1205 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc (iRegion)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1206 return VERR_INVALID_PARAMETER;
1207 }
1208 switch (enmType)
1209 {
1210 case PCI_ADDRESS_SPACE_IO:
1211 /*
1212 * Sanity check: don't allow to register more than 32K of the PCI I/O space.
1213 */
1214 AssertMsgReturn(cbRegion <= _32K,
1215 ("caller='%s'/%d: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cbRegion),
1216 VERR_INVALID_PARAMETER);
1217 break;
1218
1219 case PCI_ADDRESS_SPACE_MEM:
1220 case PCI_ADDRESS_SPACE_MEM_PREFETCH:
1221 /*
1222 * Sanity check: don't allow to register more than 512MB of the PCI MMIO space for
1223 * now. If this limit is increased beyond 2GB, adapt the aligned check below as well!
1224 */
1225 AssertMsgReturn(cbRegion <= 512 * _1M,
1226 ("caller='%s'/%d: %#x\n", pDevIns->pReg->szName, pDevIns->iInstance, cbRegion),
1227 VERR_INVALID_PARAMETER);
1228 break;
1229 default:
1230 AssertMsgFailed(("enmType=%#x is unknown\n", enmType));
1231 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc (enmType)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1232 return VERR_INVALID_PARAMETER;
1233 }
1234 if (!pfnCallback)
1235 {
1236 Assert(pfnCallback);
1237 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc (callback)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1238 return VERR_INVALID_PARAMETER;
1239 }
1240 AssertRelease(VMR3GetState(pVM) != VMSTATE_RUNNING);
1241
1242 /*
1243 * Must have a PCI device registered!
1244 */
1245 int rc;
1246 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceR3;
1247 if (pPciDev)
1248 {
1249 /*
1250 * We're currently restricted to page aligned MMIO regions.
1251 */
1252 if ( (enmType == PCI_ADDRESS_SPACE_MEM || enmType == PCI_ADDRESS_SPACE_MEM_PREFETCH)
1253 && cbRegion != RT_ALIGN_32(cbRegion, PAGE_SIZE))
1254 {
1255 Log(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: aligning cbRegion %#x -> %#x\n",
1256 pDevIns->pReg->szName, pDevIns->iInstance, cbRegion, RT_ALIGN_32(cbRegion, PAGE_SIZE)));
1257 cbRegion = RT_ALIGN_32(cbRegion, PAGE_SIZE);
1258 }
1259
1260 /*
1261 * For registering PCI MMIO memory or PCI I/O memory, the size of the region must be a power of 2!
1262 */
1263 int iLastSet = ASMBitLastSetU32(cbRegion);
1264 Assert(iLastSet > 0);
1265 uint32_t cbRegionAligned = RT_BIT_32(iLastSet - 1);
1266 if (cbRegion > cbRegionAligned)
1267 cbRegion = cbRegionAligned * 2; /* round up */
1268
1269 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusR3;
1270 Assert(pBus);
1271 pdmLock(pVM);
1272 rc = pBus->pfnIORegionRegisterR3(pBus->pDevInsR3, pPciDev, iRegion, cbRegion, enmType, pfnCallback);
1273 pdmUnlock(pVM);
1274 }
1275 else
1276 {
1277 AssertMsgFailed(("No PCI device registered!\n"));
1278 rc = VERR_PDM_NOT_PCI_DEVICE;
1279 }
1280
1281 LogFlow(("pdmR3DevHlp_PCIIORegionRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1282 return rc;
1283}
1284
1285
1286/** @interface_method_impl{PDMDEVHLPR3,pfnPCISetConfigCallbacks} */
1287static DECLCALLBACK(void) pdmR3DevHlp_PCISetConfigCallbacks(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
1288 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld)
1289{
1290 PDMDEV_ASSERT_DEVINS(pDevIns);
1291 PVM pVM = pDevIns->Internal.s.pVMR3;
1292 VM_ASSERT_EMT(pVM);
1293 LogFlow(("pdmR3DevHlp_PCISetConfigCallbacks: caller='%s'/%d: pPciDev=%p pfnRead=%p ppfnReadOld=%p pfnWrite=%p ppfnWriteOld=%p\n",
1294 pDevIns->pReg->szName, pDevIns->iInstance, pPciDev, pfnRead, ppfnReadOld, pfnWrite, ppfnWriteOld));
1295
1296 /*
1297 * Validate input and resolve defaults.
1298 */
1299 AssertPtr(pfnRead);
1300 AssertPtr(pfnWrite);
1301 AssertPtrNull(ppfnReadOld);
1302 AssertPtrNull(ppfnWriteOld);
1303 AssertPtrNull(pPciDev);
1304
1305 if (!pPciDev)
1306 pPciDev = pDevIns->Internal.s.pPciDeviceR3;
1307 AssertReleaseMsg(pPciDev, ("You must register your device first!\n"));
1308 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusR3;
1309 AssertRelease(pBus);
1310 AssertRelease(VMR3GetState(pVM) != VMSTATE_RUNNING);
1311
1312 /*
1313 * Do the job.
1314 */
1315 pdmLock(pVM);
1316 pBus->pfnSetConfigCallbacksR3(pBus->pDevInsR3, pPciDev, pfnRead, ppfnReadOld, pfnWrite, ppfnWriteOld);
1317 pdmUnlock(pVM);
1318
1319 LogFlow(("pdmR3DevHlp_PCISetConfigCallbacks: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1320}
1321
1322
1323/** @interface_method_impl{PDMDEVHLPR3,pfnPCISetIrq} */
1324static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1325{
1326 PDMDEV_ASSERT_DEVINS(pDevIns);
1327 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, iIrq, iLevel));
1328
1329 /*
1330 * Validate input.
1331 */
1332 /** @todo iIrq and iLevel checks. */
1333
1334 /*
1335 * Must have a PCI device registered!
1336 */
1337 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceR3;
1338 if (pPciDev)
1339 {
1340 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusR3; /** @todo the bus should be associated with the PCI device not the PDM device. */
1341 Assert(pBus);
1342 PVM pVM = pDevIns->Internal.s.pVMR3;
1343 pdmLock(pVM);
1344 pBus->pfnSetIrqR3(pBus->pDevInsR3, pPciDev, iIrq, iLevel);
1345 pdmUnlock(pVM);
1346 }
1347 else
1348 AssertReleaseMsgFailed(("No PCI device registered!\n"));
1349
1350 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1351}
1352
1353
1354/** @interface_method_impl{PDMDEVHLPR3,pfnPCISetIrqNoWait} */
1355static DECLCALLBACK(void) pdmR3DevHlp_PCISetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1356{
1357 pdmR3DevHlp_PCISetIrq(pDevIns, iIrq, iLevel);
1358}
1359
1360
1361/** @interface_method_impl{PDMDEVHLPR3,pfnPCIRegisterMsi} */
1362static DECLCALLBACK(int) pdmR3DevHlp_PCIRegisterMsi(PPDMDEVINS pDevIns, PPDMMSIREG pMsiReg)
1363{
1364 PDMDEV_ASSERT_DEVINS(pDevIns);
1365 LogFlow(("pdmR3DevHlp_PCIRegisterMsi: caller='%s'/%d: %d MSI vectors %d MSI-X vectors\n", pDevIns->pReg->szName, pDevIns->iInstance, pMsiReg->cMsiVectors,pMsiReg->cMsixVectors ));
1366 int rc = VINF_SUCCESS;
1367
1368 /*
1369 * Must have a PCI device registered!
1370 */
1371 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceR3;
1372 if (pPciDev)
1373 {
1374 PPDMPCIBUS pBus = pDevIns->Internal.s.pPciBusR3; /** @todo the bus should be associated with the PCI device not the PDM device. */
1375 Assert(pBus);
1376
1377 PVM pVM = pDevIns->Internal.s.pVMR3;
1378 pdmLock(pVM);
1379 if (!pBus->pfnRegisterMsiR3)
1380 rc = VERR_NOT_IMPLEMENTED;
1381 else
1382 rc = pBus->pfnRegisterMsiR3(pBus->pDevInsR3, pPciDev, pMsiReg);
1383 pdmUnlock(pVM);
1384 }
1385 else
1386 AssertReleaseMsgFailed(("No PCI device registered!\n"));
1387
1388 LogFlow(("pdmR3DevHlp_PCISetIrq: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1389 return rc;
1390}
1391
1392/** @interface_method_impl{PDMDEVHLPR3,pfnISASetIrq} */
1393static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1394{
1395 PDMDEV_ASSERT_DEVINS(pDevIns);
1396 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, iIrq, iLevel));
1397
1398 /*
1399 * Validate input.
1400 */
1401 /** @todo iIrq and iLevel checks. */
1402
1403 PVM pVM = pDevIns->Internal.s.pVMR3;
1404 PDMIsaSetIrq(pVM, iIrq, iLevel); /* (The API takes the lock.) */
1405
1406 LogFlow(("pdmR3DevHlp_ISASetIrq: caller='%s'/%d: returns void\n", pDevIns->pReg->szName, pDevIns->iInstance));
1407}
1408
1409
1410/** @interface_method_impl{PDMDEVHLPR3,pfnISASetIrqNoWait} */
1411static DECLCALLBACK(void) pdmR3DevHlp_ISASetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
1412{
1413 pdmR3DevHlp_ISASetIrq(pDevIns, iIrq, iLevel);
1414}
1415
1416
1417/** @interface_method_impl{PDMDEVHLPR3,pfnDriverAttach} */
1418static DECLCALLBACK(int) pdmR3DevHlp_DriverAttach(PPDMDEVINS pDevIns, RTUINT iLun, PPDMIBASE pBaseInterface, PPDMIBASE *ppBaseInterface, const char *pszDesc)
1419{
1420 PDMDEV_ASSERT_DEVINS(pDevIns);
1421 PVM pVM = pDevIns->Internal.s.pVMR3;
1422 VM_ASSERT_EMT(pVM);
1423 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: iLun=%d pBaseInterface=%p ppBaseInterface=%p pszDesc=%p:{%s}\n",
1424 pDevIns->pReg->szName, pDevIns->iInstance, iLun, pBaseInterface, ppBaseInterface, pszDesc, pszDesc));
1425
1426 /*
1427 * Lookup the LUN, it might already be registered.
1428 */
1429 PPDMLUN pLunPrev = NULL;
1430 PPDMLUN pLun = pDevIns->Internal.s.pLunsR3;
1431 for (; pLun; pLunPrev = pLun, pLun = pLun->pNext)
1432 if (pLun->iLun == iLun)
1433 break;
1434
1435 /*
1436 * Create the LUN if if wasn't found, else check if driver is already attached to it.
1437 */
1438 if (!pLun)
1439 {
1440 if ( !pBaseInterface
1441 || !pszDesc
1442 || !*pszDesc)
1443 {
1444 Assert(pBaseInterface);
1445 Assert(pszDesc || *pszDesc);
1446 return VERR_INVALID_PARAMETER;
1447 }
1448
1449 pLun = (PPDMLUN)MMR3HeapAlloc(pVM, MM_TAG_PDM_LUN, sizeof(*pLun));
1450 if (!pLun)
1451 return VERR_NO_MEMORY;
1452
1453 pLun->iLun = iLun;
1454 pLun->pNext = pLunPrev ? pLunPrev->pNext : NULL;
1455 pLun->pTop = NULL;
1456 pLun->pBottom = NULL;
1457 pLun->pDevIns = pDevIns;
1458 pLun->pUsbIns = NULL;
1459 pLun->pszDesc = pszDesc;
1460 pLun->pBase = pBaseInterface;
1461 if (!pLunPrev)
1462 pDevIns->Internal.s.pLunsR3 = pLun;
1463 else
1464 pLunPrev->pNext = pLun;
1465 Log(("pdmR3DevHlp_DriverAttach: Registered LUN#%d '%s' with device '%s'/%d.\n",
1466 iLun, pszDesc, pDevIns->pReg->szName, pDevIns->iInstance));
1467 }
1468 else if (pLun->pTop)
1469 {
1470 AssertMsgFailed(("Already attached! The device should keep track of such things!\n"));
1471 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_PDM_DRIVER_ALREADY_ATTACHED));
1472 return VERR_PDM_DRIVER_ALREADY_ATTACHED;
1473 }
1474 Assert(pLun->pBase == pBaseInterface);
1475
1476
1477 /*
1478 * Get the attached driver configuration.
1479 */
1480 int rc;
1481 PCFGMNODE pNode = CFGMR3GetChildF(pDevIns->Internal.s.pCfgHandle, "LUN#%u", iLun);
1482 if (pNode)
1483 rc = pdmR3DrvInstantiate(pVM, pNode, pBaseInterface, NULL /*pDrvAbove*/, pLun, ppBaseInterface);
1484 else
1485 rc = VERR_PDM_NO_ATTACHED_DRIVER;
1486
1487
1488 LogFlow(("pdmR3DevHlp_DriverAttach: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1489 return rc;
1490}
1491
1492
1493/** @interface_method_impl{PDMDEVHLPR3,pfnQueueCreate} */
1494static DECLCALLBACK(int) pdmR3DevHlp_QueueCreate(PPDMDEVINS pDevIns, RTUINT cbItem, RTUINT cItems, uint32_t cMilliesInterval,
1495 PFNPDMQUEUEDEV pfnCallback, bool fGCEnabled, const char *pszName, PPDMQUEUE *ppQueue)
1496{
1497 PDMDEV_ASSERT_DEVINS(pDevIns);
1498 LogFlow(("pdmR3DevHlp_QueueCreate: caller='%s'/%d: cbItem=%#x cItems=%#x cMilliesInterval=%u pfnCallback=%p fGCEnabled=%RTbool pszName=%p:{%s} ppQueue=%p\n",
1499 pDevIns->pReg->szName, pDevIns->iInstance, cbItem, cItems, cMilliesInterval, pfnCallback, fGCEnabled, pszName, pszName, ppQueue));
1500
1501 PVM pVM = pDevIns->Internal.s.pVMR3;
1502 VM_ASSERT_EMT(pVM);
1503
1504 if (pDevIns->iInstance > 0)
1505 {
1506 pszName = MMR3HeapAPrintf(pVM, MM_TAG_PDM_DEVICE_DESC, "%s_%u", pszName, pDevIns->iInstance);
1507 AssertLogRelReturn(pszName, VERR_NO_MEMORY);
1508 }
1509
1510 int rc = PDMR3QueueCreateDevice(pVM, pDevIns, cbItem, cItems, cMilliesInterval, pfnCallback, fGCEnabled, pszName, ppQueue);
1511
1512 LogFlow(("pdmR3DevHlp_QueueCreate: caller='%s'/%d: returns %Rrc *ppQueue=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, rc, *ppQueue));
1513 return rc;
1514}
1515
1516
1517/** @interface_method_impl{PDMDEVHLPR3,pfnCritSectInit} */
1518static DECLCALLBACK(int) pdmR3DevHlp_CritSectInit(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, RT_SRC_POS_DECL,
1519 const char *pszNameFmt, va_list va)
1520{
1521 PDMDEV_ASSERT_DEVINS(pDevIns);
1522 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: pCritSect=%p pszNameFmt=%p:{%s}\n",
1523 pDevIns->pReg->szName, pDevIns->iInstance, pCritSect, pszNameFmt, pszNameFmt));
1524
1525 PVM pVM = pDevIns->Internal.s.pVMR3;
1526 VM_ASSERT_EMT(pVM);
1527 int rc = pdmR3CritSectInitDevice(pVM, pDevIns, pCritSect, RT_SRC_POS_ARGS, pszNameFmt, va);
1528
1529 LogFlow(("pdmR3DevHlp_CritSectInit: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1530 return rc;
1531}
1532
1533
1534/** @interface_method_impl{PDMDEVHLPR3,pfnThreadCreate} */
1535static DECLCALLBACK(int) pdmR3DevHlp_ThreadCreate(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
1536 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName)
1537{
1538 PDMDEV_ASSERT_DEVINS(pDevIns);
1539 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
1540 LogFlow(("pdmR3DevHlp_ThreadCreate: caller='%s'/%d: ppThread=%p pvUser=%p pfnThread=%p pfnWakeup=%p cbStack=%#zx enmType=%d pszName=%p:{%s}\n",
1541 pDevIns->pReg->szName, pDevIns->iInstance, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName, pszName));
1542
1543 int rc = pdmR3ThreadCreateDevice(pDevIns->Internal.s.pVMR3, pDevIns, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName);
1544
1545 LogFlow(("pdmR3DevHlp_ThreadCreate: caller='%s'/%d: returns %Rrc *ppThread=%RTthrd\n", pDevIns->pReg->szName, pDevIns->iInstance,
1546 rc, *ppThread));
1547 return rc;
1548}
1549
1550
1551/** @interface_method_impl{PDMDEVHLPR3,pfnSetAsyncNotification} */
1552static DECLCALLBACK(int) pdmR3DevHlp_SetAsyncNotification(PPDMDEVINS pDevIns, PFNPDMDEVASYNCNOTIFY pfnAsyncNotify)
1553{
1554 PDMDEV_ASSERT_DEVINS(pDevIns);
1555 VM_ASSERT_EMT0(pDevIns->Internal.s.pVMR3);
1556 LogFlow(("pdmR3DevHlp_SetAsyncNotification: caller='%s'/%d: pfnAsyncNotify=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pfnAsyncNotify));
1557
1558 int rc = VINF_SUCCESS;
1559 AssertStmt(pfnAsyncNotify, rc = VERR_INVALID_PARAMETER);
1560 AssertStmt(!pDevIns->Internal.s.pfnAsyncNotify, rc = VERR_WRONG_ORDER);
1561 AssertStmt(pDevIns->Internal.s.fIntFlags & (PDMDEVINSINT_FLAGS_SUSPENDED | PDMDEVINSINT_FLAGS_RESET), rc = VERR_WRONG_ORDER);
1562 VMSTATE enmVMState = VMR3GetState(pDevIns->Internal.s.pVMR3);
1563 AssertStmt( enmVMState == VMSTATE_SUSPENDING
1564 || enmVMState == VMSTATE_SUSPENDING_EXT_LS
1565 || enmVMState == VMSTATE_SUSPENDING_LS
1566 || enmVMState == VMSTATE_RESETTING
1567 || enmVMState == VMSTATE_RESETTING_LS
1568 || enmVMState == VMSTATE_POWERING_OFF
1569 || enmVMState == VMSTATE_POWERING_OFF_LS,
1570 rc = VERR_INVALID_STATE);
1571
1572 if (RT_SUCCESS(rc))
1573 pDevIns->Internal.s.pfnAsyncNotify = pfnAsyncNotify;
1574
1575 LogFlow(("pdmR3DevHlp_SetAsyncNotification: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
1576 return rc;
1577}
1578
1579
1580/** @interface_method_impl{PDMDEVHLPR3,pfnAsyncNotificationCompleted} */
1581static DECLCALLBACK(void) pdmR3DevHlp_AsyncNotificationCompleted(PPDMDEVINS pDevIns)
1582{
1583 PDMDEV_ASSERT_DEVINS(pDevIns);
1584 PVM pVM = pDevIns->Internal.s.pVMR3;
1585
1586 VMSTATE enmVMState = VMR3GetState(pVM);
1587 if ( enmVMState == VMSTATE_SUSPENDING
1588 || enmVMState == VMSTATE_SUSPENDING_EXT_LS
1589 || enmVMState == VMSTATE_SUSPENDING_LS
1590 || enmVMState == VMSTATE_RESETTING
1591 || enmVMState == VMSTATE_RESETTING_LS
1592 || enmVMState == VMSTATE_POWERING_OFF
1593 || enmVMState == VMSTATE_POWERING_OFF_LS)
1594 {
1595 LogFlow(("pdmR3DevHlp_AsyncNotificationCompleted: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
1596 VMR3AsyncPdmNotificationWakeupU(pVM->pUVM);
1597 }
1598 else
1599 LogFlow(("pdmR3DevHlp_AsyncNotificationCompleted: caller='%s'/%d: enmVMState=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, enmVMState));
1600}
1601
1602
1603/** @interface_method_impl{PDMDEVHLPR3,pfnRTCRegister} */
1604static DECLCALLBACK(int) pdmR3DevHlp_RTCRegister(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp)
1605{
1606 PDMDEV_ASSERT_DEVINS(pDevIns);
1607 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
1608 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: pRtcReg=%p:{.u32Version=%#x, .pfnWrite=%p, .pfnRead=%p} ppRtcHlp=%p\n",
1609 pDevIns->pReg->szName, pDevIns->iInstance, pRtcReg, pRtcReg->u32Version, pRtcReg->pfnWrite,
1610 pRtcReg->pfnWrite, ppRtcHlp));
1611
1612 /*
1613 * Validate input.
1614 */
1615 if (pRtcReg->u32Version != PDM_RTCREG_VERSION)
1616 {
1617 AssertMsgFailed(("u32Version=%#x expected %#x\n", pRtcReg->u32Version,
1618 PDM_RTCREG_VERSION));
1619 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (version)\n",
1620 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1621 return VERR_INVALID_PARAMETER;
1622 }
1623 if ( !pRtcReg->pfnWrite
1624 || !pRtcReg->pfnRead)
1625 {
1626 Assert(pRtcReg->pfnWrite);
1627 Assert(pRtcReg->pfnRead);
1628 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (callbacks)\n",
1629 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1630 return VERR_INVALID_PARAMETER;
1631 }
1632
1633 if (!ppRtcHlp)
1634 {
1635 Assert(ppRtcHlp);
1636 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc (ppRtcHlp)\n",
1637 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1638 return VERR_INVALID_PARAMETER;
1639 }
1640
1641 /*
1642 * Only one DMA device.
1643 */
1644 PVM pVM = pDevIns->Internal.s.pVMR3;
1645 if (pVM->pdm.s.pRtc)
1646 {
1647 AssertMsgFailed(("Only one RTC device is supported!\n"));
1648 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc\n",
1649 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
1650 return VERR_INVALID_PARAMETER;
1651 }
1652
1653 /*
1654 * Allocate and initialize pci bus structure.
1655 */
1656 int rc = VINF_SUCCESS;
1657 PPDMRTC pRtc = (PPDMRTC)MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE, sizeof(*pRtc));
1658 if (pRtc)
1659 {
1660 pRtc->pDevIns = pDevIns;
1661 pRtc->Reg = *pRtcReg;
1662 pVM->pdm.s.pRtc = pRtc;
1663
1664 /* set the helper pointer. */
1665 *ppRtcHlp = &g_pdmR3DevRtcHlp;
1666 Log(("PDM: Registered RTC device '%s'/%d pDevIns=%p\n",
1667 pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
1668 }
1669 else
1670 rc = VERR_NO_MEMORY;
1671
1672 LogFlow(("pdmR3DevHlp_RTCRegister: caller='%s'/%d: returns %Rrc\n",
1673 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1674 return rc;
1675}
1676
1677
1678/** @interface_method_impl{PDMDEVHLPR3,pfnDMARegister} */
1679static DECLCALLBACK(int) pdmR3DevHlp_DMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser)
1680{
1681 PDMDEV_ASSERT_DEVINS(pDevIns);
1682 PVM pVM = pDevIns->Internal.s.pVMR3;
1683 VM_ASSERT_EMT(pVM);
1684 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: uChannel=%d pfnTransferHandler=%p pvUser=%p\n",
1685 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pfnTransferHandler, pvUser));
1686 int rc = VINF_SUCCESS;
1687 if (pVM->pdm.s.pDmac)
1688 pVM->pdm.s.pDmac->Reg.pfnRegister(pVM->pdm.s.pDmac->pDevIns, uChannel, pfnTransferHandler, pvUser);
1689 else
1690 {
1691 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1692 rc = VERR_PDM_NO_DMAC_INSTANCE;
1693 }
1694 LogFlow(("pdmR3DevHlp_DMARegister: caller='%s'/%d: returns %Rrc\n",
1695 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1696 return rc;
1697}
1698
1699
1700/** @interface_method_impl{PDMDEVHLPR3,pfnDMAReadMemory} */
1701static DECLCALLBACK(int) pdmR3DevHlp_DMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead)
1702{
1703 PDMDEV_ASSERT_DEVINS(pDevIns);
1704 PVM pVM = pDevIns->Internal.s.pVMR3;
1705 VM_ASSERT_EMT(pVM);
1706 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbRead=%p\n",
1707 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbRead));
1708 int rc = VINF_SUCCESS;
1709 if (pVM->pdm.s.pDmac)
1710 {
1711 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnReadMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
1712 if (pcbRead)
1713 *pcbRead = cb;
1714 }
1715 else
1716 {
1717 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1718 rc = VERR_PDM_NO_DMAC_INSTANCE;
1719 }
1720 LogFlow(("pdmR3DevHlp_DMAReadMemory: caller='%s'/%d: returns %Rrc\n",
1721 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1722 return rc;
1723}
1724
1725
1726/** @interface_method_impl{PDMDEVHLPR3,pfnDMAWriteMemory} */
1727static DECLCALLBACK(int) pdmR3DevHlp_DMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten)
1728{
1729 PDMDEV_ASSERT_DEVINS(pDevIns);
1730 PVM pVM = pDevIns->Internal.s.pVMR3;
1731 VM_ASSERT_EMT(pVM);
1732 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: uChannel=%d pvBuffer=%p off=%#x cbBlock=%#x pcbWritten=%p\n",
1733 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, pvBuffer, off, cbBlock, pcbWritten));
1734 int rc = VINF_SUCCESS;
1735 if (pVM->pdm.s.pDmac)
1736 {
1737 uint32_t cb = pVM->pdm.s.pDmac->Reg.pfnWriteMemory(pVM->pdm.s.pDmac->pDevIns, uChannel, pvBuffer, off, cbBlock);
1738 if (pcbWritten)
1739 *pcbWritten = cb;
1740 }
1741 else
1742 {
1743 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1744 rc = VERR_PDM_NO_DMAC_INSTANCE;
1745 }
1746 LogFlow(("pdmR3DevHlp_DMAWriteMemory: caller='%s'/%d: returns %Rrc\n",
1747 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1748 return rc;
1749}
1750
1751
1752/** @interface_method_impl{PDMDEVHLPR3,pfnDMASetDREQ} */
1753static DECLCALLBACK(int) pdmR3DevHlp_DMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel)
1754{
1755 PDMDEV_ASSERT_DEVINS(pDevIns);
1756 PVM pVM = pDevIns->Internal.s.pVMR3;
1757 VM_ASSERT_EMT(pVM);
1758 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: uChannel=%d uLevel=%d\n",
1759 pDevIns->pReg->szName, pDevIns->iInstance, uChannel, uLevel));
1760 int rc = VINF_SUCCESS;
1761 if (pVM->pdm.s.pDmac)
1762 pVM->pdm.s.pDmac->Reg.pfnSetDREQ(pVM->pdm.s.pDmac->pDevIns, uChannel, uLevel);
1763 else
1764 {
1765 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1766 rc = VERR_PDM_NO_DMAC_INSTANCE;
1767 }
1768 LogFlow(("pdmR3DevHlp_DMASetDREQ: caller='%s'/%d: returns %Rrc\n",
1769 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1770 return rc;
1771}
1772
1773/** @interface_method_impl{PDMDEVHLPR3,pfnDMAGetChannelMode} */
1774static DECLCALLBACK(uint8_t) pdmR3DevHlp_DMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel)
1775{
1776 PDMDEV_ASSERT_DEVINS(pDevIns);
1777 PVM pVM = pDevIns->Internal.s.pVMR3;
1778 VM_ASSERT_EMT(pVM);
1779 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: uChannel=%d\n",
1780 pDevIns->pReg->szName, pDevIns->iInstance, uChannel));
1781 uint8_t u8Mode;
1782 if (pVM->pdm.s.pDmac)
1783 u8Mode = pVM->pdm.s.pDmac->Reg.pfnGetChannelMode(pVM->pdm.s.pDmac->pDevIns, uChannel);
1784 else
1785 {
1786 AssertMsgFailed(("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1787 u8Mode = 3 << 2 /* illegal mode type */;
1788 }
1789 LogFlow(("pdmR3DevHlp_DMAGetChannelMode: caller='%s'/%d: returns %#04x\n",
1790 pDevIns->pReg->szName, pDevIns->iInstance, u8Mode));
1791 return u8Mode;
1792}
1793
1794/** @interface_method_impl{PDMDEVHLPR3,pfnDMASchedule} */
1795static DECLCALLBACK(void) pdmR3DevHlp_DMASchedule(PPDMDEVINS pDevIns)
1796{
1797 PDMDEV_ASSERT_DEVINS(pDevIns);
1798 PVM pVM = pDevIns->Internal.s.pVMR3;
1799 VM_ASSERT_EMT(pVM);
1800 LogFlow(("pdmR3DevHlp_DMASchedule: caller='%s'/%d: VM_FF_PDM_DMA %d -> 1\n",
1801 pDevIns->pReg->szName, pDevIns->iInstance, VM_FF_ISSET(pVM, VM_FF_PDM_DMA)));
1802
1803 AssertMsg(pVM->pdm.s.pDmac, ("Configuration error: No DMAC controller available. This could be related to init order too!\n"));
1804 VM_FF_SET(pVM, VM_FF_PDM_DMA);
1805 REMR3NotifyDmaPending(pVM);
1806 VMR3NotifyGlobalFFU(pVM->pUVM, VMNOTIFYFF_FLAGS_DONE_REM);
1807}
1808
1809
1810/** @interface_method_impl{PDMDEVHLPR3,pfnCMOSWrite} */
1811static DECLCALLBACK(int) pdmR3DevHlp_CMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value)
1812{
1813 PDMDEV_ASSERT_DEVINS(pDevIns);
1814 PVM pVM = pDevIns->Internal.s.pVMR3;
1815 VM_ASSERT_EMT(pVM);
1816
1817 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x u8Value=%#04x\n",
1818 pDevIns->pReg->szName, pDevIns->iInstance, iReg, u8Value));
1819 int rc;
1820 if (pVM->pdm.s.pRtc)
1821 rc = pVM->pdm.s.pRtc->Reg.pfnWrite(pVM->pdm.s.pRtc->pDevIns, iReg, u8Value);
1822 else
1823 rc = VERR_PDM_NO_RTC_INSTANCE;
1824
1825 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Rrc\n",
1826 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1827 return rc;
1828}
1829
1830
1831/** @interface_method_impl{PDMDEVHLPR3,pfnCMOSRead} */
1832static DECLCALLBACK(int) pdmR3DevHlp_CMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value)
1833{
1834 PDMDEV_ASSERT_DEVINS(pDevIns);
1835 PVM pVM = pDevIns->Internal.s.pVMR3;
1836 VM_ASSERT_EMT(pVM);
1837
1838 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: iReg=%#04x pu8Value=%p\n",
1839 pDevIns->pReg->szName, pDevIns->iInstance, iReg, pu8Value));
1840 int rc;
1841 if (pVM->pdm.s.pRtc)
1842 rc = pVM->pdm.s.pRtc->Reg.pfnRead(pVM->pdm.s.pRtc->pDevIns, iReg, pu8Value);
1843 else
1844 rc = VERR_PDM_NO_RTC_INSTANCE;
1845
1846 LogFlow(("pdmR3DevHlp_CMOSWrite: caller='%s'/%d: return %Rrc\n",
1847 pDevIns->pReg->szName, pDevIns->iInstance, rc));
1848 return rc;
1849}
1850
1851
1852/** @interface_method_impl{PDMDEVHLPR3,pfnAssertEMT} */
1853static DECLCALLBACK(bool) pdmR3DevHlp_AssertEMT(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
1854{
1855 PDMDEV_ASSERT_DEVINS(pDevIns);
1856 if (VM_IS_EMT(pDevIns->Internal.s.pVMR3))
1857 return true;
1858
1859 char szMsg[100];
1860 RTStrPrintf(szMsg, sizeof(szMsg), "AssertEMT '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance);
1861 RTAssertMsg1Weak(szMsg, iLine, pszFile, pszFunction);
1862 AssertBreakpoint();
1863 return false;
1864}
1865
1866
1867/** @interface_method_impl{PDMDEVHLPR3,pfnAssertOther} */
1868static DECLCALLBACK(bool) pdmR3DevHlp_AssertOther(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction)
1869{
1870 PDMDEV_ASSERT_DEVINS(pDevIns);
1871 if (!VM_IS_EMT(pDevIns->Internal.s.pVMR3))
1872 return true;
1873
1874 char szMsg[100];
1875 RTStrPrintf(szMsg, sizeof(szMsg), "AssertOther '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance);
1876 RTAssertMsg1Weak(szMsg, iLine, pszFile, pszFunction);
1877 AssertBreakpoint();
1878 return false;
1879}
1880
1881
1882/** @interface_method_impl{PDMDEVHLP,pfnLdrGetRCInterfaceSymbols} */
1883static DECLCALLBACK(int) pdmR3DevHlp_LdrGetRCInterfaceSymbols(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
1884 const char *pszSymPrefix, const char *pszSymList)
1885{
1886 PDMDEV_ASSERT_DEVINS(pDevIns);
1887 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
1888 LogFlow(("pdmR3DevHlp_PDMLdrGetRCInterfaceSymbols: caller='%s'/%d: pvInterface=%p cbInterface=%zu pszSymPrefix=%p:{%s} pszSymList=%p:{%s}\n",
1889 pDevIns->pReg->szName, pDevIns->iInstance, pvInterface, cbInterface, pszSymPrefix, pszSymPrefix, pszSymList, pszSymList));
1890
1891 int rc;
1892 if ( strncmp(pszSymPrefix, "dev", 3) == 0
1893 && RTStrIStr(pszSymPrefix + 3, pDevIns->pReg->szName) != NULL)
1894 {
1895 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC)
1896 rc = PDMR3LdrGetInterfaceSymbols(pDevIns->Internal.s.pVMR3,
1897 pvInterface, cbInterface,
1898 pDevIns->pReg->szRCMod, pDevIns->Internal.s.pDevR3->pszRCSearchPath,
1899 pszSymPrefix, pszSymList,
1900 false /*fRing0OrRC*/);
1901 else
1902 {
1903 AssertMsgFailed(("Not a raw-mode enabled driver\n"));
1904 rc = VERR_PERMISSION_DENIED;
1905 }
1906 }
1907 else
1908 {
1909 AssertMsgFailed(("Invalid prefix '%s' for '%s'; must start with 'dev' and contain the driver name!\n",
1910 pszSymPrefix, pDevIns->pReg->szName));
1911 rc = VERR_INVALID_NAME;
1912 }
1913
1914 LogFlow(("pdmR3DevHlp_PDMLdrGetRCInterfaceSymbols: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
1915 pDevIns->iInstance, rc));
1916 return rc;
1917}
1918
1919
1920/** @interface_method_impl{PDMDEVHLP,pfnLdrGetR0InterfaceSymbols} */
1921static DECLCALLBACK(int) pdmR3DevHlp_LdrGetR0InterfaceSymbols(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
1922 const char *pszSymPrefix, const char *pszSymList)
1923{
1924 PDMDEV_ASSERT_DEVINS(pDevIns);
1925 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
1926 LogFlow(("pdmR3DevHlp_PDMLdrGetR0InterfaceSymbols: caller='%s'/%d: pvInterface=%p cbInterface=%zu pszSymPrefix=%p:{%s} pszSymList=%p:{%s}\n",
1927 pDevIns->pReg->szName, pDevIns->iInstance, pvInterface, cbInterface, pszSymPrefix, pszSymPrefix, pszSymList, pszSymList));
1928
1929 int rc;
1930 if ( strncmp(pszSymPrefix, "dev", 3) == 0
1931 && RTStrIStr(pszSymPrefix + 3, pDevIns->pReg->szName) != NULL)
1932 {
1933 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
1934 rc = PDMR3LdrGetInterfaceSymbols(pDevIns->Internal.s.pVMR3,
1935 pvInterface, cbInterface,
1936 pDevIns->pReg->szR0Mod, pDevIns->Internal.s.pDevR3->pszR0SearchPath,
1937 pszSymPrefix, pszSymList,
1938 true /*fRing0OrRC*/);
1939 else
1940 {
1941 AssertMsgFailed(("Not a ring-0 enabled driver\n"));
1942 rc = VERR_PERMISSION_DENIED;
1943 }
1944 }
1945 else
1946 {
1947 AssertMsgFailed(("Invalid prefix '%s' for '%s'; must start with 'dev' and contain the driver name!\n",
1948 pszSymPrefix, pDevIns->pReg->szName));
1949 rc = VERR_INVALID_NAME;
1950 }
1951
1952 LogFlow(("pdmR3DevHlp_PDMLdrGetR0InterfaceSymbols: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
1953 pDevIns->iInstance, rc));
1954 return rc;
1955}
1956
1957
1958/** @interface_method_impl{PDMDEVHLP,pfnCallR0} */
1959static DECLCALLBACK(int) pdmR3DevHlp_CallR0(PPDMDEVINS pDevIns, uint32_t uOperation, uint64_t u64Arg)
1960{
1961 PDMDEV_ASSERT_DEVINS(pDevIns);
1962 PVM pVM = pDevIns->Internal.s.pVMR3;
1963 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
1964 LogFlow(("pdmR3DevHlp_CallR0: caller='%s'/%d: uOperation=%#x u64Arg=%#RX64\n",
1965 pDevIns->pReg->szName, pDevIns->iInstance, uOperation, u64Arg));
1966
1967 /*
1968 * Resolve the ring-0 entry point. There is not need to remember this like
1969 * we do for drivers since this is mainly for construction time hacks and
1970 * other things that aren't performance critical.
1971 */
1972 int rc;
1973 if (pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0)
1974 {
1975 char szSymbol[ sizeof("devR0") + sizeof(pDevIns->pReg->szName) + sizeof("ReqHandler")];
1976 strcat(strcat(strcpy(szSymbol, "devR0"), pDevIns->pReg->szName), "ReqHandler");
1977 szSymbol[sizeof("devR0") - 1] = RT_C_TO_UPPER(szSymbol[sizeof("devR0") - 1]);
1978
1979 PFNPDMDRVREQHANDLERR0 pfnReqHandlerR0;
1980 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, szSymbol, &pfnReqHandlerR0);
1981 if (RT_SUCCESS(rc))
1982 {
1983 /*
1984 * Make the ring-0 call.
1985 */
1986 PDMDEVICECALLREQHANDLERREQ Req;
1987 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
1988 Req.Hdr.cbReq = sizeof(Req);
1989 Req.pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
1990 Req.pfnReqHandlerR0 = pfnReqHandlerR0;
1991 Req.uOperation = uOperation;
1992 Req.u32Alignment = 0;
1993 Req.u64Arg = u64Arg;
1994 rc = SUPR3CallVMMR0Ex(pVM->pVMR0, NIL_VMCPUID, VMMR0_DO_PDM_DEVICE_CALL_REQ_HANDLER, 0, &Req.Hdr);
1995 }
1996 else
1997 pfnReqHandlerR0 = NIL_RTR0PTR;
1998 }
1999 else
2000 rc = VERR_ACCESS_DENIED;
2001 LogFlow(("pdmR3DevHlp_CallR0: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName,
2002 pDevIns->iInstance, rc));
2003 return rc;
2004}
2005
2006
2007/** @interface_method_impl{PDMDEVHLPR3,pfnGetVM} */
2008static DECLCALLBACK(PVM) pdmR3DevHlp_GetVM(PPDMDEVINS pDevIns)
2009{
2010 PDMDEV_ASSERT_DEVINS(pDevIns);
2011 LogFlow(("pdmR3DevHlp_GetVM: caller='%s'/%d: returns %p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns->Internal.s.pVMR3));
2012 return pDevIns->Internal.s.pVMR3;
2013}
2014
2015
2016/** @interface_method_impl{PDMDEVHLPR3,pfnGetVMCPU} */
2017static DECLCALLBACK(PVMCPU) pdmR3DevHlp_GetVMCPU(PPDMDEVINS pDevIns)
2018{
2019 PDMDEV_ASSERT_DEVINS(pDevIns);
2020 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2021 LogFlow(("pdmR3DevHlp_GetVMCPU: caller='%s'/%d for CPU %u\n", pDevIns->pReg->szName, pDevIns->iInstance, VMMGetCpuId(pDevIns->Internal.s.pVMR3)));
2022 return VMMGetCpu(pDevIns->Internal.s.pVMR3);
2023}
2024
2025
2026/** @interface_method_impl{PDMDEVHLPR3,pfnPCIBusRegister} */
2027static DECLCALLBACK(int) pdmR3DevHlp_PCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3)
2028{
2029 PDMDEV_ASSERT_DEVINS(pDevIns);
2030 PVM pVM = pDevIns->Internal.s.pVMR3;
2031 VM_ASSERT_EMT(pVM);
2032 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: pPciBusReg=%p:{.u32Version=%#x, .pfnRegisterR3=%p, .pfnIORegionRegisterR3=%p, .pfnSetIrqR3=%p, "
2033 ".pfnSaveExecR3=%p, .pfnLoadExecR3=%p, .pfnFakePCIBIOSR3=%p, .pszSetIrqRC=%p:{%s}, .pszSetIrqR0=%p:{%s}} ppPciHlpR3=%p\n",
2034 pDevIns->pReg->szName, pDevIns->iInstance, pPciBusReg, pPciBusReg->u32Version, pPciBusReg->pfnRegisterR3,
2035 pPciBusReg->pfnIORegionRegisterR3, pPciBusReg->pfnSetIrqR3, pPciBusReg->pfnSaveExecR3, pPciBusReg->pfnLoadExecR3,
2036 pPciBusReg->pfnFakePCIBIOSR3, pPciBusReg->pszSetIrqRC, pPciBusReg->pszSetIrqRC, pPciBusReg->pszSetIrqR0, pPciBusReg->pszSetIrqR0, ppPciHlpR3));
2037
2038 /*
2039 * Validate the structure.
2040 */
2041 if (pPciBusReg->u32Version != PDM_PCIBUSREG_VERSION)
2042 {
2043 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPciBusReg->u32Version, PDM_PCIBUSREG_VERSION));
2044 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2045 return VERR_INVALID_PARAMETER;
2046 }
2047 if ( !pPciBusReg->pfnRegisterR3
2048 || !pPciBusReg->pfnIORegionRegisterR3
2049 || !pPciBusReg->pfnSetIrqR3
2050 || !pPciBusReg->pfnSaveExecR3
2051 || !pPciBusReg->pfnLoadExecR3
2052 || (!pPciBusReg->pfnFakePCIBIOSR3 && !pVM->pdm.s.aPciBuses[0].pDevInsR3)) /* Only the first bus needs to do the BIOS work. */
2053 {
2054 Assert(pPciBusReg->pfnRegisterR3);
2055 Assert(pPciBusReg->pfnIORegionRegisterR3);
2056 Assert(pPciBusReg->pfnSetIrqR3);
2057 Assert(pPciBusReg->pfnSaveExecR3);
2058 Assert(pPciBusReg->pfnLoadExecR3);
2059 Assert(pPciBusReg->pfnFakePCIBIOSR3);
2060 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc (R3 callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2061 return VERR_INVALID_PARAMETER;
2062 }
2063 if ( pPciBusReg->pszSetIrqRC
2064 && !VALID_PTR(pPciBusReg->pszSetIrqRC))
2065 {
2066 Assert(VALID_PTR(pPciBusReg->pszSetIrqRC));
2067 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2068 return VERR_INVALID_PARAMETER;
2069 }
2070 if ( pPciBusReg->pszSetIrqR0
2071 && !VALID_PTR(pPciBusReg->pszSetIrqR0))
2072 {
2073 Assert(VALID_PTR(pPciBusReg->pszSetIrqR0));
2074 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2075 return VERR_INVALID_PARAMETER;
2076 }
2077 if (!ppPciHlpR3)
2078 {
2079 Assert(ppPciHlpR3);
2080 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc (ppPciHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2081 return VERR_INVALID_PARAMETER;
2082 }
2083
2084 /*
2085 * Find free PCI bus entry.
2086 */
2087 unsigned iBus = 0;
2088 for (iBus = 0; iBus < RT_ELEMENTS(pVM->pdm.s.aPciBuses); iBus++)
2089 if (!pVM->pdm.s.aPciBuses[iBus].pDevInsR3)
2090 break;
2091 if (iBus >= RT_ELEMENTS(pVM->pdm.s.aPciBuses))
2092 {
2093 AssertMsgFailed(("Too many PCI buses. Max=%u\n", RT_ELEMENTS(pVM->pdm.s.aPciBuses)));
2094 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc (pci bus)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2095 return VERR_INVALID_PARAMETER;
2096 }
2097 PPDMPCIBUS pPciBus = &pVM->pdm.s.aPciBuses[iBus];
2098
2099 /*
2100 * Resolve and init the RC bits.
2101 */
2102 if (pPciBusReg->pszSetIrqRC)
2103 {
2104 int rc = pdmR3DevGetSymbolRCLazy(pDevIns, pPciBusReg->pszSetIrqRC, &pPciBus->pfnSetIrqRC);
2105 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pPciBusReg->pszSetIrqRC, rc));
2106 if (RT_FAILURE(rc))
2107 {
2108 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2109 return rc;
2110 }
2111 pPciBus->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2112 }
2113 else
2114 {
2115 pPciBus->pfnSetIrqRC = 0;
2116 pPciBus->pDevInsRC = 0;
2117 }
2118
2119 /*
2120 * Resolve and init the R0 bits.
2121 */
2122 if (pPciBusReg->pszSetIrqR0)
2123 {
2124 int rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pPciBusReg->pszSetIrqR0, &pPciBus->pfnSetIrqR0);
2125 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pPciBusReg->pszSetIrqR0, rc));
2126 if (RT_FAILURE(rc))
2127 {
2128 LogFlow(("pdmR3DevHlp_PCIRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2129 return rc;
2130 }
2131 pPciBus->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2132 }
2133 else
2134 {
2135 pPciBus->pfnSetIrqR0 = 0;
2136 pPciBus->pDevInsR0 = 0;
2137 }
2138
2139 /*
2140 * Init the R3 bits.
2141 */
2142 pPciBus->iBus = iBus;
2143 pPciBus->pDevInsR3 = pDevIns;
2144 pPciBus->pfnRegisterR3 = pPciBusReg->pfnRegisterR3;
2145 pPciBus->pfnRegisterMsiR3 = pPciBusReg->pfnRegisterMsiR3;
2146 pPciBus->pfnIORegionRegisterR3 = pPciBusReg->pfnIORegionRegisterR3;
2147 pPciBus->pfnSetConfigCallbacksR3 = pPciBusReg->pfnSetConfigCallbacksR3;
2148 pPciBus->pfnSetIrqR3 = pPciBusReg->pfnSetIrqR3;
2149 pPciBus->pfnSaveExecR3 = pPciBusReg->pfnSaveExecR3;
2150 pPciBus->pfnLoadExecR3 = pPciBusReg->pfnLoadExecR3;
2151 pPciBus->pfnFakePCIBIOSR3 = pPciBusReg->pfnFakePCIBIOSR3;
2152
2153 Log(("PDM: Registered PCI bus device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
2154
2155 /* set the helper pointer and return. */
2156 *ppPciHlpR3 = &g_pdmR3DevPciHlp;
2157 LogFlow(("pdmR3DevHlp_PCIBusRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2158 return VINF_SUCCESS;
2159}
2160
2161
2162/** @interface_method_impl{PDMDEVHLPR3,pfnPICRegister} */
2163static DECLCALLBACK(int) pdmR3DevHlp_PICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3)
2164{
2165 PDMDEV_ASSERT_DEVINS(pDevIns);
2166 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2167 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: pPicReg=%p:{.u32Version=%#x, .pfnSetIrqR3=%p, .pfnGetInterruptR3=%p, .pszGetIrqRC=%p:{%s}, .pszGetInterruptRC=%p:{%s}, .pszGetIrqR0=%p:{%s}, .pszGetInterruptR0=%p:{%s} } ppPicHlpR3=%p\n",
2168 pDevIns->pReg->szName, pDevIns->iInstance, pPicReg, pPicReg->u32Version, pPicReg->pfnSetIrqR3, pPicReg->pfnGetInterruptR3,
2169 pPicReg->pszSetIrqRC, pPicReg->pszSetIrqRC, pPicReg->pszGetInterruptRC, pPicReg->pszGetInterruptRC,
2170 pPicReg->pszSetIrqR0, pPicReg->pszSetIrqR0, pPicReg->pszGetInterruptR0, pPicReg->pszGetInterruptR0,
2171 ppPicHlpR3));
2172
2173 /*
2174 * Validate input.
2175 */
2176 if (pPicReg->u32Version != PDM_PICREG_VERSION)
2177 {
2178 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPicReg->u32Version, PDM_PICREG_VERSION));
2179 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2180 return VERR_INVALID_PARAMETER;
2181 }
2182 if ( !pPicReg->pfnSetIrqR3
2183 || !pPicReg->pfnGetInterruptR3)
2184 {
2185 Assert(pPicReg->pfnSetIrqR3);
2186 Assert(pPicReg->pfnGetInterruptR3);
2187 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (R3 callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2188 return VERR_INVALID_PARAMETER;
2189 }
2190 if ( ( pPicReg->pszSetIrqRC
2191 || pPicReg->pszGetInterruptRC)
2192 && ( !VALID_PTR(pPicReg->pszSetIrqRC)
2193 || !VALID_PTR(pPicReg->pszGetInterruptRC))
2194 )
2195 {
2196 Assert(VALID_PTR(pPicReg->pszSetIrqRC));
2197 Assert(VALID_PTR(pPicReg->pszGetInterruptRC));
2198 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (RC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2199 return VERR_INVALID_PARAMETER;
2200 }
2201 if ( pPicReg->pszSetIrqRC
2202 && !(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC))
2203 {
2204 Assert(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_RC);
2205 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (RC flag)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2206 return VERR_INVALID_PARAMETER;
2207 }
2208 if ( pPicReg->pszSetIrqR0
2209 && !(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0))
2210 {
2211 Assert(pDevIns->pReg->fFlags & PDM_DEVREG_FLAGS_R0);
2212 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (R0 flag)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2213 return VERR_INVALID_PARAMETER;
2214 }
2215 if (!ppPicHlpR3)
2216 {
2217 Assert(ppPicHlpR3);
2218 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc (ppPicHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2219 return VERR_INVALID_PARAMETER;
2220 }
2221
2222 /*
2223 * Only one PIC device.
2224 */
2225 PVM pVM = pDevIns->Internal.s.pVMR3;
2226 if (pVM->pdm.s.Pic.pDevInsR3)
2227 {
2228 AssertMsgFailed(("Only one pic device is supported!\n"));
2229 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2230 return VERR_INVALID_PARAMETER;
2231 }
2232
2233 /*
2234 * RC stuff.
2235 */
2236 if (pPicReg->pszSetIrqRC)
2237 {
2238 int rc = pdmR3DevGetSymbolRCLazy(pDevIns, pPicReg->pszSetIrqRC, &pVM->pdm.s.Pic.pfnSetIrqRC);
2239 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pPicReg->pszSetIrqRC, rc));
2240 if (RT_SUCCESS(rc))
2241 {
2242 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pPicReg->pszGetInterruptRC, &pVM->pdm.s.Pic.pfnGetInterruptRC);
2243 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pPicReg->pszGetInterruptRC, rc));
2244 }
2245 if (RT_FAILURE(rc))
2246 {
2247 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2248 return rc;
2249 }
2250 pVM->pdm.s.Pic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2251 }
2252 else
2253 {
2254 pVM->pdm.s.Pic.pDevInsRC = 0;
2255 pVM->pdm.s.Pic.pfnSetIrqRC = 0;
2256 pVM->pdm.s.Pic.pfnGetInterruptRC = 0;
2257 }
2258
2259 /*
2260 * R0 stuff.
2261 */
2262 if (pPicReg->pszSetIrqR0)
2263 {
2264 int rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pPicReg->pszSetIrqR0, &pVM->pdm.s.Pic.pfnSetIrqR0);
2265 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pPicReg->pszSetIrqR0, rc));
2266 if (RT_SUCCESS(rc))
2267 {
2268 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pPicReg->pszGetInterruptR0, &pVM->pdm.s.Pic.pfnGetInterruptR0);
2269 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pPicReg->pszGetInterruptR0, rc));
2270 }
2271 if (RT_FAILURE(rc))
2272 {
2273 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2274 return rc;
2275 }
2276 pVM->pdm.s.Pic.pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2277 Assert(pVM->pdm.s.Pic.pDevInsR0);
2278 }
2279 else
2280 {
2281 pVM->pdm.s.Pic.pfnSetIrqR0 = 0;
2282 pVM->pdm.s.Pic.pfnGetInterruptR0 = 0;
2283 pVM->pdm.s.Pic.pDevInsR0 = 0;
2284 }
2285
2286 /*
2287 * R3 stuff.
2288 */
2289 pVM->pdm.s.Pic.pDevInsR3 = pDevIns;
2290 pVM->pdm.s.Pic.pfnSetIrqR3 = pPicReg->pfnSetIrqR3;
2291 pVM->pdm.s.Pic.pfnGetInterruptR3 = pPicReg->pfnGetInterruptR3;
2292 Log(("PDM: Registered PIC device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
2293
2294 /* set the helper pointer and return. */
2295 *ppPicHlpR3 = &g_pdmR3DevPicHlp;
2296 LogFlow(("pdmR3DevHlp_PICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2297 return VINF_SUCCESS;
2298}
2299
2300
2301/** @interface_method_impl{PDMDEVHLPR3,pfnAPICRegister} */
2302static DECLCALLBACK(int) pdmR3DevHlp_APICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3)
2303{
2304 PDMDEV_ASSERT_DEVINS(pDevIns);
2305 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2306 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: pApicReg=%p:{.u32Version=%#x, .pfnGetInterruptR3=%p, .pfnSetBaseR3=%p, .pfnGetBaseR3=%p, "
2307 ".pfnSetTPRR3=%p, .pfnGetTPRR3=%p, .pfnWriteMSR3=%p, .pfnReadMSR3=%p, .pfnBusDeliverR3=%p, .pfnLocalInterruptR3=%p, pszGetInterruptRC=%p:{%s}, pszSetBaseRC=%p:{%s}, pszGetBaseRC=%p:{%s}, "
2308 ".pszSetTPRRC=%p:{%s}, .pszGetTPRRC=%p:{%s}, .pszWriteMSRRC=%p:{%s}, .pszReadMSRRC=%p:{%s}, .pszBusDeliverRC=%p:{%s}, .pszLocalInterruptRC=%p:{%s}} ppApicHlpR3=%p\n",
2309 pDevIns->pReg->szName, pDevIns->iInstance, pApicReg, pApicReg->u32Version, pApicReg->pfnGetInterruptR3, pApicReg->pfnSetBaseR3,
2310 pApicReg->pfnGetBaseR3, pApicReg->pfnSetTPRR3, pApicReg->pfnGetTPRR3, pApicReg->pfnWriteMSRR3, pApicReg->pfnReadMSRR3, pApicReg->pfnBusDeliverR3, pApicReg->pfnLocalInterruptR3, pApicReg->pszGetInterruptRC,
2311 pApicReg->pszGetInterruptRC, pApicReg->pszSetBaseRC, pApicReg->pszSetBaseRC, pApicReg->pszGetBaseRC, pApicReg->pszGetBaseRC,
2312 pApicReg->pszSetTPRRC, pApicReg->pszSetTPRRC, pApicReg->pszGetTPRRC, pApicReg->pszGetTPRRC, pApicReg->pszWriteMSRRC, pApicReg->pszWriteMSRRC, pApicReg->pszReadMSRRC, pApicReg->pszReadMSRRC, pApicReg->pszBusDeliverRC,
2313 pApicReg->pszBusDeliverRC, pApicReg->pszLocalInterruptRC, pApicReg->pszLocalInterruptRC, ppApicHlpR3));
2314
2315 /*
2316 * Validate input.
2317 */
2318 if (pApicReg->u32Version != PDM_APICREG_VERSION)
2319 {
2320 AssertMsgFailed(("u32Version=%#x expected %#x\n", pApicReg->u32Version, PDM_APICREG_VERSION));
2321 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2322 return VERR_INVALID_PARAMETER;
2323 }
2324 if ( !pApicReg->pfnGetInterruptR3
2325 || !pApicReg->pfnHasPendingIrqR3
2326 || !pApicReg->pfnSetBaseR3
2327 || !pApicReg->pfnGetBaseR3
2328 || !pApicReg->pfnSetTPRR3
2329 || !pApicReg->pfnGetTPRR3
2330 || !pApicReg->pfnWriteMSRR3
2331 || !pApicReg->pfnReadMSRR3
2332 || !pApicReg->pfnBusDeliverR3
2333 || !pApicReg->pfnLocalInterruptR3)
2334 {
2335 Assert(pApicReg->pfnGetInterruptR3);
2336 Assert(pApicReg->pfnHasPendingIrqR3);
2337 Assert(pApicReg->pfnSetBaseR3);
2338 Assert(pApicReg->pfnGetBaseR3);
2339 Assert(pApicReg->pfnSetTPRR3);
2340 Assert(pApicReg->pfnGetTPRR3);
2341 Assert(pApicReg->pfnWriteMSRR3);
2342 Assert(pApicReg->pfnReadMSRR3);
2343 Assert(pApicReg->pfnBusDeliverR3);
2344 Assert(pApicReg->pfnLocalInterruptR3);
2345 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc (R3 callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2346 return VERR_INVALID_PARAMETER;
2347 }
2348 if ( ( pApicReg->pszGetInterruptRC
2349 || pApicReg->pszHasPendingIrqRC
2350 || pApicReg->pszSetBaseRC
2351 || pApicReg->pszGetBaseRC
2352 || pApicReg->pszSetTPRRC
2353 || pApicReg->pszGetTPRRC
2354 || pApicReg->pszWriteMSRRC
2355 || pApicReg->pszReadMSRRC
2356 || pApicReg->pszBusDeliverRC
2357 || pApicReg->pszLocalInterruptRC)
2358 && ( !VALID_PTR(pApicReg->pszGetInterruptRC)
2359 || !VALID_PTR(pApicReg->pszHasPendingIrqRC)
2360 || !VALID_PTR(pApicReg->pszSetBaseRC)
2361 || !VALID_PTR(pApicReg->pszGetBaseRC)
2362 || !VALID_PTR(pApicReg->pszSetTPRRC)
2363 || !VALID_PTR(pApicReg->pszGetTPRRC)
2364 || !VALID_PTR(pApicReg->pszWriteMSRRC)
2365 || !VALID_PTR(pApicReg->pszReadMSRRC)
2366 || !VALID_PTR(pApicReg->pszBusDeliverRC)
2367 || !VALID_PTR(pApicReg->pszLocalInterruptRC))
2368 )
2369 {
2370 Assert(VALID_PTR(pApicReg->pszGetInterruptRC));
2371 Assert(VALID_PTR(pApicReg->pszHasPendingIrqRC));
2372 Assert(VALID_PTR(pApicReg->pszSetBaseRC));
2373 Assert(VALID_PTR(pApicReg->pszGetBaseRC));
2374 Assert(VALID_PTR(pApicReg->pszSetTPRRC));
2375 Assert(VALID_PTR(pApicReg->pszGetTPRRC));
2376 Assert(VALID_PTR(pApicReg->pszReadMSRRC));
2377 Assert(VALID_PTR(pApicReg->pszWriteMSRRC));
2378 Assert(VALID_PTR(pApicReg->pszBusDeliverRC));
2379 Assert(VALID_PTR(pApicReg->pszLocalInterruptRC));
2380 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc (RC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2381 return VERR_INVALID_PARAMETER;
2382 }
2383 if ( ( pApicReg->pszGetInterruptR0
2384 || pApicReg->pszHasPendingIrqR0
2385 || pApicReg->pszSetBaseR0
2386 || pApicReg->pszGetBaseR0
2387 || pApicReg->pszSetTPRR0
2388 || pApicReg->pszGetTPRR0
2389 || pApicReg->pszWriteMSRR0
2390 || pApicReg->pszReadMSRR0
2391 || pApicReg->pszBusDeliverR0
2392 || pApicReg->pszLocalInterruptR0)
2393 && ( !VALID_PTR(pApicReg->pszGetInterruptR0)
2394 || !VALID_PTR(pApicReg->pszHasPendingIrqR0)
2395 || !VALID_PTR(pApicReg->pszSetBaseR0)
2396 || !VALID_PTR(pApicReg->pszGetBaseR0)
2397 || !VALID_PTR(pApicReg->pszSetTPRR0)
2398 || !VALID_PTR(pApicReg->pszGetTPRR0)
2399 || !VALID_PTR(pApicReg->pszReadMSRR0)
2400 || !VALID_PTR(pApicReg->pszWriteMSRR0)
2401 || !VALID_PTR(pApicReg->pszBusDeliverR0)
2402 || !VALID_PTR(pApicReg->pszLocalInterruptR0))
2403 )
2404 {
2405 Assert(VALID_PTR(pApicReg->pszGetInterruptR0));
2406 Assert(VALID_PTR(pApicReg->pszHasPendingIrqR0));
2407 Assert(VALID_PTR(pApicReg->pszSetBaseR0));
2408 Assert(VALID_PTR(pApicReg->pszGetBaseR0));
2409 Assert(VALID_PTR(pApicReg->pszSetTPRR0));
2410 Assert(VALID_PTR(pApicReg->pszGetTPRR0));
2411 Assert(VALID_PTR(pApicReg->pszReadMSRR0));
2412 Assert(VALID_PTR(pApicReg->pszWriteMSRR0));
2413 Assert(VALID_PTR(pApicReg->pszBusDeliverR0));
2414 Assert(VALID_PTR(pApicReg->pszLocalInterruptR0));
2415 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc (R0 callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2416 return VERR_INVALID_PARAMETER;
2417 }
2418 if (!ppApicHlpR3)
2419 {
2420 Assert(ppApicHlpR3);
2421 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc (ppApicHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2422 return VERR_INVALID_PARAMETER;
2423 }
2424
2425 /*
2426 * Only one APIC device. On SMP we have single logical device covering all LAPICs,
2427 * as they need to communicate and share state easily.
2428 */
2429 PVM pVM = pDevIns->Internal.s.pVMR3;
2430 if (pVM->pdm.s.Apic.pDevInsR3)
2431 {
2432 AssertMsgFailed(("Only one apic device is supported!\n"));
2433 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2434 return VERR_INVALID_PARAMETER;
2435 }
2436
2437 /*
2438 * Resolve & initialize the RC bits.
2439 */
2440 if (pApicReg->pszGetInterruptRC)
2441 {
2442 int rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszGetInterruptRC, &pVM->pdm.s.Apic.pfnGetInterruptRC);
2443 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszGetInterruptRC, rc));
2444 if (RT_SUCCESS(rc))
2445 {
2446 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszHasPendingIrqRC, &pVM->pdm.s.Apic.pfnHasPendingIrqRC);
2447 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszHasPendingIrqRC, rc));
2448 }
2449 if (RT_SUCCESS(rc))
2450 {
2451 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszSetBaseRC, &pVM->pdm.s.Apic.pfnSetBaseRC);
2452 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszSetBaseRC, rc));
2453 }
2454 if (RT_SUCCESS(rc))
2455 {
2456 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszGetBaseRC, &pVM->pdm.s.Apic.pfnGetBaseRC);
2457 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszGetBaseRC, rc));
2458 }
2459 if (RT_SUCCESS(rc))
2460 {
2461 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszSetTPRRC, &pVM->pdm.s.Apic.pfnSetTPRRC);
2462 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszSetTPRRC, rc));
2463 }
2464 if (RT_SUCCESS(rc))
2465 {
2466 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszGetTPRRC, &pVM->pdm.s.Apic.pfnGetTPRRC);
2467 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszGetTPRRC, rc));
2468 }
2469 if (RT_SUCCESS(rc))
2470 {
2471 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszWriteMSRRC, &pVM->pdm.s.Apic.pfnWriteMSRRC);
2472 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszWriteMSRRC, rc));
2473 }
2474 if (RT_SUCCESS(rc))
2475 {
2476 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszReadMSRRC, &pVM->pdm.s.Apic.pfnReadMSRRC);
2477 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszReadMSRRC, rc));
2478 }
2479 if (RT_SUCCESS(rc))
2480 {
2481 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszBusDeliverRC, &pVM->pdm.s.Apic.pfnBusDeliverRC);
2482 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszBusDeliverRC, rc));
2483 }
2484 if (RT_SUCCESS(rc))
2485 {
2486 rc = pdmR3DevGetSymbolRCLazy(pDevIns, pApicReg->pszLocalInterruptRC, &pVM->pdm.s.Apic.pfnLocalInterruptRC);
2487 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pApicReg->pszLocalInterruptRC, rc));
2488 }
2489 if (RT_FAILURE(rc))
2490 {
2491 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2492 return rc;
2493 }
2494 pVM->pdm.s.Apic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2495 }
2496 else
2497 {
2498 pVM->pdm.s.Apic.pDevInsRC = 0;
2499 pVM->pdm.s.Apic.pfnGetInterruptRC = 0;
2500 pVM->pdm.s.Apic.pfnHasPendingIrqRC = 0;
2501 pVM->pdm.s.Apic.pfnSetBaseRC = 0;
2502 pVM->pdm.s.Apic.pfnGetBaseRC = 0;
2503 pVM->pdm.s.Apic.pfnSetTPRRC = 0;
2504 pVM->pdm.s.Apic.pfnGetTPRRC = 0;
2505 pVM->pdm.s.Apic.pfnWriteMSRRC = 0;
2506 pVM->pdm.s.Apic.pfnReadMSRRC = 0;
2507 pVM->pdm.s.Apic.pfnBusDeliverRC = 0;
2508 pVM->pdm.s.Apic.pfnLocalInterruptRC = 0;
2509 }
2510
2511 /*
2512 * Resolve & initialize the R0 bits.
2513 */
2514 if (pApicReg->pszGetInterruptR0)
2515 {
2516 int rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszGetInterruptR0, &pVM->pdm.s.Apic.pfnGetInterruptR0);
2517 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszGetInterruptR0, rc));
2518 if (RT_SUCCESS(rc))
2519 {
2520 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszHasPendingIrqR0, &pVM->pdm.s.Apic.pfnHasPendingIrqR0);
2521 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszHasPendingIrqR0, rc));
2522 }
2523 if (RT_SUCCESS(rc))
2524 {
2525 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszSetBaseR0, &pVM->pdm.s.Apic.pfnSetBaseR0);
2526 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszSetBaseR0, rc));
2527 }
2528 if (RT_SUCCESS(rc))
2529 {
2530 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszGetBaseR0, &pVM->pdm.s.Apic.pfnGetBaseR0);
2531 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszGetBaseR0, rc));
2532 }
2533 if (RT_SUCCESS(rc))
2534 {
2535 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszSetTPRR0, &pVM->pdm.s.Apic.pfnSetTPRR0);
2536 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszSetTPRR0, rc));
2537 }
2538 if (RT_SUCCESS(rc))
2539 {
2540 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszGetTPRR0, &pVM->pdm.s.Apic.pfnGetTPRR0);
2541 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszGetTPRR0, rc));
2542 }
2543 if (RT_SUCCESS(rc))
2544 {
2545 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszWriteMSRR0, &pVM->pdm.s.Apic.pfnWriteMSRR0);
2546 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszWriteMSRR0, rc));
2547 }
2548 if (RT_SUCCESS(rc))
2549 {
2550 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszReadMSRR0, &pVM->pdm.s.Apic.pfnReadMSRR0);
2551 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszReadMSRR0, rc));
2552 }
2553 if (RT_SUCCESS(rc))
2554 {
2555 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszBusDeliverR0, &pVM->pdm.s.Apic.pfnBusDeliverR0);
2556 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszBusDeliverR0, rc));
2557 }
2558 if (RT_SUCCESS(rc))
2559 {
2560 rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pApicReg->pszLocalInterruptR0, &pVM->pdm.s.Apic.pfnLocalInterruptR0);
2561 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pApicReg->pszLocalInterruptR0, rc));
2562 }
2563 if (RT_FAILURE(rc))
2564 {
2565 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2566 return rc;
2567 }
2568 pVM->pdm.s.Apic.pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2569 Assert(pVM->pdm.s.Apic.pDevInsR0);
2570 }
2571 else
2572 {
2573 pVM->pdm.s.Apic.pfnGetInterruptR0 = 0;
2574 pVM->pdm.s.Apic.pfnHasPendingIrqR0 = 0;
2575 pVM->pdm.s.Apic.pfnSetBaseR0 = 0;
2576 pVM->pdm.s.Apic.pfnGetBaseR0 = 0;
2577 pVM->pdm.s.Apic.pfnSetTPRR0 = 0;
2578 pVM->pdm.s.Apic.pfnGetTPRR0 = 0;
2579 pVM->pdm.s.Apic.pfnWriteMSRR0 = 0;
2580 pVM->pdm.s.Apic.pfnReadMSRR0 = 0;
2581 pVM->pdm.s.Apic.pfnBusDeliverR0 = 0;
2582 pVM->pdm.s.Apic.pfnLocalInterruptR0 = 0;
2583 pVM->pdm.s.Apic.pDevInsR0 = 0;
2584 }
2585
2586 /*
2587 * Initialize the HC bits.
2588 */
2589 pVM->pdm.s.Apic.pDevInsR3 = pDevIns;
2590 pVM->pdm.s.Apic.pfnGetInterruptR3 = pApicReg->pfnGetInterruptR3;
2591 pVM->pdm.s.Apic.pfnHasPendingIrqR3 = pApicReg->pfnHasPendingIrqR3;
2592 pVM->pdm.s.Apic.pfnSetBaseR3 = pApicReg->pfnSetBaseR3;
2593 pVM->pdm.s.Apic.pfnGetBaseR3 = pApicReg->pfnGetBaseR3;
2594 pVM->pdm.s.Apic.pfnSetTPRR3 = pApicReg->pfnSetTPRR3;
2595 pVM->pdm.s.Apic.pfnGetTPRR3 = pApicReg->pfnGetTPRR3;
2596 pVM->pdm.s.Apic.pfnWriteMSRR3 = pApicReg->pfnWriteMSRR3;
2597 pVM->pdm.s.Apic.pfnReadMSRR3 = pApicReg->pfnReadMSRR3;
2598 pVM->pdm.s.Apic.pfnBusDeliverR3 = pApicReg->pfnBusDeliverR3;
2599 pVM->pdm.s.Apic.pfnLocalInterruptR3 = pApicReg->pfnLocalInterruptR3;
2600 Log(("PDM: Registered APIC device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
2601
2602 /* set the helper pointer and return. */
2603 *ppApicHlpR3 = &g_pdmR3DevApicHlp;
2604 LogFlow(("pdmR3DevHlp_APICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2605 return VINF_SUCCESS;
2606}
2607
2608
2609/** @interface_method_impl{PDMDEVHLPR3,pfnIOAPICRegister} */
2610static DECLCALLBACK(int) pdmR3DevHlp_IOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3)
2611{
2612 PDMDEV_ASSERT_DEVINS(pDevIns);
2613 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2614 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: pIoApicReg=%p:{.u32Version=%#x, .pfnSetIrqR3=%p, .pszSetIrqRC=%p:{%s}, .pszSetIrqR0=%p:{%s}} ppIoApicHlpR3=%p\n",
2615 pDevIns->pReg->szName, pDevIns->iInstance, pIoApicReg, pIoApicReg->u32Version, pIoApicReg->pfnSetIrqR3,
2616 pIoApicReg->pszSetIrqRC, pIoApicReg->pszSetIrqRC, pIoApicReg->pszSetIrqR0, pIoApicReg->pszSetIrqR0, ppIoApicHlpR3));
2617
2618 /*
2619 * Validate input.
2620 */
2621 if (pIoApicReg->u32Version != PDM_IOAPICREG_VERSION)
2622 {
2623 AssertMsgFailed(("u32Version=%#x expected %#x\n", pIoApicReg->u32Version, PDM_IOAPICREG_VERSION));
2624 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2625 return VERR_INVALID_PARAMETER;
2626 }
2627 if (!pIoApicReg->pfnSetIrqR3 || !pIoApicReg->pfnSendMsiR3)
2628 {
2629 Assert(pIoApicReg->pfnSetIrqR3);
2630 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (R3 callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2631 return VERR_INVALID_PARAMETER;
2632 }
2633 if ( pIoApicReg->pszSetIrqRC
2634 && !VALID_PTR(pIoApicReg->pszSetIrqRC))
2635 {
2636 Assert(VALID_PTR(pIoApicReg->pszSetIrqRC));
2637 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2638 return VERR_INVALID_PARAMETER;
2639 }
2640 if ( pIoApicReg->pszSendMsiRC
2641 && !VALID_PTR(pIoApicReg->pszSendMsiRC))
2642 {
2643 Assert(VALID_PTR(pIoApicReg->pszSendMsiRC));
2644 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2645 return VERR_INVALID_PARAMETER;
2646 }
2647 if ( pIoApicReg->pszSetIrqR0
2648 && !VALID_PTR(pIoApicReg->pszSetIrqR0))
2649 {
2650 Assert(VALID_PTR(pIoApicReg->pszSetIrqR0));
2651 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2652 return VERR_INVALID_PARAMETER;
2653 }
2654 if ( pIoApicReg->pszSendMsiR0
2655 && !VALID_PTR(pIoApicReg->pszSendMsiR0))
2656 {
2657 Assert(VALID_PTR(pIoApicReg->pszSendMsiR0));
2658 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (GC callbacks)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2659 return VERR_INVALID_PARAMETER;
2660 }
2661 if (!ppIoApicHlpR3)
2662 {
2663 Assert(ppIoApicHlpR3);
2664 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (ppApicHlp)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2665 return VERR_INVALID_PARAMETER;
2666 }
2667
2668 /*
2669 * The I/O APIC requires the APIC to be present (hacks++).
2670 * If the I/O APIC does GC stuff so must the APIC.
2671 */
2672 PVM pVM = pDevIns->Internal.s.pVMR3;
2673 if (!pVM->pdm.s.Apic.pDevInsR3)
2674 {
2675 AssertMsgFailed(("Configuration error / Init order error! No APIC!\n"));
2676 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (no APIC)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2677 return VERR_INVALID_PARAMETER;
2678 }
2679 if ( pIoApicReg->pszSetIrqRC
2680 && !pVM->pdm.s.Apic.pDevInsRC)
2681 {
2682 AssertMsgFailed(("Configuration error! APIC doesn't do GC, I/O APIC does!\n"));
2683 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (no GC APIC)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2684 return VERR_INVALID_PARAMETER;
2685 }
2686
2687 /*
2688 * Only one I/O APIC device.
2689 */
2690 if (pVM->pdm.s.IoApic.pDevInsR3)
2691 {
2692 AssertMsgFailed(("Only one ioapic device is supported!\n"));
2693 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc (only one)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2694 return VERR_INVALID_PARAMETER;
2695 }
2696
2697 /*
2698 * Resolve & initialize the GC bits.
2699 */
2700 if (pIoApicReg->pszSetIrqRC)
2701 {
2702 int rc = pdmR3DevGetSymbolRCLazy(pDevIns, pIoApicReg->pszSetIrqRC, &pVM->pdm.s.IoApic.pfnSetIrqRC);
2703 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pIoApicReg->pszSetIrqRC, rc));
2704 if (RT_FAILURE(rc))
2705 {
2706 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2707 return rc;
2708 }
2709 pVM->pdm.s.IoApic.pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
2710 }
2711 else
2712 {
2713 pVM->pdm.s.IoApic.pDevInsRC = 0;
2714 pVM->pdm.s.IoApic.pfnSetIrqRC = 0;
2715 }
2716
2717 if (pIoApicReg->pszSendMsiRC)
2718 {
2719 int rc = pdmR3DevGetSymbolRCLazy(pDevIns, pIoApicReg->pszSetIrqRC, &pVM->pdm.s.IoApic.pfnSendMsiRC);
2720 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szRCMod, pIoApicReg->pszSendMsiRC, rc));
2721 if (RT_FAILURE(rc))
2722 {
2723 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2724 return rc;
2725 }
2726 }
2727 else
2728 {
2729 pVM->pdm.s.IoApic.pfnSendMsiRC = 0;
2730 }
2731
2732 /*
2733 * Resolve & initialize the R0 bits.
2734 */
2735 if (pIoApicReg->pszSetIrqR0)
2736 {
2737 int rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pIoApicReg->pszSetIrqR0, &pVM->pdm.s.IoApic.pfnSetIrqR0);
2738 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pIoApicReg->pszSetIrqR0, rc));
2739 if (RT_FAILURE(rc))
2740 {
2741 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2742 return rc;
2743 }
2744 pVM->pdm.s.IoApic.pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
2745 Assert(pVM->pdm.s.IoApic.pDevInsR0);
2746 }
2747 else
2748 {
2749 pVM->pdm.s.IoApic.pfnSetIrqR0 = 0;
2750 pVM->pdm.s.IoApic.pDevInsR0 = 0;
2751 }
2752
2753 if (pIoApicReg->pszSendMsiR0)
2754 {
2755 int rc = pdmR3DevGetSymbolR0Lazy(pDevIns, pIoApicReg->pszSetIrqR0, &pVM->pdm.s.IoApic.pfnSendMsiR0);
2756 AssertMsgRC(rc, ("%s::%s rc=%Rrc\n", pDevIns->pReg->szR0Mod, pIoApicReg->pszSendMsiR0, rc));
2757 if (RT_FAILURE(rc))
2758 {
2759 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2760 return rc;
2761 }
2762 }
2763 else
2764 {
2765 pVM->pdm.s.IoApic.pfnSendMsiR0 = 0;
2766 }
2767
2768
2769 /*
2770 * Initialize the R3 bits.
2771 */
2772 pVM->pdm.s.IoApic.pDevInsR3 = pDevIns;
2773 pVM->pdm.s.IoApic.pfnSetIrqR3 = pIoApicReg->pfnSetIrqR3;
2774 pVM->pdm.s.IoApic.pfnSendMsiR3 = pIoApicReg->pfnSendMsiR3;
2775 Log(("PDM: Registered I/O APIC device '%s'/%d pDevIns=%p\n", pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
2776
2777 /* set the helper pointer and return. */
2778 *ppIoApicHlpR3 = &g_pdmR3DevIoApicHlp;
2779 LogFlow(("pdmR3DevHlp_IOAPICRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2780 return VINF_SUCCESS;
2781}
2782
2783
2784/** @interface_method_impl{PDMDEVHLPR3,pfnHPETRegister} */
2785static DECLCALLBACK(int) pdmR3DevHlp_HPETRegister(PPDMDEVINS pDevIns, PPDMHPETREG pHpetReg, PCPDMHPETHLPR3 *ppHpetHlpR3)
2786{
2787 PDMDEV_ASSERT_DEVINS(pDevIns);
2788 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2789 LogFlow(("pdmR3DevHlp_HPETRegister: caller='%s'/%d:\n"));
2790
2791 /*
2792 * Validate input.
2793 */
2794 if (pHpetReg->u32Version != PDM_HPETREG_VERSION)
2795 {
2796 AssertMsgFailed(("u32Version=%#x expected %#x\n", pHpetReg->u32Version, PDM_HPETREG_VERSION));
2797 LogFlow(("pdmR3DevHlp_HPETRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2798 return VERR_INVALID_PARAMETER;
2799 }
2800
2801 if (!ppHpetHlpR3)
2802 {
2803 Assert(ppHpetHlpR3);
2804 LogFlow(("pdmR3DevHlp_HPETRegister: caller='%s'/%d: returns %Rrc (ppApicHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2805 return VERR_INVALID_PARAMETER;
2806 }
2807
2808 /* set the helper pointer and return. */
2809 *ppHpetHlpR3 = &g_pdmR3DevHpetHlp;
2810 LogFlow(("pdmR3DevHlp_HPETRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2811 return VINF_SUCCESS;
2812}
2813
2814/** @interface_method_impl{PDMDEVHLPR3,pfnPciRawRegister} */
2815static DECLCALLBACK(int) pdmR3DevHlp_PciRawRegister(PPDMDEVINS pDevIns, PPDMPCIRAWREG pPciRawReg, PCPDMPCIRAWHLPR3 *ppPciRawHlpR3)
2816{
2817 PDMDEV_ASSERT_DEVINS(pDevIns);
2818 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2819 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d:\n"));
2820
2821 /*
2822 * Validate input.
2823 */
2824 if (pPciRawReg->u32Version != PDM_PCIRAWREG_VERSION)
2825 {
2826 AssertMsgFailed(("u32Version=%#x expected %#x\n", pPciRawReg->u32Version, PDM_PCIRAWREG_VERSION));
2827 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc (version)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2828 return VERR_INVALID_PARAMETER;
2829 }
2830
2831 if (!ppPciRawHlpR3)
2832 {
2833 Assert(ppPciRawHlpR3);
2834 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc (ppApicHlpR3)\n", pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2835 return VERR_INVALID_PARAMETER;
2836 }
2837
2838 /* set the helper pointer and return. */
2839 *ppPciRawHlpR3 = &g_pdmR3DevPciRawHlp;
2840 LogFlow(("pdmR3DevHlp_PciRawRegister: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, VINF_SUCCESS));
2841 return VINF_SUCCESS;
2842}
2843
2844/** @interface_method_impl{PDMDEVHLPR3,pfnDMACRegister} */
2845static DECLCALLBACK(int) pdmR3DevHlp_DMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp)
2846{
2847 PDMDEV_ASSERT_DEVINS(pDevIns);
2848 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2849 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",
2850 pDevIns->pReg->szName, pDevIns->iInstance, pDmacReg, pDmacReg->u32Version, pDmacReg->pfnRun, pDmacReg->pfnRegister,
2851 pDmacReg->pfnReadMemory, pDmacReg->pfnWriteMemory, pDmacReg->pfnSetDREQ, pDmacReg->pfnGetChannelMode, ppDmacHlp));
2852
2853 /*
2854 * Validate input.
2855 */
2856 if (pDmacReg->u32Version != PDM_DMACREG_VERSION)
2857 {
2858 AssertMsgFailed(("u32Version=%#x expected %#x\n", pDmacReg->u32Version,
2859 PDM_DMACREG_VERSION));
2860 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (version)\n",
2861 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2862 return VERR_INVALID_PARAMETER;
2863 }
2864 if ( !pDmacReg->pfnRun
2865 || !pDmacReg->pfnRegister
2866 || !pDmacReg->pfnReadMemory
2867 || !pDmacReg->pfnWriteMemory
2868 || !pDmacReg->pfnSetDREQ
2869 || !pDmacReg->pfnGetChannelMode)
2870 {
2871 Assert(pDmacReg->pfnRun);
2872 Assert(pDmacReg->pfnRegister);
2873 Assert(pDmacReg->pfnReadMemory);
2874 Assert(pDmacReg->pfnWriteMemory);
2875 Assert(pDmacReg->pfnSetDREQ);
2876 Assert(pDmacReg->pfnGetChannelMode);
2877 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (callbacks)\n",
2878 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2879 return VERR_INVALID_PARAMETER;
2880 }
2881
2882 if (!ppDmacHlp)
2883 {
2884 Assert(ppDmacHlp);
2885 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc (ppDmacHlp)\n",
2886 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2887 return VERR_INVALID_PARAMETER;
2888 }
2889
2890 /*
2891 * Only one DMA device.
2892 */
2893 PVM pVM = pDevIns->Internal.s.pVMR3;
2894 if (pVM->pdm.s.pDmac)
2895 {
2896 AssertMsgFailed(("Only one DMA device is supported!\n"));
2897 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc\n",
2898 pDevIns->pReg->szName, pDevIns->iInstance, VERR_INVALID_PARAMETER));
2899 return VERR_INVALID_PARAMETER;
2900 }
2901
2902 /*
2903 * Allocate and initialize pci bus structure.
2904 */
2905 int rc = VINF_SUCCESS;
2906 PPDMDMAC pDmac = (PPDMDMAC)MMR3HeapAlloc(pDevIns->Internal.s.pVMR3, MM_TAG_PDM_DEVICE, sizeof(*pDmac));
2907 if (pDmac)
2908 {
2909 pDmac->pDevIns = pDevIns;
2910 pDmac->Reg = *pDmacReg;
2911 pVM->pdm.s.pDmac = pDmac;
2912
2913 /* set the helper pointer. */
2914 *ppDmacHlp = &g_pdmR3DevDmacHlp;
2915 Log(("PDM: Registered DMAC device '%s'/%d pDevIns=%p\n",
2916 pDevIns->pReg->szName, pDevIns->iInstance, pDevIns));
2917 }
2918 else
2919 rc = VERR_NO_MEMORY;
2920
2921 LogFlow(("pdmR3DevHlp_DMACRegister: caller='%s'/%d: returns %Rrc\n",
2922 pDevIns->pReg->szName, pDevIns->iInstance, rc));
2923 return rc;
2924}
2925
2926
2927/**
2928 * @copydoc PDMDEVHLPR3::pfnRegisterVMMDevHeap
2929 */
2930static DECLCALLBACK(int) pdmR3DevHlp_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize)
2931{
2932 PDMDEV_ASSERT_DEVINS(pDevIns);
2933 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2934
2935 int rc = PDMR3RegisterVMMDevHeap(pDevIns->Internal.s.pVMR3, GCPhys, pvHeap, cbSize);
2936 return rc;
2937}
2938
2939
2940/**
2941 * @copydoc PDMDEVHLPR3::pfnUnregisterVMMDevHeap
2942 */
2943static DECLCALLBACK(int) pdmR3DevHlp_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys)
2944{
2945 PDMDEV_ASSERT_DEVINS(pDevIns);
2946 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
2947
2948 int rc = PDMR3UnregisterVMMDevHeap(pDevIns->Internal.s.pVMR3, GCPhys);
2949 return rc;
2950}
2951
2952
2953/** @interface_method_impl{PDMDEVHLPR3,pfnVMReset} */
2954static DECLCALLBACK(int) pdmR3DevHlp_VMReset(PPDMDEVINS pDevIns)
2955{
2956 PDMDEV_ASSERT_DEVINS(pDevIns);
2957 PVM pVM = pDevIns->Internal.s.pVMR3;
2958 VM_ASSERT_EMT(pVM);
2959 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: VM_FF_RESET %d -> 1\n",
2960 pDevIns->pReg->szName, pDevIns->iInstance, VM_FF_ISSET(pVM, VM_FF_RESET)));
2961
2962 /*
2963 * We postpone this operation because we're likely to be inside a I/O instruction
2964 * and the EIP will be updated when we return.
2965 * We still return VINF_EM_RESET to break out of any execution loops and force FF evaluation.
2966 */
2967 bool fHaltOnReset;
2968 int rc = CFGMR3QueryBool(CFGMR3GetChild(CFGMR3GetRoot(pVM), "PDM"), "HaltOnReset", &fHaltOnReset);
2969 if (RT_SUCCESS(rc) && fHaltOnReset)
2970 {
2971 Log(("pdmR3DevHlp_VMReset: Halt On Reset!\n"));
2972 rc = VINF_EM_HALT;
2973 }
2974 else
2975 {
2976 VM_FF_SET(pVM, VM_FF_RESET);
2977 rc = VINF_EM_RESET;
2978 }
2979
2980 LogFlow(("pdmR3DevHlp_VMReset: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
2981 return rc;
2982}
2983
2984
2985/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspend} */
2986static DECLCALLBACK(int) pdmR3DevHlp_VMSuspend(PPDMDEVINS pDevIns)
2987{
2988 int rc;
2989 PDMDEV_ASSERT_DEVINS(pDevIns);
2990 PVM pVM = pDevIns->Internal.s.pVMR3;
2991 VM_ASSERT_EMT(pVM);
2992 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d:\n",
2993 pDevIns->pReg->szName, pDevIns->iInstance));
2994
2995 /** @todo Always take the SMP path - fewer code paths. */
2996 if (pVM->cCpus > 1)
2997 {
2998 /* We own the IOM lock here and could cause a deadlock by waiting for a VCPU that is blocking on the IOM lock. */
2999 rc = VMR3ReqCallNoWaitU(pVM->pUVM, VMCPUID_ANY_QUEUE, (PFNRT)VMR3Suspend, 1, pVM);
3000 AssertRC(rc);
3001 rc = VINF_EM_SUSPEND;
3002 }
3003 else
3004 rc = VMR3Suspend(pVM);
3005
3006 LogFlow(("pdmR3DevHlp_VMSuspend: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
3007 return rc;
3008}
3009
3010
3011/**
3012 * Worker for pdmR3DevHlp_VMSuspendSaveAndPowerOff that is invoked via a queued
3013 * EMT request to avoid deadlocks.
3014 *
3015 * @returns VBox status code fit for scheduling.
3016 * @param pVM The VM handle.
3017 * @param pDevIns The device that triggered this action.
3018 */
3019static DECLCALLBACK(int) pdmR3DevHlp_VMSuspendSaveAndPowerOffWorker(PVM pVM, PPDMDEVINS pDevIns)
3020{
3021 /*
3022 * Suspend the VM first then do the saving.
3023 */
3024 int rc = VMR3Suspend(pVM);
3025 if (RT_SUCCESS(rc))
3026 {
3027 rc = pVM->pUVM->pVmm2UserMethods->pfnSaveState(pVM->pUVM->pVmm2UserMethods, pVM);
3028
3029 /*
3030 * On success, power off the VM, on failure we'll leave it suspended.
3031 */
3032 if (RT_SUCCESS(rc))
3033 {
3034 rc = VMR3PowerOff(pVM);
3035 if (RT_FAILURE(rc))
3036 LogRel(("%s/SSP: VMR3PowerOff failed: %Rrc\n", pDevIns->pReg->szName, rc));
3037 }
3038 else
3039 LogRel(("%s/SSP: pfnSaveState failed: %Rrc\n", pDevIns->pReg->szName, rc));
3040 }
3041 else
3042 LogRel(("%s/SSP: Suspend failed: %Rrc\n", pDevIns->pReg->szName, rc));
3043 return rc;
3044}
3045
3046
3047/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspendSaveAndPowerOff} */
3048static DECLCALLBACK(int) pdmR3DevHlp_VMSuspendSaveAndPowerOff(PPDMDEVINS pDevIns)
3049{
3050 PDMDEV_ASSERT_DEVINS(pDevIns);
3051 PVM pVM = pDevIns->Internal.s.pVMR3;
3052 VM_ASSERT_EMT(pVM);
3053 LogFlow(("pdmR3DevHlp_VMSuspendSaveAndPowerOff: caller='%s'/%d:\n",
3054 pDevIns->pReg->szName, pDevIns->iInstance));
3055
3056 int rc;
3057 if ( pVM->pUVM->pVmm2UserMethods
3058 && pVM->pUVM->pVmm2UserMethods->pfnSaveState)
3059 {
3060 rc = VMR3ReqCallNoWaitU(pVM->pUVM, VMCPUID_ANY_QUEUE, (PFNRT)pdmR3DevHlp_VMSuspendSaveAndPowerOffWorker, 2, pVM, pDevIns);
3061 if (RT_SUCCESS(rc))
3062 {
3063 LogRel(("%s: Suspending, Saving and Powering Off the VM\n", pDevIns->pReg->szName));
3064 rc = VINF_EM_SUSPEND;
3065 }
3066 }
3067 else
3068 rc = VERR_NOT_SUPPORTED;
3069
3070 LogFlow(("pdmR3DevHlp_VMSuspendSaveAndPowerOff: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
3071 return rc;
3072}
3073
3074
3075/** @interface_method_impl{PDMDEVHLPR3,pfnVMPowerOff} */
3076static DECLCALLBACK(int) pdmR3DevHlp_VMPowerOff(PPDMDEVINS pDevIns)
3077{
3078 int rc;
3079 PDMDEV_ASSERT_DEVINS(pDevIns);
3080 PVM pVM = pDevIns->Internal.s.pVMR3;
3081 VM_ASSERT_EMT(pVM);
3082 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d:\n",
3083 pDevIns->pReg->szName, pDevIns->iInstance));
3084
3085 /** @todo Always take the SMP path - fewer code paths. */
3086 if (pVM->cCpus > 1)
3087 {
3088 /* We own the IOM lock here and could cause a deadlock by waiting for a VCPU that is blocking on the IOM lock. */
3089 rc = VMR3ReqCallNoWaitU(pVM->pUVM, VMCPUID_ANY_QUEUE, (PFNRT)VMR3PowerOff, 1, pVM);
3090 AssertRC(rc);
3091 /* Set the VCPU state to stopped here as well to make sure no
3092 * inconsistency with the EM state occurs.
3093 */
3094 VMCPU_SET_STATE(VMMGetCpu(pVM), VMCPUSTATE_STOPPED);
3095 rc = VINF_EM_OFF;
3096 }
3097 else
3098 rc = VMR3PowerOff(pVM);
3099
3100 LogFlow(("pdmR3DevHlp_VMPowerOff: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
3101 return rc;
3102}
3103
3104
3105/** @interface_method_impl{PDMDEVHLPR3,pfnA20IsEnabled} */
3106static DECLCALLBACK(bool) pdmR3DevHlp_A20IsEnabled(PPDMDEVINS pDevIns)
3107{
3108 PDMDEV_ASSERT_DEVINS(pDevIns);
3109 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3110
3111 bool fRc = PGMPhysIsA20Enabled(VMMGetCpu(pDevIns->Internal.s.pVMR3));
3112
3113 LogFlow(("pdmR3DevHlp_A20IsEnabled: caller='%s'/%d: returns %d\n", pDevIns->pReg->szName, pDevIns->iInstance, fRc));
3114 return fRc;
3115}
3116
3117
3118/** @interface_method_impl{PDMDEVHLPR3,pfnA20Set} */
3119static DECLCALLBACK(void) pdmR3DevHlp_A20Set(PPDMDEVINS pDevIns, bool fEnable)
3120{
3121 PDMDEV_ASSERT_DEVINS(pDevIns);
3122 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3123 LogFlow(("pdmR3DevHlp_A20Set: caller='%s'/%d: fEnable=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, fEnable));
3124 PGMR3PhysSetA20(VMMGetCpu(pDevIns->Internal.s.pVMR3), fEnable);
3125}
3126
3127
3128/** @interface_method_impl{PDMDEVHLPR3,pfnGetCpuId} */
3129static DECLCALLBACK(void) pdmR3DevHlp_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
3130 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
3131{
3132 PDMDEV_ASSERT_DEVINS(pDevIns);
3133 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
3134
3135 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: iLeaf=%d pEax=%p pEbx=%p pEcx=%p pEdx=%p\n",
3136 pDevIns->pReg->szName, pDevIns->iInstance, iLeaf, pEax, pEbx, pEcx, pEdx));
3137 AssertPtr(pEax); AssertPtr(pEbx); AssertPtr(pEcx); AssertPtr(pEdx);
3138
3139 CPUMGetGuestCpuId(VMMGetCpu(pDevIns->Internal.s.pVMR3), iLeaf, pEax, pEbx, pEcx, pEdx);
3140
3141 LogFlow(("pdmR3DevHlp_GetCpuId: caller='%s'/%d: returns void - *pEax=%#x *pEbx=%#x *pEcx=%#x *pEdx=%#x\n",
3142 pDevIns->pReg->szName, pDevIns->iInstance, *pEax, *pEbx, *pEcx, *pEdx));
3143}
3144
3145
3146/**
3147 * The device helper structure for trusted devices.
3148 */
3149const PDMDEVHLPR3 g_pdmR3DevHlpTrusted =
3150{
3151 PDM_DEVHLPR3_VERSION,
3152 pdmR3DevHlp_IOPortRegister,
3153 pdmR3DevHlp_IOPortRegisterRC,
3154 pdmR3DevHlp_IOPortRegisterR0,
3155 pdmR3DevHlp_IOPortDeregister,
3156 pdmR3DevHlp_MMIORegister,
3157 pdmR3DevHlp_MMIORegisterRC,
3158 pdmR3DevHlp_MMIORegisterR0,
3159 pdmR3DevHlp_MMIODeregister,
3160 pdmR3DevHlp_MMIO2Register,
3161 pdmR3DevHlp_MMIO2Deregister,
3162 pdmR3DevHlp_MMIO2Map,
3163 pdmR3DevHlp_MMIO2Unmap,
3164 pdmR3DevHlp_MMHyperMapMMIO2,
3165 pdmR3DevHlp_MMIO2MapKernel,
3166 pdmR3DevHlp_ROMRegister,
3167 pdmR3DevHlp_ROMProtectShadow,
3168 pdmR3DevHlp_SSMRegister,
3169 pdmR3DevHlp_TMTimerCreate,
3170 pdmR3DevHlp_TMUtcNow,
3171 pdmR3DevHlp_PhysRead,
3172 pdmR3DevHlp_PhysWrite,
3173 pdmR3DevHlp_PhysGCPhys2CCPtr,
3174 pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly,
3175 pdmR3DevHlp_PhysReleasePageMappingLock,
3176 pdmR3DevHlp_PhysReadGCVirt,
3177 pdmR3DevHlp_PhysWriteGCVirt,
3178 pdmR3DevHlp_PhysGCPtr2GCPhys,
3179 pdmR3DevHlp_MMHeapAlloc,
3180 pdmR3DevHlp_MMHeapAllocZ,
3181 pdmR3DevHlp_MMHeapFree,
3182 pdmR3DevHlp_VMState,
3183 pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet,
3184 pdmR3DevHlp_VMSetError,
3185 pdmR3DevHlp_VMSetErrorV,
3186 pdmR3DevHlp_VMSetRuntimeError,
3187 pdmR3DevHlp_VMSetRuntimeErrorV,
3188 pdmR3DevHlp_DBGFStopV,
3189 pdmR3DevHlp_DBGFInfoRegister,
3190 pdmR3DevHlp_STAMRegister,
3191 pdmR3DevHlp_STAMRegisterF,
3192 pdmR3DevHlp_STAMRegisterV,
3193 pdmR3DevHlp_PCIRegister,
3194 pdmR3DevHlp_PCIRegisterMsi,
3195 pdmR3DevHlp_PCIIORegionRegister,
3196 pdmR3DevHlp_PCISetConfigCallbacks,
3197 pdmR3DevHlp_PCISetIrq,
3198 pdmR3DevHlp_PCISetIrqNoWait,
3199 pdmR3DevHlp_ISASetIrq,
3200 pdmR3DevHlp_ISASetIrqNoWait,
3201 pdmR3DevHlp_DriverAttach,
3202 pdmR3DevHlp_QueueCreate,
3203 pdmR3DevHlp_CritSectInit,
3204 pdmR3DevHlp_ThreadCreate,
3205 pdmR3DevHlp_SetAsyncNotification,
3206 pdmR3DevHlp_AsyncNotificationCompleted,
3207 pdmR3DevHlp_RTCRegister,
3208 pdmR3DevHlp_PCIBusRegister,
3209 pdmR3DevHlp_PICRegister,
3210 pdmR3DevHlp_APICRegister,
3211 pdmR3DevHlp_IOAPICRegister,
3212 pdmR3DevHlp_HPETRegister,
3213 pdmR3DevHlp_PciRawRegister,
3214 pdmR3DevHlp_DMACRegister,
3215 pdmR3DevHlp_DMARegister,
3216 pdmR3DevHlp_DMAReadMemory,
3217 pdmR3DevHlp_DMAWriteMemory,
3218 pdmR3DevHlp_DMASetDREQ,
3219 pdmR3DevHlp_DMAGetChannelMode,
3220 pdmR3DevHlp_DMASchedule,
3221 pdmR3DevHlp_CMOSWrite,
3222 pdmR3DevHlp_CMOSRead,
3223 pdmR3DevHlp_AssertEMT,
3224 pdmR3DevHlp_AssertOther,
3225 pdmR3DevHlp_LdrGetRCInterfaceSymbols,
3226 pdmR3DevHlp_LdrGetR0InterfaceSymbols,
3227 pdmR3DevHlp_CallR0,
3228 0,
3229 0,
3230 0,
3231 0,
3232 0,
3233 0,
3234 0,
3235 0,
3236 0,
3237 0,
3238 pdmR3DevHlp_GetVM,
3239 pdmR3DevHlp_GetVMCPU,
3240 pdmR3DevHlp_RegisterVMMDevHeap,
3241 pdmR3DevHlp_UnregisterVMMDevHeap,
3242 pdmR3DevHlp_VMReset,
3243 pdmR3DevHlp_VMSuspend,
3244 pdmR3DevHlp_VMSuspendSaveAndPowerOff,
3245 pdmR3DevHlp_VMPowerOff,
3246 pdmR3DevHlp_A20IsEnabled,
3247 pdmR3DevHlp_A20Set,
3248 pdmR3DevHlp_GetCpuId,
3249 pdmR3DevHlp_TMTimeVirtGet,
3250 pdmR3DevHlp_TMTimeVirtGetFreq,
3251 pdmR3DevHlp_TMTimeVirtGetNano,
3252 PDM_DEVHLPR3_VERSION /* the end */
3253};
3254
3255
3256
3257
3258/** @interface_method_impl{PDMDEVHLPR3,pfnGetVM} */
3259static DECLCALLBACK(PVM) pdmR3DevHlp_Untrusted_GetVM(PPDMDEVINS pDevIns)
3260{
3261 PDMDEV_ASSERT_DEVINS(pDevIns);
3262 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3263 return NULL;
3264}
3265
3266
3267/** @interface_method_impl{PDMDEVHLPR3,pfnGetVMCPU} */
3268static DECLCALLBACK(PVMCPU) pdmR3DevHlp_Untrusted_GetVMCPU(PPDMDEVINS pDevIns)
3269{
3270 PDMDEV_ASSERT_DEVINS(pDevIns);
3271 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3272 return NULL;
3273}
3274
3275
3276/** @interface_method_impl{PDMDEVHLPR3,pfnRegisterVMMDevHeap} */
3277static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_RegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbSize)
3278{
3279 PDMDEV_ASSERT_DEVINS(pDevIns);
3280 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3281 return VERR_ACCESS_DENIED;
3282}
3283
3284
3285/** @interface_method_impl{PDMDEVHLPR3,pfnUnregisterVMMDevHeap} */
3286static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_UnregisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys)
3287{
3288 PDMDEV_ASSERT_DEVINS(pDevIns);
3289 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3290 return VERR_ACCESS_DENIED;
3291}
3292
3293
3294/** @interface_method_impl{PDMDEVHLPR3,pfnVMReset} */
3295static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMReset(PPDMDEVINS pDevIns)
3296{
3297 PDMDEV_ASSERT_DEVINS(pDevIns);
3298 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3299 return VERR_ACCESS_DENIED;
3300}
3301
3302
3303/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspend} */
3304static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspend(PPDMDEVINS pDevIns)
3305{
3306 PDMDEV_ASSERT_DEVINS(pDevIns);
3307 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3308 return VERR_ACCESS_DENIED;
3309}
3310
3311
3312/** @interface_method_impl{PDMDEVHLPR3,pfnVMSuspendSaveAndPowerOff} */
3313static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMSuspendSaveAndPowerOff(PPDMDEVINS pDevIns)
3314{
3315 PDMDEV_ASSERT_DEVINS(pDevIns);
3316 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3317 return VERR_ACCESS_DENIED;
3318}
3319
3320
3321/** @interface_method_impl{PDMDEVHLPR3,pfnVMPowerOff} */
3322static DECLCALLBACK(int) pdmR3DevHlp_Untrusted_VMPowerOff(PPDMDEVINS pDevIns)
3323{
3324 PDMDEV_ASSERT_DEVINS(pDevIns);
3325 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3326 return VERR_ACCESS_DENIED;
3327}
3328
3329
3330/** @interface_method_impl{PDMDEVHLPR3,pfnA20IsEnabled} */
3331static DECLCALLBACK(bool) pdmR3DevHlp_Untrusted_A20IsEnabled(PPDMDEVINS pDevIns)
3332{
3333 PDMDEV_ASSERT_DEVINS(pDevIns);
3334 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3335 return false;
3336}
3337
3338
3339/** @interface_method_impl{PDMDEVHLPR3,pfnA20Set} */
3340static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_A20Set(PPDMDEVINS pDevIns, bool fEnable)
3341{
3342 PDMDEV_ASSERT_DEVINS(pDevIns);
3343 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3344 NOREF(fEnable);
3345}
3346
3347
3348/** @interface_method_impl{PDMDEVHLPR3,pfnGetCpuId} */
3349static DECLCALLBACK(void) pdmR3DevHlp_Untrusted_GetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf,
3350 uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
3351{
3352 PDMDEV_ASSERT_DEVINS(pDevIns);
3353 AssertReleaseMsgFailed(("Untrusted device called trusted helper! '%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
3354}
3355
3356
3357/**
3358 * The device helper structure for non-trusted devices.
3359 */
3360const PDMDEVHLPR3 g_pdmR3DevHlpUnTrusted =
3361{
3362 PDM_DEVHLPR3_VERSION,
3363 pdmR3DevHlp_IOPortRegister,
3364 pdmR3DevHlp_IOPortRegisterRC,
3365 pdmR3DevHlp_IOPortRegisterR0,
3366 pdmR3DevHlp_IOPortDeregister,
3367 pdmR3DevHlp_MMIORegister,
3368 pdmR3DevHlp_MMIORegisterRC,
3369 pdmR3DevHlp_MMIORegisterR0,
3370 pdmR3DevHlp_MMIODeregister,
3371 pdmR3DevHlp_MMIO2Register,
3372 pdmR3DevHlp_MMIO2Deregister,
3373 pdmR3DevHlp_MMIO2Map,
3374 pdmR3DevHlp_MMIO2Unmap,
3375 pdmR3DevHlp_MMHyperMapMMIO2,
3376 pdmR3DevHlp_MMIO2MapKernel,
3377 pdmR3DevHlp_ROMRegister,
3378 pdmR3DevHlp_ROMProtectShadow,
3379 pdmR3DevHlp_SSMRegister,
3380 pdmR3DevHlp_TMTimerCreate,
3381 pdmR3DevHlp_TMUtcNow,
3382 pdmR3DevHlp_PhysRead,
3383 pdmR3DevHlp_PhysWrite,
3384 pdmR3DevHlp_PhysGCPhys2CCPtr,
3385 pdmR3DevHlp_PhysGCPhys2CCPtrReadOnly,
3386 pdmR3DevHlp_PhysReleasePageMappingLock,
3387 pdmR3DevHlp_PhysReadGCVirt,
3388 pdmR3DevHlp_PhysWriteGCVirt,
3389 pdmR3DevHlp_PhysGCPtr2GCPhys,
3390 pdmR3DevHlp_MMHeapAlloc,
3391 pdmR3DevHlp_MMHeapAllocZ,
3392 pdmR3DevHlp_MMHeapFree,
3393 pdmR3DevHlp_VMState,
3394 pdmR3DevHlp_VMTeleportedAndNotFullyResumedYet,
3395 pdmR3DevHlp_VMSetError,
3396 pdmR3DevHlp_VMSetErrorV,
3397 pdmR3DevHlp_VMSetRuntimeError,
3398 pdmR3DevHlp_VMSetRuntimeErrorV,
3399 pdmR3DevHlp_DBGFStopV,
3400 pdmR3DevHlp_DBGFInfoRegister,
3401 pdmR3DevHlp_STAMRegister,
3402 pdmR3DevHlp_STAMRegisterF,
3403 pdmR3DevHlp_STAMRegisterV,
3404 pdmR3DevHlp_PCIRegister,
3405 pdmR3DevHlp_PCIRegisterMsi,
3406 pdmR3DevHlp_PCIIORegionRegister,
3407 pdmR3DevHlp_PCISetConfigCallbacks,
3408 pdmR3DevHlp_PCISetIrq,
3409 pdmR3DevHlp_PCISetIrqNoWait,
3410 pdmR3DevHlp_ISASetIrq,
3411 pdmR3DevHlp_ISASetIrqNoWait,
3412 pdmR3DevHlp_DriverAttach,
3413 pdmR3DevHlp_QueueCreate,
3414 pdmR3DevHlp_CritSectInit,
3415 pdmR3DevHlp_ThreadCreate,
3416 pdmR3DevHlp_SetAsyncNotification,
3417 pdmR3DevHlp_AsyncNotificationCompleted,
3418 pdmR3DevHlp_RTCRegister,
3419 pdmR3DevHlp_PCIBusRegister,
3420 pdmR3DevHlp_PICRegister,
3421 pdmR3DevHlp_APICRegister,
3422 pdmR3DevHlp_IOAPICRegister,
3423 pdmR3DevHlp_HPETRegister,
3424 pdmR3DevHlp_PciRawRegister,
3425 pdmR3DevHlp_DMACRegister,
3426 pdmR3DevHlp_DMARegister,
3427 pdmR3DevHlp_DMAReadMemory,
3428 pdmR3DevHlp_DMAWriteMemory,
3429 pdmR3DevHlp_DMASetDREQ,
3430 pdmR3DevHlp_DMAGetChannelMode,
3431 pdmR3DevHlp_DMASchedule,
3432 pdmR3DevHlp_CMOSWrite,
3433 pdmR3DevHlp_CMOSRead,
3434 pdmR3DevHlp_AssertEMT,
3435 pdmR3DevHlp_AssertOther,
3436 pdmR3DevHlp_LdrGetRCInterfaceSymbols,
3437 pdmR3DevHlp_LdrGetR0InterfaceSymbols,
3438 pdmR3DevHlp_CallR0,
3439 0,
3440 0,
3441 0,
3442 0,
3443 0,
3444 0,
3445 0,
3446 0,
3447 0,
3448 0,
3449 pdmR3DevHlp_Untrusted_GetVM,
3450 pdmR3DevHlp_Untrusted_GetVMCPU,
3451 pdmR3DevHlp_Untrusted_RegisterVMMDevHeap,
3452 pdmR3DevHlp_Untrusted_UnregisterVMMDevHeap,
3453 pdmR3DevHlp_Untrusted_VMReset,
3454 pdmR3DevHlp_Untrusted_VMSuspend,
3455 pdmR3DevHlp_Untrusted_VMSuspendSaveAndPowerOff,
3456 pdmR3DevHlp_Untrusted_VMPowerOff,
3457 pdmR3DevHlp_Untrusted_A20IsEnabled,
3458 pdmR3DevHlp_Untrusted_A20Set,
3459 pdmR3DevHlp_Untrusted_GetCpuId,
3460 pdmR3DevHlp_TMTimeVirtGet,
3461 pdmR3DevHlp_TMTimeVirtGetFreq,
3462 pdmR3DevHlp_TMTimeVirtGetNano,
3463 PDM_DEVHLPR3_VERSION /* the end */
3464};
3465
3466
3467
3468/**
3469 * Queue consumer callback for internal component.
3470 *
3471 * @returns Success indicator.
3472 * If false the item will not be removed and the flushing will stop.
3473 * @param pVM The VM handle.
3474 * @param pItem The item to consume. Upon return this item will be freed.
3475 */
3476DECLCALLBACK(bool) pdmR3DevHlpQueueConsumer(PVM pVM, PPDMQUEUEITEMCORE pItem)
3477{
3478 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)pItem;
3479 LogFlow(("pdmR3DevHlpQueueConsumer: enmOp=%d pDevIns=%p\n", pTask->enmOp, pTask->pDevInsR3));
3480 switch (pTask->enmOp)
3481 {
3482 case PDMDEVHLPTASKOP_ISA_SET_IRQ:
3483 PDMIsaSetIrq(pVM, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
3484 break;
3485
3486 case PDMDEVHLPTASKOP_PCI_SET_IRQ:
3487 pdmR3DevHlp_PCISetIrq(pTask->pDevInsR3, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
3488 break;
3489
3490 case PDMDEVHLPTASKOP_IOAPIC_SET_IRQ:
3491 PDMIoApicSetIrq(pVM, pTask->u.SetIRQ.iIrq, pTask->u.SetIRQ.iLevel);
3492 break;
3493
3494 default:
3495 AssertReleaseMsgFailed(("Invalid operation %d\n", pTask->enmOp));
3496 break;
3497 }
3498 return true;
3499}
3500
3501/** @} */
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