VirtualBox

source: vbox/trunk/include/VBox/vmm/pdmdev.h@ 62822

Last change on this file since 62822 was 61776, checked in by vboxsync, 9 years ago

CPUM,APIC: Per-CPU APIC CPUID feature bit and MSR_IA32_APICBASE GP mask adjustments.

  • Changed the PDMAPICHLPR3::pfnChangeFeature to pfnSetFeatureLevel, removing the RC and R0 versions.
  • Only use pfnSetFeatureLevel from the APIC constructor to communicate to CPUM the max APIC feature level, not to globally flip CPUID[1].EDX[9].
  • Renamed APIC enmOriginalMode to enmMaxMode, changing the type of it and the corresponding config values to PDMAPICMODE. This makes the above simpler and eliminates two conversion functions. It also makes APICMODE private to the APIC again.
  • Introduced CPUMSetGuestCpuIdPerCpuApicFeature for the per-CPU APIC feature bit management.
  • Introduced CPUMCPUIDLEAF_F_CONTAINS_APIC which works same as CPUMCPUIDLEAF_F_CONTAINS_OSXSAVE and CPUMCPUIDLEAF_F_CONTAINS_APIC_ID. Updated existing CPU profiles with this.
  • Made the patch manager helper function actually handle CPUMCPUIDLEAF_F_CONTAINS_APIC and CPUMCPUIDLEAF_F_CONTAINS_OSXSAVE (the latter previously relied on CPUMSetGuestCpuIdFeature/CPUMClearGuestCpuIdFeature from CPUMSetGuestCR4).
  • Pushed CPUMSetGuestCpuIdFeature, CPUMGetGuestCpuIdFeature and CPUMClearGuestCpuIdFeature down to ring-3 only (now CPUMR3*). The latter two function are deprecated.
  • Added call to CPUMSetGuestCpuIdPerCpuApicFeature from load function just in case the APIC is disabled by the guest at the time of saving.
  • CPUMSetGuestCpuIdFeature ensures we've got a MSR_IA32_APICBASE register when enabling the APIC.
  • CPUMSetGuestCpuIdFeature adjust the MSR_IA32_APICBASE GP mask when enabling x2APIC so setting MSR_IA32_APICBASE_EXTD does not trap.
  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 209.4 KB
Line 
1/** @file
2 * PDM - Pluggable Device Manager, Devices.
3 */
4
5/*
6 * Copyright (C) 2006-2016 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___VBox_vmm_pdmdev_h
27#define ___VBox_vmm_pdmdev_h
28
29#include <VBox/vmm/pdmqueue.h>
30#include <VBox/vmm/pdmcritsect.h>
31#include <VBox/vmm/pdmthread.h>
32#include <VBox/vmm/pdmifs.h>
33#include <VBox/vmm/pdmins.h>
34#include <VBox/vmm/pdmcommon.h>
35#include <VBox/vmm/iom.h>
36#include <VBox/vmm/tm.h>
37#include <VBox/vmm/ssm.h>
38#include <VBox/vmm/cfgm.h>
39#include <VBox/vmm/dbgf.h>
40#include <VBox/err.h>
41#include <VBox/pci.h>
42#include <VBox/sup.h>
43#include <iprt/stdarg.h>
44
45
46RT_C_DECLS_BEGIN
47
48/** @defgroup grp_pdm_device The PDM Devices API
49 * @ingroup grp_pdm
50 * @{
51 */
52
53/**
54 * Construct a device instance for a VM.
55 *
56 * @returns VBox status.
57 * @param pDevIns The device instance data. If the registration structure
58 * is needed, it can be accessed thru pDevIns->pReg.
59 * @param iInstance Instance number. Use this to figure out which registers
60 * and such to use. The instance number is also found in
61 * pDevIns->iInstance, but since it's likely to be
62 * frequently used PDM passes it as parameter.
63 * @param pCfg Configuration node handle for the driver. This is
64 * expected to be in high demand in the constructor and is
65 * therefore passed as an argument. When using it at other
66 * times, it can be found in pDevIns->pCfg.
67 */
68typedef DECLCALLBACK(int) FNPDMDEVCONSTRUCT(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg);
69/** Pointer to a FNPDMDEVCONSTRUCT() function. */
70typedef FNPDMDEVCONSTRUCT *PFNPDMDEVCONSTRUCT;
71
72/**
73 * Destruct a device instance.
74 *
75 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
76 * resources can be freed correctly.
77 *
78 * @returns VBox status.
79 * @param pDevIns The device instance data.
80 *
81 * @remarks The device critical section is not entered. The routine may delete
82 * the critical section, so the caller cannot exit it.
83 */
84typedef DECLCALLBACK(int) FNPDMDEVDESTRUCT(PPDMDEVINS pDevIns);
85/** Pointer to a FNPDMDEVDESTRUCT() function. */
86typedef FNPDMDEVDESTRUCT *PFNPDMDEVDESTRUCT;
87
88/**
89 * Device relocation callback.
90 *
91 * This is called when the instance data has been relocated in raw-mode context
92 * (RC). It is also called when the RC hypervisor selects changes. The device
93 * must fixup all necessary pointers and re-query all interfaces to other RC
94 * devices and drivers.
95 *
96 * Before the RC code is executed the first time, this function will be called
97 * with a 0 delta so RC pointer calculations can be one in one place.
98 *
99 * @param pDevIns Pointer to the device instance.
100 * @param offDelta The relocation delta relative to the old location.
101 *
102 * @remarks A relocation CANNOT fail.
103 *
104 * @remarks The device critical section is not entered. The relocations should
105 * not normally require any locking.
106 */
107typedef DECLCALLBACK(void) FNPDMDEVRELOCATE(PPDMDEVINS pDevIns, RTGCINTPTR offDelta);
108/** Pointer to a FNPDMDEVRELOCATE() function. */
109typedef FNPDMDEVRELOCATE *PFNPDMDEVRELOCATE;
110
111/**
112 * Power On notification.
113 *
114 * @returns VBox status.
115 * @param pDevIns The device instance data.
116 *
117 * @remarks Caller enters the device critical section.
118 */
119typedef DECLCALLBACK(void) FNPDMDEVPOWERON(PPDMDEVINS pDevIns);
120/** Pointer to a FNPDMDEVPOWERON() function. */
121typedef FNPDMDEVPOWERON *PFNPDMDEVPOWERON;
122
123/**
124 * Reset notification.
125 *
126 * @returns VBox status.
127 * @param pDevIns The device instance data.
128 *
129 * @remarks Caller enters the device critical section.
130 */
131typedef DECLCALLBACK(void) FNPDMDEVRESET(PPDMDEVINS pDevIns);
132/** Pointer to a FNPDMDEVRESET() function. */
133typedef FNPDMDEVRESET *PFNPDMDEVRESET;
134
135/**
136 * Soft reset notification.
137 *
138 * This is mainly for emulating the 286 style protected mode exits, in which
139 * most devices should remain in their current state.
140 *
141 * @returns VBox status.
142 * @param pDevIns The device instance data.
143 * @param fFlags PDMVMRESET_F_XXX (only bits relevant to soft resets).
144 *
145 * @remarks Caller enters the device critical section.
146 */
147typedef DECLCALLBACK(void) FNPDMDEVSOFTRESET(PPDMDEVINS pDevIns, uint32_t fFlags);
148/** Pointer to a FNPDMDEVSOFTRESET() function. */
149typedef FNPDMDEVSOFTRESET *PFNPDMDEVSOFTRESET;
150
151/** @name PDMVMRESET_F_XXX - VM reset flags.
152 * These flags are used both for FNPDMDEVSOFTRESET and for hardware signalling
153 * reset via PDMDevHlpVMReset.
154 * @{ */
155/** Unknown reason. */
156#define PDMVMRESET_F_UNKNOWN UINT32_C(0x00000000)
157/** GIM triggered reset. */
158#define PDMVMRESET_F_GIM UINT32_C(0x00000001)
159/** The last source always causing hard resets. */
160#define PDMVMRESET_F_LAST_ALWAYS_HARD PDMVMRESET_F_GIM
161/** ACPI triggered reset. */
162#define PDMVMRESET_F_ACPI UINT32_C(0x0000000c)
163/** PS/2 system port A (92h) reset. */
164#define PDMVMRESET_F_PORT_A UINT32_C(0x0000000d)
165/** Keyboard reset. */
166#define PDMVMRESET_F_KBD UINT32_C(0x0000000e)
167/** Tripple fault. */
168#define PDMVMRESET_F_TRIPLE_FAULT UINT32_C(0x0000000f)
169/** Reset source mask. */
170#define PDMVMRESET_F_SRC_MASK UINT32_C(0x0000000f)
171/** @} */
172
173/**
174 * Suspend notification.
175 *
176 * @returns VBox status.
177 * @param pDevIns The device instance data.
178 * @thread EMT(0)
179 *
180 * @remarks Caller enters the device critical section.
181 */
182typedef DECLCALLBACK(void) FNPDMDEVSUSPEND(PPDMDEVINS pDevIns);
183/** Pointer to a FNPDMDEVSUSPEND() function. */
184typedef FNPDMDEVSUSPEND *PFNPDMDEVSUSPEND;
185
186/**
187 * Resume notification.
188 *
189 * @returns VBox status.
190 * @param pDevIns The device instance data.
191 *
192 * @remarks Caller enters the device critical section.
193 */
194typedef DECLCALLBACK(void) FNPDMDEVRESUME(PPDMDEVINS pDevIns);
195/** Pointer to a FNPDMDEVRESUME() function. */
196typedef FNPDMDEVRESUME *PFNPDMDEVRESUME;
197
198/**
199 * Power Off notification.
200 *
201 * This is always called when VMR3PowerOff is called.
202 * There will be no callback when hot plugging devices.
203 *
204 * @param pDevIns The device instance data.
205 * @thread EMT(0)
206 *
207 * @remarks Caller enters the device critical section.
208 */
209typedef DECLCALLBACK(void) FNPDMDEVPOWEROFF(PPDMDEVINS pDevIns);
210/** Pointer to a FNPDMDEVPOWEROFF() function. */
211typedef FNPDMDEVPOWEROFF *PFNPDMDEVPOWEROFF;
212
213/**
214 * Attach command.
215 *
216 * This is called to let the device attach to a driver for a specified LUN
217 * at runtime. This is not called during VM construction, the device
218 * constructor has to attach to all the available drivers.
219 *
220 * This is like plugging in the keyboard or mouse after turning on the PC.
221 *
222 * @returns VBox status code.
223 * @param pDevIns The device instance.
224 * @param iLUN The logical unit which is being attached.
225 * @param fFlags Flags, combination of the PDM_TACH_FLAGS_* \#defines.
226 *
227 * @remarks Caller enters the device critical section.
228 */
229typedef DECLCALLBACK(int) FNPDMDEVATTACH(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags);
230/** Pointer to a FNPDMDEVATTACH() function. */
231typedef FNPDMDEVATTACH *PFNPDMDEVATTACH;
232
233/**
234 * Detach notification.
235 *
236 * This is called when a driver is detaching itself from a LUN of the device.
237 * The device should adjust its state to reflect this.
238 *
239 * This is like unplugging the network cable to use it for the laptop or
240 * something while the PC is still running.
241 *
242 * @param pDevIns The device instance.
243 * @param iLUN The logical unit which is being detached.
244 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
245 *
246 * @remarks Caller enters the device critical section.
247 */
248typedef DECLCALLBACK(void) FNPDMDEVDETACH(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags);
249/** Pointer to a FNPDMDEVDETACH() function. */
250typedef FNPDMDEVDETACH *PFNPDMDEVDETACH;
251
252/**
253 * Query the base interface of a logical unit.
254 *
255 * @returns VBOX status code.
256 * @param pDevIns The device instance.
257 * @param iLUN The logicial unit to query.
258 * @param ppBase Where to store the pointer to the base interface of the LUN.
259 *
260 * @remarks The device critical section is not entered.
261 */
262typedef DECLCALLBACK(int) FNPDMDEVQUERYINTERFACE(PPDMDEVINS pDevIns, unsigned iLUN, PPDMIBASE *ppBase);
263/** Pointer to a FNPDMDEVQUERYINTERFACE() function. */
264typedef FNPDMDEVQUERYINTERFACE *PFNPDMDEVQUERYINTERFACE;
265
266/**
267 * Init complete notification (after ring-0 & RC init since 5.1).
268 *
269 * This can be done to do communication with other devices and other
270 * initialization which requires everything to be in place.
271 *
272 * @returns VBOX status code.
273 * @param pDevIns The device instance.
274 *
275 * @remarks Caller enters the device critical section.
276 */
277typedef DECLCALLBACK(int) FNPDMDEVINITCOMPLETE(PPDMDEVINS pDevIns);
278/** Pointer to a FNPDMDEVINITCOMPLETE() function. */
279typedef FNPDMDEVINITCOMPLETE *PFNPDMDEVINITCOMPLETE;
280
281
282/**
283 * The context of a pfnMemSetup call.
284 */
285typedef enum PDMDEVMEMSETUPCTX
286{
287 /** Invalid zero value. */
288 PDMDEVMEMSETUPCTX_INVALID = 0,
289 /** After construction. */
290 PDMDEVMEMSETUPCTX_AFTER_CONSTRUCTION,
291 /** After reset. */
292 PDMDEVMEMSETUPCTX_AFTER_RESET,
293 /** Type size hack. */
294 PDMDEVMEMSETUPCTX_32BIT_HACK = 0x7fffffff
295} PDMDEVMEMSETUPCTX;
296
297
298/**
299 * PDM Device Registration Structure.
300 *
301 * This structure is used when registering a device from VBoxInitDevices() in HC
302 * Ring-3. PDM will continue use till the VM is terminated.
303 */
304typedef struct PDMDEVREG
305{
306 /** Structure version. PDM_DEVREG_VERSION defines the current version. */
307 uint32_t u32Version;
308 /** Device name. */
309 char szName[32];
310 /** Name of the raw-mode context module (no path).
311 * Only evalutated if PDM_DEVREG_FLAGS_RC is set. */
312 char szRCMod[32];
313 /** Name of the ring-0 module (no path).
314 * Only evalutated if PDM_DEVREG_FLAGS_R0 is set. */
315 char szR0Mod[32];
316 /** The description of the device. The UTF-8 string pointed to shall, like this structure,
317 * remain unchanged from registration till VM destruction. */
318 const char *pszDescription;
319
320 /** Flags, combination of the PDM_DEVREG_FLAGS_* \#defines. */
321 uint32_t fFlags;
322 /** Device class(es), combination of the PDM_DEVREG_CLASS_* \#defines. */
323 uint32_t fClass;
324 /** Maximum number of instances (per VM). */
325 uint32_t cMaxInstances;
326 /** Size of the instance data. */
327 uint32_t cbInstance;
328
329 /** Construct instance - required. */
330 PFNPDMDEVCONSTRUCT pfnConstruct;
331 /** Destruct instance - optional.
332 * Critical section NOT entered (will be destroyed). */
333 PFNPDMDEVDESTRUCT pfnDestruct;
334 /** Relocation command - optional.
335 * Critical section NOT entered. */
336 PFNPDMDEVRELOCATE pfnRelocate;
337
338 /**
339 * Memory setup callback.
340 *
341 * @param pDevIns The device instance data.
342 * @param enmCtx Indicates the context of the call.
343 * @remarks The critical section is entered prior to calling this method.
344 */
345 DECLR3CALLBACKMEMBER(void, pfnMemSetup, (PPDMDEVINS pDevIns, PDMDEVMEMSETUPCTX enmCtx));
346
347 /** Power on notification - optional.
348 * Critical section is entered. */
349 PFNPDMDEVPOWERON pfnPowerOn;
350 /** Reset notification - optional.
351 * Critical section is entered. */
352 PFNPDMDEVRESET pfnReset;
353 /** Suspend notification - optional.
354 * Critical section is entered. */
355 PFNPDMDEVSUSPEND pfnSuspend;
356 /** Resume notification - optional.
357 * Critical section is entered. */
358 PFNPDMDEVRESUME pfnResume;
359 /** Attach command - optional.
360 * Critical section is entered. */
361 PFNPDMDEVATTACH pfnAttach;
362 /** Detach notification - optional.
363 * Critical section is entered. */
364 PFNPDMDEVDETACH pfnDetach;
365 /** Query a LUN base interface - optional.
366 * Critical section is NOT entered. */
367 PFNPDMDEVQUERYINTERFACE pfnQueryInterface;
368 /** Init complete notification - optional.
369 * Critical section is entered. */
370 PFNPDMDEVINITCOMPLETE pfnInitComplete;
371 /** Power off notification - optional.
372 * Critical section is entered. */
373 PFNPDMDEVPOWEROFF pfnPowerOff;
374 /** Software system reset notification - optional.
375 * Critical section is entered. */
376 PFNPDMDEVSOFTRESET pfnSoftReset;
377 /** Initialization safty marker. */
378 uint32_t u32VersionEnd;
379} PDMDEVREG;
380/** Pointer to a PDM Device Structure. */
381typedef PDMDEVREG *PPDMDEVREG;
382/** Const pointer to a PDM Device Structure. */
383typedef PDMDEVREG const *PCPDMDEVREG;
384
385/** Current DEVREG version number. */
386#define PDM_DEVREG_VERSION PDM_VERSION_MAKE(0xffff, 2, 1)
387
388/** PDM Device Flags.
389 * @{ */
390/** This flag is used to indicate that the device has a RC component. */
391#define PDM_DEVREG_FLAGS_RC 0x00000001
392/** This flag is used to indicate that the device has a R0 component. */
393#define PDM_DEVREG_FLAGS_R0 0x00000002
394
395/** @def PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT
396 * The bit count for the current host. */
397#if HC_ARCH_BITS == 32
398# define PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT 0x00000010
399#elif HC_ARCH_BITS == 64
400# define PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT 0x00000020
401#else
402# error Unsupported HC_ARCH_BITS value.
403#endif
404/** The host bit count mask. */
405#define PDM_DEVREG_FLAGS_HOST_BITS_MASK 0x00000030
406
407/** The device support only 32-bit guests. */
408#define PDM_DEVREG_FLAGS_GUEST_BITS_32 0x00000100
409/** The device support only 64-bit guests. */
410#define PDM_DEVREG_FLAGS_GUEST_BITS_64 0x00000200
411/** The device support both 32-bit & 64-bit guests. */
412#define PDM_DEVREG_FLAGS_GUEST_BITS_32_64 0x00000300
413/** @def PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT
414 * The guest bit count for the current compilation. */
415#if GC_ARCH_BITS == 32
416# define PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT PDM_DEVREG_FLAGS_GUEST_BITS_32
417#elif GC_ARCH_BITS == 64
418# define PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT PDM_DEVREG_FLAGS_GUEST_BITS_32_64
419#else
420# error Unsupported GC_ARCH_BITS value.
421#endif
422/** The guest bit count mask. */
423#define PDM_DEVREG_FLAGS_GUEST_BITS_MASK 0x00000300
424
425/** A convenience. */
426#define PDM_DEVREG_FLAGS_DEFAULT_BITS (PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT | PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT)
427
428/** Indicates that the devices support PAE36 on a 32-bit guest. */
429#define PDM_DEVREG_FLAGS_PAE36 0x00001000
430
431/** Indicates that the device needs to be notified before the drivers when suspending. */
432#define PDM_DEVREG_FLAGS_FIRST_SUSPEND_NOTIFICATION 0x00002000
433
434/** Indicates that the device needs to be notified before the drivers when powering off. */
435#define PDM_DEVREG_FLAGS_FIRST_POWEROFF_NOTIFICATION 0x00004000
436
437/** Indicates that the device needs to be notified before the drivers when resetting. */
438#define PDM_DEVREG_FLAGS_FIRST_RESET_NOTIFICATION 0x00008000
439/** @} */
440
441
442/** PDM Device Classes.
443 * The order is important, lower bit earlier instantiation.
444 * @{ */
445/** Architecture device. */
446#define PDM_DEVREG_CLASS_ARCH RT_BIT(0)
447/** Architecture BIOS device. */
448#define PDM_DEVREG_CLASS_ARCH_BIOS RT_BIT(1)
449/** PCI bus brigde. */
450#define PDM_DEVREG_CLASS_BUS_PCI RT_BIT(2)
451/** ISA bus brigde. */
452#define PDM_DEVREG_CLASS_BUS_ISA RT_BIT(3)
453/** Input device (mouse, keyboard, joystick, HID, ...). */
454#define PDM_DEVREG_CLASS_INPUT RT_BIT(4)
455/** Interrupt controller (PIC). */
456#define PDM_DEVREG_CLASS_PIC RT_BIT(5)
457/** Interval controoler (PIT). */
458#define PDM_DEVREG_CLASS_PIT RT_BIT(6)
459/** RTC/CMOS. */
460#define PDM_DEVREG_CLASS_RTC RT_BIT(7)
461/** DMA controller. */
462#define PDM_DEVREG_CLASS_DMA RT_BIT(8)
463/** VMM Device. */
464#define PDM_DEVREG_CLASS_VMM_DEV RT_BIT(9)
465/** Graphics device, like VGA. */
466#define PDM_DEVREG_CLASS_GRAPHICS RT_BIT(10)
467/** Storage controller device. */
468#define PDM_DEVREG_CLASS_STORAGE RT_BIT(11)
469/** Network interface controller. */
470#define PDM_DEVREG_CLASS_NETWORK RT_BIT(12)
471/** Audio. */
472#define PDM_DEVREG_CLASS_AUDIO RT_BIT(13)
473/** USB HIC. */
474#define PDM_DEVREG_CLASS_BUS_USB RT_BIT(14)
475/** ACPI. */
476#define PDM_DEVREG_CLASS_ACPI RT_BIT(15)
477/** Serial controller device. */
478#define PDM_DEVREG_CLASS_SERIAL RT_BIT(16)
479/** Parallel controller device */
480#define PDM_DEVREG_CLASS_PARALLEL RT_BIT(17)
481/** Host PCI pass-through device */
482#define PDM_DEVREG_CLASS_HOST_DEV RT_BIT(18)
483/** Misc devices (always last). */
484#define PDM_DEVREG_CLASS_MISC RT_BIT(31)
485/** @} */
486
487
488/** @name IRQ Level for use with the *SetIrq APIs.
489 * @{
490 */
491/** Assert the IRQ (can assume value 1). */
492#define PDM_IRQ_LEVEL_HIGH RT_BIT(0)
493/** Deassert the IRQ (can assume value 0). */
494#define PDM_IRQ_LEVEL_LOW 0
495/** flip-flop - deassert and then assert the IRQ again immediately. */
496#define PDM_IRQ_LEVEL_FLIP_FLOP (RT_BIT(1) | PDM_IRQ_LEVEL_HIGH)
497/** @} */
498
499/**
500 * Registration record for MSI.
501 */
502typedef struct PDMMSIREG
503{
504 /** Number of MSI interrupt vectors, 0 if MSI not supported */
505 uint16_t cMsiVectors;
506 /** Offset of MSI capability */
507 uint8_t iMsiCapOffset;
508 /** Offset of next capability to MSI */
509 uint8_t iMsiNextOffset;
510 /** If we support 64-bit MSI addressing */
511 bool fMsi64bit;
512
513 /** Number of MSI-X interrupt vectors, 0 if MSI-X not supported */
514 uint16_t cMsixVectors;
515 /** Offset of MSI-X capability */
516 uint8_t iMsixCapOffset;
517 /** Offset of next capability to MSI-X */
518 uint8_t iMsixNextOffset;
519 /** Value of PCI BAR (base addresss register) assigned by device for MSI-X page access */
520 uint8_t iMsixBar;
521} PDMMSIREG;
522typedef PDMMSIREG *PPDMMSIREG;
523
524/**
525 * PCI Bus registration structure.
526 * All the callbacks, except the PCIBIOS hack, are working on PCI devices.
527 */
528typedef struct PDMPCIBUSREG
529{
530 /** Structure version number. PDM_PCIBUSREG_VERSION defines the current version. */
531 uint32_t u32Version;
532
533 /**
534 * Registers the device with the default PCI bus.
535 *
536 * @returns VBox status code.
537 * @param pDevIns Device instance of the PCI Bus.
538 * @param pPciDev The PCI device structure.
539 * Any PCI enabled device must keep this in it's instance data!
540 * Fill in the PCI data config before registration, please.
541 * @param pszName Pointer to device name (permanent, readonly). For debugging, not unique.
542 * @param iDev The device number ((dev << 3) | function) the device should have on the bus.
543 * If negative, the pci bus device will assign one.
544 * @remarks Caller enters the PDM critical section.
545 */
546 DECLR3CALLBACKMEMBER(int, pfnRegisterR3,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, const char *pszName, int iDev));
547
548 /**
549 * Initialize MSI support in a PCI device.
550 *
551 * @returns VBox status code.
552 * @param pDevIns Device instance of the PCI Bus.
553 * @param pPciDev The PCI device structure.
554 * @param pMsiReg MSI registration structure
555 * @remarks Caller enters the PDM critical section.
556 */
557 DECLR3CALLBACKMEMBER(int, pfnRegisterMsiR3,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PPDMMSIREG pMsiReg));
558
559 /**
560 * Registers a I/O region (memory mapped or I/O ports) for a PCI device.
561 *
562 * @returns VBox status code.
563 * @param pDevIns Device instance of the PCI Bus.
564 * @param pPciDev The PCI device structure.
565 * @param iRegion The region number.
566 * @param cbRegion Size of the region.
567 * @param iType PCI_ADDRESS_SPACE_MEM, PCI_ADDRESS_SPACE_IO or PCI_ADDRESS_SPACE_MEM_PREFETCH.
568 * @param pfnCallback Callback for doing the mapping.
569 * @remarks Caller enters the PDM critical section.
570 */
571 DECLR3CALLBACKMEMBER(int, pfnIORegionRegisterR3,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, int iRegion, uint32_t cbRegion, PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback));
572
573 /**
574 * Register PCI configuration space read/write callbacks.
575 *
576 * @param pDevIns Device instance of the PCI Bus.
577 * @param pPciDev The PCI device structure.
578 * @param pfnRead Pointer to the user defined PCI config read function.
579 * @param ppfnReadOld Pointer to function pointer which will receive the old (default)
580 * PCI config read function. This way, user can decide when (and if)
581 * to call default PCI config read function. Can be NULL.
582 * @param pfnWrite Pointer to the user defined PCI config write function.
583 * @param pfnWriteOld Pointer to function pointer which will receive the old (default)
584 * PCI config write function. This way, user can decide when (and if)
585 * to call default PCI config write function. Can be NULL.
586 * @remarks Caller enters the PDM critical section.
587 * @thread EMT
588 */
589 DECLR3CALLBACKMEMBER(void, pfnSetConfigCallbacksR3,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
590 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld));
591
592 /**
593 * Set the IRQ for a PCI device.
594 *
595 * @param pDevIns Device instance of the PCI Bus.
596 * @param pPciDev The PCI device structure.
597 * @param iIrq IRQ number to set.
598 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
599 * @param uTagSrc The IRQ tag and source (for tracing).
600 * @remarks Caller enters the PDM critical section.
601 */
602 DECLR3CALLBACKMEMBER(void, pfnSetIrqR3,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, int iIrq, int iLevel, uint32_t uTagSrc));
603
604 /**
605 * Called to perform the job of the bios.
606 * This is only called for the first PCI Bus - it is expected to
607 * service all the PCI buses.
608 *
609 * @returns VBox status.
610 * @param pDevIns Device instance of the first bus.
611 * @remarks Caller enters the PDM critical section.
612 */
613 DECLR3CALLBACKMEMBER(int, pfnFakePCIBIOSR3,(PPDMDEVINS pDevIns));
614
615 /** The name of the SetIrq RC entry point. */
616 const char *pszSetIrqRC;
617
618 /** The name of the SetIrq R0 entry point. */
619 const char *pszSetIrqR0;
620
621} PDMPCIBUSREG;
622/** Pointer to a PCI bus registration structure. */
623typedef PDMPCIBUSREG *PPDMPCIBUSREG;
624
625/** Current PDMPCIBUSREG version number. */
626#define PDM_PCIBUSREG_VERSION PDM_VERSION_MAKE(0xfffe, 4, 0)
627
628/**
629 * PCI Bus RC helpers.
630 */
631typedef struct PDMPCIHLPRC
632{
633 /** Structure version. PDM_PCIHLPRC_VERSION defines the current version. */
634 uint32_t u32Version;
635
636 /**
637 * Set an ISA IRQ.
638 *
639 * @param pDevIns PCI device instance.
640 * @param iIrq IRQ number to set.
641 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
642 * @param uTagSrc The IRQ tag and source (for tracing).
643 * @thread EMT only.
644 */
645 DECLRCCALLBACKMEMBER(void, pfnIsaSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
646
647 /**
648 * Set an I/O-APIC IRQ.
649 *
650 * @param pDevIns PCI device instance.
651 * @param iIrq IRQ number to set.
652 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
653 * @param uTagSrc The IRQ tag and source (for tracing).
654 * @thread EMT only.
655 */
656 DECLRCCALLBACKMEMBER(void, pfnIoApicSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
657
658 /**
659 * Send an MSI.
660 *
661 * @param pDevIns PCI device instance.
662 * @param GCPhys Physical address MSI request was written.
663 * @param uValue Value written.
664 * @param uTagSrc The IRQ tag and source (for tracing).
665 * @thread EMT only.
666 */
667 DECLRCCALLBACKMEMBER(void, pfnIoApicSendMsi,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue, uint32_t uTagSrc));
668
669
670 /**
671 * Acquires the PDM lock.
672 *
673 * @returns VINF_SUCCESS on success.
674 * @returns rc if we failed to acquire the lock.
675 * @param pDevIns The PCI device instance.
676 * @param rc What to return if we fail to acquire the lock.
677 */
678 DECLRCCALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
679
680 /**
681 * Releases the PDM lock.
682 *
683 * @param pDevIns The PCI device instance.
684 */
685 DECLRCCALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
686
687 /** Just a safety precaution. */
688 uint32_t u32TheEnd;
689} PDMPCIHLPRC;
690/** Pointer to PCI helpers. */
691typedef RCPTRTYPE(PDMPCIHLPRC *) PPDMPCIHLPRC;
692/** Pointer to const PCI helpers. */
693typedef RCPTRTYPE(const PDMPCIHLPRC *) PCPDMPCIHLPRC;
694
695/** Current PDMPCIHLPRC version number. */
696#define PDM_PCIHLPRC_VERSION PDM_VERSION_MAKE(0xfffd, 3, 0)
697
698
699/**
700 * PCI Bus R0 helpers.
701 */
702typedef struct PDMPCIHLPR0
703{
704 /** Structure version. PDM_PCIHLPR0_VERSION defines the current version. */
705 uint32_t u32Version;
706
707 /**
708 * Set an ISA IRQ.
709 *
710 * @param pDevIns PCI device instance.
711 * @param iIrq IRQ number to set.
712 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
713 * @param uTagSrc The IRQ tag and source (for tracing).
714 * @thread EMT only.
715 */
716 DECLR0CALLBACKMEMBER(void, pfnIsaSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
717
718 /**
719 * Set an I/O-APIC IRQ.
720 *
721 * @param pDevIns PCI device instance.
722 * @param iIrq IRQ number to set.
723 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
724 * @param uTagSrc The IRQ tag and source (for tracing).
725 * @thread EMT only.
726 */
727 DECLR0CALLBACKMEMBER(void, pfnIoApicSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
728
729 /**
730 * Send an MSI.
731 *
732 * @param pDevIns PCI device instance.
733 * @param GCPhys Physical address MSI request was written.
734 * @param uValue Value written.
735 * @param uTagSrc The IRQ tag and source (for tracing).
736 * @thread EMT only.
737 */
738 DECLR0CALLBACKMEMBER(void, pfnIoApicSendMsi,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue, uint32_t uTagSrc));
739
740
741 /**
742 * Acquires the PDM lock.
743 *
744 * @returns VINF_SUCCESS on success.
745 * @returns rc if we failed to acquire the lock.
746 * @param pDevIns The PCI device instance.
747 * @param rc What to return if we fail to acquire the lock.
748 */
749 DECLR0CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
750
751 /**
752 * Releases the PDM lock.
753 *
754 * @param pDevIns The PCI device instance.
755 */
756 DECLR0CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
757
758 /** Just a safety precaution. */
759 uint32_t u32TheEnd;
760} PDMPCIHLPR0;
761/** Pointer to PCI helpers. */
762typedef R0PTRTYPE(PDMPCIHLPR0 *) PPDMPCIHLPR0;
763/** Pointer to const PCI helpers. */
764typedef R0PTRTYPE(const PDMPCIHLPR0 *) PCPDMPCIHLPR0;
765
766/** Current PDMPCIHLPR0 version number. */
767#define PDM_PCIHLPR0_VERSION PDM_VERSION_MAKE(0xfffc, 3, 0)
768
769/**
770 * PCI device helpers.
771 */
772typedef struct PDMPCIHLPR3
773{
774 /** Structure version. PDM_PCIHLPR3_VERSION defines the current version. */
775 uint32_t u32Version;
776
777 /**
778 * Set an ISA IRQ.
779 *
780 * @param pDevIns The PCI device instance.
781 * @param iIrq IRQ number to set.
782 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
783 * @param uTagSrc The IRQ tag and source (for tracing).
784 */
785 DECLR3CALLBACKMEMBER(void, pfnIsaSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
786
787 /**
788 * Set an I/O-APIC IRQ.
789 *
790 * @param pDevIns The PCI device instance.
791 * @param iIrq IRQ number to set.
792 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
793 * @param uTagSrc The IRQ tag and source (for tracing).
794 */
795 DECLR3CALLBACKMEMBER(void, pfnIoApicSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
796
797 /**
798 * Send an MSI.
799 *
800 * @param pDevIns PCI device instance.
801 * @param GCPhys Physical address MSI request was written.
802 * @param uValue Value written.
803 * @param uTagSrc The IRQ tag and source (for tracing).
804 */
805 DECLR3CALLBACKMEMBER(void, pfnIoApicSendMsi,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue, uint32_t uTagSrc));
806
807 /**
808 * Checks if the given address is an MMIO2 base address or not.
809 *
810 * @returns true/false accordingly.
811 * @param pDevIns The PCI device instance.
812 * @param pOwner The owner of the memory, optional.
813 * @param GCPhys The address to check.
814 */
815 DECLR3CALLBACKMEMBER(bool, pfnIsMMIO2Base,(PPDMDEVINS pDevIns, PPDMDEVINS pOwner, RTGCPHYS GCPhys));
816
817 /**
818 * Gets the address of the RC PCI Bus helpers.
819 *
820 * This should be called at both construction and relocation time
821 * to obtain the correct address of the RC helpers.
822 *
823 * @returns RC pointer to the PCI Bus helpers.
824 * @param pDevIns Device instance of the PCI Bus.
825 * @thread EMT only.
826 */
827 DECLR3CALLBACKMEMBER(PCPDMPCIHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
828
829 /**
830 * Gets the address of the R0 PCI Bus helpers.
831 *
832 * This should be called at both construction and relocation time
833 * to obtain the correct address of the R0 helpers.
834 *
835 * @returns R0 pointer to the PCI Bus helpers.
836 * @param pDevIns Device instance of the PCI Bus.
837 * @thread EMT only.
838 */
839 DECLR3CALLBACKMEMBER(PCPDMPCIHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
840
841 /**
842 * Acquires the PDM lock.
843 *
844 * @returns VINF_SUCCESS on success.
845 * @returns Fatal error on failure.
846 * @param pDevIns The PCI device instance.
847 * @param rc Dummy for making the interface identical to the RC and R0 versions.
848 */
849 DECLR3CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
850
851 /**
852 * Releases the PDM lock.
853 *
854 * @param pDevIns The PCI device instance.
855 */
856 DECLR3CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
857
858 /** Just a safety precaution. */
859 uint32_t u32TheEnd;
860} PDMPCIHLPR3;
861/** Pointer to PCI helpers. */
862typedef R3PTRTYPE(PDMPCIHLPR3 *) PPDMPCIHLPR3;
863/** Pointer to const PCI helpers. */
864typedef R3PTRTYPE(const PDMPCIHLPR3 *) PCPDMPCIHLPR3;
865
866/** Current PDMPCIHLPR3 version number. */
867#define PDM_PCIHLPR3_VERSION PDM_VERSION_MAKE(0xfffb, 3, 0)
868
869
870/**
871 * Programmable Interrupt Controller registration structure.
872 */
873typedef struct PDMPICREG
874{
875 /** Structure version number. PDM_PICREG_VERSION defines the current version. */
876 uint32_t u32Version;
877
878 /**
879 * Set the an IRQ.
880 *
881 * @param pDevIns Device instance of the PIC.
882 * @param iIrq IRQ number to set.
883 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
884 * @param uTagSrc The IRQ tag and source (for tracing).
885 * @remarks Caller enters the PDM critical section.
886 */
887 DECLR3CALLBACKMEMBER(void, pfnSetIrqR3,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
888
889 /**
890 * Get a pending interrupt.
891 *
892 * @returns Pending interrupt number.
893 * @param pDevIns Device instance of the PIC.
894 * @param puTagSrc Where to return the IRQ tag and source.
895 * @remarks Caller enters the PDM critical section.
896 */
897 DECLR3CALLBACKMEMBER(int, pfnGetInterruptR3,(PPDMDEVINS pDevIns, uint32_t *puTagSrc));
898
899 /** The name of the RC SetIrq entry point. */
900 const char *pszSetIrqRC;
901 /** The name of the RC GetInterrupt entry point. */
902 const char *pszGetInterruptRC;
903
904 /** The name of the R0 SetIrq entry point. */
905 const char *pszSetIrqR0;
906 /** The name of the R0 GetInterrupt entry point. */
907 const char *pszGetInterruptR0;
908} PDMPICREG;
909/** Pointer to a PIC registration structure. */
910typedef PDMPICREG *PPDMPICREG;
911
912/** Current PDMPICREG version number. */
913#define PDM_PICREG_VERSION PDM_VERSION_MAKE(0xfffa, 2, 0)
914
915/**
916 * PIC RC helpers.
917 */
918typedef struct PDMPICHLPRC
919{
920 /** Structure version. PDM_PICHLPRC_VERSION defines the current version. */
921 uint32_t u32Version;
922
923 /**
924 * Set the interrupt force action flag.
925 *
926 * @param pDevIns Device instance of the PIC.
927 */
928 DECLRCCALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns));
929
930 /**
931 * Clear the interrupt force action flag.
932 *
933 * @param pDevIns Device instance of the PIC.
934 */
935 DECLRCCALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns));
936
937 /**
938 * Acquires the PDM lock.
939 *
940 * @returns VINF_SUCCESS on success.
941 * @returns rc if we failed to acquire the lock.
942 * @param pDevIns The PIC device instance.
943 * @param rc What to return if we fail to acquire the lock.
944 */
945 DECLRCCALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
946
947 /**
948 * Releases the PDM lock.
949 *
950 * @param pDevIns The PIC device instance.
951 */
952 DECLRCCALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
953
954 /** Just a safety precaution. */
955 uint32_t u32TheEnd;
956} PDMPICHLPRC;
957
958/** Pointer to PIC RC helpers. */
959typedef RCPTRTYPE(PDMPICHLPRC *) PPDMPICHLPRC;
960/** Pointer to const PIC RC helpers. */
961typedef RCPTRTYPE(const PDMPICHLPRC *) PCPDMPICHLPRC;
962
963/** Current PDMPICHLPRC version number. */
964#define PDM_PICHLPRC_VERSION PDM_VERSION_MAKE(0xfff9, 2, 0)
965
966
967/**
968 * PIC R0 helpers.
969 */
970typedef struct PDMPICHLPR0
971{
972 /** Structure version. PDM_PICHLPR0_VERSION defines the current version. */
973 uint32_t u32Version;
974
975 /**
976 * Set the interrupt force action flag.
977 *
978 * @param pDevIns Device instance of the PIC.
979 */
980 DECLR0CALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns));
981
982 /**
983 * Clear the interrupt force action flag.
984 *
985 * @param pDevIns Device instance of the PIC.
986 */
987 DECLR0CALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns));
988
989 /**
990 * Acquires the PDM lock.
991 *
992 * @returns VINF_SUCCESS on success.
993 * @returns rc if we failed to acquire the lock.
994 * @param pDevIns The PIC device instance.
995 * @param rc What to return if we fail to acquire the lock.
996 */
997 DECLR0CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
998
999 /**
1000 * Releases the PDM lock.
1001 *
1002 * @param pDevIns The PCI device instance.
1003 */
1004 DECLR0CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1005
1006 /** Just a safety precaution. */
1007 uint32_t u32TheEnd;
1008} PDMPICHLPR0;
1009
1010/** Pointer to PIC R0 helpers. */
1011typedef R0PTRTYPE(PDMPICHLPR0 *) PPDMPICHLPR0;
1012/** Pointer to const PIC R0 helpers. */
1013typedef R0PTRTYPE(const PDMPICHLPR0 *) PCPDMPICHLPR0;
1014
1015/** Current PDMPICHLPR0 version number. */
1016#define PDM_PICHLPR0_VERSION PDM_VERSION_MAKE(0xfff8, 1, 0)
1017
1018/**
1019 * PIC R3 helpers.
1020 */
1021typedef struct PDMPICHLPR3
1022{
1023 /** Structure version. PDM_PICHLP_VERSION defines the current version. */
1024 uint32_t u32Version;
1025
1026 /**
1027 * Set the interrupt force action flag.
1028 *
1029 * @param pDevIns Device instance of the PIC.
1030 */
1031 DECLR3CALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns));
1032
1033 /**
1034 * Clear the interrupt force action flag.
1035 *
1036 * @param pDevIns Device instance of the PIC.
1037 */
1038 DECLR3CALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns));
1039
1040 /**
1041 * Acquires the PDM lock.
1042 *
1043 * @returns VINF_SUCCESS on success.
1044 * @returns Fatal error on failure.
1045 * @param pDevIns The PIC device instance.
1046 * @param rc Dummy for making the interface identical to the RC and R0 versions.
1047 */
1048 DECLR3CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1049
1050 /**
1051 * Releases the PDM lock.
1052 *
1053 * @param pDevIns The PIC device instance.
1054 */
1055 DECLR3CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1056
1057 /**
1058 * Gets the address of the RC PIC helpers.
1059 *
1060 * This should be called at both construction and relocation time
1061 * to obtain the correct address of the RC helpers.
1062 *
1063 * @returns RC pointer to the PIC helpers.
1064 * @param pDevIns Device instance of the PIC.
1065 */
1066 DECLR3CALLBACKMEMBER(PCPDMPICHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
1067
1068 /**
1069 * Gets the address of the R0 PIC helpers.
1070 *
1071 * This should be called at both construction and relocation time
1072 * to obtain the correct address of the R0 helpers.
1073 *
1074 * @returns R0 pointer to the PIC helpers.
1075 * @param pDevIns Device instance of the PIC.
1076 */
1077 DECLR3CALLBACKMEMBER(PCPDMPICHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
1078
1079 /** Just a safety precaution. */
1080 uint32_t u32TheEnd;
1081} PDMPICHLPR3;
1082
1083/** Pointer to PIC R3 helpers. */
1084typedef R3PTRTYPE(PDMPICHLPR3 *) PPDMPICHLPR3;
1085/** Pointer to const PIC R3 helpers. */
1086typedef R3PTRTYPE(const PDMPICHLPR3 *) PCPDMPICHLPR3;
1087
1088/** Current PDMPICHLPR3 version number. */
1089#define PDM_PICHLPR3_VERSION PDM_VERSION_MAKE(0xfff7, 1, 0)
1090
1091
1092
1093/**
1094 * Firmware registration structure.
1095 */
1096typedef struct PDMFWREG
1097{
1098 /** Struct version+magic number (PDM_FWREG_VERSION). */
1099 uint32_t u32Version;
1100
1101 /**
1102 * Checks whether this is a hard or soft reset.
1103 *
1104 * The current definition of soft reset is what the PC BIOS does when CMOS[0xF]
1105 * is 5, 9 or 0xA.
1106 *
1107 * @returns true if hard reset, false if soft.
1108 * @param pDevIns Device instance of the firmware.
1109 * @param fFlags PDMRESET_F_XXX passed to the PDMDevHlpVMReset API.
1110 */
1111 DECLR3CALLBACKMEMBER(bool, pfnIsHardReset,(PPDMDEVINS pDevIns, uint32_t fFlags));
1112
1113 /** Just a safety precaution. */
1114 uint32_t u32TheEnd;
1115} PDMFWREG;
1116/** Pointer to a FW registration structure. */
1117typedef PDMFWREG *PPDMFWREG;
1118/** Pointer to a const FW registration structure. */
1119typedef PDMFWREG const *PCPDMFWREG;
1120
1121/** Current PDMFWREG version number. */
1122#define PDM_FWREG_VERSION PDM_VERSION_MAKE(0xffdd, 1, 0)
1123
1124/**
1125 * Firmware R3 helpers.
1126 */
1127typedef struct PDMFWHLPR3
1128{
1129 /** Structure version. PDM_FWHLP_VERSION defines the current version. */
1130 uint32_t u32Version;
1131
1132 /** Just a safety precaution. */
1133 uint32_t u32TheEnd;
1134} PDMFWHLPR3;
1135
1136/** Pointer to FW R3 helpers. */
1137typedef R3PTRTYPE(PDMFWHLPR3 *) PPDMFWHLPR3;
1138/** Pointer to const FW R3 helpers. */
1139typedef R3PTRTYPE(const PDMFWHLPR3 *) PCPDMFWHLPR3;
1140
1141/** Current PDMFWHLPR3 version number. */
1142#define PDM_FWHLPR3_VERSION PDM_VERSION_MAKE(0xffdb, 1, 0)
1143
1144
1145/**
1146 * Advanced Programmable Interrupt Controller registration structure.
1147 */
1148typedef struct PDMAPICREG
1149{
1150 /** Structure version number. PDM_APICREG_VERSION defines the current version. */
1151 uint32_t u32Version;
1152
1153 /**
1154 * Get a pending interrupt.
1155 *
1156 * @returns VBox status code.
1157 * @param pDevIns Device instance of the APIC.
1158 * @param pVCpu The cross context virtual CPU structure.
1159 * @param pu8Vector Where to store the vector.
1160 * @param pu32TagSrc Where to return the tag source (tracing
1161 * purposes).
1162 * @remarks Caller enters the PDM critical section.
1163 */
1164 DECLR3CALLBACKMEMBER(int, pfnGetInterruptR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint8_t *pu8Vector, uint32_t *pu32TagSrc));
1165
1166 /**
1167 * Set the APIC base.
1168 *
1169 * @param pDevIns Device instance of the APIC.
1170 * @param pVCpu The cross context virtual CPU structure.
1171 * @param u64BaseMsr The base MSR value.
1172 * @remarks Caller enters the PDM critical section (might not be the case with
1173 * the new APIC code)
1174 */
1175 DECLR3CALLBACKMEMBER(VBOXSTRICTRC, pfnSetBaseMsrR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint64_t u64BaseMsr));
1176
1177 /**
1178 * Get the APIC base.
1179 *
1180 * @returns Current base.
1181 * @param pDevIns Device instance of the APIC.
1182 * @param pVCpu The cross context virtual CPU structure.
1183 * @remarks Caller enters the PDM critical section.
1184 */
1185 DECLR3CALLBACKMEMBER(uint64_t, pfnGetBaseMsrR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu));
1186
1187 /**
1188 * Set the TPR (task priority register).
1189 *
1190 * @param pDevIns Device instance of the APIC.
1191 * @param pVCpu The cross context virtual CPU structure.
1192 * @param u8Tpr The new TPR.
1193 * @remarks Caller enters the PDM critical section.
1194 */
1195 DECLR3CALLBACKMEMBER(void, pfnSetTprR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint8_t u8Tpr));
1196
1197 /**
1198 * Get the TPR (task priority register).
1199 *
1200 * @returns The current TPR.
1201 * @param pDevIns Device instance of the APIC.
1202 * @param pVCpu The cross context virtual CPU structure.
1203 * @param pfPending Where to store if there is an interrupt pending
1204 * (optional, can be NULL).
1205 * @param pu8PendingIntr Where to store the pending interrupt vector
1206 * (optional, can be NULL).
1207 * @remarks Caller enters the PDM critical section.
1208 */
1209 DECLR3CALLBACKMEMBER(uint8_t, pfnGetTprR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, bool *pfPending, uint8_t *pu8PendingIntr));
1210
1211 /**
1212 * Write to a MSR in APIC range.
1213 *
1214 * @returns Strict VBox status code.
1215 * @param pDevIns Device instance of the APIC.
1216 * @param pVCpu The cross context virtual CPU structure.
1217 * @param u32Reg The MSR begin written to.
1218 * @param u64Value The value to write.
1219 *
1220 * @remarks Unlike the other callbacks, the PDM lock is not taken before
1221 * calling this method.
1222 */
1223 DECLR3CALLBACKMEMBER(VBOXSTRICTRC, pfnWriteMsrR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint32_t u32Reg, uint64_t u64Value));
1224
1225 /**
1226 * Read from a MSR in APIC range.
1227 *
1228 * @returns Strict VBox status code.
1229 * @param pDevIns Device instance of the APIC.
1230 * @param pVCpu The cross context virtual CPU structure.
1231 * @param u32Reg MSR to read.
1232 * @param pu64Value Where to return the read value.
1233 *
1234 * @remarks Unlike the other callbacks, the PDM lock is not taken before
1235 * calling this method.
1236 */
1237 DECLR3CALLBACKMEMBER(VBOXSTRICTRC, pfnReadMsrR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint32_t u32Reg, uint64_t *pu64Value));
1238
1239 /**
1240 * Private interface between the IOAPIC and APIC.
1241 *
1242 * This is a low-level, APIC/IOAPIC implementation specific interface which
1243 * is registered with PDM only because it makes life so much simpler right
1244 * now (GC bits). This is a bad bad hack! The correct way of doing this
1245 * would involve some way of querying GC interfaces and relocating them.
1246 * Perhaps doing some kind of device init in GC...
1247 *
1248 * @returns VBox status code.
1249 * @param pDevIns Device instance of the APIC.
1250 * @param uDest The destination mask.
1251 * @param uDestMode The destination mode, see XAPICDESTMODE.
1252 * @param uDeliveryMode The delivery mode, see XAPICDELIVERYMODE.
1253 * @param uVector The interrupt vector.
1254 * @param uPolarity The input pin polarity.
1255 * @param uTriggerMode The trigger mode, see XAPICTRIGGERMODE.
1256 * @param uTagSrc The IRQ tag and source (for tracing).
1257 * @remarks Caller enters the PDM critical section.
1258 */
1259 DECLR3CALLBACKMEMBER(int, pfnBusDeliverR3,(PPDMDEVINS pDevIns, uint8_t uDest, uint8_t uDestMode, uint8_t uDeliveryMode,
1260 uint8_t uVector, uint8_t uPolarity, uint8_t uTriggerMode, uint32_t uTagSrc));
1261
1262 /**
1263 * Deliver a signal to CPU's local interrupt pins (LINT0/LINT1).
1264 *
1265 * Used for virtual wire mode when interrupts from the PIC are passed through
1266 * LAPIC.
1267 *
1268 * @returns Strict VBox status code.
1269 * @param pDevIns Device instance of the APIC.
1270 * @param pVCpu The cross context virtual CPU structure.
1271 * @param u8Pin Local pin number (0 or 1 for current CPUs).
1272 * @param u8Level The level.
1273 * @param rcRZ The return code if the operation cannot be
1274 * performed in the current context.
1275 * @remarks Caller enters the PDM critical section
1276 */
1277 DECLR3CALLBACKMEMBER(VBOXSTRICTRC, pfnLocalInterruptR3,(PPDMDEVINS pDevIns, PVMCPU pVCpu, uint8_t u8Pin, uint8_t u8Level,
1278 int rcRZ));
1279
1280 /**
1281 * Get the APIC timer frequency (in Hz).
1282 *
1283 * @returns The frequency of the APIC timer.
1284 * @param pDevIns Device instance of the APIC.
1285 */
1286 DECLR3CALLBACKMEMBER(uint64_t, pfnGetTimerFreqR3,(PPDMDEVINS pDevIns));
1287
1288 /** The name of the RC GetInterrupt entry point. */
1289 const char *pszGetInterruptRC;
1290 /** The name of the RC SetBaseMsr entry point. */
1291 const char *pszSetBaseMsrRC;
1292 /** The name of the RC GetBaseMsr entry point. */
1293 const char *pszGetBaseMsrRC;
1294 /** The name of the RC SetTpr entry point. */
1295 const char *pszSetTprRC;
1296 /** The name of the RC GetTpr entry point. */
1297 const char *pszGetTprRC;
1298 /** The name of the RC WriteMsr entry point. */
1299 const char *pszWriteMsrRC;
1300 /** The name of the RC ReadMsr entry point. */
1301 const char *pszReadMsrRC;
1302 /** The name of the RC BusDeliver entry point. */
1303 const char *pszBusDeliverRC;
1304 /** The name of the RC LocalInterrupt entry point. */
1305 const char *pszLocalInterruptRC;
1306 /** The name of the RC GetTimerFreq entry point. */
1307 const char *pszGetTimerFreqRC;
1308
1309 /** The name of the R0 GetInterrupt entry point. */
1310 const char *pszGetInterruptR0;
1311 /** The name of the R0 SetBaseMsr entry point. */
1312 const char *pszSetBaseMsrR0;
1313 /** The name of the R0 GetBaseMsr entry point. */
1314 const char *pszGetBaseMsrR0;
1315 /** The name of the R0 SetTpr entry point. */
1316 const char *pszSetTprR0;
1317 /** The name of the R0 GetTpr entry point. */
1318 const char *pszGetTprR0;
1319 /** The name of the R0 WriteMsr entry point. */
1320 const char *pszWriteMsrR0;
1321 /** The name of the R0 ReadMsr entry point. */
1322 const char *pszReadMsrR0;
1323 /** The name of the R0 BusDeliver entry point. */
1324 const char *pszBusDeliverR0;
1325 /** The name of the R0 LocalInterrupt entry point. */
1326 const char *pszLocalInterruptR0;
1327 /** The name of the R0 GetTimerFreq entry point. */
1328 const char *pszGetTimerFreqR0;
1329} PDMAPICREG;
1330/** Pointer to an APIC registration structure. */
1331typedef PDMAPICREG *PPDMAPICREG;
1332
1333/** Current PDMAPICREG version number. */
1334#define PDM_APICREG_VERSION PDM_VERSION_MAKE(0xfff6, 4, 0)
1335
1336
1337/**
1338 * APIC mode argument for pfnChangeFeature.
1339 *
1340 * Also used in saved-states, don't change existing values.
1341 */
1342typedef enum PDMAPICMODE
1343{
1344 /** Invalid 0 entry. */
1345 PDMAPICMODE_INVALID = 0,
1346 /** No APIC. */
1347 PDMAPICMODE_NONE,
1348 /** Standard APIC (X86_CPUID_FEATURE_EDX_APIC). */
1349 PDMAPICMODE_APIC,
1350 /** Intel X2APIC (X86_CPUID_FEATURE_ECX_X2APIC). */
1351 PDMAPICMODE_X2APIC,
1352 /** The usual 32-bit paranoia. */
1353 PDMAPICMODE_32BIT_HACK = 0x7fffffff
1354} PDMAPICMODE;
1355
1356/**
1357 * APIC irq argument for pfnSetInterruptFF and pfnClearInterruptFF.
1358 */
1359typedef enum PDMAPICIRQ
1360{
1361 /** Invalid 0 entry. */
1362 PDMAPICIRQ_INVALID = 0,
1363 /** Normal hardware interrupt. */
1364 PDMAPICIRQ_HARDWARE,
1365 /** NMI. */
1366 PDMAPICIRQ_NMI,
1367 /** SMI. */
1368 PDMAPICIRQ_SMI,
1369 /** ExtINT (HW interrupt via PIC). */
1370 PDMAPICIRQ_EXTINT,
1371 /** Interrupt arrived, needs to be updated to the IRR. */
1372 PDMAPICIRQ_UPDATE_PENDING,
1373 /** The usual 32-bit paranoia. */
1374 PDMAPICIRQ_32BIT_HACK = 0x7fffffff
1375} PDMAPICIRQ;
1376
1377
1378/**
1379 * APIC RC helpers.
1380 */
1381typedef struct PDMAPICHLPRC
1382{
1383 /** Structure version. PDM_APICHLPRC_VERSION defines the current version. */
1384 uint32_t u32Version;
1385
1386 /**
1387 * Set the interrupt force action flag.
1388 *
1389 * @param pDevIns Device instance of the APIC.
1390 * @param enmType IRQ type.
1391 * @param idCpu Virtual CPU to set flag upon.
1392 */
1393 DECLRCCALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1394
1395 /**
1396 * Clear the interrupt force action flag.
1397 *
1398 * @param pDevIns Device instance of the APIC.
1399 * @param enmType IRQ type.
1400 * @param idCpu Virtual CPU to clear flag upon.
1401 */
1402 DECLRCCALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1403
1404 /**
1405 * Broadcasts an EOI for an interrupt vector to the I/O APICs.
1406 *
1407 * @returns VBox status code.
1408 * @param pDevIns The APIC device instance.
1409 * @param u8Vector The interrupt vector.
1410 */
1411 DECLRCCALLBACKMEMBER(int, pfnBusBroadcastEoi,(PPDMDEVINS pDevIns, uint8_t u8Vector));
1412
1413 /**
1414 * Calculates an IRQ tag for a timer, IPI or similar event.
1415 *
1416 * @returns The IRQ tag.
1417 * @param pDevIns Device instance of the APIC.
1418 * @param u8Level PDM_IRQ_LEVEL_HIGH or PDM_IRQ_LEVEL_FLIP_FLOP.
1419 */
1420 DECLRCCALLBACKMEMBER(uint32_t, pfnCalcIrqTag,(PPDMDEVINS pDevIns, uint8_t u8Level));
1421
1422 /**
1423 * Acquires the PDM lock.
1424 *
1425 * @returns VINF_SUCCESS on success.
1426 * @returns rc if we failed to acquire the lock.
1427 * @param pDevIns The APIC device instance.
1428 * @param rc What to return if we fail to acquire the lock.
1429 */
1430 DECLRCCALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1431
1432 /**
1433 * Releases the PDM lock.
1434 *
1435 * @param pDevIns The APIC device instance.
1436 */
1437 DECLRCCALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1438
1439 /**
1440 * Get the virtual CPU id corresponding to the current EMT.
1441 *
1442 * @param pDevIns The APIC device instance.
1443 */
1444 DECLRCCALLBACKMEMBER(VMCPUID, pfnGetCpuId,(PPDMDEVINS pDevIns));
1445
1446 /** Just a safety precaution. */
1447 uint32_t u32TheEnd;
1448} PDMAPICHLPRC;
1449/** Pointer to APIC GC helpers. */
1450typedef RCPTRTYPE(PDMAPICHLPRC *) PPDMAPICHLPRC;
1451/** Pointer to const APIC helpers. */
1452typedef RCPTRTYPE(const PDMAPICHLPRC *) PCPDMAPICHLPRC;
1453
1454/** Current PDMAPICHLPRC version number. */
1455#define PDM_APICHLPRC_VERSION PDM_VERSION_MAKE(0xfff5, 5, 0)
1456
1457
1458/**
1459 * APIC R0 helpers.
1460 */
1461typedef struct PDMAPICHLPR0
1462{
1463 /** Structure version. PDM_APICHLPR0_VERSION defines the current version. */
1464 uint32_t u32Version;
1465
1466 /**
1467 * Set the interrupt force action flag.
1468 *
1469 * @param pDevIns Device instance of the APIC.
1470 * @param enmType IRQ type.
1471 * @param idCpu Virtual CPU to set flag upon.
1472 */
1473 DECLR0CALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1474
1475 /**
1476 * Clear the interrupt force action flag.
1477 *
1478 * @param pDevIns Device instance of the APIC.
1479 * @param enmType IRQ type.
1480 * @param idCpu Virtual CPU to clear flag upon.
1481 */
1482 DECLR0CALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1483
1484 /**
1485 * Broadcasts an EOI for an interrupt vector to the I/O APICs.
1486 *
1487 * @returns VBox status code.
1488 * @param pDevIns The APIC device instance.
1489 * @param u8Vector The interrupt vector.
1490 */
1491 DECLR0CALLBACKMEMBER(int, pfnBusBroadcastEoi,(PPDMDEVINS pDevIns, uint8_t u8Vector));
1492
1493 /**
1494 * Calculates an IRQ tag for a timer, IPI or similar event.
1495 *
1496 * @returns The IRQ tag.
1497 * @param pDevIns Device instance of the APIC.
1498 * @param u8Level PDM_IRQ_LEVEL_HIGH or PDM_IRQ_LEVEL_FLIP_FLOP.
1499 */
1500 DECLR0CALLBACKMEMBER(uint32_t, pfnCalcIrqTag,(PPDMDEVINS pDevIns, uint8_t u8Level));
1501
1502 /**
1503 * Acquires the PDM lock.
1504 *
1505 * @returns VINF_SUCCESS on success.
1506 * @returns rc if we failed to acquire the lock.
1507 * @param pDevIns The APIC device instance.
1508 * @param rc What to return if we fail to acquire the lock.
1509 */
1510 DECLR0CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1511
1512 /**
1513 * Releases the PDM lock.
1514 *
1515 * @param pDevIns The APIC device instance.
1516 */
1517 DECLR0CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1518
1519 /**
1520 * Get the virtual CPU id corresponding to the current EMT.
1521 *
1522 * @param pDevIns The APIC device instance.
1523 */
1524 DECLR0CALLBACKMEMBER(VMCPUID, pfnGetCpuId,(PPDMDEVINS pDevIns));
1525
1526 /** Just a safety precaution. */
1527 uint32_t u32TheEnd;
1528} PDMAPICHLPR0;
1529/** Pointer to APIC GC helpers. */
1530typedef RCPTRTYPE(PDMAPICHLPR0 *) PPDMAPICHLPR0;
1531/** Pointer to const APIC helpers. */
1532typedef R0PTRTYPE(const PDMAPICHLPR0 *) PCPDMAPICHLPR0;
1533
1534/** Current PDMAPICHLPR0 version number. */
1535#define PDM_APICHLPR0_VERSION PDM_VERSION_MAKE(0xfff4, 5, 0)
1536
1537/**
1538 * APIC R3 helpers.
1539 */
1540typedef struct PDMAPICHLPR3
1541{
1542 /** Structure version. PDM_APICHLPR3_VERSION defines the current version. */
1543 uint32_t u32Version;
1544
1545 /**
1546 * Set the interrupt force action flag.
1547 *
1548 * @param pDevIns Device instance of the APIC.
1549 * @param enmType IRQ type.
1550 * @param idCpu Virtual CPU to set flag upon.
1551 */
1552 DECLR3CALLBACKMEMBER(void, pfnSetInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1553
1554 /**
1555 * Clear the interrupt force action flag.
1556 *
1557 * @param pDevIns Device instance of the APIC.
1558 * @param enmType IRQ type.
1559 * @param idCpu Virtual CPU to clear flag upon.
1560 */
1561 DECLR3CALLBACKMEMBER(void, pfnClearInterruptFF,(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu));
1562
1563 /**
1564 * Broadcasts an EOI for an interrupt vector to the I/O APICs.
1565 *
1566 * @returns VBox status code.
1567 * @param pDevIns The APIC device instance.
1568 * @param u8Vector The interrupt vector.
1569 */
1570 DECLR3CALLBACKMEMBER(int, pfnBusBroadcastEoi,(PPDMDEVINS pDevIns, uint8_t u8Vector));
1571
1572 /**
1573 * Calculates an IRQ tag for a timer, IPI or similar event.
1574 *
1575 * @returns The IRQ tag.
1576 * @param pDevIns Device instance of the APIC.
1577 * @param u8Level PDM_IRQ_LEVEL_HIGH or PDM_IRQ_LEVEL_FLIP_FLOP.
1578 */
1579 DECLR3CALLBACKMEMBER(uint32_t, pfnCalcIrqTag,(PPDMDEVINS pDevIns, uint8_t u8Level));
1580
1581 /**
1582 * Modifies APIC-related bits in the CPUID feature mask and preps MSRs.
1583 *
1584 * @param pDevIns Device instance of the APIC.
1585 * @param enmMode Max supported APIC mode.
1586 */
1587 DECLR3CALLBACKMEMBER(void, pfnSetFeatureLevel,(PPDMDEVINS pDevIns, PDMAPICMODE enmMode));
1588
1589 /**
1590 * Get the virtual CPU id corresponding to the current EMT.
1591 *
1592 * @param pDevIns The APIC device instance.
1593 */
1594 DECLR3CALLBACKMEMBER(VMCPUID, pfnGetCpuId,(PPDMDEVINS pDevIns));
1595
1596 /**
1597 * Sends Startup IPI to given virtual CPU.
1598 *
1599 * @param pDevIns The APIC device instance.
1600 * @param idCpu Virtual CPU to perform Startup IPI on.
1601 * @param uVector Startup IPI vector.
1602 */
1603 DECLR3CALLBACKMEMBER(void, pfnSendStartupIpi,(PPDMDEVINS pDevIns, VMCPUID idCpu, uint32_t uVector));
1604
1605 /**
1606 * Sends INIT IPI to given virtual CPU, should result in reset and
1607 * halting till Startup IPI.
1608 *
1609 * @param pDevIns The APIC device instance.
1610 * @param idCpu Virtual CPU to perform INIT IPI on.
1611 */
1612 DECLR3CALLBACKMEMBER(void, pfnSendInitIpi,(PPDMDEVINS pDevIns, VMCPUID idCpu));
1613
1614 /**
1615 * Gets the address of the RC APIC helpers.
1616 *
1617 * This should be called at both construction and relocation time
1618 * to obtain the correct address of the RC helpers.
1619 *
1620 * @returns GC pointer to the APIC helpers.
1621 * @param pDevIns Device instance of the APIC.
1622 */
1623 DECLR3CALLBACKMEMBER(PCPDMAPICHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
1624
1625 /**
1626 * Gets the address of the R0 APIC helpers.
1627 *
1628 * This should be called at both construction and relocation time
1629 * to obtain the correct address of the R0 helpers.
1630 *
1631 * @returns R0 pointer to the APIC helpers.
1632 * @param pDevIns Device instance of the APIC.
1633 */
1634 DECLR3CALLBACKMEMBER(PCPDMAPICHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
1635
1636 /**
1637 * Get the critical section used to synchronize the PICs, PCI and stuff.
1638 *
1639 * @returns Ring-3 pointer to the critical section.
1640 * @param pDevIns The APIC device instance.
1641 */
1642 DECLR3CALLBACKMEMBER(R3PTRTYPE(PPDMCRITSECT), pfnGetR3CritSect,(PPDMDEVINS pDevIns));
1643
1644 /**
1645 * Get the critical section used to synchronize the PICs, PCI and stuff.
1646 *
1647 * @returns Raw-mode context pointer to the critical section.
1648 * @param pDevIns The APIC device instance.
1649 */
1650 DECLR3CALLBACKMEMBER(RCPTRTYPE(PPDMCRITSECT), pfnGetRCCritSect,(PPDMDEVINS pDevIns));
1651
1652 /**
1653 * Get the critical section used to synchronize the PICs, PCI and stuff.
1654 *
1655 * @returns Ring-0 pointer to the critical section.
1656 * @param pDevIns The APIC device instance.
1657 */
1658 DECLR3CALLBACKMEMBER(R0PTRTYPE(PPDMCRITSECT), pfnGetR0CritSect,(PPDMDEVINS pDevIns));
1659
1660 /** Just a safety precaution. */
1661 uint32_t u32TheEnd;
1662} PDMAPICHLPR3;
1663/** Pointer to APIC helpers. */
1664typedef R3PTRTYPE(PDMAPICHLPR3 *) PPDMAPICHLPR3;
1665/** Pointer to const APIC helpers. */
1666typedef R3PTRTYPE(const PDMAPICHLPR3 *) PCPDMAPICHLPR3;
1667
1668/** Current PDMAPICHLP version number. */
1669#define PDM_APICHLPR3_VERSION PDM_VERSION_MAKE(0xfff3, 4, 0)
1670
1671
1672/**
1673 * I/O APIC registration structure.
1674 */
1675typedef struct PDMIOAPICREG
1676{
1677 /** Struct version+magic number (PDM_IOAPICREG_VERSION). */
1678 uint32_t u32Version;
1679
1680 /**
1681 * Set an IRQ.
1682 *
1683 * @param pDevIns Device instance of the I/O APIC.
1684 * @param iIrq IRQ number to set.
1685 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
1686 * @param uTagSrc The IRQ tag and source (for tracing).
1687 * @remarks Caller enters the PDM critical section
1688 */
1689 DECLR3CALLBACKMEMBER(void, pfnSetIrqR3,(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc));
1690
1691 /** The name of the RC SetIrq entry point. */
1692 const char *pszSetIrqRC;
1693
1694 /** The name of the R0 SetIrq entry point. */
1695 const char *pszSetIrqR0;
1696
1697 /**
1698 * Send a MSI.
1699 *
1700 * @param pDevIns Device instance of the I/O APIC.
1701 * @param GCPhys Request address.
1702 * @param uValue Request value.
1703 * @param uTagSrc The IRQ tag and source (for tracing).
1704 * @remarks Caller enters the PDM critical section
1705 */
1706 DECLR3CALLBACKMEMBER(void, pfnSendMsiR3,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue, uint32_t uTagSrc));
1707
1708 /** The name of the RC SendMsi entry point. */
1709 const char *pszSendMsiRC;
1710
1711 /** The name of the R0 SendMsi entry point. */
1712 const char *pszSendMsiR0;
1713
1714 /**
1715 * Set the EOI for an interrupt vector.
1716 *
1717 * @returns VBox status code.
1718 * @param pDevIns Device instance of the I/O APIC.
1719 * @param u8Vector The vector.
1720 * @remarks Caller enters the PDM critical section
1721 */
1722 DECLR3CALLBACKMEMBER(int, pfnSetEoiR3,(PPDMDEVINS pDevIns, uint8_t u8Vector));
1723
1724 /** The name of the RC SetEoi entry point. */
1725 const char *pszSetEoiRC;
1726
1727 /** The name of the R0 SetEoi entry point. */
1728 const char *pszSetEoiR0;
1729} PDMIOAPICREG;
1730/** Pointer to an APIC registration structure. */
1731typedef PDMIOAPICREG *PPDMIOAPICREG;
1732
1733/** Current PDMAPICREG version number. */
1734#define PDM_IOAPICREG_VERSION PDM_VERSION_MAKE(0xfff2, 5, 0)
1735
1736
1737/**
1738 * IOAPIC RC helpers.
1739 */
1740typedef struct PDMIOAPICHLPRC
1741{
1742 /** Structure version. PDM_IOAPICHLPRC_VERSION defines the current version. */
1743 uint32_t u32Version;
1744
1745 /**
1746 * Private interface between the IOAPIC and APIC.
1747 *
1748 * See comments about this hack on PDMAPICREG::pfnBusDeliverR3.
1749 *
1750 * @returns status code.
1751 * @param pDevIns Device instance of the IOAPIC.
1752 * @param u8Dest See APIC implementation.
1753 * @param u8DestMode See APIC implementation.
1754 * @param u8DeliveryMode See APIC implementation.
1755 * @param iVector See APIC implementation.
1756 * @param u8Polarity See APIC implementation.
1757 * @param u8TriggerMode See APIC implementation.
1758 * @param uTagSrc The IRQ tag and source (for tracing).
1759 */
1760 DECLRCCALLBACKMEMBER(int, pfnApicBusDeliver,(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
1761 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode, uint32_t uTagSrc));
1762
1763 /**
1764 * Acquires the PDM lock.
1765 *
1766 * @returns VINF_SUCCESS on success.
1767 * @returns rc if we failed to acquire the lock.
1768 * @param pDevIns The IOAPIC device instance.
1769 * @param rc What to return if we fail to acquire the lock.
1770 */
1771 DECLRCCALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1772
1773 /**
1774 * Releases the PDM lock.
1775 *
1776 * @param pDevIns The IOAPIC device instance.
1777 */
1778 DECLRCCALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1779
1780 /** Just a safety precaution. */
1781 uint32_t u32TheEnd;
1782} PDMIOAPICHLPRC;
1783/** Pointer to IOAPIC RC helpers. */
1784typedef RCPTRTYPE(PDMIOAPICHLPRC *) PPDMIOAPICHLPRC;
1785/** Pointer to const IOAPIC helpers. */
1786typedef RCPTRTYPE(const PDMIOAPICHLPRC *) PCPDMIOAPICHLPRC;
1787
1788/** Current PDMIOAPICHLPRC version number. */
1789#define PDM_IOAPICHLPRC_VERSION PDM_VERSION_MAKE(0xfff1, 2, 0)
1790
1791
1792/**
1793 * IOAPIC R0 helpers.
1794 */
1795typedef struct PDMIOAPICHLPR0
1796{
1797 /** Structure version. PDM_IOAPICHLPR0_VERSION defines the current version. */
1798 uint32_t u32Version;
1799
1800 /**
1801 * Private interface between the IOAPIC and APIC.
1802 *
1803 * See comments about this hack on PDMAPICREG::pfnBusDeliverR3.
1804 *
1805 * @returns status code.
1806 * @param pDevIns Device instance of the IOAPIC.
1807 * @param u8Dest See APIC implementation.
1808 * @param u8DestMode See APIC implementation.
1809 * @param u8DeliveryMode See APIC implementation.
1810 * @param iVector See APIC implementation.
1811 * @param u8Polarity See APIC implementation.
1812 * @param u8TriggerMode See APIC implementation.
1813 * @param uTagSrc The IRQ tag and source (for tracing).
1814 */
1815 DECLR0CALLBACKMEMBER(int, pfnApicBusDeliver,(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
1816 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode, uint32_t uTagSrc));
1817
1818 /**
1819 * Acquires the PDM lock.
1820 *
1821 * @returns VINF_SUCCESS on success.
1822 * @returns rc if we failed to acquire the lock.
1823 * @param pDevIns The IOAPIC device instance.
1824 * @param rc What to return if we fail to acquire the lock.
1825 */
1826 DECLR0CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1827
1828 /**
1829 * Releases the PDM lock.
1830 *
1831 * @param pDevIns The IOAPIC device instance.
1832 */
1833 DECLR0CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1834
1835 /** Just a safety precaution. */
1836 uint32_t u32TheEnd;
1837} PDMIOAPICHLPR0;
1838/** Pointer to IOAPIC R0 helpers. */
1839typedef R0PTRTYPE(PDMIOAPICHLPR0 *) PPDMIOAPICHLPR0;
1840/** Pointer to const IOAPIC helpers. */
1841typedef R0PTRTYPE(const PDMIOAPICHLPR0 *) PCPDMIOAPICHLPR0;
1842
1843/** Current PDMIOAPICHLPR0 version number. */
1844#define PDM_IOAPICHLPR0_VERSION PDM_VERSION_MAKE(0xfff0, 2, 0)
1845
1846/**
1847 * IOAPIC R3 helpers.
1848 */
1849typedef struct PDMIOAPICHLPR3
1850{
1851 /** Structure version. PDM_IOAPICHLPR3_VERSION defines the current version. */
1852 uint32_t u32Version;
1853
1854 /**
1855 * Private interface between the IOAPIC and APIC.
1856 *
1857 * See comments about this hack on PDMAPICREG::pfnBusDeliverR3.
1858 *
1859 * @returns status code
1860 * @param pDevIns Device instance of the IOAPIC.
1861 * @param u8Dest See APIC implementation.
1862 * @param u8DestMode See APIC implementation.
1863 * @param u8DeliveryMode See APIC implementation.
1864 * @param iVector See APIC implementation.
1865 * @param u8Polarity See APIC implementation.
1866 * @param u8TriggerMode See APIC implementation.
1867 * @param uTagSrc The IRQ tag and source (for tracing).
1868 */
1869 DECLR3CALLBACKMEMBER(int, pfnApicBusDeliver,(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
1870 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode, uint32_t uTagSrc));
1871
1872 /**
1873 * Acquires the PDM lock.
1874 *
1875 * @returns VINF_SUCCESS on success.
1876 * @returns Fatal error on failure.
1877 * @param pDevIns The IOAPIC device instance.
1878 * @param rc Dummy for making the interface identical to the GC and R0 versions.
1879 */
1880 DECLR3CALLBACKMEMBER(int, pfnLock,(PPDMDEVINS pDevIns, int rc));
1881
1882 /**
1883 * Releases the PDM lock.
1884 *
1885 * @param pDevIns The IOAPIC device instance.
1886 */
1887 DECLR3CALLBACKMEMBER(void, pfnUnlock,(PPDMDEVINS pDevIns));
1888
1889 /**
1890 * Gets the address of the RC IOAPIC helpers.
1891 *
1892 * This should be called at both construction and relocation time
1893 * to obtain the correct address of the RC helpers.
1894 *
1895 * @returns RC pointer to the IOAPIC helpers.
1896 * @param pDevIns Device instance of the IOAPIC.
1897 */
1898 DECLR3CALLBACKMEMBER(PCPDMIOAPICHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
1899
1900 /**
1901 * Gets the address of the R0 IOAPIC helpers.
1902 *
1903 * This should be called at both construction and relocation time
1904 * to obtain the correct address of the R0 helpers.
1905 *
1906 * @returns R0 pointer to the IOAPIC helpers.
1907 * @param pDevIns Device instance of the IOAPIC.
1908 */
1909 DECLR3CALLBACKMEMBER(PCPDMIOAPICHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
1910
1911 /** Just a safety precaution. */
1912 uint32_t u32TheEnd;
1913} PDMIOAPICHLPR3;
1914/** Pointer to IOAPIC R3 helpers. */
1915typedef R3PTRTYPE(PDMIOAPICHLPR3 *) PPDMIOAPICHLPR3;
1916/** Pointer to const IOAPIC helpers. */
1917typedef R3PTRTYPE(const PDMIOAPICHLPR3 *) PCPDMIOAPICHLPR3;
1918
1919/** Current PDMIOAPICHLPR3 version number. */
1920#define PDM_IOAPICHLPR3_VERSION PDM_VERSION_MAKE(0xffef, 2, 0)
1921
1922
1923/**
1924 * HPET registration structure.
1925 */
1926typedef struct PDMHPETREG
1927{
1928 /** Struct version+magic number (PDM_HPETREG_VERSION). */
1929 uint32_t u32Version;
1930
1931} PDMHPETREG;
1932/** Pointer to an HPET registration structure. */
1933typedef PDMHPETREG *PPDMHPETREG;
1934
1935/** Current PDMHPETREG version number. */
1936#define PDM_HPETREG_VERSION PDM_VERSION_MAKE(0xffe2, 1, 0)
1937
1938/**
1939 * HPET RC helpers.
1940 *
1941 * @remarks Keep this around in case HPET will need PDM interaction in again RC
1942 * at some later point.
1943 */
1944typedef struct PDMHPETHLPRC
1945{
1946 /** Structure version. PDM_HPETHLPRC_VERSION defines the current version. */
1947 uint32_t u32Version;
1948
1949 /** Just a safety precaution. */
1950 uint32_t u32TheEnd;
1951} PDMHPETHLPRC;
1952
1953/** Pointer to HPET RC helpers. */
1954typedef RCPTRTYPE(PDMHPETHLPRC *) PPDMHPETHLPRC;
1955/** Pointer to const HPET RC helpers. */
1956typedef RCPTRTYPE(const PDMHPETHLPRC *) PCPDMHPETHLPRC;
1957
1958/** Current PDMHPETHLPRC version number. */
1959#define PDM_HPETHLPRC_VERSION PDM_VERSION_MAKE(0xffee, 2, 0)
1960
1961
1962/**
1963 * HPET R0 helpers.
1964 *
1965 * @remarks Keep this around in case HPET will need PDM interaction in again R0
1966 * at some later point.
1967 */
1968typedef struct PDMHPETHLPR0
1969{
1970 /** Structure version. PDM_HPETHLPR0_VERSION defines the current version. */
1971 uint32_t u32Version;
1972
1973 /** Just a safety precaution. */
1974 uint32_t u32TheEnd;
1975} PDMHPETHLPR0;
1976
1977/** Pointer to HPET R0 helpers. */
1978typedef R0PTRTYPE(PDMHPETHLPR0 *) PPDMHPETHLPR0;
1979/** Pointer to const HPET R0 helpers. */
1980typedef R0PTRTYPE(const PDMHPETHLPR0 *) PCPDMHPETHLPR0;
1981
1982/** Current PDMHPETHLPR0 version number. */
1983#define PDM_HPETHLPR0_VERSION PDM_VERSION_MAKE(0xffed, 2, 0)
1984
1985/**
1986 * HPET R3 helpers.
1987 */
1988typedef struct PDMHPETHLPR3
1989{
1990 /** Structure version. PDM_HPETHLP_VERSION defines the current version. */
1991 uint32_t u32Version;
1992
1993 /**
1994 * Gets the address of the RC HPET helpers.
1995 *
1996 * This should be called at both construction and relocation time
1997 * to obtain the correct address of the RC helpers.
1998 *
1999 * @returns RC pointer to the HPET helpers.
2000 * @param pDevIns Device instance of the HPET.
2001 */
2002 DECLR3CALLBACKMEMBER(PCPDMHPETHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
2003
2004 /**
2005 * Gets the address of the R0 HPET helpers.
2006 *
2007 * This should be called at both construction and relocation time
2008 * to obtain the correct address of the R0 helpers.
2009 *
2010 * @returns R0 pointer to the HPET helpers.
2011 * @param pDevIns Device instance of the HPET.
2012 */
2013 DECLR3CALLBACKMEMBER(PCPDMHPETHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
2014
2015 /**
2016 * Set legacy mode on PIT and RTC.
2017 *
2018 * @returns VINF_SUCCESS on success.
2019 * @returns rc if we failed to set legacy mode.
2020 * @param pDevIns Device instance of the HPET.
2021 * @param fActivated Whether legacy mode is activated or deactivated.
2022 */
2023 DECLR3CALLBACKMEMBER(int, pfnSetLegacyMode,(PPDMDEVINS pDevIns, bool fActivated));
2024
2025
2026 /**
2027 * Set IRQ, bypassing ISA bus override rules.
2028 *
2029 * @returns VINF_SUCCESS on success.
2030 * @returns rc if we failed to set legacy mode.
2031 * @param pDevIns Device instance of the HPET.
2032 * @param iIrq IRQ number to set.
2033 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
2034 */
2035 DECLR3CALLBACKMEMBER(int, pfnSetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
2036
2037 /** Just a safety precaution. */
2038 uint32_t u32TheEnd;
2039} PDMHPETHLPR3;
2040
2041/** Pointer to HPET R3 helpers. */
2042typedef R3PTRTYPE(PDMHPETHLPR3 *) PPDMHPETHLPR3;
2043/** Pointer to const HPET R3 helpers. */
2044typedef R3PTRTYPE(const PDMHPETHLPR3 *) PCPDMHPETHLPR3;
2045
2046/** Current PDMHPETHLPR3 version number. */
2047#define PDM_HPETHLPR3_VERSION PDM_VERSION_MAKE(0xffec, 2, 0)
2048
2049
2050/**
2051 * Raw PCI device registration structure.
2052 */
2053typedef struct PDMPCIRAWREG
2054{
2055 /** Struct version+magic number (PDM_PCIRAWREG_VERSION). */
2056 uint32_t u32Version;
2057 /** Just a safety precaution. */
2058 uint32_t u32TheEnd;
2059} PDMPCIRAWREG;
2060/** Pointer to a raw PCI registration structure. */
2061typedef PDMPCIRAWREG *PPDMPCIRAWREG;
2062
2063/** Current PDMPCIRAWREG version number. */
2064#define PDM_PCIRAWREG_VERSION PDM_VERSION_MAKE(0xffe1, 1, 0)
2065
2066/**
2067 * Raw PCI device raw-mode context helpers.
2068 */
2069typedef struct PDMPCIRAWHLPRC
2070{
2071 /** Structure version and magic number (PDM_PCIRAWHLPRC_VERSION). */
2072 uint32_t u32Version;
2073 /** Just a safety precaution. */
2074 uint32_t u32TheEnd;
2075} PDMPCIRAWHLPRC;
2076/** Pointer to a raw PCI deviec raw-mode context helper structure. */
2077typedef RCPTRTYPE(PDMPCIRAWHLPRC *) PPDMPCIRAWHLPRC;
2078/** Pointer to a const raw PCI deviec raw-mode context helper structure. */
2079typedef RCPTRTYPE(const PDMPCIRAWHLPRC *) PCPDMPCIRAWHLPRC;
2080
2081/** Current PDMPCIRAWHLPRC version number. */
2082#define PDM_PCIRAWHLPRC_VERSION PDM_VERSION_MAKE(0xffe0, 1, 0)
2083
2084/**
2085 * Raw PCI device ring-0 context helpers.
2086 */
2087typedef struct PDMPCIRAWHLPR0
2088{
2089 /** Structure version and magic number (PDM_PCIRAWHLPR0_VERSION). */
2090 uint32_t u32Version;
2091 /** Just a safety precaution. */
2092 uint32_t u32TheEnd;
2093} PDMPCIRAWHLPR0;
2094/** Pointer to a raw PCI deviec ring-0 context helper structure. */
2095typedef R0PTRTYPE(PDMPCIRAWHLPR0 *) PPDMPCIRAWHLPR0;
2096/** Pointer to a const raw PCI deviec ring-0 context helper structure. */
2097typedef R0PTRTYPE(const PDMPCIRAWHLPR0 *) PCPDMPCIRAWHLPR0;
2098
2099/** Current PDMPCIRAWHLPR0 version number. */
2100#define PDM_PCIRAWHLPR0_VERSION PDM_VERSION_MAKE(0xffdf, 1, 0)
2101
2102
2103/**
2104 * Raw PCI device ring-3 context helpers.
2105 */
2106typedef struct PDMPCIRAWHLPR3
2107{
2108 /** Undefined structure version and magic number. */
2109 uint32_t u32Version;
2110
2111 /**
2112 * Gets the address of the RC raw PCI device helpers.
2113 *
2114 * This should be called at both construction and relocation time to obtain
2115 * the correct address of the RC helpers.
2116 *
2117 * @returns RC pointer to the raw PCI device helpers.
2118 * @param pDevIns Device instance of the raw PCI device.
2119 */
2120 DECLR3CALLBACKMEMBER(PCPDMPCIRAWHLPRC, pfnGetRCHelpers,(PPDMDEVINS pDevIns));
2121
2122 /**
2123 * Gets the address of the R0 raw PCI device helpers.
2124 *
2125 * This should be called at both construction and relocation time to obtain
2126 * the correct address of the R0 helpers.
2127 *
2128 * @returns R0 pointer to the raw PCI device helpers.
2129 * @param pDevIns Device instance of the raw PCI device.
2130 */
2131 DECLR3CALLBACKMEMBER(PCPDMPCIRAWHLPR0, pfnGetR0Helpers,(PPDMDEVINS pDevIns));
2132
2133 /** Just a safety precaution. */
2134 uint32_t u32TheEnd;
2135} PDMPCIRAWHLPR3;
2136/** Pointer to raw PCI R3 helpers. */
2137typedef R3PTRTYPE(PDMPCIRAWHLPR3 *) PPDMPCIRAWHLPR3;
2138/** Pointer to const raw PCI R3 helpers. */
2139typedef R3PTRTYPE(const PDMPCIRAWHLPR3 *) PCPDMPCIRAWHLPR3;
2140
2141/** Current PDMPCIRAWHLPR3 version number. */
2142#define PDM_PCIRAWHLPR3_VERSION PDM_VERSION_MAKE(0xffde, 1, 0)
2143
2144
2145#ifdef IN_RING3
2146
2147/**
2148 * DMA Transfer Handler.
2149 *
2150 * @returns Number of bytes transferred.
2151 * @param pDevIns Device instance of the DMA.
2152 * @param pvUser User pointer.
2153 * @param uChannel Channel number.
2154 * @param off DMA position.
2155 * @param cb Block size.
2156 * @remarks The device lock is not taken, however, the DMA device lock is held.
2157 */
2158typedef DECLCALLBACK(uint32_t) FNDMATRANSFERHANDLER(PPDMDEVINS pDevIns, void *pvUser, unsigned uChannel, uint32_t off, uint32_t cb);
2159/** Pointer to a FNDMATRANSFERHANDLER(). */
2160typedef FNDMATRANSFERHANDLER *PFNDMATRANSFERHANDLER;
2161
2162/**
2163 * DMA Controller registration structure.
2164 */
2165typedef struct PDMDMAREG
2166{
2167 /** Structure version number. PDM_DMACREG_VERSION defines the current version. */
2168 uint32_t u32Version;
2169
2170 /**
2171 * Execute pending transfers.
2172 *
2173 * @returns A more work indiciator. I.e. 'true' if there is more to be done, and 'false' if all is done.
2174 * @param pDevIns Device instance of the DMAC.
2175 * @remarks No locks held, called on EMT(0) as a form of serialization.
2176 */
2177 DECLR3CALLBACKMEMBER(bool, pfnRun,(PPDMDEVINS pDevIns));
2178
2179 /**
2180 * Register transfer function for DMA channel.
2181 *
2182 * @param pDevIns Device instance of the DMAC.
2183 * @param uChannel Channel number.
2184 * @param pfnTransferHandler Device specific transfer function.
2185 * @param pvUSer User pointer to be passed to the callback.
2186 * @remarks No locks held, called on an EMT.
2187 */
2188 DECLR3CALLBACKMEMBER(void, pfnRegister,(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser));
2189
2190 /**
2191 * Read memory
2192 *
2193 * @returns Number of bytes read.
2194 * @param pDevIns Device instance of the DMAC.
2195 * @param pvBuffer Pointer to target buffer.
2196 * @param off DMA position.
2197 * @param cbBlock Block size.
2198 * @remarks No locks held, called on an EMT.
2199 */
2200 DECLR3CALLBACKMEMBER(uint32_t, pfnReadMemory,(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock));
2201
2202 /**
2203 * Write memory
2204 *
2205 * @returns Number of bytes written.
2206 * @param pDevIns Device instance of the DMAC.
2207 * @param pvBuffer Memory to write.
2208 * @param off DMA position.
2209 * @param cbBlock Block size.
2210 * @remarks No locks held, called on an EMT.
2211 */
2212 DECLR3CALLBACKMEMBER(uint32_t, pfnWriteMemory,(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock));
2213
2214 /**
2215 * Set the DREQ line.
2216 *
2217 * @param pDevIns Device instance of the DMAC.
2218 * @param uChannel Channel number.
2219 * @param uLevel Level of the line.
2220 * @remarks No locks held, called on an EMT.
2221 */
2222 DECLR3CALLBACKMEMBER(void, pfnSetDREQ,(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel));
2223
2224 /**
2225 * Get channel mode
2226 *
2227 * @returns Channel mode.
2228 * @param pDevIns Device instance of the DMAC.
2229 * @param uChannel Channel number.
2230 * @remarks No locks held, called on an EMT.
2231 */
2232 DECLR3CALLBACKMEMBER(uint8_t, pfnGetChannelMode,(PPDMDEVINS pDevIns, unsigned uChannel));
2233
2234} PDMDMACREG;
2235/** Pointer to a DMAC registration structure. */
2236typedef PDMDMACREG *PPDMDMACREG;
2237
2238/** Current PDMDMACREG version number. */
2239#define PDM_DMACREG_VERSION PDM_VERSION_MAKE(0xffeb, 1, 0)
2240
2241
2242/**
2243 * DMA Controller device helpers.
2244 */
2245typedef struct PDMDMACHLP
2246{
2247 /** Structure version. PDM_DMACHLP_VERSION defines the current version. */
2248 uint32_t u32Version;
2249
2250 /* to-be-defined */
2251
2252} PDMDMACHLP;
2253/** Pointer to DMAC helpers. */
2254typedef PDMDMACHLP *PPDMDMACHLP;
2255/** Pointer to const DMAC helpers. */
2256typedef const PDMDMACHLP *PCPDMDMACHLP;
2257
2258/** Current PDMDMACHLP version number. */
2259#define PDM_DMACHLP_VERSION PDM_VERSION_MAKE(0xffea, 1, 0)
2260
2261#endif /* IN_RING3 */
2262
2263
2264
2265/**
2266 * RTC registration structure.
2267 */
2268typedef struct PDMRTCREG
2269{
2270 /** Structure version number. PDM_RTCREG_VERSION defines the current version. */
2271 uint32_t u32Version;
2272 uint32_t u32Alignment; /**< structure size alignment. */
2273
2274 /**
2275 * Write to a CMOS register and update the checksum if necessary.
2276 *
2277 * @returns VBox status code.
2278 * @param pDevIns Device instance of the RTC.
2279 * @param iReg The CMOS register index.
2280 * @param u8Value The CMOS register value.
2281 * @remarks Caller enters the device critical section.
2282 */
2283 DECLR3CALLBACKMEMBER(int, pfnWrite,(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value));
2284
2285 /**
2286 * Read a CMOS register.
2287 *
2288 * @returns VBox status code.
2289 * @param pDevIns Device instance of the RTC.
2290 * @param iReg The CMOS register index.
2291 * @param pu8Value Where to store the CMOS register value.
2292 * @remarks Caller enters the device critical section.
2293 */
2294 DECLR3CALLBACKMEMBER(int, pfnRead,(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value));
2295
2296} PDMRTCREG;
2297/** Pointer to a RTC registration structure. */
2298typedef PDMRTCREG *PPDMRTCREG;
2299/** Pointer to a const RTC registration structure. */
2300typedef const PDMRTCREG *PCPDMRTCREG;
2301
2302/** Current PDMRTCREG version number. */
2303#define PDM_RTCREG_VERSION PDM_VERSION_MAKE(0xffe9, 2, 0)
2304
2305
2306/**
2307 * RTC device helpers.
2308 */
2309typedef struct PDMRTCHLP
2310{
2311 /** Structure version. PDM_RTCHLP_VERSION defines the current version. */
2312 uint32_t u32Version;
2313
2314 /* to-be-defined */
2315
2316} PDMRTCHLP;
2317/** Pointer to RTC helpers. */
2318typedef PDMRTCHLP *PPDMRTCHLP;
2319/** Pointer to const RTC helpers. */
2320typedef const PDMRTCHLP *PCPDMRTCHLP;
2321
2322/** Current PDMRTCHLP version number. */
2323#define PDM_RTCHLP_VERSION PDM_VERSION_MAKE(0xffe8, 1, 0)
2324
2325
2326
2327#ifdef IN_RING3
2328
2329/**
2330 * PDM Device API.
2331 */
2332typedef struct PDMDEVHLPR3
2333{
2334 /** Structure version. PDM_DEVHLPR3_VERSION defines the current version. */
2335 uint32_t u32Version;
2336
2337 /**
2338 * Register a number of I/O ports with a device.
2339 *
2340 * These callbacks are of course for the host context (HC).
2341 * Register HC handlers before guest context (GC) handlers! There must be a
2342 * HC handler for every GC handler!
2343 *
2344 * @returns VBox status.
2345 * @param pDevIns The device instance to register the ports with.
2346 * @param Port First port number in the range.
2347 * @param cPorts Number of ports to register.
2348 * @param pvUser User argument.
2349 * @param pfnOut Pointer to function which is gonna handle OUT operations.
2350 * @param pfnIn Pointer to function which is gonna handle IN operations.
2351 * @param pfnOutStr Pointer to function which is gonna handle string OUT operations.
2352 * @param pfnInStr Pointer to function which is gonna handle string IN operations.
2353 * @param pszDesc Pointer to description string. This must not be freed.
2354 * @remarks Caller enters the device critical section prior to invoking the
2355 * registered callback methods.
2356 */
2357 DECLR3CALLBACKMEMBER(int, pfnIOPortRegister,(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTHCPTR pvUser,
2358 PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn,
2359 PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc));
2360
2361 /**
2362 * Register a number of I/O ports with a device for RC.
2363 *
2364 * These callbacks are for the raw-mode context (RC). Register ring-3 context
2365 * (R3) handlers before raw-mode context handlers! There must be a R3 handler
2366 * for every RC handler!
2367 *
2368 * @returns VBox status.
2369 * @param pDevIns The device instance to register the ports with
2370 * and which RC module to resolve the names
2371 * against.
2372 * @param Port First port number in the range.
2373 * @param cPorts Number of ports to register.
2374 * @param pvUser User argument.
2375 * @param pszOut Name of the RC function which is gonna handle OUT operations.
2376 * @param pszIn Name of the RC function which is gonna handle IN operations.
2377 * @param pszOutStr Name of the RC function which is gonna handle string OUT operations.
2378 * @param pszInStr Name of the RC function which is gonna handle string IN operations.
2379 * @param pszDesc Pointer to description string. This must not be freed.
2380 * @remarks Caller enters the device critical section prior to invoking the
2381 * registered callback methods.
2382 */
2383 DECLR3CALLBACKMEMBER(int, pfnIOPortRegisterRC,(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTRCPTR pvUser,
2384 const char *pszOut, const char *pszIn,
2385 const char *pszOutStr, const char *pszInStr, const char *pszDesc));
2386
2387 /**
2388 * Register a number of I/O ports with a device.
2389 *
2390 * These callbacks are of course for the ring-0 host context (R0).
2391 * Register R3 (HC) handlers before R0 (R0) handlers! There must be a R3 (HC) handler for every R0 handler!
2392 *
2393 * @returns VBox status.
2394 * @param pDevIns The device instance to register the ports with.
2395 * @param Port First port number in the range.
2396 * @param cPorts Number of ports to register.
2397 * @param pvUser User argument. (if pointer, then it must be in locked memory!)
2398 * @param pszOut Name of the R0 function which is gonna handle OUT operations.
2399 * @param pszIn Name of the R0 function which is gonna handle IN operations.
2400 * @param pszOutStr Name of the R0 function which is gonna handle string OUT operations.
2401 * @param pszInStr Name of the R0 function which is gonna handle string IN operations.
2402 * @param pszDesc Pointer to description string. This must not be freed.
2403 * @remarks Caller enters the device critical section prior to invoking the
2404 * registered callback methods.
2405 */
2406 DECLR3CALLBACKMEMBER(int, pfnIOPortRegisterR0,(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTR0PTR pvUser,
2407 const char *pszOut, const char *pszIn,
2408 const char *pszOutStr, const char *pszInStr, const char *pszDesc));
2409
2410 /**
2411 * Deregister I/O ports.
2412 *
2413 * This naturally affects both guest context (GC), ring-0 (R0) and ring-3 (R3/HC) handlers.
2414 *
2415 * @returns VBox status.
2416 * @param pDevIns The device instance owning the ports.
2417 * @param Port First port number in the range.
2418 * @param cPorts Number of ports to deregister.
2419 */
2420 DECLR3CALLBACKMEMBER(int, pfnIOPortDeregister,(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts));
2421
2422 /**
2423 * Register a Memory Mapped I/O (MMIO) region.
2424 *
2425 * These callbacks are of course for the ring-3 context (R3). Register HC
2426 * handlers before raw-mode context (RC) and ring-0 context (R0) handlers! There
2427 * must be a R3 handler for every RC and R0 handler!
2428 *
2429 * @returns VBox status.
2430 * @param pDevIns The device instance to register the MMIO with.
2431 * @param GCPhysStart First physical address in the range.
2432 * @param cbRange The size of the range (in bytes).
2433 * @param pvUser User argument.
2434 * @param pfnWrite Pointer to function which is gonna handle Write operations.
2435 * @param pfnRead Pointer to function which is gonna handle Read operations.
2436 * @param pfnFill Pointer to function which is gonna handle Fill/memset operations. (optional)
2437 * @param fFlags Flags, IOMMMIO_FLAGS_XXX.
2438 * @param pszDesc Pointer to description string. This must not be freed.
2439 * @remarks Caller enters the device critical section prior to invoking the
2440 * registered callback methods.
2441 */
2442 DECLR3CALLBACKMEMBER(int, pfnMMIORegister,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTHCPTR pvUser,
2443 PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, PFNIOMMMIOFILL pfnFill,
2444 uint32_t fFlags, const char *pszDesc));
2445
2446 /**
2447 * Register a Memory Mapped I/O (MMIO) region for RC.
2448 *
2449 * These callbacks are for the raw-mode context (RC). Register ring-3 context
2450 * (R3) handlers before guest context handlers! There must be a R3 handler for
2451 * every RC handler!
2452 *
2453 * @returns VBox status.
2454 * @param pDevIns The device instance to register the MMIO with.
2455 * @param GCPhysStart First physical address in the range.
2456 * @param cbRange The size of the range (in bytes).
2457 * @param pvUser User argument.
2458 * @param pszWrite Name of the RC function which is gonna handle Write operations.
2459 * @param pszRead Name of the RC function which is gonna handle Read operations.
2460 * @param pszFill Name of the RC function which is gonna handle Fill/memset operations. (optional)
2461 * @remarks Caller enters the device critical section prior to invoking the
2462 * registered callback methods.
2463 */
2464 DECLR3CALLBACKMEMBER(int, pfnMMIORegisterRC,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTRCPTR pvUser,
2465 const char *pszWrite, const char *pszRead, const char *pszFill));
2466
2467 /**
2468 * Register a Memory Mapped I/O (MMIO) region for R0.
2469 *
2470 * These callbacks are for the ring-0 host context (R0). Register ring-3
2471 * constext (R3) handlers before R0 handlers! There must be a R3 handler for
2472 * every R0 handler!
2473 *
2474 * @returns VBox status.
2475 * @param pDevIns The device instance to register the MMIO with.
2476 * @param GCPhysStart First physical address in the range.
2477 * @param cbRange The size of the range (in bytes).
2478 * @param pvUser User argument. (if pointer, then it must be in locked memory!)
2479 * @param pszWrite Name of the RC function which is gonna handle Write operations.
2480 * @param pszRead Name of the RC function which is gonna handle Read operations.
2481 * @param pszFill Name of the RC function which is gonna handle Fill/memset operations. (optional)
2482 * @remarks Caller enters the device critical section prior to invoking the
2483 * registered callback methods.
2484 */
2485 DECLR3CALLBACKMEMBER(int, pfnMMIORegisterR0,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTR0PTR pvUser,
2486 const char *pszWrite, const char *pszRead, const char *pszFill));
2487
2488 /**
2489 * Deregister a Memory Mapped I/O (MMIO) region.
2490 *
2491 * This naturally affects both guest context (GC), ring-0 (R0) and ring-3 (R3/HC) handlers.
2492 *
2493 * @returns VBox status.
2494 * @param pDevIns The device instance owning the MMIO region(s).
2495 * @param GCPhysStart First physical address in the range.
2496 * @param cbRange The size of the range (in bytes).
2497 */
2498 DECLR3CALLBACKMEMBER(int, pfnMMIODeregister,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange));
2499
2500 /**
2501 * Allocate and register a MMIO2 region.
2502 *
2503 * As mentioned elsewhere, MMIO2 is just RAM spelled differently. It's
2504 * RAM associated with a device. It is also non-shared memory with a
2505 * permanent ring-3 mapping and page backing (presently).
2506 *
2507 * @returns VBox status.
2508 * @param pDevIns The device instance.
2509 * @param iRegion The region number. Use the PCI region number as
2510 * this must be known to the PCI bus device too. If
2511 * it's not associated with the PCI device, then
2512 * any number up to UINT8_MAX is fine.
2513 * @param cb The size (in bytes) of the region.
2514 * @param fFlags Reserved for future use, must be zero.
2515 * @param ppv Where to store the address of the ring-3 mapping
2516 * of the memory.
2517 * @param pszDesc Pointer to description string. This must not be
2518 * freed.
2519 * @thread EMT.
2520 */
2521 DECLR3CALLBACKMEMBER(int, pfnMMIO2Register,(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc));
2522
2523 /**
2524 * Deregisters and frees a MMIO2 region.
2525 *
2526 * Any physical (and virtual) access handlers registered for the region must
2527 * be deregistered before calling this function.
2528 *
2529 * @returns VBox status code.
2530 * @param pDevIns The device instance.
2531 * @param iRegion The region number used during registration.
2532 * @thread EMT.
2533 */
2534 DECLR3CALLBACKMEMBER(int, pfnMMIO2Deregister,(PPDMDEVINS pDevIns, uint32_t iRegion));
2535
2536 /**
2537 * Maps a MMIO2 region into the physical memory space.
2538 *
2539 * A MMIO2 range may overlap with base memory if a lot of RAM
2540 * is configured for the VM, in which case we'll drop the base
2541 * memory pages. Presently we will make no attempt to preserve
2542 * anything that happens to be present in the base memory that
2543 * is replaced, this is of course incorrect but it's too much
2544 * effort.
2545 *
2546 * @returns VBox status code.
2547 * @param pDevIns The device instance.
2548 * @param iRegion The region number used during registration.
2549 * @param GCPhys The physical address to map it at.
2550 * @thread EMT.
2551 */
2552 DECLR3CALLBACKMEMBER(int, pfnMMIO2Map,(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys));
2553
2554 /**
2555 * Unmaps a MMIO2 region previously mapped using pfnMMIO2Map.
2556 *
2557 * @returns VBox status code.
2558 * @param pDevIns The device instance.
2559 * @param iRegion The region number used during registration.
2560 * @param GCPhys The physical address it's currently mapped at.
2561 * @thread EMT.
2562 */
2563 DECLR3CALLBACKMEMBER(int, pfnMMIO2Unmap,(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys));
2564
2565 /**
2566 * Maps a portion of an MMIO2 region into the hypervisor region.
2567 *
2568 * Callers of this API must never deregister the MMIO2 region before the
2569 * VM is powered off.
2570 *
2571 * @return VBox status code.
2572 * @param pDevIns The device owning the MMIO2 memory.
2573 * @param iRegion The region.
2574 * @param off The offset into the region. Will be rounded down
2575 * to closest page boundary.
2576 * @param cb The number of bytes to map. Will be rounded up
2577 * to the closest page boundary.
2578 * @param pszDesc Mapping description.
2579 * @param pRCPtr Where to store the RC address.
2580 */
2581 DECLR3CALLBACKMEMBER(int, pfnMMHyperMapMMIO2,(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
2582 const char *pszDesc, PRTRCPTR pRCPtr));
2583
2584 /**
2585 * Maps a portion of an MMIO2 region into kernel space (host).
2586 *
2587 * The kernel mapping will become invalid when the MMIO2 memory is deregistered
2588 * or the VM is terminated.
2589 *
2590 * @return VBox status code.
2591 * @param pDevIns The device owning the MMIO2 memory.
2592 * @param iRegion The region.
2593 * @param off The offset into the region. Must be page
2594 * aligned.
2595 * @param cb The number of bytes to map. Must be page
2596 * aligned.
2597 * @param pszDesc Mapping description.
2598 * @param pR0Ptr Where to store the R0 address.
2599 */
2600 DECLR3CALLBACKMEMBER(int, pfnMMIO2MapKernel,(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
2601 const char *pszDesc, PRTR0PTR pR0Ptr));
2602
2603 /**
2604 * Register a ROM (BIOS) region.
2605 *
2606 * It goes without saying that this is read-only memory. The memory region must be
2607 * in unassigned memory. I.e. from the top of the address space or on the PC in
2608 * the 0xa0000-0xfffff range.
2609 *
2610 * @returns VBox status.
2611 * @param pDevIns The device instance owning the ROM region.
2612 * @param GCPhysStart First physical address in the range.
2613 * Must be page aligned!
2614 * @param cbRange The size of the range (in bytes).
2615 * Must be page aligned!
2616 * @param pvBinary Pointer to the binary data backing the ROM image.
2617 * @param cbBinary The size of the binary pointer. This must
2618 * be equal or smaller than @a cbRange.
2619 * @param fFlags Shadow ROM flags, PGMPHYS_ROM_FLAGS_* in pgm.h.
2620 * @param pszDesc Pointer to description string. This must not be freed.
2621 *
2622 * @remark There is no way to remove the rom, automatically on device cleanup or
2623 * manually from the device yet. At present I doubt we need such features...
2624 */
2625 DECLR3CALLBACKMEMBER(int, pfnROMRegister,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange,
2626 const void *pvBinary, uint32_t cbBinary, uint32_t fFlags, const char *pszDesc));
2627
2628 /**
2629 * Changes the protection of shadowed ROM mapping.
2630 *
2631 * This is intented for use by the system BIOS, chipset or device in question to
2632 * change the protection of shadowed ROM code after init and on reset.
2633 *
2634 * @param pDevIns The device instance.
2635 * @param GCPhysStart Where the mapping starts.
2636 * @param cbRange The size of the mapping.
2637 * @param enmProt The new protection type.
2638 */
2639 DECLR3CALLBACKMEMBER(int, pfnROMProtectShadow,(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, PGMROMPROT enmProt));
2640
2641 /**
2642 * Register a save state data unit.
2643 *
2644 * @returns VBox status.
2645 * @param pDevIns The device instance.
2646 * @param uVersion Data layout version number.
2647 * @param cbGuess The approximate amount of data in the unit.
2648 * Only for progress indicators.
2649 * @param pszBefore Name of data unit which we should be put in
2650 * front of. Optional (NULL).
2651 *
2652 * @param pfnLivePrep Prepare live save callback, optional.
2653 * @param pfnLiveExec Execute live save callback, optional.
2654 * @param pfnLiveVote Vote live save callback, optional.
2655 *
2656 * @param pfnSavePrep Prepare save callback, optional.
2657 * @param pfnSaveExec Execute save callback, optional.
2658 * @param pfnSaveDone Done save callback, optional.
2659 *
2660 * @param pfnLoadPrep Prepare load callback, optional.
2661 * @param pfnLoadExec Execute load callback, optional.
2662 * @param pfnLoadDone Done load callback, optional.
2663 * @remarks Caller enters the device critical section prior to invoking the
2664 * registered callback methods.
2665 */
2666 DECLR3CALLBACKMEMBER(int, pfnSSMRegister,(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess, const char *pszBefore,
2667 PFNSSMDEVLIVEPREP pfnLivePrep, PFNSSMDEVLIVEEXEC pfnLiveExec, PFNSSMDEVLIVEVOTE pfnLiveVote,
2668 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
2669 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone));
2670
2671 /**
2672 * Creates a timer.
2673 *
2674 * @returns VBox status.
2675 * @param pDevIns The device instance.
2676 * @param enmClock The clock to use on this timer.
2677 * @param pfnCallback Callback function.
2678 * @param pvUser User argument for the callback.
2679 * @param fFlags Flags, see TMTIMER_FLAGS_*.
2680 * @param pszDesc Pointer to description string which must stay around
2681 * until the timer is fully destroyed (i.e. a bit after TMTimerDestroy()).
2682 * @param ppTimer Where to store the timer on success.
2683 * @remarks Caller enters the device critical section prior to invoking the
2684 * callback.
2685 */
2686 DECLR3CALLBACKMEMBER(int, pfnTMTimerCreate,(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback,
2687 void *pvUser, uint32_t fFlags, const char *pszDesc, PPTMTIMERR3 ppTimer));
2688
2689 /**
2690 * Get the real world UTC time adjusted for VM lag, user offset and warpdrive.
2691 *
2692 * @returns pTime.
2693 * @param pDevIns The device instance.
2694 * @param pTime Where to store the time.
2695 */
2696 DECLR3CALLBACKMEMBER(PRTTIMESPEC, pfnTMUtcNow,(PPDMDEVINS pDevIns, PRTTIMESPEC pTime));
2697
2698 /**
2699 * Read physical memory.
2700 *
2701 * @returns VINF_SUCCESS (for now).
2702 * @param pDevIns The device instance.
2703 * @param GCPhys Physical address start reading from.
2704 * @param pvBuf Where to put the read bits.
2705 * @param cbRead How many bytes to read.
2706 * @thread Any thread, but the call may involve the emulation thread.
2707 */
2708 DECLR3CALLBACKMEMBER(int, pfnPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
2709
2710 /**
2711 * Write to physical memory.
2712 *
2713 * @returns VINF_SUCCESS for now, and later maybe VERR_EM_MEMORY.
2714 * @param pDevIns The device instance.
2715 * @param GCPhys Physical address to write to.
2716 * @param pvBuf What to write.
2717 * @param cbWrite How many bytes to write.
2718 * @thread Any thread, but the call may involve the emulation thread.
2719 */
2720 DECLR3CALLBACKMEMBER(int, pfnPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
2721
2722 /**
2723 * Requests the mapping of a guest page into ring-3.
2724 *
2725 * When you're done with the page, call pfnPhysReleasePageMappingLock() ASAP to
2726 * release it.
2727 *
2728 * This API will assume your intention is to write to the page, and will
2729 * therefore replace shared and zero pages. If you do not intend to modify the
2730 * page, use the pfnPhysGCPhys2CCPtrReadOnly() API.
2731 *
2732 * @returns VBox status code.
2733 * @retval VINF_SUCCESS on success.
2734 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical
2735 * backing or if the page has any active access handlers. The caller
2736 * must fall back on using PGMR3PhysWriteExternal.
2737 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
2738 *
2739 * @param pDevIns The device instance.
2740 * @param GCPhys The guest physical address of the page that
2741 * should be mapped.
2742 * @param fFlags Flags reserved for future use, MBZ.
2743 * @param ppv Where to store the address corresponding to
2744 * GCPhys.
2745 * @param pLock Where to store the lock information that
2746 * pfnPhysReleasePageMappingLock needs.
2747 *
2748 * @remark Avoid calling this API from within critical sections (other than the
2749 * PGM one) because of the deadlock risk when we have to delegating the
2750 * task to an EMT.
2751 * @thread Any.
2752 */
2753 DECLR3CALLBACKMEMBER(int, pfnPhysGCPhys2CCPtr,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, void **ppv,
2754 PPGMPAGEMAPLOCK pLock));
2755
2756 /**
2757 * Requests the mapping of a guest page into ring-3, external threads.
2758 *
2759 * When you're done with the page, call pfnPhysReleasePageMappingLock() ASAP to
2760 * release it.
2761 *
2762 * @returns VBox status code.
2763 * @retval VINF_SUCCESS on success.
2764 * @retval VERR_PGM_PHYS_PAGE_RESERVED it it's a valid page but has no physical
2765 * backing or if the page as an active ALL access handler. The caller
2766 * must fall back on using PGMPhysRead.
2767 * @retval VERR_PGM_INVALID_GC_PHYSICAL_ADDRESS if it's not a valid physical address.
2768 *
2769 * @param pDevIns The device instance.
2770 * @param GCPhys The guest physical address of the page that
2771 * should be mapped.
2772 * @param fFlags Flags reserved for future use, MBZ.
2773 * @param ppv Where to store the address corresponding to
2774 * GCPhys.
2775 * @param pLock Where to store the lock information that
2776 * pfnPhysReleasePageMappingLock needs.
2777 *
2778 * @remark Avoid calling this API from within critical sections.
2779 * @thread Any.
2780 */
2781 DECLR3CALLBACKMEMBER(int, pfnPhysGCPhys2CCPtrReadOnly,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags,
2782 void const **ppv, PPGMPAGEMAPLOCK pLock));
2783
2784 /**
2785 * Release the mapping of a guest page.
2786 *
2787 * This is the counter part of pfnPhysGCPhys2CCPtr and
2788 * pfnPhysGCPhys2CCPtrReadOnly.
2789 *
2790 * @param pDevIns The device instance.
2791 * @param pLock The lock structure initialized by the mapping
2792 * function.
2793 */
2794 DECLR3CALLBACKMEMBER(void, pfnPhysReleasePageMappingLock,(PPDMDEVINS pDevIns, PPGMPAGEMAPLOCK pLock));
2795
2796 /**
2797 * Read guest physical memory by virtual address.
2798 *
2799 * @param pDevIns The device instance.
2800 * @param pvDst Where to put the read bits.
2801 * @param GCVirtSrc Guest virtual address to start reading from.
2802 * @param cb How many bytes to read.
2803 * @thread The emulation thread.
2804 */
2805 DECLR3CALLBACKMEMBER(int, pfnPhysReadGCVirt,(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb));
2806
2807 /**
2808 * Write to guest physical memory by virtual address.
2809 *
2810 * @param pDevIns The device instance.
2811 * @param GCVirtDst Guest virtual address to write to.
2812 * @param pvSrc What to write.
2813 * @param cb How many bytes to write.
2814 * @thread The emulation thread.
2815 */
2816 DECLR3CALLBACKMEMBER(int, pfnPhysWriteGCVirt,(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb));
2817
2818 /**
2819 * Convert a guest virtual address to a guest physical address.
2820 *
2821 * @returns VBox status code.
2822 * @param pDevIns The device instance.
2823 * @param GCPtr Guest virtual address.
2824 * @param pGCPhys Where to store the GC physical address
2825 * corresponding to GCPtr.
2826 * @thread The emulation thread.
2827 * @remark Careful with page boundaries.
2828 */
2829 DECLR3CALLBACKMEMBER(int, pfnPhysGCPtr2GCPhys, (PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys));
2830
2831 /**
2832 * Allocate memory which is associated with current VM instance
2833 * and automatically freed on it's destruction.
2834 *
2835 * @returns Pointer to allocated memory. The memory is *NOT* zero-ed.
2836 * @param pDevIns The device instance.
2837 * @param cb Number of bytes to allocate.
2838 */
2839 DECLR3CALLBACKMEMBER(void *, pfnMMHeapAlloc,(PPDMDEVINS pDevIns, size_t cb));
2840
2841 /**
2842 * Allocate memory which is associated with current VM instance
2843 * and automatically freed on it's destruction. The memory is ZEROed.
2844 *
2845 * @returns Pointer to allocated memory. The memory is *NOT* zero-ed.
2846 * @param pDevIns The device instance.
2847 * @param cb Number of bytes to allocate.
2848 */
2849 DECLR3CALLBACKMEMBER(void *, pfnMMHeapAllocZ,(PPDMDEVINS pDevIns, size_t cb));
2850
2851 /**
2852 * Free memory allocated with pfnMMHeapAlloc() and pfnMMHeapAllocZ().
2853 *
2854 * @param pDevIns The device instance.
2855 * @param pv Pointer to the memory to free.
2856 */
2857 DECLR3CALLBACKMEMBER(void, pfnMMHeapFree,(PPDMDEVINS pDevIns, void *pv));
2858
2859 /**
2860 * Gets the VM state.
2861 *
2862 * @returns VM state.
2863 * @param pDevIns The device instance.
2864 * @thread Any thread (just keep in mind that it's volatile info).
2865 */
2866 DECLR3CALLBACKMEMBER(VMSTATE, pfnVMState, (PPDMDEVINS pDevIns));
2867
2868 /**
2869 * Checks if the VM was teleported and hasn't been fully resumed yet.
2870 *
2871 * @returns true / false.
2872 * @param pDevIns The device instance.
2873 * @thread Any thread.
2874 */
2875 DECLR3CALLBACKMEMBER(bool, pfnVMTeleportedAndNotFullyResumedYet,(PPDMDEVINS pDevIns));
2876
2877 /**
2878 * Set the VM error message
2879 *
2880 * @returns rc.
2881 * @param pDevIns The device instance.
2882 * @param rc VBox status code.
2883 * @param SRC_POS Use RT_SRC_POS.
2884 * @param pszFormat Error message format string.
2885 * @param ... Error message arguments.
2886 */
2887 DECLR3CALLBACKMEMBER(int, pfnVMSetError,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
2888 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(6, 7));
2889
2890 /**
2891 * Set the VM error message
2892 *
2893 * @returns rc.
2894 * @param pDevIns The device instance.
2895 * @param rc VBox status code.
2896 * @param SRC_POS Use RT_SRC_POS.
2897 * @param pszFormat Error message format string.
2898 * @param va Error message arguments.
2899 */
2900 DECLR3CALLBACKMEMBER(int, pfnVMSetErrorV,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
2901 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(6, 0));
2902
2903 /**
2904 * Set the VM runtime error message
2905 *
2906 * @returns VBox status code.
2907 * @param pDevIns The device instance.
2908 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
2909 * @param pszErrorId Error ID string.
2910 * @param pszFormat Error message format string.
2911 * @param ... Error message arguments.
2912 */
2913 DECLR3CALLBACKMEMBER(int, pfnVMSetRuntimeError,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
2914 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(4, 5));
2915
2916 /**
2917 * Set the VM runtime error message
2918 *
2919 * @returns VBox status code.
2920 * @param pDevIns The device instance.
2921 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
2922 * @param pszErrorId Error ID string.
2923 * @param pszFormat Error message format string.
2924 * @param va Error message arguments.
2925 */
2926 DECLR3CALLBACKMEMBER(int, pfnVMSetRuntimeErrorV,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
2927 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(4, 0));
2928
2929 /**
2930 * Stops the VM and enters the debugger to look at the guest state.
2931 *
2932 * Use the PDMDeviceDBGFStop() inline function with the RT_SRC_POS macro instead of
2933 * invoking this function directly.
2934 *
2935 * @returns VBox status code which must be passed up to the VMM.
2936 * @param pDevIns The device instance.
2937 * @param pszFile Filename of the assertion location.
2938 * @param iLine The linenumber of the assertion location.
2939 * @param pszFunction Function of the assertion location.
2940 * @param pszFormat Message. (optional)
2941 * @param args Message parameters.
2942 */
2943 DECLR3CALLBACKMEMBER(int, pfnDBGFStopV,(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction,
2944 const char *pszFormat, va_list args) RT_IPRT_FORMAT_ATTR(5, 0));
2945
2946 /**
2947 * Register a info handler with DBGF,
2948 *
2949 * @returns VBox status code.
2950 * @param pDevIns The device instance.
2951 * @param pszName The identifier of the info.
2952 * @param pszDesc The description of the info and any arguments
2953 * the handler may take.
2954 * @param pfnHandler The handler function to be called to display the
2955 * info.
2956 */
2957 DECLR3CALLBACKMEMBER(int, pfnDBGFInfoRegister,(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler));
2958
2959 /**
2960 * Registers a set of registers for a device.
2961 *
2962 * The @a pvUser argument of the getter and setter callbacks will be
2963 * @a pDevIns. The register names will be prefixed by the device name followed
2964 * immediately by the instance number.
2965 *
2966 * @returns VBox status code.
2967 * @param pDevIns The device instance.
2968 * @param paRegisters The register descriptors.
2969 *
2970 * @remarks The device critical section is NOT entered prior to working the
2971 * callbacks registered via this helper!
2972 */
2973 DECLR3CALLBACKMEMBER(int, pfnDBGFRegRegister,(PPDMDEVINS pDevIns, PCDBGFREGDESC paRegisters));
2974
2975 /**
2976 * Gets the trace buffer handle.
2977 *
2978 * This is used by the macros found in VBox/vmm/dbgftrace.h and is not
2979 * really inteded for direct usage, thus no inline wrapper function.
2980 *
2981 * @returns Trace buffer handle or NIL_RTTRACEBUF.
2982 * @param pDevIns The device instance.
2983 */
2984 DECLR3CALLBACKMEMBER(RTTRACEBUF, pfnDBGFTraceBuf,(PPDMDEVINS pDevIns));
2985
2986 /**
2987 * Registers a statistics sample if statistics are enabled.
2988 *
2989 * @param pDevIns Device instance of the DMA.
2990 * @param pvSample Pointer to the sample.
2991 * @param enmType Sample type. This indicates what pvSample is
2992 * pointing at.
2993 * @param pszName Sample name. The name is on this form
2994 * "/<component>/<sample>". Further nesting is
2995 * possible.
2996 * @param enmUnit Sample unit.
2997 * @param pszDesc Sample description.
2998 */
2999 DECLR3CALLBACKMEMBER(void, pfnSTAMRegister,(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName, STAMUNIT enmUnit, const char *pszDesc));
3000
3001 /**
3002 * Same as pfnSTAMRegister except that the name is specified in a
3003 * RTStrPrintf like fashion.
3004 *
3005 * @returns VBox status.
3006 * @param pDevIns Device instance of the DMA.
3007 * @param pvSample Pointer to the sample.
3008 * @param enmType Sample type. This indicates what pvSample is
3009 * pointing at.
3010 * @param enmVisibility Visibility type specifying whether unused
3011 * statistics should be visible or not.
3012 * @param enmUnit Sample unit.
3013 * @param pszDesc Sample description.
3014 * @param pszName The sample name format string.
3015 * @param ... Arguments to the format string.
3016 */
3017 DECLR3CALLBACKMEMBER(void, pfnSTAMRegisterF,(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType,
3018 STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc,
3019 const char *pszName, ...) RT_IPRT_FORMAT_ATTR(7, 8));
3020
3021 /**
3022 * Same as pfnSTAMRegister except that the name is specified in a
3023 * RTStrPrintfV like fashion.
3024 *
3025 * @returns VBox status.
3026 * @param pDevIns Device instance of the DMA.
3027 * @param pvSample Pointer to the sample.
3028 * @param enmType Sample type. This indicates what pvSample is
3029 * pointing at.
3030 * @param enmVisibility Visibility type specifying whether unused
3031 * statistics should be visible or not.
3032 * @param enmUnit Sample unit.
3033 * @param pszDesc Sample description.
3034 * @param pszName The sample name format string.
3035 * @param args Arguments to the format string.
3036 */
3037 DECLR3CALLBACKMEMBER(void, pfnSTAMRegisterV,(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType,
3038 STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc,
3039 const char *pszName, va_list args) RT_IPRT_FORMAT_ATTR(7, 0));
3040
3041 /**
3042 * Registers the device with the default PCI bus.
3043 *
3044 * @returns VBox status code.
3045 * @param pDevIns The device instance.
3046 * @param pPciDev The PCI device structure.
3047 * Any PCI enabled device must keep this in it's instance data!
3048 * Fill in the PCI data config before registration, please.
3049 * @remark This is the simple interface, a Ex interface will be created if
3050 * more features are needed later.
3051 */
3052 DECLR3CALLBACKMEMBER(int, pfnPCIRegister,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev));
3053
3054 /**
3055 * Initialize MSI support in a PCI device.
3056 *
3057 * @returns VBox status code.
3058 * @param pDevIns The device instance.
3059 * @param pMsiReg MSI registartion structure.
3060 */
3061 DECLR3CALLBACKMEMBER(int, pfnPCIRegisterMsi,(PPDMDEVINS pDevIns, PPDMMSIREG pMsiReg));
3062
3063 /**
3064 * Registers a I/O region (memory mapped or I/O ports) for a PCI device.
3065 *
3066 * @returns VBox status code.
3067 * @param pDevIns The device instance.
3068 * @param iRegion The region number.
3069 * @param cbRegion Size of the region.
3070 * @param enmType PCI_ADDRESS_SPACE_MEM, PCI_ADDRESS_SPACE_IO or PCI_ADDRESS_SPACE_MEM_PREFETCH.
3071 * @param pfnCallback Callback for doing the mapping.
3072 * @remarks The callback will be invoked holding the PDM lock. The device lock
3073 * is NOT take because that is very likely be a lock order violation.
3074 */
3075 DECLR3CALLBACKMEMBER(int, pfnPCIIORegionRegister,(PPDMDEVINS pDevIns, int iRegion, uint32_t cbRegion,
3076 PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback));
3077
3078 /**
3079 * Register PCI configuration space read/write callbacks.
3080 *
3081 * @param pDevIns The device instance.
3082 * @param pPciDev The PCI device structure.
3083 * If NULL the default PCI device for this device instance is used.
3084 * @param pfnRead Pointer to the user defined PCI config read function.
3085 * @param ppfnReadOld Pointer to function pointer which will receive the old (default)
3086 * PCI config read function. This way, user can decide when (and if)
3087 * to call default PCI config read function. Can be NULL.
3088 * @param pfnWrite Pointer to the user defined PCI config write function.
3089 * @param ppfnWriteOld Pointer to function pointer which will receive
3090 * the old (default) PCI config write function.
3091 * This way, user can decide when (and if) to call
3092 * default PCI config write function. Can be NULL.
3093 * @remarks The callbacks will be invoked holding the PDM lock. The device lock
3094 * is NOT take because that is very likely be a lock order violation.
3095 * @thread EMT
3096 */
3097 DECLR3CALLBACKMEMBER(void, pfnPCISetConfigCallbacks,(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
3098 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld));
3099
3100 /**
3101 * Bus master physical memory read.
3102 *
3103 * @returns VINF_SUCCESS or VERR_PGM_PCI_PHYS_READ_BM_DISABLED, later maybe
3104 * VERR_EM_MEMORY. The informational status shall NOT be propagated!
3105 * @param pDevIns The device instance.
3106 * @param GCPhys Physical address start reading from.
3107 * @param pvBuf Where to put the read bits.
3108 * @param cbRead How many bytes to read.
3109 * @thread Any thread, but the call may involve the emulation thread.
3110 */
3111 DECLR3CALLBACKMEMBER(int, pfnPCIPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
3112
3113 /**
3114 * Bus master physical memory write.
3115 *
3116 * @returns VINF_SUCCESS or VERR_PGM_PCI_PHYS_WRITE_BM_DISABLED, later maybe
3117 * VERR_EM_MEMORY. The informational status shall NOT be propagated!
3118 * @param pDevIns The device instance.
3119 * @param GCPhys Physical address to write to.
3120 * @param pvBuf What to write.
3121 * @param cbWrite How many bytes to write.
3122 * @thread Any thread, but the call may involve the emulation thread.
3123 */
3124 DECLR3CALLBACKMEMBER(int, pfnPCIPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
3125
3126 /**
3127 * Set the IRQ for a PCI device.
3128 *
3129 * @param pDevIns The device instance.
3130 * @param iIrq IRQ number to set.
3131 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
3132 * @thread Any thread, but will involve the emulation thread.
3133 */
3134 DECLR3CALLBACKMEMBER(void, pfnPCISetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3135
3136 /**
3137 * Set the IRQ for a PCI device, but don't wait for EMT to process
3138 * the request when not called from EMT.
3139 *
3140 * @param pDevIns The device instance.
3141 * @param iIrq IRQ number to set.
3142 * @param iLevel IRQ level.
3143 * @thread Any thread, but will involve the emulation thread.
3144 */
3145 DECLR3CALLBACKMEMBER(void, pfnPCISetIrqNoWait,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3146
3147 /**
3148 * Set ISA IRQ for a device.
3149 *
3150 * @param pDevIns The device instance.
3151 * @param iIrq IRQ number to set.
3152 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
3153 * @thread Any thread, but will involve the emulation thread.
3154 */
3155 DECLR3CALLBACKMEMBER(void, pfnISASetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3156
3157 /**
3158 * Set the ISA IRQ for a device, but don't wait for EMT to process
3159 * the request when not called from EMT.
3160 *
3161 * @param pDevIns The device instance.
3162 * @param iIrq IRQ number to set.
3163 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
3164 * @thread Any thread, but will involve the emulation thread.
3165 */
3166 DECLR3CALLBACKMEMBER(void, pfnISASetIrqNoWait,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3167
3168 /**
3169 * Attaches a driver (chain) to the device.
3170 *
3171 * The first call for a LUN this will serve as a registartion of the LUN. The pBaseInterface and
3172 * the pszDesc string will be registered with that LUN and kept around for PDMR3QueryDeviceLun().
3173 *
3174 * @returns VBox status code.
3175 * @param pDevIns The device instance.
3176 * @param iLun The logical unit to attach.
3177 * @param pBaseInterface Pointer to the base interface for that LUN. (device side / down)
3178 * @param ppBaseInterface Where to store the pointer to the base interface. (driver side / up)
3179 * @param pszDesc Pointer to a string describing the LUN. This string must remain valid
3180 * for the live of the device instance.
3181 */
3182 DECLR3CALLBACKMEMBER(int, pfnDriverAttach,(PPDMDEVINS pDevIns, uint32_t iLun, PPDMIBASE pBaseInterface,
3183 PPDMIBASE *ppBaseInterface, const char *pszDesc));
3184
3185 /**
3186 * Detaches an attached driver (chain) from the device again.
3187 *
3188 * @returns VBox status code.
3189 * @param pDevIns The device instance.
3190 * @param pDrvIns The driver instance to detach.
3191 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
3192 */
3193 DECLR3CALLBACKMEMBER(int, pfnDriverDetach,(PPDMDEVINS pDevIns, PPDMDRVINS pDrvIns, uint32_t fFlags));
3194
3195 /**
3196 * Create a queue.
3197 *
3198 * @returns VBox status code.
3199 * @param pDevIns The device instance.
3200 * @param cbItem The size of a queue item.
3201 * @param cItems The number of items in the queue.
3202 * @param cMilliesInterval The number of milliseconds between polling the queue.
3203 * If 0 then the emulation thread will be notified whenever an item arrives.
3204 * @param pfnCallback The consumer function.
3205 * @param fRZEnabled Set if the queue should work in RC and R0.
3206 * @param pszName The queue base name. The instance number will be
3207 * appended automatically.
3208 * @param ppQueue Where to store the queue handle on success.
3209 * @thread The emulation thread.
3210 * @remarks The device critical section will NOT be entered before calling the
3211 * callback. No locks will be held, but for now it's safe to assume
3212 * that only one EMT will do queue callbacks at any one time.
3213 */
3214 DECLR3CALLBACKMEMBER(int, pfnQueueCreate,(PPDMDEVINS pDevIns, size_t cbItem, uint32_t cItems, uint32_t cMilliesInterval,
3215 PFNPDMQUEUEDEV pfnCallback, bool fRZEnabled, const char *pszName, PPDMQUEUE *ppQueue));
3216
3217 /**
3218 * Initializes a PDM critical section.
3219 *
3220 * The PDM critical sections are derived from the IPRT critical sections, but
3221 * works in RC and R0 as well.
3222 *
3223 * @returns VBox status code.
3224 * @param pDevIns The device instance.
3225 * @param pCritSect Pointer to the critical section.
3226 * @param SRC_POS Use RT_SRC_POS.
3227 * @param pszNameFmt Format string for naming the critical section.
3228 * For statistics and lock validation.
3229 * @param va Arguments for the format string.
3230 */
3231 DECLR3CALLBACKMEMBER(int, pfnCritSectInit,(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, RT_SRC_POS_DECL,
3232 const char *pszNameFmt, va_list va) RT_IPRT_FORMAT_ATTR(6, 0));
3233
3234 /**
3235 * Gets the NOP critical section.
3236 *
3237 * @returns The ring-3 address of the NOP critical section.
3238 * @param pDevIns The device instance.
3239 */
3240 DECLR3CALLBACKMEMBER(PPDMCRITSECT, pfnCritSectGetNop,(PPDMDEVINS pDevIns));
3241
3242 /**
3243 * Gets the NOP critical section.
3244 *
3245 * @returns The ring-0 address of the NOP critical section.
3246 * @param pDevIns The device instance.
3247 */
3248 DECLR3CALLBACKMEMBER(R0PTRTYPE(PPDMCRITSECT), pfnCritSectGetNopR0,(PPDMDEVINS pDevIns));
3249
3250 /**
3251 * Gets the NOP critical section.
3252 *
3253 * @returns The raw-mode context address of the NOP critical section.
3254 * @param pDevIns The device instance.
3255 */
3256 DECLR3CALLBACKMEMBER(RCPTRTYPE(PPDMCRITSECT), pfnCritSectGetNopRC,(PPDMDEVINS pDevIns));
3257
3258 /**
3259 * Changes the device level critical section from the automatically created
3260 * default to one desired by the device constructor.
3261 *
3262 * @returns VBox status code.
3263 * @param pDevIns The device instance.
3264 * @param pCritSect The critical section to use. NULL is not
3265 * valid, instead use the NOP critical
3266 * section.
3267 */
3268 DECLR3CALLBACKMEMBER(int, pfnSetDeviceCritSect,(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect));
3269
3270 /**
3271 * Creates a PDM thread.
3272 *
3273 * This differs from the RTThreadCreate() API in that PDM takes care of suspending,
3274 * resuming, and destroying the thread as the VM state changes.
3275 *
3276 * @returns VBox status code.
3277 * @param pDevIns The device instance.
3278 * @param ppThread Where to store the thread 'handle'.
3279 * @param pvUser The user argument to the thread function.
3280 * @param pfnThread The thread function.
3281 * @param pfnWakeup The wakup callback. This is called on the EMT
3282 * thread when a state change is pending.
3283 * @param cbStack See RTThreadCreate.
3284 * @param enmType See RTThreadCreate.
3285 * @param pszName See RTThreadCreate.
3286 * @remarks The device critical section will NOT be entered prior to invoking
3287 * the function pointers.
3288 */
3289 DECLR3CALLBACKMEMBER(int, pfnThreadCreate,(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
3290 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName));
3291
3292 /**
3293 * Set up asynchronous handling of a suspend, reset or power off notification.
3294 *
3295 * This shall only be called when getting the notification. It must be called
3296 * for each one.
3297 *
3298 * @returns VBox status code.
3299 * @param pDevIns The device instance.
3300 * @param pfnAsyncNotify The callback.
3301 * @thread EMT(0)
3302 * @remarks The caller will enter the device critical section prior to invoking
3303 * the callback.
3304 */
3305 DECLR3CALLBACKMEMBER(int, pfnSetAsyncNotification, (PPDMDEVINS pDevIns, PFNPDMDEVASYNCNOTIFY pfnAsyncNotify));
3306
3307 /**
3308 * Notify EMT(0) that the device has completed the asynchronous notification
3309 * handling.
3310 *
3311 * This can be called at any time, spurious calls will simply be ignored.
3312 *
3313 * @param pDevIns The device instance.
3314 * @thread Any
3315 */
3316 DECLR3CALLBACKMEMBER(void, pfnAsyncNotificationCompleted, (PPDMDEVINS pDevIns));
3317
3318 /**
3319 * Register the RTC device.
3320 *
3321 * @returns VBox status code.
3322 * @param pDevIns The device instance.
3323 * @param pRtcReg Pointer to a RTC registration structure.
3324 * @param ppRtcHlp Where to store the pointer to the helper
3325 * functions.
3326 */
3327 DECLR3CALLBACKMEMBER(int, pfnRTCRegister,(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp));
3328
3329 /**
3330 * Register the PCI Bus.
3331 *
3332 * @returns VBox status code.
3333 * @param pDevIns The device instance.
3334 * @param pPciBusReg Pointer to PCI bus registration structure.
3335 * @param ppPciHlpR3 Where to store the pointer to the PCI Bus
3336 * helpers.
3337 */
3338 DECLR3CALLBACKMEMBER(int, pfnPCIBusRegister,(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3));
3339
3340 /**
3341 * Register the PIC device.
3342 *
3343 * @returns VBox status code.
3344 * @param pDevIns The device instance.
3345 * @param pPicReg Pointer to a PIC registration structure.
3346 * @param ppPicHlpR3 Where to store the pointer to the PIC HC
3347 * helpers.
3348 */
3349 DECLR3CALLBACKMEMBER(int, pfnPICRegister,(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3));
3350
3351 /**
3352 * Register the APIC device.
3353 *
3354 * @returns VBox status code.
3355 * @param pDevIns The device instance.
3356 * @param pApicReg Pointer to a APIC registration structure.
3357 * @param ppApicHlpR3 Where to store the pointer to the APIC helpers.
3358 */
3359 DECLR3CALLBACKMEMBER(int, pfnAPICRegister,(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3));
3360
3361 /**
3362 * Register the I/O APIC device.
3363 *
3364 * @returns VBox status code.
3365 * @param pDevIns The device instance.
3366 * @param pIoApicReg Pointer to a I/O APIC registration structure.
3367 * @param ppIoApicHlpR3 Where to store the pointer to the IOAPIC
3368 * helpers.
3369 */
3370 DECLR3CALLBACKMEMBER(int, pfnIOAPICRegister,(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3));
3371
3372 /**
3373 * Register the HPET device.
3374 *
3375 * @returns VBox status code.
3376 * @param pDevIns The device instance.
3377 * @param pHpetReg Pointer to a HPET registration structure.
3378 * @param ppHpetHlpR3 Where to store the pointer to the HPET
3379 * helpers.
3380 */
3381 DECLR3CALLBACKMEMBER(int, pfnHPETRegister,(PPDMDEVINS pDevIns, PPDMHPETREG pHpetReg, PCPDMHPETHLPR3 *ppHpetHlpR3));
3382
3383 /**
3384 * Register a raw PCI device.
3385 *
3386 * @returns VBox status code.
3387 * @param pDevIns The device instance.
3388 * @param pPciRawReg Pointer to a raw PCI registration structure.
3389 * @param ppPciRawHlpR3 Where to store the pointer to the raw PCI
3390 * device helpers.
3391 */
3392 DECLR3CALLBACKMEMBER(int, pfnPciRawRegister,(PPDMDEVINS pDevIns, PPDMPCIRAWREG pPciRawReg, PCPDMPCIRAWHLPR3 *ppPciRawHlpR3));
3393
3394 /**
3395 * Register the DMA device.
3396 *
3397 * @returns VBox status code.
3398 * @param pDevIns The device instance.
3399 * @param pDmacReg Pointer to a DMAC registration structure.
3400 * @param ppDmacHlp Where to store the pointer to the DMA helpers.
3401 */
3402 DECLR3CALLBACKMEMBER(int, pfnDMACRegister,(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp));
3403
3404 /**
3405 * Register transfer function for DMA channel.
3406 *
3407 * @returns VBox status code.
3408 * @param pDevIns The device instance.
3409 * @param uChannel Channel number.
3410 * @param pfnTransferHandler Device specific transfer callback function.
3411 * @param pvUser User pointer to pass to the callback.
3412 * @thread EMT
3413 */
3414 DECLR3CALLBACKMEMBER(int, pfnDMARegister,(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser));
3415
3416 /**
3417 * Read memory.
3418 *
3419 * @returns VBox status code.
3420 * @param pDevIns The device instance.
3421 * @param uChannel Channel number.
3422 * @param pvBuffer Pointer to target buffer.
3423 * @param off DMA position.
3424 * @param cbBlock Block size.
3425 * @param pcbRead Where to store the number of bytes which was
3426 * read. optional.
3427 * @thread EMT
3428 */
3429 DECLR3CALLBACKMEMBER(int, pfnDMAReadMemory,(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead));
3430
3431 /**
3432 * Write memory.
3433 *
3434 * @returns VBox status code.
3435 * @param pDevIns The device instance.
3436 * @param uChannel Channel number.
3437 * @param pvBuffer Memory to write.
3438 * @param off DMA position.
3439 * @param cbBlock Block size.
3440 * @param pcbWritten Where to store the number of bytes which was
3441 * written. optional.
3442 * @thread EMT
3443 */
3444 DECLR3CALLBACKMEMBER(int, pfnDMAWriteMemory,(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten));
3445
3446 /**
3447 * Set the DREQ line.
3448 *
3449 * @returns VBox status code.
3450 * @param pDevIns Device instance.
3451 * @param uChannel Channel number.
3452 * @param uLevel Level of the line.
3453 * @thread EMT
3454 */
3455 DECLR3CALLBACKMEMBER(int, pfnDMASetDREQ,(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel));
3456
3457 /**
3458 * Get channel mode.
3459 *
3460 * @returns Channel mode. See specs.
3461 * @param pDevIns The device instance.
3462 * @param uChannel Channel number.
3463 * @thread EMT
3464 */
3465 DECLR3CALLBACKMEMBER(uint8_t, pfnDMAGetChannelMode,(PPDMDEVINS pDevIns, unsigned uChannel));
3466
3467 /**
3468 * Schedule DMA execution.
3469 *
3470 * @param pDevIns The device instance.
3471 * @thread Any thread.
3472 */
3473 DECLR3CALLBACKMEMBER(void, pfnDMASchedule,(PPDMDEVINS pDevIns));
3474
3475 /**
3476 * Write CMOS value and update the checksum(s).
3477 *
3478 * @returns VBox status code.
3479 * @param pDevIns The device instance.
3480 * @param iReg The CMOS register index.
3481 * @param u8Value The CMOS register value.
3482 * @thread EMT
3483 */
3484 DECLR3CALLBACKMEMBER(int, pfnCMOSWrite,(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value));
3485
3486 /**
3487 * Read CMOS value.
3488 *
3489 * @returns VBox status code.
3490 * @param pDevIns The device instance.
3491 * @param iReg The CMOS register index.
3492 * @param pu8Value Where to store the CMOS register value.
3493 * @thread EMT
3494 */
3495 DECLR3CALLBACKMEMBER(int, pfnCMOSRead,(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value));
3496
3497 /**
3498 * Assert that the current thread is the emulation thread.
3499 *
3500 * @returns True if correct.
3501 * @returns False if wrong.
3502 * @param pDevIns The device instance.
3503 * @param pszFile Filename of the assertion location.
3504 * @param iLine The linenumber of the assertion location.
3505 * @param pszFunction Function of the assertion location.
3506 */
3507 DECLR3CALLBACKMEMBER(bool, pfnAssertEMT,(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction));
3508
3509 /**
3510 * Assert that the current thread is NOT the emulation thread.
3511 *
3512 * @returns True if correct.
3513 * @returns False if wrong.
3514 * @param pDevIns The device instance.
3515 * @param pszFile Filename of the assertion location.
3516 * @param iLine The linenumber of the assertion location.
3517 * @param pszFunction Function of the assertion location.
3518 */
3519 DECLR3CALLBACKMEMBER(bool, pfnAssertOther,(PPDMDEVINS pDevIns, const char *pszFile, unsigned iLine, const char *pszFunction));
3520
3521 /**
3522 * Resolves the symbol for a raw-mode context interface.
3523 *
3524 * @returns VBox status code.
3525 * @param pDevIns The device instance.
3526 * @param pvInterface The interface structure.
3527 * @param cbInterface The size of the interface structure.
3528 * @param pszSymPrefix What to prefix the symbols in the list with
3529 * before resolving them. This must start with
3530 * 'dev' and contain the driver name.
3531 * @param pszSymList List of symbols corresponding to the interface.
3532 * There is generally a there is generally a define
3533 * holding this list associated with the interface
3534 * definition (INTERFACE_SYM_LIST). For more
3535 * details see PDMR3LdrGetInterfaceSymbols.
3536 * @thread EMT
3537 */
3538 DECLR3CALLBACKMEMBER(int, pfnLdrGetRCInterfaceSymbols,(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
3539 const char *pszSymPrefix, const char *pszSymList));
3540
3541 /**
3542 * Resolves the symbol for a ring-0 context interface.
3543 *
3544 * @returns VBox status code.
3545 * @param pDevIns The device instance.
3546 * @param pvInterface The interface structure.
3547 * @param cbInterface The size of the interface structure.
3548 * @param pszSymPrefix What to prefix the symbols in the list with
3549 * before resolving them. This must start with
3550 * 'dev' and contain the driver name.
3551 * @param pszSymList List of symbols corresponding to the interface.
3552 * There is generally a there is generally a define
3553 * holding this list associated with the interface
3554 * definition (INTERFACE_SYM_LIST). For more
3555 * details see PDMR3LdrGetInterfaceSymbols.
3556 * @thread EMT
3557 */
3558 DECLR3CALLBACKMEMBER(int, pfnLdrGetR0InterfaceSymbols,(PPDMDEVINS pDevIns, void *pvInterface, size_t cbInterface,
3559 const char *pszSymPrefix, const char *pszSymList));
3560
3561 /**
3562 * Call the ring-0 request handler routine of the device.
3563 *
3564 * For this to work, the device must be ring-0 enabled and export a request
3565 * handler function. The name of the function must be the device name in
3566 * the PDMDRVREG struct prefixed with 'drvR0' and suffixed with
3567 * 'ReqHandler'. The device name will be captialized. It shall take the
3568 * exact same arguments as this function and be declared using
3569 * PDMBOTHCBDECL. See FNPDMDEVREQHANDLERR0.
3570 *
3571 * Unlike PDMDrvHlpCallR0, this is current unsuitable for more than a call
3572 * or two as the handler address will be resolved on each invocation. This
3573 * is the reason for the EMT only restriction as well.
3574 *
3575 * @returns VBox status code.
3576 * @retval VERR_SYMBOL_NOT_FOUND if the device doesn't export the required
3577 * handler function.
3578 * @retval VERR_ACCESS_DENIED if the device isn't ring-0 capable.
3579 *
3580 * @param pDevIns The device instance.
3581 * @param uOperation The operation to perform.
3582 * @param u64Arg 64-bit integer argument.
3583 * @thread EMT
3584 */
3585 DECLR3CALLBACKMEMBER(int, pfnCallR0,(PPDMDEVINS pDevIns, uint32_t uOperation, uint64_t u64Arg));
3586
3587 /**
3588 * Gets the reason for the most recent VM suspend.
3589 *
3590 * @returns The suspend reason. VMSUSPENDREASON_INVALID is returned if no
3591 * suspend has been made or if the pDevIns is invalid.
3592 * @param pDevIns The device instance.
3593 */
3594 DECLR3CALLBACKMEMBER(VMSUSPENDREASON, pfnVMGetSuspendReason,(PPDMDEVINS pDevIns));
3595
3596 /**
3597 * Gets the reason for the most recent VM resume.
3598 *
3599 * @returns The resume reason. VMRESUMEREASON_INVALID is returned if no
3600 * resume has been made or if the pDevIns is invalid.
3601 * @param pDevIns The device instance.
3602 */
3603 DECLR3CALLBACKMEMBER(VMRESUMEREASON, pfnVMGetResumeReason,(PPDMDEVINS pDevIns));
3604
3605
3606 /** Space reserved for future members.
3607 * @{ */
3608 DECLR3CALLBACKMEMBER(void, pfnReserved1,(void));
3609 DECLR3CALLBACKMEMBER(void, pfnReserved2,(void));
3610 DECLR3CALLBACKMEMBER(void, pfnReserved3,(void));
3611 DECLR3CALLBACKMEMBER(void, pfnReserved4,(void));
3612 DECLR3CALLBACKMEMBER(void, pfnReserved5,(void));
3613 DECLR3CALLBACKMEMBER(void, pfnReserved6,(void));
3614 DECLR3CALLBACKMEMBER(void, pfnReserved7,(void));
3615 /*DECLR3CALLBACKMEMBER(void, pfnReserved8,(void));
3616 DECLR3CALLBACKMEMBER(void, pfnReserved9,(void));*/
3617 /*DECLR3CALLBACKMEMBER(void, pfnReserved10,(void));*/
3618 /** @} */
3619
3620
3621 /** API available to trusted devices only.
3622 *
3623 * These APIs are providing unrestricted access to the guest and the VM,
3624 * or they are interacting intimately with PDM.
3625 *
3626 * @{
3627 */
3628
3629 /**
3630 * Gets the user mode VM handle. Restricted API.
3631 *
3632 * @returns User mode VM Handle.
3633 * @param pDevIns The device instance.
3634 */
3635 DECLR3CALLBACKMEMBER(PUVM, pfnGetUVM,(PPDMDEVINS pDevIns));
3636
3637 /**
3638 * Gets the global VM handle. Restricted API.
3639 *
3640 * @returns VM Handle.
3641 * @param pDevIns The device instance.
3642 */
3643 DECLR3CALLBACKMEMBER(PVM, pfnGetVM,(PPDMDEVINS pDevIns));
3644
3645 /**
3646 * Gets the VMCPU handle. Restricted API.
3647 *
3648 * @returns VMCPU Handle.
3649 * @param pDevIns The device instance.
3650 */
3651 DECLR3CALLBACKMEMBER(PVMCPU, pfnGetVMCPU,(PPDMDEVINS pDevIns));
3652
3653 /**
3654 * The the VM CPU ID of the current thread (restricted API).
3655 *
3656 * @returns The VMCPUID of the calling thread, NIL_CPUID if not EMT.
3657 * @param pDevIns The device instance.
3658 */
3659 DECLR3CALLBACKMEMBER(VMCPUID, pfnGetCurrentCpuId,(PPDMDEVINS pDevIns));
3660
3661 /**
3662 * Registers the VMM device heap or notifies about mapping/unmapping.
3663 *
3664 * This interface serves three purposes:
3665 *
3666 * -# Register the VMM device heap during device construction
3667 * for the HM to use.
3668 * -# Notify PDM/HM that it's mapped into guest address
3669 * space (i.e. usable).
3670 * -# Notify PDM/HM that it is being unmapped from the guest
3671 * address space (i.e. not usable).
3672 *
3673 * @returns VBox status code.
3674 * @param pDevIns The device instance.
3675 * @param GCPhys The physical address if mapped, NIL_RTGCPHYS if
3676 * not mapped.
3677 * @param pvHeap Ring 3 heap pointer.
3678 * @param cbHeap Size of the heap.
3679 * @thread EMT.
3680 */
3681 DECLR3CALLBACKMEMBER(int, pfnRegisterVMMDevHeap,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbHeap));
3682
3683 /**
3684 * Registers the firmware (BIOS, EFI) device with PDM.
3685 *
3686 * The firmware provides a callback table and gets a special PDM helper table.
3687 * There can only be one firmware device for a VM.
3688 *
3689 * @returns VBox status code.
3690 * @param pDevIns The device instance.
3691 * @param pFwReg Firmware registration structure.
3692 * @param ppFwHlp Where to return the firmware helper structure.
3693 * @remarks Only valid during device construction.
3694 * @thread EMT(0)
3695 */
3696 DECLR3CALLBACKMEMBER(int, pfnFirmwareRegister,(PPDMDEVINS pDevIns, PCPDMFWREG pFwReg, PCPDMFWHLPR3 *ppFwHlp));
3697
3698 /**
3699 * Resets the VM.
3700 *
3701 * @returns The appropriate VBox status code to pass around on reset.
3702 * @param pDevIns The device instance.
3703 * @param fFlags PDMVMRESET_F_XXX flags.
3704 * @thread The emulation thread.
3705 */
3706 DECLR3CALLBACKMEMBER(int, pfnVMReset,(PPDMDEVINS pDevIns, uint32_t fFlags));
3707
3708 /**
3709 * Suspends the VM.
3710 *
3711 * @returns The appropriate VBox status code to pass around on suspend.
3712 * @param pDevIns The device instance.
3713 * @thread The emulation thread.
3714 */
3715 DECLR3CALLBACKMEMBER(int, pfnVMSuspend,(PPDMDEVINS pDevIns));
3716
3717 /**
3718 * Suspends, saves and powers off the VM.
3719 *
3720 * @returns The appropriate VBox status code to pass around.
3721 * @param pDevIns The device instance.
3722 * @thread An emulation thread.
3723 */
3724 DECLR3CALLBACKMEMBER(int, pfnVMSuspendSaveAndPowerOff,(PPDMDEVINS pDevIns));
3725
3726 /**
3727 * Power off the VM.
3728 *
3729 * @returns The appropriate VBox status code to pass around on power off.
3730 * @param pDevIns The device instance.
3731 * @thread The emulation thread.
3732 */
3733 DECLR3CALLBACKMEMBER(int, pfnVMPowerOff,(PPDMDEVINS pDevIns));
3734
3735 /**
3736 * Checks if the Gate A20 is enabled or not.
3737 *
3738 * @returns true if A20 is enabled.
3739 * @returns false if A20 is disabled.
3740 * @param pDevIns The device instance.
3741 * @thread The emulation thread.
3742 */
3743 DECLR3CALLBACKMEMBER(bool, pfnA20IsEnabled,(PPDMDEVINS pDevIns));
3744
3745 /**
3746 * Enables or disables the Gate A20.
3747 *
3748 * @param pDevIns The device instance.
3749 * @param fEnable Set this flag to enable the Gate A20; clear it
3750 * to disable.
3751 * @thread The emulation thread.
3752 */
3753 DECLR3CALLBACKMEMBER(void, pfnA20Set,(PPDMDEVINS pDevIns, bool fEnable));
3754
3755 /**
3756 * Get the specified CPUID leaf for the virtual CPU associated with the calling
3757 * thread.
3758 *
3759 * @param pDevIns The device instance.
3760 * @param iLeaf The CPUID leaf to get.
3761 * @param pEax Where to store the EAX value.
3762 * @param pEbx Where to store the EBX value.
3763 * @param pEcx Where to store the ECX value.
3764 * @param pEdx Where to store the EDX value.
3765 * @thread EMT.
3766 */
3767 DECLR3CALLBACKMEMBER(void, pfnGetCpuId,(PPDMDEVINS pDevIns, uint32_t iLeaf, uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx));
3768
3769 /**
3770 * Get the current virtual clock time in a VM. The clock frequency must be
3771 * queried separately.
3772 *
3773 * @returns Current clock time.
3774 * @param pDevIns The device instance.
3775 */
3776 DECLR3CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGet,(PPDMDEVINS pDevIns));
3777
3778 /**
3779 * Get the frequency of the virtual clock.
3780 *
3781 * @returns The clock frequency (not variable at run-time).
3782 * @param pDevIns The device instance.
3783 */
3784 DECLR3CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetFreq,(PPDMDEVINS pDevIns));
3785
3786 /**
3787 * Get the current virtual clock time in a VM, in nanoseconds.
3788 *
3789 * @returns Current clock time (in ns).
3790 * @param pDevIns The device instance.
3791 */
3792 DECLR3CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetNano,(PPDMDEVINS pDevIns));
3793
3794 /**
3795 * Gets the support driver session.
3796 *
3797 * This is intended for working with the semaphore API.
3798 *
3799 * @returns Support driver session handle.
3800 * @param pDevIns The device instance.
3801 */
3802 DECLR3CALLBACKMEMBER(PSUPDRVSESSION, pfnGetSupDrvSession,(PPDMDEVINS pDevIns));
3803
3804 /** @} */
3805
3806 /** Just a safety precaution. (PDM_DEVHLPR3_VERSION) */
3807 uint32_t u32TheEnd;
3808} PDMDEVHLPR3;
3809#endif /* !IN_RING3 */
3810/** Pointer to the R3 PDM Device API. */
3811typedef R3PTRTYPE(struct PDMDEVHLPR3 *) PPDMDEVHLPR3;
3812/** Pointer to the R3 PDM Device API, const variant. */
3813typedef R3PTRTYPE(const struct PDMDEVHLPR3 *) PCPDMDEVHLPR3;
3814
3815/** Current PDMDEVHLPR3 version number. */
3816#define PDM_DEVHLPR3_VERSION PDM_VERSION_MAKE(0xffe7, 16, 0)
3817
3818
3819/**
3820 * PDM Device API - RC Variant.
3821 */
3822typedef struct PDMDEVHLPRC
3823{
3824 /** Structure version. PDM_DEVHLPRC_VERSION defines the current version. */
3825 uint32_t u32Version;
3826
3827 /**
3828 * Bus master physical memory read.
3829 *
3830 * @returns VINF_SUCCESS or VERR_PGM_PCI_PHYS_READ_BM_DISABLED, later maybe
3831 * VERR_EM_MEMORY. The informational status shall NOT be propagated!
3832 * @param pDevIns The device instance.
3833 * @param GCPhys Physical address start reading from.
3834 * @param pvBuf Where to put the read bits.
3835 * @param cbRead How many bytes to read.
3836 * @thread Any thread, but the call may involve the emulation thread.
3837 */
3838 DECLRCCALLBACKMEMBER(int, pfnPCIPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
3839
3840 /**
3841 * Bus master physical memory write.
3842 *
3843 * @returns VINF_SUCCESS or VERR_PGM_PCI_PHYS_WRITE_BM_DISABLED, later maybe
3844 * VERR_EM_MEMORY. The informational status shall NOT be propagated!
3845 * @param pDevIns The device instance.
3846 * @param GCPhys Physical address to write to.
3847 * @param pvBuf What to write.
3848 * @param cbWrite How many bytes to write.
3849 * @thread Any thread, but the call may involve the emulation thread.
3850 */
3851 DECLRCCALLBACKMEMBER(int, pfnPCIPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
3852
3853 /**
3854 * Set the IRQ for a PCI device.
3855 *
3856 * @param pDevIns Device instance.
3857 * @param iIrq IRQ number to set.
3858 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
3859 * @thread Any thread, but will involve the emulation thread.
3860 */
3861 DECLRCCALLBACKMEMBER(void, pfnPCISetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3862
3863 /**
3864 * Set ISA IRQ for a device.
3865 *
3866 * @param pDevIns Device instance.
3867 * @param iIrq IRQ number to set.
3868 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
3869 * @thread Any thread, but will involve the emulation thread.
3870 */
3871 DECLRCCALLBACKMEMBER(void, pfnISASetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
3872
3873 /**
3874 * Read physical memory.
3875 *
3876 * @returns VINF_SUCCESS (for now).
3877 * @param pDevIns Device instance.
3878 * @param GCPhys Physical address start reading from.
3879 * @param pvBuf Where to put the read bits.
3880 * @param cbRead How many bytes to read.
3881 */
3882 DECLRCCALLBACKMEMBER(int, pfnPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
3883
3884 /**
3885 * Write to physical memory.
3886 *
3887 * @returns VINF_SUCCESS for now, and later maybe VERR_EM_MEMORY.
3888 * @param pDevIns Device instance.
3889 * @param GCPhys Physical address to write to.
3890 * @param pvBuf What to write.
3891 * @param cbWrite How many bytes to write.
3892 */
3893 DECLRCCALLBACKMEMBER(int, pfnPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
3894
3895 /**
3896 * Checks if the Gate A20 is enabled or not.
3897 *
3898 * @returns true if A20 is enabled.
3899 * @returns false if A20 is disabled.
3900 * @param pDevIns Device instance.
3901 * @thread The emulation thread.
3902 */
3903 DECLRCCALLBACKMEMBER(bool, pfnA20IsEnabled,(PPDMDEVINS pDevIns));
3904
3905 /**
3906 * Gets the VM state.
3907 *
3908 * @returns VM state.
3909 * @param pDevIns The device instance.
3910 * @thread Any thread (just keep in mind that it's volatile info).
3911 */
3912 DECLRCCALLBACKMEMBER(VMSTATE, pfnVMState, (PPDMDEVINS pDevIns));
3913
3914 /**
3915 * Set the VM error message
3916 *
3917 * @returns rc.
3918 * @param pDevIns Driver instance.
3919 * @param rc VBox status code.
3920 * @param SRC_POS Use RT_SRC_POS.
3921 * @param pszFormat Error message format string.
3922 * @param ... Error message arguments.
3923 */
3924 DECLRCCALLBACKMEMBER(int, pfnVMSetError,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
3925 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(6, 7));
3926
3927 /**
3928 * Set the VM error message
3929 *
3930 * @returns rc.
3931 * @param pDevIns Driver instance.
3932 * @param rc VBox status code.
3933 * @param SRC_POS Use RT_SRC_POS.
3934 * @param pszFormat Error message format string.
3935 * @param va Error message arguments.
3936 */
3937 DECLRCCALLBACKMEMBER(int, pfnVMSetErrorV,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
3938 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(6, 0));
3939
3940 /**
3941 * Set the VM runtime error message
3942 *
3943 * @returns VBox status code.
3944 * @param pDevIns Device instance.
3945 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
3946 * @param pszErrorId Error ID string.
3947 * @param pszFormat Error message format string.
3948 * @param ... Error message arguments.
3949 */
3950 DECLRCCALLBACKMEMBER(int, pfnVMSetRuntimeError,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
3951 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(4, 5));
3952
3953 /**
3954 * Set the VM runtime error message
3955 *
3956 * @returns VBox status code.
3957 * @param pDevIns Device instance.
3958 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
3959 * @param pszErrorId Error ID string.
3960 * @param pszFormat Error message format string.
3961 * @param va Error message arguments.
3962 */
3963 DECLRCCALLBACKMEMBER(int, pfnVMSetRuntimeErrorV,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
3964 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(4, 0));
3965
3966 /**
3967 * Set parameters for pending MMIO patch operation
3968 *
3969 * @returns VBox status code.
3970 * @param pDevIns Device instance.
3971 * @param GCPhys MMIO physical address
3972 * @param pCachedData GC pointer to cached data
3973 */
3974 DECLRCCALLBACKMEMBER(int, pfnPATMSetMMIOPatchInfo,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTGCPTR pCachedData));
3975
3976 /**
3977 * Gets the VM handle. Restricted API.
3978 *
3979 * @returns VM Handle.
3980 * @param pDevIns Device instance.
3981 */
3982 DECLRCCALLBACKMEMBER(PVM, pfnGetVM,(PPDMDEVINS pDevIns));
3983
3984 /**
3985 * Gets the VMCPU handle. Restricted API.
3986 *
3987 * @returns VMCPU Handle.
3988 * @param pDevIns The device instance.
3989 */
3990 DECLRCCALLBACKMEMBER(PVMCPU, pfnGetVMCPU,(PPDMDEVINS pDevIns));
3991
3992 /**
3993 * The the VM CPU ID of the current thread (restricted API).
3994 *
3995 * @returns The VMCPUID of the calling thread, NIL_CPUID if not EMT.
3996 * @param pDevIns The device instance.
3997 */
3998 DECLRCCALLBACKMEMBER(VMCPUID, pfnGetCurrentCpuId,(PPDMDEVINS pDevIns));
3999
4000 /**
4001 * Get the current virtual clock time in a VM. The clock frequency must be
4002 * queried separately.
4003 *
4004 * @returns Current clock time.
4005 * @param pDevIns The device instance.
4006 */
4007 DECLRCCALLBACKMEMBER(uint64_t, pfnTMTimeVirtGet,(PPDMDEVINS pDevIns));
4008
4009 /**
4010 * Get the frequency of the virtual clock.
4011 *
4012 * @returns The clock frequency (not variable at run-time).
4013 * @param pDevIns The device instance.
4014 */
4015 DECLRCCALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetFreq,(PPDMDEVINS pDevIns));
4016
4017 /**
4018 * Get the current virtual clock time in a VM, in nanoseconds.
4019 *
4020 * @returns Current clock time (in ns).
4021 * @param pDevIns The device instance.
4022 */
4023 DECLRCCALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetNano,(PPDMDEVINS pDevIns));
4024
4025 /**
4026 * Gets the trace buffer handle.
4027 *
4028 * This is used by the macros found in VBox/vmm/dbgftrace.h and is not
4029 * really inteded for direct usage, thus no inline wrapper function.
4030 *
4031 * @returns Trace buffer handle or NIL_RTTRACEBUF.
4032 * @param pDevIns The device instance.
4033 */
4034 DECLRCCALLBACKMEMBER(RTTRACEBUF, pfnDBGFTraceBuf,(PPDMDEVINS pDevIns));
4035
4036 /** Just a safety precaution. */
4037 uint32_t u32TheEnd;
4038} PDMDEVHLPRC;
4039/** Pointer PDM Device RC API. */
4040typedef RCPTRTYPE(struct PDMDEVHLPRC *) PPDMDEVHLPRC;
4041/** Pointer PDM Device RC API. */
4042typedef RCPTRTYPE(const struct PDMDEVHLPRC *) PCPDMDEVHLPRC;
4043
4044/** Current PDMDEVHLP version number. */
4045#define PDM_DEVHLPRC_VERSION PDM_VERSION_MAKE(0xffe6, 4, 1)
4046
4047
4048/**
4049 * PDM Device API - R0 Variant.
4050 */
4051typedef struct PDMDEVHLPR0
4052{
4053 /** Structure version. PDM_DEVHLPR0_VERSION defines the current version. */
4054 uint32_t u32Version;
4055
4056 /**
4057 * Bus master physical memory read.
4058 *
4059 * @returns VINF_SUCCESS or VERR_PDM_NOT_PCI_BUS_MASTER, later maybe
4060 * VERR_EM_MEMORY.
4061 * @param pDevIns The device instance.
4062 * @param GCPhys Physical address start reading from.
4063 * @param pvBuf Where to put the read bits.
4064 * @param cbRead How many bytes to read.
4065 * @thread Any thread, but the call may involve the emulation thread.
4066 */
4067 DECLR0CALLBACKMEMBER(int, pfnPCIPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
4068
4069 /**
4070 * Bus master physical memory write.
4071 *
4072 * @returns VINF_SUCCESS or VERR_PDM_NOT_PCI_BUS_MASTER, later maybe
4073 * VERR_EM_MEMORY.
4074 * @param pDevIns The device instance.
4075 * @param GCPhys Physical address to write to.
4076 * @param pvBuf What to write.
4077 * @param cbWrite How many bytes to write.
4078 * @thread Any thread, but the call may involve the emulation thread.
4079 */
4080 DECLR0CALLBACKMEMBER(int, pfnPCIPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
4081
4082 /**
4083 * Set the IRQ for a PCI device.
4084 *
4085 * @param pDevIns Device instance.
4086 * @param iIrq IRQ number to set.
4087 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
4088 * @thread Any thread, but will involve the emulation thread.
4089 */
4090 DECLR0CALLBACKMEMBER(void, pfnPCISetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
4091
4092 /**
4093 * Set ISA IRQ for a device.
4094 *
4095 * @param pDevIns Device instance.
4096 * @param iIrq IRQ number to set.
4097 * @param iLevel IRQ level. See the PDM_IRQ_LEVEL_* \#defines.
4098 * @thread Any thread, but will involve the emulation thread.
4099 */
4100 DECLR0CALLBACKMEMBER(void, pfnISASetIrq,(PPDMDEVINS pDevIns, int iIrq, int iLevel));
4101
4102 /**
4103 * Read physical memory.
4104 *
4105 * @returns VINF_SUCCESS (for now).
4106 * @param pDevIns Device instance.
4107 * @param GCPhys Physical address start reading from.
4108 * @param pvBuf Where to put the read bits.
4109 * @param cbRead How many bytes to read.
4110 */
4111 DECLR0CALLBACKMEMBER(int, pfnPhysRead,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead));
4112
4113 /**
4114 * Write to physical memory.
4115 *
4116 * @returns VINF_SUCCESS for now, and later maybe VERR_EM_MEMORY.
4117 * @param pDevIns Device instance.
4118 * @param GCPhys Physical address to write to.
4119 * @param pvBuf What to write.
4120 * @param cbWrite How many bytes to write.
4121 */
4122 DECLR0CALLBACKMEMBER(int, pfnPhysWrite,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite));
4123
4124 /**
4125 * Checks if the Gate A20 is enabled or not.
4126 *
4127 * @returns true if A20 is enabled.
4128 * @returns false if A20 is disabled.
4129 * @param pDevIns Device instance.
4130 * @thread The emulation thread.
4131 */
4132 DECLR0CALLBACKMEMBER(bool, pfnA20IsEnabled,(PPDMDEVINS pDevIns));
4133
4134 /**
4135 * Gets the VM state.
4136 *
4137 * @returns VM state.
4138 * @param pDevIns The device instance.
4139 * @thread Any thread (just keep in mind that it's volatile info).
4140 */
4141 DECLR0CALLBACKMEMBER(VMSTATE, pfnVMState, (PPDMDEVINS pDevIns));
4142
4143 /**
4144 * Set the VM error message
4145 *
4146 * @returns rc.
4147 * @param pDevIns Driver instance.
4148 * @param rc VBox status code.
4149 * @param SRC_POS Use RT_SRC_POS.
4150 * @param pszFormat Error message format string.
4151 * @param ... Error message arguments.
4152 */
4153 DECLR0CALLBACKMEMBER(int, pfnVMSetError,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
4154 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(6, 7));
4155
4156 /**
4157 * Set the VM error message
4158 *
4159 * @returns rc.
4160 * @param pDevIns Driver instance.
4161 * @param rc VBox status code.
4162 * @param SRC_POS Use RT_SRC_POS.
4163 * @param pszFormat Error message format string.
4164 * @param va Error message arguments.
4165 */
4166 DECLR0CALLBACKMEMBER(int, pfnVMSetErrorV,(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL,
4167 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(6, 0));
4168
4169 /**
4170 * Set the VM runtime error message
4171 *
4172 * @returns VBox status code.
4173 * @param pDevIns Device instance.
4174 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
4175 * @param pszErrorId Error ID string.
4176 * @param pszFormat Error message format string.
4177 * @param ... Error message arguments.
4178 */
4179 DECLR0CALLBACKMEMBER(int, pfnVMSetRuntimeError,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
4180 const char *pszFormat, ...) RT_IPRT_FORMAT_ATTR(4, 5));
4181
4182 /**
4183 * Set the VM runtime error message
4184 *
4185 * @returns VBox status code.
4186 * @param pDevIns Device instance.
4187 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
4188 * @param pszErrorId Error ID string.
4189 * @param pszFormat Error message format string.
4190 * @param va Error message arguments.
4191 */
4192 DECLR0CALLBACKMEMBER(int, pfnVMSetRuntimeErrorV,(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
4193 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(4, 0));
4194
4195 /**
4196 * Set parameters for pending MMIO patch operation
4197 *
4198 * @returns rc.
4199 * @param pDevIns Device instance.
4200 * @param GCPhys MMIO physical address
4201 * @param pCachedData GC pointer to cached data
4202 */
4203 DECLR0CALLBACKMEMBER(int, pfnPATMSetMMIOPatchInfo,(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTGCPTR pCachedData));
4204
4205 /**
4206 * Gets the VM handle. Restricted API.
4207 *
4208 * @returns VM Handle.
4209 * @param pDevIns Device instance.
4210 */
4211 DECLR0CALLBACKMEMBER(PVM, pfnGetVM,(PPDMDEVINS pDevIns));
4212
4213 /**
4214 * Checks if our current CPU state allows for IO block emulation fallback to the recompiler
4215 *
4216 * @returns true = yes, false = no
4217 * @param pDevIns Device instance.
4218 */
4219 DECLR0CALLBACKMEMBER(bool, pfnCanEmulateIoBlock,(PPDMDEVINS pDevIns));
4220
4221 /**
4222 * Gets the VMCPU handle. Restricted API.
4223 *
4224 * @returns VMCPU Handle.
4225 * @param pDevIns The device instance.
4226 */
4227 DECLR0CALLBACKMEMBER(PVMCPU, pfnGetVMCPU,(PPDMDEVINS pDevIns));
4228
4229 /**
4230 * The the VM CPU ID of the current thread (restricted API).
4231 *
4232 * @returns The VMCPUID of the calling thread, NIL_CPUID if not EMT.
4233 * @param pDevIns The device instance.
4234 */
4235 DECLR0CALLBACKMEMBER(VMCPUID, pfnGetCurrentCpuId,(PPDMDEVINS pDevIns));
4236
4237 /**
4238 * Get the current virtual clock time in a VM. The clock frequency must be
4239 * queried separately.
4240 *
4241 * @returns Current clock time.
4242 * @param pDevIns The device instance.
4243 */
4244 DECLR0CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGet,(PPDMDEVINS pDevIns));
4245
4246 /**
4247 * Get the frequency of the virtual clock.
4248 *
4249 * @returns The clock frequency (not variable at run-time).
4250 * @param pDevIns The device instance.
4251 */
4252 DECLR0CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetFreq,(PPDMDEVINS pDevIns));
4253
4254 /**
4255 * Get the current virtual clock time in a VM, in nanoseconds.
4256 *
4257 * @returns Current clock time (in ns).
4258 * @param pDevIns The device instance.
4259 */
4260 DECLR0CALLBACKMEMBER(uint64_t, pfnTMTimeVirtGetNano,(PPDMDEVINS pDevIns));
4261
4262 /**
4263 * Gets the trace buffer handle.
4264 *
4265 * This is used by the macros found in VBox/vmm/dbgftrace.h and is not
4266 * really inteded for direct usage, thus no inline wrapper function.
4267 *
4268 * @returns Trace buffer handle or NIL_RTTRACEBUF.
4269 * @param pDevIns The device instance.
4270 */
4271 DECLR0CALLBACKMEMBER(RTTRACEBUF, pfnDBGFTraceBuf,(PPDMDEVINS pDevIns));
4272
4273 /** Just a safety precaution. */
4274 uint32_t u32TheEnd;
4275} PDMDEVHLPR0;
4276/** Pointer PDM Device R0 API. */
4277typedef R0PTRTYPE(struct PDMDEVHLPR0 *) PPDMDEVHLPR0;
4278/** Pointer PDM Device GC API. */
4279typedef R0PTRTYPE(const struct PDMDEVHLPR0 *) PCPDMDEVHLPR0;
4280
4281/** Current PDMDEVHLP version number. */
4282#define PDM_DEVHLPR0_VERSION PDM_VERSION_MAKE(0xffe5, 4, 1)
4283
4284
4285
4286/**
4287 * PDM Device Instance.
4288 */
4289typedef struct PDMDEVINS
4290{
4291 /** Structure version. PDM_DEVINS_VERSION defines the current version. */
4292 uint32_t u32Version;
4293 /** Device instance number. */
4294 uint32_t iInstance;
4295
4296 /** Pointer the GC PDM Device API. */
4297 PCPDMDEVHLPRC pHlpRC;
4298 /** Pointer to device instance data. */
4299 RTRCPTR pvInstanceDataRC;
4300 /** The critical section for the device, see pCritSectXR3. */
4301 RCPTRTYPE(PPDMCRITSECT) pCritSectRoRC;
4302 /** Alignment padding. */
4303 RTRCPTR pAlignmentRC;
4304
4305 /** Pointer the R0 PDM Device API. */
4306 PCPDMDEVHLPR0 pHlpR0;
4307 /** Pointer to device instance data (R0). */
4308 RTR0PTR pvInstanceDataR0;
4309 /** The critical section for the device, see pCritSectXR3. */
4310 R0PTRTYPE(PPDMCRITSECT) pCritSectRoR0;
4311
4312 /** Pointer the HC PDM Device API. */
4313 PCPDMDEVHLPR3 pHlpR3;
4314 /** Pointer to device instance data. */
4315 RTR3PTR pvInstanceDataR3;
4316 /** The critical section for the device.
4317 *
4318 * TM and IOM will enter this critical section before calling into the device
4319 * code. PDM will when doing power on, power off, reset, suspend and resume
4320 * notifications. SSM will currently not, but this will be changed later on.
4321 *
4322 * The device gets a critical section automatically assigned to it before
4323 * the constructor is called. If the constructor wishes to use a different
4324 * critical section, it calls PDMDevHlpSetDeviceCritSect() to change it
4325 * very early on.
4326 */
4327 R3PTRTYPE(PPDMCRITSECT) pCritSectRoR3;
4328
4329 /** Pointer to device registration structure. */
4330 R3PTRTYPE(PCPDMDEVREG) pReg;
4331 /** Configuration handle. */
4332 R3PTRTYPE(PCFGMNODE) pCfg;
4333
4334 /** The base interface of the device.
4335 *
4336 * The device constructor initializes this if it has any
4337 * device level interfaces to export. To obtain this interface
4338 * call PDMR3QueryDevice(). */
4339 PDMIBASE IBase;
4340
4341 /** Tracing indicator. */
4342 uint32_t fTracing;
4343 /** The tracing ID of this device. */
4344 uint32_t idTracing;
4345#if HC_ARCH_BITS == 32
4346 /** Align the internal data more naturally. */
4347 uint32_t au32Padding[HC_ARCH_BITS == 32 ? 13 : 0];
4348#endif
4349
4350 /** Internal data. */
4351 union
4352 {
4353#ifdef PDMDEVINSINT_DECLARED
4354 PDMDEVINSINT s;
4355#endif
4356 uint8_t padding[HC_ARCH_BITS == 32 ? 72 : 112 + 0x28];
4357 } Internal;
4358
4359 /** Device instance data. The size of this area is defined
4360 * in the PDMDEVREG::cbInstanceData field. */
4361 char achInstanceData[8];
4362} PDMDEVINS;
4363
4364/** Current PDMDEVINS version number. */
4365#define PDM_DEVINS_VERSION PDM_VERSION_MAKE(0xffe4, 3, 0)
4366
4367/** Converts a pointer to the PDMDEVINS::IBase to a pointer to PDMDEVINS. */
4368#define PDMIBASE_2_PDMDEV(pInterface) ( (PPDMDEVINS)((char *)(pInterface) - RT_OFFSETOF(PDMDEVINS, IBase)) )
4369
4370/**
4371 * Checks the structure versions of the device instance and device helpers,
4372 * returning if they are incompatible.
4373 *
4374 * This is for use in the constructor.
4375 *
4376 * @param pDevIns The device instance pointer.
4377 */
4378#define PDMDEV_CHECK_VERSIONS_RETURN(pDevIns) \
4379 do \
4380 { \
4381 PPDMDEVINS pDevInsTypeCheck = (pDevIns); NOREF(pDevInsTypeCheck); \
4382 AssertLogRelMsgReturn(PDM_VERSION_ARE_COMPATIBLE((pDevIns)->u32Version, PDM_DEVINS_VERSION), \
4383 ("DevIns=%#x mine=%#x\n", (pDevIns)->u32Version, PDM_DEVINS_VERSION), \
4384 VERR_PDM_DEVINS_VERSION_MISMATCH); \
4385 AssertLogRelMsgReturn(PDM_VERSION_ARE_COMPATIBLE((pDevIns)->pHlpR3->u32Version, PDM_DEVHLPR3_VERSION), \
4386 ("DevHlp=%#x mine=%#x\n", (pDevIns)->pHlpR3->u32Version, PDM_DEVHLPR3_VERSION), \
4387 VERR_PDM_DEVHLPR3_VERSION_MISMATCH); \
4388 } while (0)
4389
4390/**
4391 * Quietly checks the structure versions of the device instance and device
4392 * helpers, returning if they are incompatible.
4393 *
4394 * This is for use in the destructor.
4395 *
4396 * @param pDevIns The device instance pointer.
4397 */
4398#define PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns) \
4399 do \
4400 { \
4401 PPDMDEVINS pDevInsTypeCheck = (pDevIns); NOREF(pDevInsTypeCheck); \
4402 if (RT_LIKELY(PDM_VERSION_ARE_COMPATIBLE((pDevIns)->u32Version, PDM_DEVINS_VERSION) )) \
4403 { /* likely */ } else return VERR_PDM_DEVINS_VERSION_MISMATCH; \
4404 if (RT_LIKELY(PDM_VERSION_ARE_COMPATIBLE((pDevIns)->pHlpR3->u32Version, PDM_DEVHLPR3_VERSION) )) \
4405 { /* likely */ } else return VERR_PDM_DEVHLPR3_VERSION_MISMATCH; \
4406 } while (0)
4407
4408/**
4409 * Wrapper around CFGMR3ValidateConfig for the root config for use in the
4410 * constructor - returns on failure.
4411 *
4412 * This should be invoked after having initialized the instance data
4413 * sufficiently for the correct operation of the destructor. The destructor is
4414 * always called!
4415 *
4416 * @param pDevIns Pointer to the PDM device instance.
4417 * @param pszValidValues Patterns describing the valid value names. See
4418 * RTStrSimplePatternMultiMatch for details on the
4419 * pattern syntax.
4420 * @param pszValidNodes Patterns describing the valid node (key) names.
4421 * Pass empty string if no valid nodes.
4422 */
4423#define PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns, pszValidValues, pszValidNodes) \
4424 do \
4425 { \
4426 int rcValCfg = CFGMR3ValidateConfig((pDevIns)->pCfg, "/", pszValidValues, pszValidNodes, \
4427 (pDevIns)->pReg->szName, (pDevIns)->iInstance); \
4428 if (RT_SUCCESS(rcValCfg)) \
4429 { /* likely */ } else return rcValCfg; \
4430 } while (0)
4431
4432/** @def PDMDEV_ASSERT_EMT
4433 * Assert that the current thread is the emulation thread.
4434 */
4435#ifdef VBOX_STRICT
4436# define PDMDEV_ASSERT_EMT(pDevIns) pDevIns->pHlpR3->pfnAssertEMT(pDevIns, __FILE__, __LINE__, __FUNCTION__)
4437#else
4438# define PDMDEV_ASSERT_EMT(pDevIns) do { } while (0)
4439#endif
4440
4441/** @def PDMDEV_ASSERT_OTHER
4442 * Assert that the current thread is NOT the emulation thread.
4443 */
4444#ifdef VBOX_STRICT
4445# define PDMDEV_ASSERT_OTHER(pDevIns) pDevIns->pHlpR3->pfnAssertOther(pDevIns, __FILE__, __LINE__, __FUNCTION__)
4446#else
4447# define PDMDEV_ASSERT_OTHER(pDevIns) do { } while (0)
4448#endif
4449
4450/** @def PDMDEV_ASSERT_VMLOCK_OWNER
4451 * Assert that the current thread is owner of the VM lock.
4452 */
4453#ifdef VBOX_STRICT
4454# define PDMDEV_ASSERT_VMLOCK_OWNER(pDevIns) pDevIns->pHlpR3->pfnAssertVMLock(pDevIns, __FILE__, __LINE__, __FUNCTION__)
4455#else
4456# define PDMDEV_ASSERT_VMLOCK_OWNER(pDevIns) do { } while (0)
4457#endif
4458
4459/** @def PDMDEV_SET_ERROR
4460 * Set the VM error. See PDMDevHlpVMSetError() for printf like message formatting.
4461 */
4462#define PDMDEV_SET_ERROR(pDevIns, rc, pszError) \
4463 PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, "%s", pszError)
4464
4465/** @def PDMDEV_SET_RUNTIME_ERROR
4466 * Set the VM runtime error. See PDMDevHlpVMSetRuntimeError() for printf like message formatting.
4467 */
4468#define PDMDEV_SET_RUNTIME_ERROR(pDevIns, fFlags, pszErrorId, pszError) \
4469 PDMDevHlpVMSetRuntimeError(pDevIns, fFlags, pszErrorId, "%s", pszError)
4470
4471/** @def PDMDEVINS_2_RCPTR
4472 * Converts a PDM Device instance pointer a RC PDM Device instance pointer.
4473 */
4474#define PDMDEVINS_2_RCPTR(pDevIns) ( (RCPTRTYPE(PPDMDEVINS))((RTGCUINTPTR)(pDevIns)->pvInstanceDataRC - RT_OFFSETOF(PDMDEVINS, achInstanceData)) )
4475
4476/** @def PDMDEVINS_2_R3PTR
4477 * Converts a PDM Device instance pointer a R3 PDM Device instance pointer.
4478 */
4479#define PDMDEVINS_2_R3PTR(pDevIns) ( (R3PTRTYPE(PPDMDEVINS))((RTHCUINTPTR)(pDevIns)->pvInstanceDataR3 - RT_OFFSETOF(PDMDEVINS, achInstanceData)) )
4480
4481/** @def PDMDEVINS_2_R0PTR
4482 * Converts a PDM Device instance pointer a R0 PDM Device instance pointer.
4483 */
4484#define PDMDEVINS_2_R0PTR(pDevIns) ( (R0PTRTYPE(PPDMDEVINS))((RTR0UINTPTR)(pDevIns)->pvInstanceDataR0 - RT_OFFSETOF(PDMDEVINS, achInstanceData)) )
4485
4486
4487#ifdef IN_RING3
4488
4489/**
4490 * @copydoc PDMDEVHLPR3::pfnIOPortRegister
4491 */
4492DECLINLINE(int) PDMDevHlpIOPortRegister(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTHCPTR pvUser,
4493 PFNIOMIOPORTOUT pfnOut, PFNIOMIOPORTIN pfnIn,
4494 PFNIOMIOPORTOUTSTRING pfnOutStr, PFNIOMIOPORTINSTRING pfnInStr, const char *pszDesc)
4495{
4496 return pDevIns->pHlpR3->pfnIOPortRegister(pDevIns, Port, cPorts, pvUser, pfnOut, pfnIn, pfnOutStr, pfnInStr, pszDesc);
4497}
4498
4499/**
4500 * @copydoc PDMDEVHLPR3::pfnIOPortRegisterRC
4501 */
4502DECLINLINE(int) PDMDevHlpIOPortRegisterRC(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTRCPTR pvUser,
4503 const char *pszOut, const char *pszIn, const char *pszOutStr,
4504 const char *pszInStr, const char *pszDesc)
4505{
4506 return pDevIns->pHlpR3->pfnIOPortRegisterRC(pDevIns, Port, cPorts, pvUser, pszOut, pszIn, pszOutStr, pszInStr, pszDesc);
4507}
4508
4509/**
4510 * @copydoc PDMDEVHLPR3::pfnIOPortRegisterR0
4511 */
4512DECLINLINE(int) PDMDevHlpIOPortRegisterR0(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts, RTR0PTR pvUser,
4513 const char *pszOut, const char *pszIn, const char *pszOutStr,
4514 const char *pszInStr, const char *pszDesc)
4515{
4516 return pDevIns->pHlpR3->pfnIOPortRegisterR0(pDevIns, Port, cPorts, pvUser, pszOut, pszIn, pszOutStr, pszInStr, pszDesc);
4517}
4518
4519/**
4520 * @copydoc PDMDEVHLPR3::pfnIOPortDeregister
4521 */
4522DECLINLINE(int) PDMDevHlpIOPortDeregister(PPDMDEVINS pDevIns, RTIOPORT Port, RTIOPORT cPorts)
4523{
4524 return pDevIns->pHlpR3->pfnIOPortDeregister(pDevIns, Port, cPorts);
4525}
4526
4527/**
4528 * Register a Memory Mapped I/O (MMIO) region.
4529 *
4530 * These callbacks are of course for the ring-3 context (R3). Register HC
4531 * handlers before raw-mode context (RC) and ring-0 context (R0) handlers! There
4532 * must be a R3 handler for every RC and R0 handler!
4533 *
4534 * @returns VBox status.
4535 * @param pDevIns The device instance to register the MMIO with.
4536 * @param GCPhysStart First physical address in the range.
4537 * @param cbRange The size of the range (in bytes).
4538 * @param pvUser User argument.
4539 * @param fFlags Flags, IOMMMIO_FLAGS_XXX.
4540 * @param pfnWrite Pointer to function which is gonna handle Write operations.
4541 * @param pfnRead Pointer to function which is gonna handle Read operations.
4542 * @param pszDesc Pointer to description string. This must not be freed.
4543 */
4544DECLINLINE(int) PDMDevHlpMMIORegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTHCPTR pvUser,
4545 uint32_t fFlags, PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead, const char *pszDesc)
4546{
4547 return pDevIns->pHlpR3->pfnMMIORegister(pDevIns, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, NULL /*pfnFill*/,
4548 fFlags, pszDesc);
4549}
4550
4551/**
4552 * Register a Memory Mapped I/O (MMIO) region for RC.
4553 *
4554 * These callbacks are for the raw-mode context (RC). Register ring-3 context
4555 * (R3) handlers before guest context handlers! There must be a R3 handler for
4556 * every RC handler!
4557 *
4558 * @returns VBox status.
4559 * @param pDevIns The device instance to register the MMIO with.
4560 * @param GCPhysStart First physical address in the range.
4561 * @param cbRange The size of the range (in bytes).
4562 * @param pvUser User argument.
4563 * @param pszWrite Name of the RC function which is gonna handle Write operations.
4564 * @param pszRead Name of the RC function which is gonna handle Read operations.
4565 */
4566DECLINLINE(int) PDMDevHlpMMIORegisterRC(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTRCPTR pvUser,
4567 const char *pszWrite, const char *pszRead)
4568{
4569 return pDevIns->pHlpR3->pfnMMIORegisterRC(pDevIns, GCPhysStart, cbRange, pvUser, pszWrite, pszRead, NULL /*pszFill*/);
4570}
4571
4572/**
4573 * Register a Memory Mapped I/O (MMIO) region for R0.
4574 *
4575 * These callbacks are for the ring-0 host context (R0). Register ring-3
4576 * constext (R3) handlers before R0 handlers! There must be a R3 handler for
4577 * every R0 handler!
4578 *
4579 * @returns VBox status.
4580 * @param pDevIns The device instance to register the MMIO with.
4581 * @param GCPhysStart First physical address in the range.
4582 * @param cbRange The size of the range (in bytes).
4583 * @param pvUser User argument. (if pointer, then it must be in locked memory!)
4584 * @param pszWrite Name of the RC function which is gonna handle Write operations.
4585 * @param pszRead Name of the RC function which is gonna handle Read operations.
4586 * @remarks Caller enters the device critical section prior to invoking the
4587 * registered callback methods.
4588 */
4589DECLINLINE(int) PDMDevHlpMMIORegisterR0(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTR0PTR pvUser,
4590 const char *pszWrite, const char *pszRead)
4591{
4592 return pDevIns->pHlpR3->pfnMMIORegisterR0(pDevIns, GCPhysStart, cbRange, pvUser, pszWrite, pszRead, NULL /*pszFill*/);
4593}
4594
4595/**
4596 * @copydoc PDMDEVHLPR3::pfnMMIORegister
4597 */
4598DECLINLINE(int) PDMDevHlpMMIORegisterEx(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTHCPTR pvUser,
4599 uint32_t fFlags, PFNIOMMMIOWRITE pfnWrite, PFNIOMMMIOREAD pfnRead,
4600 PFNIOMMMIOFILL pfnFill, const char *pszDesc)
4601{
4602 return pDevIns->pHlpR3->pfnMMIORegister(pDevIns, GCPhysStart, cbRange, pvUser, pfnWrite, pfnRead, pfnFill,
4603 fFlags, pszDesc);
4604}
4605
4606/**
4607 * @copydoc PDMDEVHLPR3::pfnMMIORegisterRC
4608 */
4609DECLINLINE(int) PDMDevHlpMMIORegisterRCEx(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTRCPTR pvUser,
4610 const char *pszWrite, const char *pszRead, const char *pszFill)
4611{
4612 return pDevIns->pHlpR3->pfnMMIORegisterRC(pDevIns, GCPhysStart, cbRange, pvUser, pszWrite, pszRead, pszFill);
4613}
4614
4615/**
4616 * @copydoc PDMDEVHLPR3::pfnMMIORegisterR0
4617 */
4618DECLINLINE(int) PDMDevHlpMMIORegisterR0Ex(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, RTR0PTR pvUser,
4619 const char *pszWrite, const char *pszRead, const char *pszFill)
4620{
4621 return pDevIns->pHlpR3->pfnMMIORegisterR0(pDevIns, GCPhysStart, cbRange, pvUser, pszWrite, pszRead, pszFill);
4622}
4623
4624/**
4625 * @copydoc PDMDEVHLPR3::pfnMMIODeregister
4626 */
4627DECLINLINE(int) PDMDevHlpMMIODeregister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange)
4628{
4629 return pDevIns->pHlpR3->pfnMMIODeregister(pDevIns, GCPhysStart, cbRange);
4630}
4631
4632/**
4633 * @copydoc PDMDEVHLPR3::pfnMMIO2Register
4634 */
4635DECLINLINE(int) PDMDevHlpMMIO2Register(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS cb, uint32_t fFlags, void **ppv, const char *pszDesc)
4636{
4637 return pDevIns->pHlpR3->pfnMMIO2Register(pDevIns, iRegion, cb, fFlags, ppv, pszDesc);
4638}
4639
4640/**
4641 * @copydoc PDMDEVHLPR3::pfnMMIO2Deregister
4642 */
4643DECLINLINE(int) PDMDevHlpMMIO2Deregister(PPDMDEVINS pDevIns, uint32_t iRegion)
4644{
4645 return pDevIns->pHlpR3->pfnMMIO2Deregister(pDevIns, iRegion);
4646}
4647
4648/**
4649 * @copydoc PDMDEVHLPR3::pfnMMIO2Map
4650 */
4651DECLINLINE(int) PDMDevHlpMMIO2Map(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
4652{
4653 return pDevIns->pHlpR3->pfnMMIO2Map(pDevIns, iRegion, GCPhys);
4654}
4655
4656/**
4657 * @copydoc PDMDEVHLPR3::pfnMMIO2Unmap
4658 */
4659DECLINLINE(int) PDMDevHlpMMIO2Unmap(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS GCPhys)
4660{
4661 return pDevIns->pHlpR3->pfnMMIO2Unmap(pDevIns, iRegion, GCPhys);
4662}
4663
4664/**
4665 * @copydoc PDMDEVHLPR3::pfnMMHyperMapMMIO2
4666 */
4667DECLINLINE(int) PDMDevHlpMMHyperMapMMIO2(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
4668 const char *pszDesc, PRTRCPTR pRCPtr)
4669{
4670 return pDevIns->pHlpR3->pfnMMHyperMapMMIO2(pDevIns, iRegion, off, cb, pszDesc, pRCPtr);
4671}
4672
4673/**
4674 * @copydoc PDMDEVHLPR3::pfnMMIO2MapKernel
4675 */
4676DECLINLINE(int) PDMDevHlpMMIO2MapKernel(PPDMDEVINS pDevIns, uint32_t iRegion, RTGCPHYS off, RTGCPHYS cb,
4677 const char *pszDesc, PRTR0PTR pR0Ptr)
4678{
4679 return pDevIns->pHlpR3->pfnMMIO2MapKernel(pDevIns, iRegion, off, cb, pszDesc, pR0Ptr);
4680}
4681
4682/**
4683 * @copydoc PDMDEVHLPR3::pfnROMRegister
4684 */
4685DECLINLINE(int) PDMDevHlpROMRegister(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange,
4686 const void *pvBinary, uint32_t cbBinary, uint32_t fFlags, const char *pszDesc)
4687{
4688 return pDevIns->pHlpR3->pfnROMRegister(pDevIns, GCPhysStart, cbRange, pvBinary, cbBinary, fFlags, pszDesc);
4689}
4690
4691/**
4692 * @copydoc PDMDEVHLPR3::pfnROMProtectShadow
4693 */
4694DECLINLINE(int) PDMDevHlpROMProtectShadow(PPDMDEVINS pDevIns, RTGCPHYS GCPhysStart, uint32_t cbRange, PGMROMPROT enmProt)
4695{
4696 return pDevIns->pHlpR3->pfnROMProtectShadow(pDevIns, GCPhysStart, cbRange, enmProt);
4697}
4698
4699/**
4700 * Register a save state data unit.
4701 *
4702 * @returns VBox status.
4703 * @param pDevIns The device instance.
4704 * @param uVersion Data layout version number.
4705 * @param cbGuess The approximate amount of data in the unit.
4706 * Only for progress indicators.
4707 * @param pfnSaveExec Execute save callback, optional.
4708 * @param pfnLoadExec Execute load callback, optional.
4709 */
4710DECLINLINE(int) PDMDevHlpSSMRegister(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess,
4711 PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVLOADEXEC pfnLoadExec)
4712{
4713 return pDevIns->pHlpR3->pfnSSMRegister(pDevIns, uVersion, cbGuess, NULL /*pszBefore*/,
4714 NULL /*pfnLivePrep*/, NULL /*pfnLiveExec*/, NULL /*pfnLiveDone*/,
4715 NULL /*pfnSavePrep*/, pfnSaveExec, NULL /*pfnSaveDone*/,
4716 NULL /*pfnLoadPrep*/, pfnLoadExec, NULL /*pfnLoadDone*/);
4717}
4718
4719/**
4720 * Register a save state data unit with a live save callback as well.
4721 *
4722 * @returns VBox status.
4723 * @param pDevIns The device instance.
4724 * @param uVersion Data layout version number.
4725 * @param cbGuess The approximate amount of data in the unit.
4726 * Only for progress indicators.
4727 * @param pfnLiveExec Execute live callback, optional.
4728 * @param pfnSaveExec Execute save callback, optional.
4729 * @param pfnLoadExec Execute load callback, optional.
4730 */
4731DECLINLINE(int) PDMDevHlpSSMRegister3(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess,
4732 FNSSMDEVLIVEEXEC pfnLiveExec, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVLOADEXEC pfnLoadExec)
4733{
4734 return pDevIns->pHlpR3->pfnSSMRegister(pDevIns, uVersion, cbGuess, NULL /*pszBefore*/,
4735 NULL /*pfnLivePrep*/, pfnLiveExec, NULL /*pfnLiveDone*/,
4736 NULL /*pfnSavePrep*/, pfnSaveExec, NULL /*pfnSaveDone*/,
4737 NULL /*pfnLoadPrep*/, pfnLoadExec, NULL /*pfnLoadDone*/);
4738}
4739
4740/**
4741 * @copydoc PDMDEVHLPR3::pfnSSMRegister
4742 */
4743DECLINLINE(int) PDMDevHlpSSMRegisterEx(PPDMDEVINS pDevIns, uint32_t uVersion, size_t cbGuess, const char *pszBefore,
4744 PFNSSMDEVLIVEPREP pfnLivePrep, PFNSSMDEVLIVEEXEC pfnLiveExec, PFNSSMDEVLIVEVOTE pfnLiveVote,
4745 PFNSSMDEVSAVEPREP pfnSavePrep, PFNSSMDEVSAVEEXEC pfnSaveExec, PFNSSMDEVSAVEDONE pfnSaveDone,
4746 PFNSSMDEVLOADPREP pfnLoadPrep, PFNSSMDEVLOADEXEC pfnLoadExec, PFNSSMDEVLOADDONE pfnLoadDone)
4747{
4748 return pDevIns->pHlpR3->pfnSSMRegister(pDevIns, uVersion, cbGuess, pszBefore,
4749 pfnLivePrep, pfnLiveExec, pfnLiveVote,
4750 pfnSavePrep, pfnSaveExec, pfnSaveDone,
4751 pfnLoadPrep, pfnLoadExec, pfnLoadDone);
4752}
4753
4754/**
4755 * @copydoc PDMDEVHLPR3::pfnTMTimerCreate
4756 */
4757DECLINLINE(int) PDMDevHlpTMTimerCreate(PPDMDEVINS pDevIns, TMCLOCK enmClock, PFNTMTIMERDEV pfnCallback, void *pvUser, uint32_t fFlags,
4758 const char *pszDesc, PPTMTIMERR3 ppTimer)
4759{
4760 return pDevIns->pHlpR3->pfnTMTimerCreate(pDevIns, enmClock, pfnCallback, pvUser, fFlags, pszDesc, ppTimer);
4761}
4762
4763/**
4764 * @copydoc PDMDEVHLPR3::pfnTMUtcNow
4765 */
4766DECLINLINE(PRTTIMESPEC) PDMDevHlpTMUtcNow(PPDMDEVINS pDevIns, PRTTIMESPEC pTime)
4767{
4768 return pDevIns->pHlpR3->pfnTMUtcNow(pDevIns, pTime);
4769}
4770
4771#endif /* IN_RING3 */
4772
4773/**
4774 * @copydoc PDMDEVHLPR3::pfnPhysRead
4775 */
4776DECLINLINE(int) PDMDevHlpPhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
4777{
4778 return pDevIns->CTX_SUFF(pHlp)->pfnPhysRead(pDevIns, GCPhys, pvBuf, cbRead);
4779}
4780
4781/**
4782 * @copydoc PDMDEVHLPR3::pfnPhysWrite
4783 */
4784DECLINLINE(int) PDMDevHlpPhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
4785{
4786 return pDevIns->CTX_SUFF(pHlp)->pfnPhysWrite(pDevIns, GCPhys, pvBuf, cbWrite);
4787}
4788
4789#ifdef IN_RING3
4790
4791/**
4792 * @copydoc PDMDEVHLPR3::pfnPhysGCPhys2CCPtr
4793 */
4794DECLINLINE(int) PDMDevHlpPhysGCPhys2CCPtr(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, void **ppv, PPGMPAGEMAPLOCK pLock)
4795{
4796 return pDevIns->CTX_SUFF(pHlp)->pfnPhysGCPhys2CCPtr(pDevIns, GCPhys, fFlags, ppv, pLock);
4797}
4798
4799/**
4800 * @copydoc PDMDEVHLPR3::pfnPhysGCPhys2CCPtrReadOnly
4801 */
4802DECLINLINE(int) PDMDevHlpPhysGCPhys2CCPtrReadOnly(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t fFlags, void const **ppv,
4803 PPGMPAGEMAPLOCK pLock)
4804{
4805 return pDevIns->CTX_SUFF(pHlp)->pfnPhysGCPhys2CCPtrReadOnly(pDevIns, GCPhys, fFlags, ppv, pLock);
4806}
4807
4808/**
4809 * @copydoc PDMDEVHLPR3::pfnPhysReleasePageMappingLock
4810 */
4811DECLINLINE(void) PDMDevHlpPhysReleasePageMappingLock(PPDMDEVINS pDevIns, PPGMPAGEMAPLOCK pLock)
4812{
4813 pDevIns->CTX_SUFF(pHlp)->pfnPhysReleasePageMappingLock(pDevIns, pLock);
4814}
4815
4816/**
4817 * @copydoc PDMDEVHLPR3::pfnPhysReadGCVirt
4818 */
4819DECLINLINE(int) PDMDevHlpPhysReadGCVirt(PPDMDEVINS pDevIns, void *pvDst, RTGCPTR GCVirtSrc, size_t cb)
4820{
4821 return pDevIns->pHlpR3->pfnPhysReadGCVirt(pDevIns, pvDst, GCVirtSrc, cb);
4822}
4823
4824/**
4825 * @copydoc PDMDEVHLPR3::pfnPhysWriteGCVirt
4826 */
4827DECLINLINE(int) PDMDevHlpPhysWriteGCVirt(PPDMDEVINS pDevIns, RTGCPTR GCVirtDst, const void *pvSrc, size_t cb)
4828{
4829 return pDevIns->pHlpR3->pfnPhysWriteGCVirt(pDevIns, GCVirtDst, pvSrc, cb);
4830}
4831
4832/**
4833 * @copydoc PDMDEVHLPR3::pfnPhysGCPtr2GCPhys
4834 */
4835DECLINLINE(int) PDMDevHlpPhysGCPtr2GCPhys(PPDMDEVINS pDevIns, RTGCPTR GCPtr, PRTGCPHYS pGCPhys)
4836{
4837 return pDevIns->pHlpR3->pfnPhysGCPtr2GCPhys(pDevIns, GCPtr, pGCPhys);
4838}
4839
4840/**
4841 * @copydoc PDMDEVHLPR3::pfnMMHeapAlloc
4842 */
4843DECLINLINE(void *) PDMDevHlpMMHeapAlloc(PPDMDEVINS pDevIns, size_t cb)
4844{
4845 return pDevIns->pHlpR3->pfnMMHeapAlloc(pDevIns, cb);
4846}
4847
4848/**
4849 * @copydoc PDMDEVHLPR3::pfnMMHeapAllocZ
4850 */
4851DECLINLINE(void *) PDMDevHlpMMHeapAllocZ(PPDMDEVINS pDevIns, size_t cb)
4852{
4853 return pDevIns->pHlpR3->pfnMMHeapAllocZ(pDevIns, cb);
4854}
4855
4856/**
4857 * @copydoc PDMDEVHLPR3::pfnMMHeapFree
4858 */
4859DECLINLINE(void) PDMDevHlpMMHeapFree(PPDMDEVINS pDevIns, void *pv)
4860{
4861 pDevIns->pHlpR3->pfnMMHeapFree(pDevIns, pv);
4862}
4863#endif /* IN_RING3 */
4864
4865/**
4866 * @copydoc PDMDEVHLPR3::pfnVMState
4867 */
4868DECLINLINE(VMSTATE) PDMDevHlpVMState(PPDMDEVINS pDevIns)
4869{
4870 return pDevIns->CTX_SUFF(pHlp)->pfnVMState(pDevIns);
4871}
4872
4873#ifdef IN_RING3
4874/**
4875 * @copydoc PDMDEVHLPR3::pfnVMTeleportedAndNotFullyResumedYet
4876 */
4877DECLINLINE(bool) PDMDevHlpVMTeleportedAndNotFullyResumedYet(PPDMDEVINS pDevIns)
4878{
4879 return pDevIns->pHlpR3->pfnVMTeleportedAndNotFullyResumedYet(pDevIns);
4880}
4881#endif /* IN_RING3 */
4882
4883/**
4884 * @copydoc PDMDEVHLPR3::pfnVMSetError
4885 */
4886DECLINLINE(int) RT_IPRT_FORMAT_ATTR(6, 7) PDMDevHlpVMSetError(PPDMDEVINS pDevIns, const int rc, RT_SRC_POS_DECL,
4887 const char *pszFormat, ...)
4888{
4889 va_list va;
4890 va_start(va, pszFormat);
4891 pDevIns->CTX_SUFF(pHlp)->pfnVMSetErrorV(pDevIns, rc, RT_SRC_POS_ARGS, pszFormat, va);
4892 va_end(va);
4893 return rc;
4894}
4895
4896/**
4897 * @copydoc PDMDEVHLPR3::pfnVMSetRuntimeError
4898 */
4899DECLINLINE(int) RT_IPRT_FORMAT_ATTR(4, 5) PDMDevHlpVMSetRuntimeError(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId,
4900 const char *pszFormat, ...)
4901{
4902 va_list va;
4903 int rc;
4904 va_start(va, pszFormat);
4905 rc = pDevIns->CTX_SUFF(pHlp)->pfnVMSetRuntimeErrorV(pDevIns, fFlags, pszErrorId, pszFormat, va);
4906 va_end(va);
4907 return rc;
4908}
4909
4910/**
4911 * VBOX_STRICT wrapper for pHlp->pfnDBGFStopV.
4912 *
4913 * @returns VBox status code which must be passed up to the VMM. This will be
4914 * VINF_SUCCESS in non-strict builds.
4915 * @param pDevIns The device instance.
4916 * @param SRC_POS Use RT_SRC_POS.
4917 * @param pszFormat Message. (optional)
4918 * @param ... Message parameters.
4919 */
4920DECLINLINE(int) RT_IPRT_FORMAT_ATTR(5, 6) PDMDevHlpDBGFStop(PPDMDEVINS pDevIns, RT_SRC_POS_DECL, const char *pszFormat, ...)
4921{
4922#ifdef VBOX_STRICT
4923# ifdef IN_RING3
4924 int rc;
4925 va_list args;
4926 va_start(args, pszFormat);
4927 rc = pDevIns->pHlpR3->pfnDBGFStopV(pDevIns, RT_SRC_POS_ARGS, pszFormat, args);
4928 va_end(args);
4929 return rc;
4930# else
4931 NOREF(pDevIns);
4932 NOREF(pszFile);
4933 NOREF(iLine);
4934 NOREF(pszFunction);
4935 NOREF(pszFormat);
4936 return VINF_EM_DBG_STOP;
4937# endif
4938#else
4939 NOREF(pDevIns);
4940 NOREF(pszFile);
4941 NOREF(iLine);
4942 NOREF(pszFunction);
4943 NOREF(pszFormat);
4944 return VINF_SUCCESS;
4945#endif
4946}
4947
4948#ifdef IN_RING3
4949
4950/**
4951 * @copydoc PDMDEVHLPR3::pfnDBGFInfoRegister
4952 */
4953DECLINLINE(int) PDMDevHlpDBGFInfoRegister(PPDMDEVINS pDevIns, const char *pszName, const char *pszDesc, PFNDBGFHANDLERDEV pfnHandler)
4954{
4955 return pDevIns->pHlpR3->pfnDBGFInfoRegister(pDevIns, pszName, pszDesc, pfnHandler);
4956}
4957
4958/**
4959 * @copydoc PDMDEVHLPR3::pfnDBGFRegRegister
4960 */
4961DECLINLINE(int) PDMDevHlpDBGFRegRegister(PPDMDEVINS pDevIns, PCDBGFREGDESC paRegisters)
4962{
4963 return pDevIns->pHlpR3->pfnDBGFRegRegister(pDevIns, paRegisters);
4964}
4965
4966/**
4967 * @copydoc PDMDEVHLPR3::pfnSTAMRegister
4968 */
4969DECLINLINE(void) PDMDevHlpSTAMRegister(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType, const char *pszName, STAMUNIT enmUnit, const char *pszDesc)
4970{
4971 pDevIns->pHlpR3->pfnSTAMRegister(pDevIns, pvSample, enmType, pszName, enmUnit, pszDesc);
4972}
4973
4974/**
4975 * @copydoc PDMDEVHLPR3::pfnSTAMRegisterF
4976 */
4977DECLINLINE(void) RT_IPRT_FORMAT_ATTR(7, 8) PDMDevHlpSTAMRegisterF(PPDMDEVINS pDevIns, void *pvSample, STAMTYPE enmType,
4978 STAMVISIBILITY enmVisibility, STAMUNIT enmUnit,
4979 const char *pszDesc, const char *pszName, ...)
4980{
4981 va_list va;
4982 va_start(va, pszName);
4983 pDevIns->pHlpR3->pfnSTAMRegisterV(pDevIns, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, va);
4984 va_end(va);
4985}
4986
4987/**
4988 * @copydoc PDMDEVHLPR3::pfnPCIRegister
4989 */
4990DECLINLINE(int) PDMDevHlpPCIRegister(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev)
4991{
4992 return pDevIns->pHlpR3->pfnPCIRegister(pDevIns, pPciDev);
4993}
4994
4995/**
4996 * @copydoc PDMDEVHLPR3::pfnPCIIORegionRegister
4997 */
4998DECLINLINE(int) PDMDevHlpPCIIORegionRegister(PPDMDEVINS pDevIns, int iRegion, uint32_t cbRegion,
4999 PCIADDRESSSPACE enmType, PFNPCIIOREGIONMAP pfnCallback)
5000{
5001 return pDevIns->pHlpR3->pfnPCIIORegionRegister(pDevIns, iRegion, cbRegion, enmType, pfnCallback);
5002}
5003
5004/**
5005 * @copydoc PDMDEVHLPR3::pfnPCIRegisterMsi
5006 */
5007DECLINLINE(int) PDMDevHlpPCIRegisterMsi(PPDMDEVINS pDevIns, PPDMMSIREG pMsiReg)
5008{
5009 return pDevIns->pHlpR3->pfnPCIRegisterMsi(pDevIns, pMsiReg);
5010}
5011
5012/**
5013 * @copydoc PDMDEVHLPR3::pfnPCISetConfigCallbacks
5014 */
5015DECLINLINE(void) PDMDevHlpPCISetConfigCallbacks(PPDMDEVINS pDevIns, PPCIDEVICE pPciDev, PFNPCICONFIGREAD pfnRead, PPFNPCICONFIGREAD ppfnReadOld,
5016 PFNPCICONFIGWRITE pfnWrite, PPFNPCICONFIGWRITE ppfnWriteOld)
5017{
5018 pDevIns->pHlpR3->pfnPCISetConfigCallbacks(pDevIns, pPciDev, pfnRead, ppfnReadOld, pfnWrite, ppfnWriteOld);
5019}
5020
5021#endif /* IN_RING3 */
5022
5023/**
5024 * @copydoc PDMDEVHLPR3::pfnPCIPhysRead
5025 */
5026DECLINLINE(int) PDMDevHlpPCIPhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
5027{
5028 return pDevIns->CTX_SUFF(pHlp)->pfnPCIPhysRead(pDevIns, GCPhys, pvBuf, cbRead);
5029}
5030
5031/**
5032 * @copydoc PDMDEVHLPR3::pfnPCIPhysWrite
5033 */
5034DECLINLINE(int) PDMDevHlpPCIPhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
5035{
5036 return pDevIns->CTX_SUFF(pHlp)->pfnPCIPhysWrite(pDevIns, GCPhys, pvBuf, cbWrite);
5037}
5038
5039/**
5040 * @copydoc PDMDEVHLPR3::pfnPCISetIrq
5041 */
5042DECLINLINE(void) PDMDevHlpPCISetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
5043{
5044 pDevIns->CTX_SUFF(pHlp)->pfnPCISetIrq(pDevIns, iIrq, iLevel);
5045}
5046
5047/**
5048 * @copydoc PDMDEVHLPR3::pfnPCISetIrqNoWait
5049 */
5050DECLINLINE(void) PDMDevHlpPCISetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
5051{
5052 pDevIns->CTX_SUFF(pHlp)->pfnPCISetIrq(pDevIns, iIrq, iLevel);
5053}
5054
5055/**
5056 * @copydoc PDMDEVHLPR3::pfnISASetIrq
5057 */
5058DECLINLINE(void) PDMDevHlpISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
5059{
5060 pDevIns->CTX_SUFF(pHlp)->pfnISASetIrq(pDevIns, iIrq, iLevel);
5061}
5062
5063/**
5064 * @copydoc PDMDEVHLPR3::pfnISASetIrqNoWait
5065 */
5066DECLINLINE(void) PDMDevHlpISASetIrqNoWait(PPDMDEVINS pDevIns, int iIrq, int iLevel)
5067{
5068 pDevIns->CTX_SUFF(pHlp)->pfnISASetIrq(pDevIns, iIrq, iLevel);
5069}
5070
5071#ifdef IN_RING3
5072
5073/**
5074 * @copydoc PDMDEVHLPR3::pfnDriverAttach
5075 */
5076DECLINLINE(int) PDMDevHlpDriverAttach(PPDMDEVINS pDevIns, uint32_t iLun, PPDMIBASE pBaseInterface, PPDMIBASE *ppBaseInterface, const char *pszDesc)
5077{
5078 return pDevIns->pHlpR3->pfnDriverAttach(pDevIns, iLun, pBaseInterface, ppBaseInterface, pszDesc);
5079}
5080
5081/**
5082 * @copydoc PDMDEVHLPR3::pfnDriverDetach
5083 */
5084DECLINLINE(int) PDMDevHlpDriverDetach(PPDMDEVINS pDevIns, PPDMDRVINS pDrvIns, uint32_t fFlags)
5085{
5086 return pDevIns->pHlpR3->pfnDriverDetach(pDevIns, pDrvIns, fFlags);
5087}
5088
5089/**
5090 * @copydoc PDMDEVHLPR3::pfnQueueCreate
5091 */
5092DECLINLINE(int) PDMDevHlpQueueCreate(PPDMDEVINS pDevIns, size_t cbItem, uint32_t cItems, uint32_t cMilliesInterval,
5093 PFNPDMQUEUEDEV pfnCallback, bool fRZEnabled, const char *pszName, PPDMQUEUE *ppQueue)
5094{
5095 return pDevIns->pHlpR3->pfnQueueCreate(pDevIns, cbItem, cItems, cMilliesInterval, pfnCallback, fRZEnabled, pszName, ppQueue);
5096}
5097
5098/**
5099 * Initializes a PDM critical section.
5100 *
5101 * The PDM critical sections are derived from the IPRT critical sections, but
5102 * works in RC and R0 as well.
5103 *
5104 * @returns VBox status code.
5105 * @param pDevIns The device instance.
5106 * @param pCritSect Pointer to the critical section.
5107 * @param SRC_POS Use RT_SRC_POS.
5108 * @param pszNameFmt Format string for naming the critical section.
5109 * For statistics and lock validation.
5110 * @param ... Arguments for the format string.
5111 */
5112DECLINLINE(int) RT_IPRT_FORMAT_ATTR(6, 7) PDMDevHlpCritSectInit(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect, RT_SRC_POS_DECL,
5113 const char *pszNameFmt, ...)
5114{
5115 int rc;
5116 va_list va;
5117 va_start(va, pszNameFmt);
5118 rc = pDevIns->pHlpR3->pfnCritSectInit(pDevIns, pCritSect, RT_SRC_POS_ARGS, pszNameFmt, va);
5119 va_end(va);
5120 return rc;
5121}
5122
5123/**
5124 * @copydoc PDMDEVHLPR3::pfnCritSectGetNop
5125 */
5126DECLINLINE(PPDMCRITSECT) PDMDevHlpCritSectGetNop(PPDMDEVINS pDevIns)
5127{
5128 return pDevIns->pHlpR3->pfnCritSectGetNop(pDevIns);
5129}
5130
5131/**
5132 * @copydoc PDMDEVHLPR3::pfnCritSectGetNopR0
5133 */
5134DECLINLINE(R0PTRTYPE(PPDMCRITSECT)) PDMDevHlpCritSectGetNopR0(PPDMDEVINS pDevIns)
5135{
5136 return pDevIns->pHlpR3->pfnCritSectGetNopR0(pDevIns);
5137}
5138
5139/**
5140 * @copydoc PDMDEVHLPR3::pfnCritSectGetNopRC
5141 */
5142DECLINLINE(RCPTRTYPE(PPDMCRITSECT)) PDMDevHlpCritSectGetNopRC(PPDMDEVINS pDevIns)
5143{
5144 return pDevIns->pHlpR3->pfnCritSectGetNopRC(pDevIns);
5145}
5146
5147/**
5148 * @copydoc PDMDEVHLPR3::pfnSetDeviceCritSect
5149 */
5150DECLINLINE(int) PDMDevHlpSetDeviceCritSect(PPDMDEVINS pDevIns, PPDMCRITSECT pCritSect)
5151{
5152 return pDevIns->pHlpR3->pfnSetDeviceCritSect(pDevIns, pCritSect);
5153}
5154
5155/**
5156 * @copydoc PDMDEVHLPR3::pfnThreadCreate
5157 */
5158DECLINLINE(int) PDMDevHlpThreadCreate(PPDMDEVINS pDevIns, PPPDMTHREAD ppThread, void *pvUser, PFNPDMTHREADDEV pfnThread,
5159 PFNPDMTHREADWAKEUPDEV pfnWakeup, size_t cbStack, RTTHREADTYPE enmType, const char *pszName)
5160{
5161 return pDevIns->pHlpR3->pfnThreadCreate(pDevIns, ppThread, pvUser, pfnThread, pfnWakeup, cbStack, enmType, pszName);
5162}
5163
5164/**
5165 * @copydoc PDMDEVHLPR3::pfnSetAsyncNotification
5166 */
5167DECLINLINE(int) PDMDevHlpSetAsyncNotification(PPDMDEVINS pDevIns, PFNPDMDEVASYNCNOTIFY pfnAsyncNotify)
5168{
5169 return pDevIns->pHlpR3->pfnSetAsyncNotification(pDevIns, pfnAsyncNotify);
5170}
5171
5172/**
5173 * @copydoc PDMDEVHLPR3::pfnAsyncNotificationCompleted
5174 */
5175DECLINLINE(void) PDMDevHlpAsyncNotificationCompleted(PPDMDEVINS pDevIns)
5176{
5177 pDevIns->pHlpR3->pfnAsyncNotificationCompleted(pDevIns);
5178}
5179
5180/**
5181 * @copydoc PDMDEVHLPR3::pfnA20Set
5182 */
5183DECLINLINE(void) PDMDevHlpA20Set(PPDMDEVINS pDevIns, bool fEnable)
5184{
5185 pDevIns->pHlpR3->pfnA20Set(pDevIns, fEnable);
5186}
5187
5188/**
5189 * @copydoc PDMDEVHLPR3::pfnRTCRegister
5190 */
5191DECLINLINE(int) PDMDevHlpRTCRegister(PPDMDEVINS pDevIns, PCPDMRTCREG pRtcReg, PCPDMRTCHLP *ppRtcHlp)
5192{
5193 return pDevIns->pHlpR3->pfnRTCRegister(pDevIns, pRtcReg, ppRtcHlp);
5194}
5195
5196/**
5197 * @copydoc PDMDEVHLPR3::pfnPCIBusRegister
5198 */
5199DECLINLINE(int) PDMDevHlpPCIBusRegister(PPDMDEVINS pDevIns, PPDMPCIBUSREG pPciBusReg, PCPDMPCIHLPR3 *ppPciHlpR3)
5200{
5201 return pDevIns->pHlpR3->pfnPCIBusRegister(pDevIns, pPciBusReg, ppPciHlpR3);
5202}
5203
5204/**
5205 * @copydoc PDMDEVHLPR3::pfnPICRegister
5206 */
5207DECLINLINE(int) PDMDevHlpPICRegister(PPDMDEVINS pDevIns, PPDMPICREG pPicReg, PCPDMPICHLPR3 *ppPicHlpR3)
5208{
5209 return pDevIns->pHlpR3->pfnPICRegister(pDevIns, pPicReg, ppPicHlpR3);
5210}
5211
5212/**
5213 * @copydoc PDMDEVHLPR3::pfnAPICRegister
5214 */
5215DECLINLINE(int) PDMDevHlpAPICRegister(PPDMDEVINS pDevIns, PPDMAPICREG pApicReg, PCPDMAPICHLPR3 *ppApicHlpR3)
5216{
5217 return pDevIns->pHlpR3->pfnAPICRegister(pDevIns, pApicReg, ppApicHlpR3);
5218}
5219
5220/**
5221 * @copydoc PDMDEVHLPR3::pfnIOAPICRegister
5222 */
5223DECLINLINE(int) PDMDevHlpIOAPICRegister(PPDMDEVINS pDevIns, PPDMIOAPICREG pIoApicReg, PCPDMIOAPICHLPR3 *ppIoApicHlpR3)
5224{
5225 return pDevIns->pHlpR3->pfnIOAPICRegister(pDevIns, pIoApicReg, ppIoApicHlpR3);
5226}
5227
5228/**
5229 * @copydoc PDMDEVHLPR3::pfnHPETRegister
5230 */
5231DECLINLINE(int) PDMDevHlpHPETRegister(PPDMDEVINS pDevIns, PPDMHPETREG pHpetReg, PCPDMHPETHLPR3 *ppHpetHlpR3)
5232{
5233 return pDevIns->pHlpR3->pfnHPETRegister(pDevIns, pHpetReg, ppHpetHlpR3);
5234}
5235
5236/**
5237 * @copydoc PDMDEVHLPR3::pfnPciRawRegister
5238 */
5239DECLINLINE(int) PDMDevHlpPciRawRegister(PPDMDEVINS pDevIns, PPDMPCIRAWREG pPciRawReg, PCPDMPCIRAWHLPR3 *ppPciRawHlpR3)
5240{
5241 return pDevIns->pHlpR3->pfnPciRawRegister(pDevIns, pPciRawReg, ppPciRawHlpR3);
5242}
5243
5244/**
5245 * @copydoc PDMDEVHLPR3::pfnDMACRegister
5246 */
5247DECLINLINE(int) PDMDevHlpDMACRegister(PPDMDEVINS pDevIns, PPDMDMACREG pDmacReg, PCPDMDMACHLP *ppDmacHlp)
5248{
5249 return pDevIns->pHlpR3->pfnDMACRegister(pDevIns, pDmacReg, ppDmacHlp);
5250}
5251
5252/**
5253 * @copydoc PDMDEVHLPR3::pfnDMARegister
5254 */
5255DECLINLINE(int) PDMDevHlpDMARegister(PPDMDEVINS pDevIns, unsigned uChannel, PFNDMATRANSFERHANDLER pfnTransferHandler, void *pvUser)
5256{
5257 return pDevIns->pHlpR3->pfnDMARegister(pDevIns, uChannel, pfnTransferHandler, pvUser);
5258}
5259
5260/**
5261 * @copydoc PDMDEVHLPR3::pfnDMAReadMemory
5262 */
5263DECLINLINE(int) PDMDevHlpDMAReadMemory(PPDMDEVINS pDevIns, unsigned uChannel, void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbRead)
5264{
5265 return pDevIns->pHlpR3->pfnDMAReadMemory(pDevIns, uChannel, pvBuffer, off, cbBlock, pcbRead);
5266}
5267
5268/**
5269 * @copydoc PDMDEVHLPR3::pfnDMAWriteMemory
5270 */
5271DECLINLINE(int) PDMDevHlpDMAWriteMemory(PPDMDEVINS pDevIns, unsigned uChannel, const void *pvBuffer, uint32_t off, uint32_t cbBlock, uint32_t *pcbWritten)
5272{
5273 return pDevIns->pHlpR3->pfnDMAWriteMemory(pDevIns, uChannel, pvBuffer, off, cbBlock, pcbWritten);
5274}
5275
5276/**
5277 * @copydoc PDMDEVHLPR3::pfnDMASetDREQ
5278 */
5279DECLINLINE(int) PDMDevHlpDMASetDREQ(PPDMDEVINS pDevIns, unsigned uChannel, unsigned uLevel)
5280{
5281 return pDevIns->pHlpR3->pfnDMASetDREQ(pDevIns, uChannel, uLevel);
5282}
5283
5284/**
5285 * @copydoc PDMDEVHLPR3::pfnDMAGetChannelMode
5286 */
5287DECLINLINE(uint8_t) PDMDevHlpDMAGetChannelMode(PPDMDEVINS pDevIns, unsigned uChannel)
5288{
5289 return pDevIns->pHlpR3->pfnDMAGetChannelMode(pDevIns, uChannel);
5290}
5291
5292/**
5293 * @copydoc PDMDEVHLPR3::pfnDMASchedule
5294 */
5295DECLINLINE(void) PDMDevHlpDMASchedule(PPDMDEVINS pDevIns)
5296{
5297 pDevIns->pHlpR3->pfnDMASchedule(pDevIns);
5298}
5299
5300/**
5301 * @copydoc PDMDEVHLPR3::pfnCMOSWrite
5302 */
5303DECLINLINE(int) PDMDevHlpCMOSWrite(PPDMDEVINS pDevIns, unsigned iReg, uint8_t u8Value)
5304{
5305 return pDevIns->pHlpR3->pfnCMOSWrite(pDevIns, iReg, u8Value);
5306}
5307
5308/**
5309 * @copydoc PDMDEVHLPR3::pfnCMOSRead
5310 */
5311DECLINLINE(int) PDMDevHlpCMOSRead(PPDMDEVINS pDevIns, unsigned iReg, uint8_t *pu8Value)
5312{
5313 return pDevIns->pHlpR3->pfnCMOSRead(pDevIns, iReg, pu8Value);
5314}
5315
5316/**
5317 * @copydoc PDMDEVHLPR3::pfnCallR0
5318 */
5319DECLINLINE(int) PDMDevHlpCallR0(PPDMDEVINS pDevIns, uint32_t uOperation, uint64_t u64Arg)
5320{
5321 return pDevIns->pHlpR3->pfnCallR0(pDevIns, uOperation, u64Arg);
5322}
5323
5324/**
5325 * @copydoc PDMDEVHLPR3::pfnVMGetSuspendReason
5326 */
5327DECLINLINE(VMSUSPENDREASON) PDMDevHlpVMGetSuspendReason(PPDMDEVINS pDevIns)
5328{
5329 return pDevIns->pHlpR3->pfnVMGetSuspendReason(pDevIns);
5330}
5331
5332/**
5333 * @copydoc PDMDEVHLPR3::pfnVMGetResumeReason
5334 */
5335DECLINLINE(VMRESUMEREASON) PDMDevHlpVMGetResumeReason(PPDMDEVINS pDevIns)
5336{
5337 return pDevIns->pHlpR3->pfnVMGetResumeReason(pDevIns);
5338}
5339
5340/**
5341 * @copydoc PDMDEVHLPR3::pfnGetUVM
5342 */
5343DECLINLINE(PUVM) PDMDevHlpGetUVM(PPDMDEVINS pDevIns)
5344{
5345 return pDevIns->CTX_SUFF(pHlp)->pfnGetUVM(pDevIns);
5346}
5347
5348#endif /* IN_RING3 */
5349
5350/**
5351 * @copydoc PDMDEVHLPR3::pfnGetVM
5352 */
5353DECLINLINE(PVM) PDMDevHlpGetVM(PPDMDEVINS pDevIns)
5354{
5355 return pDevIns->CTX_SUFF(pHlp)->pfnGetVM(pDevIns);
5356}
5357
5358/**
5359 * @copydoc PDMDEVHLPR3::pfnGetVMCPU
5360 */
5361DECLINLINE(PVMCPU) PDMDevHlpGetVMCPU(PPDMDEVINS pDevIns)
5362{
5363 return pDevIns->CTX_SUFF(pHlp)->pfnGetVMCPU(pDevIns);
5364}
5365
5366/**
5367 * @copydoc PDMDEVHLPR3::pfnGetCurrentCpuId
5368 */
5369DECLINLINE(VMCPUID) PDMDevHlpGetCurrentCpuId(PPDMDEVINS pDevIns)
5370{
5371 return pDevIns->CTX_SUFF(pHlp)->pfnGetCurrentCpuId(pDevIns);
5372}
5373
5374/**
5375 * @copydoc PDMDEVHLPR3::pfnTMTimeVirtGet
5376 */
5377DECLINLINE(uint64_t) PDMDevHlpTMTimeVirtGet(PPDMDEVINS pDevIns)
5378{
5379 return pDevIns->CTX_SUFF(pHlp)->pfnTMTimeVirtGet(pDevIns);
5380}
5381
5382/**
5383 * @copydoc PDMDEVHLPR3::pfnTMTimeVirtGetFreq
5384 */
5385DECLINLINE(uint64_t) PDMDevHlpTMTimeVirtGetFreq(PPDMDEVINS pDevIns)
5386{
5387 return pDevIns->CTX_SUFF(pHlp)->pfnTMTimeVirtGetFreq(pDevIns);
5388}
5389
5390/**
5391 * @copydoc PDMDEVHLPR3::pfnTMTimeVirtGetFreq
5392 */
5393DECLINLINE(uint64_t) PDMDevHlpTMTimeVirtGetNano(PPDMDEVINS pDevIns)
5394{
5395 return pDevIns->CTX_SUFF(pHlp)->pfnTMTimeVirtGetNano(pDevIns);
5396}
5397
5398#ifdef IN_RING3
5399
5400/**
5401 * @copydoc PDMDEVHLPR3::pfnRegisterVMMDevHeap
5402 */
5403DECLINLINE(int) PDMDevHlpRegisterVMMDevHeap(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTR3PTR pvHeap, unsigned cbHeap)
5404{
5405 return pDevIns->pHlpR3->pfnRegisterVMMDevHeap(pDevIns, GCPhys, pvHeap, cbHeap);
5406}
5407
5408/**
5409 * @copydoc PDMDEVHLPR3::pfnFirmwareRegister
5410 */
5411DECLINLINE(int) PDMDevHlpFirmwareRegister(PPDMDEVINS pDevIns, PCPDMFWREG pFwReg, PCPDMFWHLPR3 *ppFwHlp)
5412{
5413 return pDevIns->pHlpR3->pfnFirmwareRegister(pDevIns, pFwReg, ppFwHlp);
5414}
5415
5416/**
5417 * @copydoc PDMDEVHLPR3::pfnVMReset
5418 */
5419DECLINLINE(int) PDMDevHlpVMReset(PPDMDEVINS pDevIns, uint32_t fFlags)
5420{
5421 return pDevIns->pHlpR3->pfnVMReset(pDevIns, fFlags);
5422}
5423
5424/**
5425 * @copydoc PDMDEVHLPR3::pfnVMSuspend
5426 */
5427DECLINLINE(int) PDMDevHlpVMSuspend(PPDMDEVINS pDevIns)
5428{
5429 return pDevIns->pHlpR3->pfnVMSuspend(pDevIns);
5430}
5431
5432/**
5433 * @copydoc PDMDEVHLPR3::pfnVMSuspendSaveAndPowerOff
5434 */
5435DECLINLINE(int) PDMDevHlpVMSuspendSaveAndPowerOff(PPDMDEVINS pDevIns)
5436{
5437 return pDevIns->pHlpR3->pfnVMSuspendSaveAndPowerOff(pDevIns);
5438}
5439
5440/**
5441 * @copydoc PDMDEVHLPR3::pfnVMPowerOff
5442 */
5443DECLINLINE(int) PDMDevHlpVMPowerOff(PPDMDEVINS pDevIns)
5444{
5445 return pDevIns->pHlpR3->pfnVMPowerOff(pDevIns);
5446}
5447
5448#endif /* IN_RING3 */
5449
5450/**
5451 * @copydoc PDMDEVHLPR3::pfnA20IsEnabled
5452 */
5453DECLINLINE(bool) PDMDevHlpA20IsEnabled(PPDMDEVINS pDevIns)
5454{
5455 return pDevIns->CTX_SUFF(pHlp)->pfnA20IsEnabled(pDevIns);
5456}
5457
5458#ifdef IN_RING3
5459
5460/**
5461 * @copydoc PDMDEVHLPR3::pfnGetCpuId
5462 */
5463DECLINLINE(void) PDMDevHlpGetCpuId(PPDMDEVINS pDevIns, uint32_t iLeaf, uint32_t *pEax, uint32_t *pEbx, uint32_t *pEcx, uint32_t *pEdx)
5464{
5465 pDevIns->pHlpR3->pfnGetCpuId(pDevIns, iLeaf, pEax, pEbx, pEcx, pEdx);
5466}
5467
5468/**
5469 * @copydoc PDMDEVHLPR3::pfnGetSupDrvSession
5470 */
5471DECLINLINE(PSUPDRVSESSION) PDMDevHlpGetSupDrvSession(PPDMDEVINS pDevIns)
5472{
5473 return pDevIns->pHlpR3->pfnGetSupDrvSession(pDevIns);
5474}
5475
5476#endif /* IN_RING3 */
5477#ifdef IN_RING0
5478
5479/**
5480 * @copydoc PDMDEVHLPR0::pfnCanEmulateIoBlock
5481 */
5482DECLINLINE(bool) PDMDevHlpCanEmulateIoBlock(PPDMDEVINS pDevIns)
5483{
5484 return pDevIns->CTX_SUFF(pHlp)->pfnCanEmulateIoBlock(pDevIns);
5485}
5486
5487#endif /* IN_RING0 */
5488
5489
5490
5491
5492/** Pointer to callbacks provided to the VBoxDeviceRegister() call. */
5493typedef struct PDMDEVREGCB *PPDMDEVREGCB;
5494
5495/**
5496 * Callbacks for VBoxDeviceRegister().
5497 */
5498typedef struct PDMDEVREGCB
5499{
5500 /** Interface version.
5501 * This is set to PDM_DEVREG_CB_VERSION. */
5502 uint32_t u32Version;
5503
5504 /**
5505 * Registers a device with the current VM instance.
5506 *
5507 * @returns VBox status code.
5508 * @param pCallbacks Pointer to the callback table.
5509 * @param pReg Pointer to the device registration record.
5510 * This data must be permanent and readonly.
5511 */
5512 DECLR3CALLBACKMEMBER(int, pfnRegister,(PPDMDEVREGCB pCallbacks, PCPDMDEVREG pReg));
5513} PDMDEVREGCB;
5514
5515/** Current version of the PDMDEVREGCB structure. */
5516#define PDM_DEVREG_CB_VERSION PDM_VERSION_MAKE(0xffe3, 1, 0)
5517
5518
5519/**
5520 * The VBoxDevicesRegister callback function.
5521 *
5522 * PDM will invoke this function after loading a device module and letting
5523 * the module decide which devices to register and how to handle conflicts.
5524 *
5525 * @returns VBox status code.
5526 * @param pCallbacks Pointer to the callback table.
5527 * @param u32Version VBox version number.
5528 */
5529typedef DECLCALLBACK(int) FNPDMVBOXDEVICESREGISTER(PPDMDEVREGCB pCallbacks, uint32_t u32Version);
5530
5531/** @} */
5532
5533RT_C_DECLS_END
5534
5535#endif
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