VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/DevACPI.cpp@ 26831

Last change on this file since 26831 was 26614, checked in by vboxsync, 15 years ago

CPU hotplug: Replace container objects with noop's if the attached CPU can't never show up in the VM

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 92.9 KB
Line 
1/* $Id: DevACPI.cpp 26614 2010-02-17 15:03:18Z vboxsync $ */
2/** @file
3 * DevACPI - Advanced Configuration and Power Interface (ACPI) Device.
4 */
5
6/*
7 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_DEV_ACPI
26#include <VBox/pdmdev.h>
27#include <VBox/pgm.h>
28#include <VBox/log.h>
29#include <VBox/param.h>
30#include <iprt/assert.h>
31#include <iprt/asm.h>
32#ifdef IN_RING3
33# include <iprt/alloc.h>
34# include <iprt/string.h>
35# include <iprt/uuid.h>
36#endif /* IN_RING3 */
37
38#include "../Builtins.h"
39
40#ifdef LOG_ENABLED
41# define DEBUG_ACPI
42#endif
43
44#if defined(IN_RING3) && !defined(VBOX_DEVICE_STRUCT_TESTCASE)
45int acpiPrepareDsdt(PPDMDEVINS pDevIns, void* *ppPtr, size_t *puDsdtLen);
46int acpiCleanupDsdt(PPDMDEVINS pDevIns, void* pPtr);
47
48int acpiPrepareSsdt(PPDMDEVINS pDevIns, void* *ppPtr, size_t *puSsdtLen);
49int acpiCleanupSsdt(PPDMDEVINS pDevIns, void* pPtr);
50#endif /* !IN_RING3 */
51
52
53
54/*******************************************************************************
55* Defined Constants And Macros *
56*******************************************************************************/
57#define DEBUG_HEX 0x3000
58#define DEBUG_CHR 0x3001
59
60#define PM_TMR_FREQ 3579545
61/* Default base for PM PIIX4 device */
62#define PM_PORT_BASE 0x4000
63/* Port offsets in PM device */
64enum
65{
66 PM1a_EVT_OFFSET = 0x00,
67 PM1b_EVT_OFFSET = -1, /**< not supported */
68 PM1a_CTL_OFFSET = 0x04,
69 PM1b_CTL_OFFSET = -1, /**< not supported */
70 PM2_CTL_OFFSET = -1, /**< not supported */
71 PM_TMR_OFFSET = 0x08,
72 GPE0_OFFSET = 0x20,
73 GPE1_OFFSET = -1 /**< not supported */
74};
75
76#define BAT_INDEX 0x00004040
77#define BAT_DATA 0x00004044
78#define SYSI_INDEX 0x00004048
79#define SYSI_DATA 0x0000404c
80#define ACPI_RESET_BLK 0x00004050
81
82/* PM1x status register bits */
83#define TMR_STS RT_BIT(0)
84#define RSR1_STS (RT_BIT(1) | RT_BIT(2) | RT_BIT(3))
85#define BM_STS RT_BIT(4)
86#define GBL_STS RT_BIT(5)
87#define RSR2_STS (RT_BIT(6) | RT_BIT(7))
88#define PWRBTN_STS RT_BIT(8)
89#define SLPBTN_STS RT_BIT(9)
90#define RTC_STS RT_BIT(10)
91#define IGN_STS RT_BIT(11)
92#define RSR3_STS (RT_BIT(12) | RT_BIT(13) | RT_BIT(14))
93#define WAK_STS RT_BIT(15)
94#define RSR_STS (RSR1_STS | RSR2_STS | RSR3_STS)
95
96/* PM1x enable register bits */
97#define TMR_EN RT_BIT(0)
98#define RSR1_EN (RT_BIT(1) | RT_BIT(2) | RT_BIT(3) | RT_BIT(4))
99#define GBL_EN RT_BIT(5)
100#define RSR2_EN (RT_BIT(6) | RT_BIT(7))
101#define PWRBTN_EN RT_BIT(8)
102#define SLPBTN_EN RT_BIT(9)
103#define RTC_EN RT_BIT(10)
104#define RSR3_EN (RT_BIT(11) | RT_BIT(12) | RT_BIT(13) | RT_BIT(14) | RT_BIT(15))
105#define RSR_EN (RSR1_EN | RSR2_EN | RSR3_EN)
106#define IGN_EN 0
107
108/* PM1x control register bits */
109#define SCI_EN RT_BIT(0)
110#define BM_RLD RT_BIT(1)
111#define GBL_RLS RT_BIT(2)
112#define RSR1_CNT (RT_BIT(3) | RT_BIT(4) | RT_BIT(5) | RT_BIT(6) | RT_BIT(7) | RT_BIT(8))
113#define IGN_CNT RT_BIT(9)
114#define SLP_TYPx_SHIFT 10
115#define SLP_TYPx_MASK 7
116#define SLP_EN RT_BIT(13)
117#define RSR2_CNT (RT_BIT(14) | RT_BIT(15))
118#define RSR_CNT (RSR1_CNT | RSR2_CNT)
119
120#define GPE0_BATTERY_INFO_CHANGED RT_BIT(0)
121
122enum
123{
124 BAT_STATUS_STATE = 0x00, /**< BST battery state */
125 BAT_STATUS_PRESENT_RATE = 0x01, /**< BST battery present rate */
126 BAT_STATUS_REMAINING_CAPACITY = 0x02, /**< BST battery remaining capacity */
127 BAT_STATUS_PRESENT_VOLTAGE = 0x03, /**< BST battery present voltage */
128 BAT_INFO_UNITS = 0x04, /**< BIF power unit */
129 BAT_INFO_DESIGN_CAPACITY = 0x05, /**< BIF design capacity */
130 BAT_INFO_LAST_FULL_CHARGE_CAPACITY = 0x06, /**< BIF last full charge capacity */
131 BAT_INFO_TECHNOLOGY = 0x07, /**< BIF battery technology */
132 BAT_INFO_DESIGN_VOLTAGE = 0x08, /**< BIF design voltage */
133 BAT_INFO_DESIGN_CAPACITY_OF_WARNING = 0x09, /**< BIF design capacity of warning */
134 BAT_INFO_DESIGN_CAPACITY_OF_LOW = 0x0A, /**< BIF design capacity of low */
135 BAT_INFO_CAPACITY_GRANULARITY_1 = 0x0B, /**< BIF battery capacity granularity 1 */
136 BAT_INFO_CAPACITY_GRANULARITY_2 = 0x0C, /**< BIF battery capacity granularity 2 */
137 BAT_DEVICE_STATUS = 0x0D, /**< STA device status */
138 BAT_POWER_SOURCE = 0x0E, /**< PSR power source */
139 BAT_INDEX_LAST
140};
141
142enum
143{
144 CPU_EVENT_TYPE_ADD = 0x01, /**< Event type add */
145 CPU_EVENT_TYPE_REMOVE = 0x03 /**< Event type remove */
146};
147
148enum
149{
150 SYSTEM_INFO_INDEX_LOW_MEMORY_LENGTH = 0,
151 SYSTEM_INFO_INDEX_USE_IOAPIC = 1,
152 SYSTEM_INFO_INDEX_HPET_STATUS = 2,
153 SYSTEM_INFO_INDEX_SMC_STATUS = 3,
154 SYSTEM_INFO_INDEX_FDC_STATUS = 4,
155 SYSTEM_INFO_INDEX_CPU0_STATUS = 5,
156 SYSTEM_INFO_INDEX_CPU1_STATUS = 6,
157 SYSTEM_INFO_INDEX_CPU2_STATUS = 7,
158 SYSTEM_INFO_INDEX_CPU3_STATUS = 8,
159 SYSTEM_INFO_INDEX_HIGH_MEMORY_LENGTH= 9,
160 SYSTEM_INFO_INDEX_RTC_STATUS = 10,
161 SYSTEM_INFO_INDEX_CPU_LOCKED = 11, /**< Contains a flag indicating whether the CPU is locked or not */
162 SYSTEM_INFO_INDEX_CPU_LOCK_CHECK = 12, /**< For which CPU the lock status should be checked */
163 SYSTEM_INFO_INDEX_CPU_EVENT_TYPE = 13, /**< Type of the CPU hot-plug event */
164 SYSTEM_INFO_INDEX_CPU_EVENT = 14, /**< The CPU id the event is for */
165 SYSTEM_INFO_INDEX_END = 15,
166 SYSTEM_INFO_INDEX_INVALID = 0x80,
167 SYSTEM_INFO_INDEX_VALID = 0x200
168};
169
170#define AC_OFFLINE 0
171#define AC_ONLINE 1
172
173#define BAT_TECH_PRIMARY 1
174#define BAT_TECH_SECONDARY 2
175
176#define STA_DEVICE_PRESENT_MASK RT_BIT(0) /**< present */
177#define STA_DEVICE_ENABLED_MASK RT_BIT(1) /**< enabled and decodes its resources */
178#define STA_DEVICE_SHOW_IN_UI_MASK RT_BIT(2) /**< should be shown in UI */
179#define STA_DEVICE_FUNCTIONING_PROPERLY_MASK RT_BIT(3) /**< functioning properly */
180#define STA_BATTERY_PRESENT_MASK RT_BIT(4) /**< the battery is present */
181
182
183/*******************************************************************************
184* Structures and Typedefs *
185*******************************************************************************/
186/**
187 * The ACPI device state.
188 */
189typedef struct ACPIState
190{
191 PCIDevice dev;
192 uint16_t pm1a_en;
193 uint16_t pm1a_sts;
194 uint16_t pm1a_ctl;
195 /** Number of logical CPUs in guest */
196 uint16_t cCpus;
197 int64_t pm_timer_initial;
198 PTMTIMERR3 tsR3;
199 PTMTIMERR0 tsR0;
200 PTMTIMERRC tsRC;
201
202 uint32_t gpe0_en;
203 uint32_t gpe0_sts;
204
205 unsigned int uBatteryIndex;
206 uint32_t au8BatteryInfo[13];
207
208 unsigned int uSystemInfoIndex;
209 uint64_t u64RamSize;
210 /** The number of bytes above 4GB. */
211 uint64_t cbRamHigh;
212 /** The number of bytes below 4GB. */
213 uint32_t cbRamLow;
214
215 /** Current ACPI S* state. We support S0 and S5 */
216 uint32_t uSleepState;
217 uint8_t au8RSDPPage[0x1000];
218 /** This is a workaround for incorrect index field handling by Intels ACPICA.
219 * The system info _INI method writes to offset 0x200. We either observe a
220 * write request to index 0x80 (in that case we don't change the index) or a
221 * write request to offset 0x200 (in that case we divide the index value by
222 * 4. Note that the _STA method is sometimes called prior to the _INI method
223 * (ACPI spec 6.3.7, _STA). See the special case for BAT_DEVICE_STATUS in
224 * acpiBatIndexWrite() for handling this. */
225 uint8_t u8IndexShift;
226 /** provide an I/O-APIC */
227 uint8_t u8UseIOApic;
228 /** provide a floppy controller */
229 bool fUseFdc;
230 /** If High Precision Event Timer device should be supported */
231 bool fUseHpet;
232 /** If System Management Controller device should be supported */
233 bool fUseSmc;
234 /** the guest handled the last power button event */
235 bool fPowerButtonHandled;
236 /** If ACPI CPU device should be shown */
237 bool fShowCpu;
238 /** If Real Time Clock ACPI object to be shown */
239 bool fShowRtc;
240 /** I/O port address of PM device. */
241 RTIOPORT uPmIoPortBase;
242 /** Flag whether the GC part of the device is enabled. */
243 bool fGCEnabled;
244 /** Flag whether the R0 part of the device is enabled. */
245 bool fR0Enabled;
246 /** Array of flags of attached CPUs */
247 VMCPUSET CpuSetAttached;
248 /** Which CPU to check for the locked status. */
249 uint32_t idCpuLockCheck;
250 /** Mask of locked CPUs (used by the guest) */
251 VMCPUSET CpuSetLocked;
252 /** The CPU event type */
253 uint32_t u32CpuEventType;
254 /** The CPU id affected */
255 uint32_t u32CpuEvent;
256 /** Flag whether CPU hot plugging is enabled */
257 bool fCpuHotPlug;
258 /** Aligning IBase. */
259 bool afAlignment[4];
260
261 /** ACPI port base interface. */
262 PDMIBASE IBase;
263 /** ACPI port interface. */
264 PDMIACPIPORT IACPIPort;
265 /** Pointer to the device instance. */
266 PPDMDEVINSR3 pDevIns;
267 /** Pointer to the driver base interface */
268 R3PTRTYPE(PPDMIBASE) pDrvBase;
269 /** Pointer to the driver connector interface */
270 R3PTRTYPE(PPDMIACPICONNECTOR) pDrv;
271
272 /* Pointer to default PCI config read function */
273 R3PTRTYPE(PFNPCICONFIGREAD) pfnAcpiPciConfigRead;
274 /* Pointer to default PCI config write function */
275 R3PTRTYPE(PFNPCICONFIGWRITE) pfnAcpiPciConfigWrite;
276} ACPIState;
277
278#pragma pack(1)
279
280/** Generic Address Structure (see ACPIspec 3.0, 5.2.3.1) */
281struct ACPIGENADDR
282{
283 uint8_t u8AddressSpaceId; /**< 0=sys, 1=IO, 2=PCICfg, 3=emb, 4=SMBus */
284 uint8_t u8RegisterBitWidth; /**< size in bits of the given register */
285 uint8_t u8RegisterBitOffset; /**< bit offset of register */
286 uint8_t u8AccessSize; /**< 1=byte, 2=word, 3=dword, 4=qword */
287 uint64_t u64Address; /**< 64-bit address of register */
288};
289AssertCompileSize(ACPIGENADDR, 12);
290
291/** Root System Description Pointer */
292struct ACPITBLRSDP
293{
294 uint8_t au8Signature[8]; /**< 'RSD PTR ' */
295 uint8_t u8Checksum; /**< checksum for the first 20 bytes */
296 uint8_t au8OemId[6]; /**< OEM-supplied identifier */
297 uint8_t u8Revision; /**< revision number, currently 2 */
298#define ACPI_REVISION 2 /**< ACPI 3.0 */
299 uint32_t u32RSDT; /**< phys addr of RSDT */
300 uint32_t u32Length; /**< bytes of this table */
301 uint64_t u64XSDT; /**< 64-bit phys addr of XSDT */
302 uint8_t u8ExtChecksum; /**< checksum of entire table */
303 uint8_t u8Reserved[3]; /**< reserved */
304};
305AssertCompileSize(ACPITBLRSDP, 36);
306
307/** System Description Table Header */
308struct ACPITBLHEADER
309{
310 uint8_t au8Signature[4]; /**< table identifier */
311 uint32_t u32Length; /**< length of the table including header */
312 uint8_t u8Revision; /**< revision number */
313 uint8_t u8Checksum; /**< all fields inclusive this add to zero */
314 uint8_t au8OemId[6]; /**< OEM-supplied string */
315 uint8_t au8OemTabId[8]; /**< to identify the particular data table */
316 uint32_t u32OemRevision; /**< OEM-supplied revision number */
317 uint8_t au8CreatorId[4]; /**< ID for the ASL compiler */
318 uint32_t u32CreatorRev; /**< revision for the ASL compiler */
319};
320AssertCompileSize(ACPITBLHEADER, 36);
321
322/** Root System Description Table */
323struct ACPITBLRSDT
324{
325 ACPITBLHEADER header;
326 uint32_t u32Entry[1]; /**< array of phys. addresses to other tables */
327};
328AssertCompileSize(ACPITBLRSDT, 40);
329
330/** Extended System Description Table */
331struct ACPITBLXSDT
332{
333 ACPITBLHEADER header;
334 uint64_t u64Entry[1]; /**< array of phys. addresses to other tables */
335};
336AssertCompileSize(ACPITBLXSDT, 44);
337
338/** Fixed ACPI Description Table */
339struct ACPITBLFADT
340{
341 ACPITBLHEADER header;
342 uint32_t u32FACS; /**< phys. address of FACS */
343 uint32_t u32DSDT; /**< phys. address of DSDT */
344 uint8_t u8IntModel; /**< was eleminated in ACPI 2.0 */
345#define INT_MODEL_DUAL_PIC 1 /**< for ACPI 2+ */
346#define INT_MODEL_MULTIPLE_APIC 2
347 uint8_t u8PreferredPMProfile; /**< preferred power management profile */
348 uint16_t u16SCIInt; /**< system vector the SCI is wired in 8259 mode */
349#define SCI_INT 9
350 uint32_t u32SMICmd; /**< system port address of SMI command port */
351#define SMI_CMD 0x0000442e
352 uint8_t u8AcpiEnable; /**< SMICmd val to disable ownship of ACPIregs */
353#define ACPI_ENABLE 0xa1
354 uint8_t u8AcpiDisable; /**< SMICmd val to re-enable ownship of ACPIregs */
355#define ACPI_DISABLE 0xa0
356 uint8_t u8S4BIOSReq; /**< SMICmd val to enter S4BIOS state */
357 uint8_t u8PStateCnt; /**< SMICmd val to assume processor performance
358 state control responsibility */
359 uint32_t u32PM1aEVTBLK; /**< port addr of PM1a event regs block */
360 uint32_t u32PM1bEVTBLK; /**< port addr of PM1b event regs block */
361 uint32_t u32PM1aCTLBLK; /**< port addr of PM1a control regs block */
362 uint32_t u32PM1bCTLBLK; /**< port addr of PM1b control regs block */
363 uint32_t u32PM2CTLBLK; /**< port addr of PM2 control regs block */
364 uint32_t u32PMTMRBLK; /**< port addr of PMTMR regs block */
365 uint32_t u32GPE0BLK; /**< port addr of gen-purp event 0 regs block */
366 uint32_t u32GPE1BLK; /**< port addr of gen-purp event 1 regs block */
367 uint8_t u8PM1EVTLEN; /**< bytes decoded by PM1a_EVT_BLK. >= 4 */
368 uint8_t u8PM1CTLLEN; /**< bytes decoded by PM1b_CNT_BLK. >= 2 */
369 uint8_t u8PM2CTLLEN; /**< bytes decoded by PM2_CNT_BLK. >= 1 or 0 */
370 uint8_t u8PMTMLEN; /**< bytes decoded by PM_TMR_BLK. ==4 */
371 uint8_t u8GPE0BLKLEN; /**< bytes decoded by GPE0_BLK. %2==0 */
372#define GPE0_BLK_LEN 2
373 uint8_t u8GPE1BLKLEN; /**< bytes decoded by GPE1_BLK. %2==0 */
374#define GPE1_BLK_LEN 0
375 uint8_t u8GPE1BASE; /**< offset of GPE1 based events */
376#define GPE1_BASE 0
377 uint8_t u8CSTCNT; /**< SMICmd val to indicate OS supp for C states */
378 uint16_t u16PLVL2LAT; /**< us to enter/exit C2. >100 => unsupported */
379#define P_LVL2_LAT 101 /**< C2 state not supported */
380 uint16_t u16PLVL3LAT; /**< us to enter/exit C3. >1000 => unsupported */
381#define P_LVL3_LAT 1001 /**< C3 state not supported */
382 uint16_t u16FlushSize; /**< # of flush strides to read to flush dirty
383 lines from any processors memory caches */
384#define FLUSH_SIZE 0 /**< Ignored if WBVIND set in FADT_FLAGS */
385 uint16_t u16FlushStride; /**< cache line width */
386#define FLUSH_STRIDE 0 /**< Ignored if WBVIND set in FADT_FLAGS */
387 uint8_t u8DutyOffset;
388 uint8_t u8DutyWidth;
389 uint8_t u8DayAlarm; /**< RTC CMOS RAM index of day-of-month alarm */
390 uint8_t u8MonAlarm; /**< RTC CMOS RAM index of month-of-year alarm */
391 uint8_t u8Century; /**< RTC CMOS RAM index of century */
392 uint16_t u16IAPCBOOTARCH; /**< IA-PC boot architecture flags */
393#define IAPC_BOOT_ARCH_LEGACY_DEV RT_BIT(0) /**< legacy devices present such as LPT
394 (COM too?) */
395#define IAPC_BOOT_ARCH_8042 RT_BIT(1) /**< legacy keyboard device present */
396#define IAPC_BOOT_ARCH_NO_VGA RT_BIT(2) /**< VGA not present */
397 uint8_t u8Must0_0; /**< must be 0 */
398 uint32_t u32Flags; /**< fixed feature flags */
399#define FADT_FL_WBINVD RT_BIT(0) /**< emulation of WBINVD available */
400#define FADT_FL_WBINVD_FLUSH RT_BIT(1)
401#define FADT_FL_PROC_C1 RT_BIT(2) /**< 1=C1 supported on all processors */
402#define FADT_FL_P_LVL2_UP RT_BIT(3) /**< 1=C2 works on SMP and UNI systems */
403#define FADT_FL_PWR_BUTTON RT_BIT(4) /**< 1=power button handled as ctrl method dev */
404#define FADT_FL_SLP_BUTTON RT_BIT(5) /**< 1=sleep button handled as ctrl method dev */
405#define FADT_FL_FIX_RTC RT_BIT(6) /**< 0=RTC wake status in fixed register */
406#define FADT_FL_RTC_S4 RT_BIT(7) /**< 1=RTC can wake system from S4 */
407#define FADT_FL_TMR_VAL_EXT RT_BIT(8) /**< 1=TMR_VAL implemented as 32 bit */
408#define FADT_FL_DCK_CAP RT_BIT(9) /**< 0=system cannot support docking */
409#define FADT_FL_RESET_REG_SUP RT_BIT(10) /**< 1=system supports system resets */
410#define FADT_FL_SEALED_CASE RT_BIT(11) /**< 1=case is sealed */
411#define FADT_FL_HEADLESS RT_BIT(12) /**< 1=system cannot detect moni/keyb/mouse */
412#define FADT_FL_CPU_SW_SLP RT_BIT(13)
413#define FADT_FL_PCI_EXT_WAK RT_BIT(14) /**< 1=system supports PCIEXP_WAKE_STS */
414#define FADT_FL_USE_PLATFORM_CLOCK RT_BIT(15) /**< 1=system has ACPI PM timer */
415#define FADT_FL_S4_RTC_STS_VALID RT_BIT(16) /**< 1=RTC_STS flag is valid when waking from S4 */
416#define FADT_FL_REMOVE_POWER_ON_CAPABLE RT_BIT(17) /**< 1=platform can remote power on */
417#define FADT_FL_FORCE_APIC_CLUSTER_MODEL RT_BIT(18)
418#define FADT_FL_FORCE_APIC_PHYS_DEST_MODE RT_BIT(19)
419
420 /** Start of the ACPI 2.0 extension. */
421 ACPIGENADDR ResetReg; /**< ext addr of reset register */
422 uint8_t u8ResetVal; /**< ResetReg value to reset the system */
423#define ACPI_RESET_REG_VAL 0x10
424 uint8_t au8Must0_1[3]; /**< must be 0 */
425 uint64_t u64XFACS; /**< 64-bit phys address of FACS */
426 uint64_t u64XDSDT; /**< 64-bit phys address of DSDT */
427 ACPIGENADDR X_PM1aEVTBLK; /**< ext addr of PM1a event regs block */
428 ACPIGENADDR X_PM1bEVTBLK; /**< ext addr of PM1b event regs block */
429 ACPIGENADDR X_PM1aCTLBLK; /**< ext addr of PM1a control regs block */
430 ACPIGENADDR X_PM1bCTLBLK; /**< ext addr of PM1b control regs block */
431 ACPIGENADDR X_PM2CTLBLK; /**< ext addr of PM2 control regs block */
432 ACPIGENADDR X_PMTMRBLK; /**< ext addr of PMTMR control regs block */
433 ACPIGENADDR X_GPE0BLK; /**< ext addr of GPE1 regs block */
434 ACPIGENADDR X_GPE1BLK; /**< ext addr of GPE1 regs block */
435};
436AssertCompileSize(ACPITBLFADT, 244);
437#define ACPITBLFADT_VERSION1_SIZE RT_OFFSETOF(ACPITBLFADT, ResetReg)
438
439/** Firmware ACPI Control Structure */
440struct ACPITBLFACS
441{
442 uint8_t au8Signature[4]; /**< 'FACS' */
443 uint32_t u32Length; /**< bytes of entire FACS structure >= 64 */
444 uint32_t u32HWSignature; /**< systems HW signature at last boot */
445 uint32_t u32FWVector; /**< address of waking vector */
446 uint32_t u32GlobalLock; /**< global lock to sync HW/SW */
447 uint32_t u32Flags; /**< FACS flags */
448 uint64_t u64X_FWVector; /**< 64-bit waking vector */
449 uint8_t u8Version; /**< version of this table */
450 uint8_t au8Reserved[31]; /**< zero */
451};
452AssertCompileSize(ACPITBLFACS, 64);
453
454/** Processor Local APIC Structure */
455struct ACPITBLLAPIC
456{
457 uint8_t u8Type; /**< 0 = LAPIC */
458 uint8_t u8Length; /**< 8 */
459 uint8_t u8ProcId; /**< processor ID */
460 uint8_t u8ApicId; /**< local APIC ID */
461 uint32_t u32Flags; /**< Flags */
462#define LAPIC_ENABLED 0x1
463};
464AssertCompileSize(ACPITBLLAPIC, 8);
465
466/** I/O APIC Structure */
467struct ACPITBLIOAPIC
468{
469 uint8_t u8Type; /**< 1 == I/O APIC */
470 uint8_t u8Length; /**< 12 */
471 uint8_t u8IOApicId; /**< I/O APIC ID */
472 uint8_t u8Reserved; /**< 0 */
473 uint32_t u32Address; /**< phys address to access I/O APIC */
474 uint32_t u32GSIB; /**< global system interrupt number to start */
475};
476AssertCompileSize(ACPITBLIOAPIC, 12);
477
478
479/** HPET Descriptor Structure */
480struct ACPITBLHPET
481{
482 ACPITBLHEADER aHeader;
483 uint32_t u32Id; /**< hardware ID of event timer block
484 [31:16] PCI vendor ID of first timer block
485 [15] legacy replacement IRQ routing capable
486 [14] reserved
487 [13] COUNT_SIZE_CAP counter size
488 [12:8] number of comparators in first timer block
489 [7:0] hardware rev ID */
490 ACPIGENADDR HpetAddr; /**< lower 32-bit base address */
491 uint8_t u32Number; /**< sequence number starting at 0 */
492 uint16_t u32MinTick; /**< minimum clock ticks which can be set without
493 lost interrupts while the counter is programmed
494 to operate in periodic mode. Unit: clock tick. */
495 uint8_t u8Attributes; /**< page protextion and OEM attribute. */
496};
497AssertCompileSize(ACPITBLHPET, 56);
498
499# ifdef IN_RING3 /** @todo r=bird: Move this down to where it's used. */
500
501# define PCAT_COMPAT 0x1 /**< system has also a dual-8259 setup */
502
503/**
504 * Multiple APIC Description Table.
505 *
506 * This structure looks somewhat convoluted due layout of MADT table in MP case.
507 * There extpected to be multiple LAPIC records for each CPU, thus we cannot
508 * use regular C structure and proxy to raw memory instead.
509 */
510class AcpiTableMADT
511{
512 /**
513 * All actual data stored in dynamically allocated memory pointed by this field.
514 */
515 uint8_t *m_pbData;
516 /**
517 * Number of CPU entries in this MADT.
518 */
519 uint32_t m_cCpus;
520
521public:
522 /**
523 * Address of ACPI header
524 */
525 inline ACPITBLHEADER *header_addr(void) const
526 {
527 return (ACPITBLHEADER *)m_pbData;
528 }
529
530 /**
531 * Address of local APIC for each CPU. Note that different CPUs address different LAPICs,
532 * although address is the same for all of them.
533 */
534 inline uint32_t *u32LAPIC_addr(void) const
535 {
536 return (uint32_t *)(header_addr() + 1);
537 }
538
539 /**
540 * Address of APIC flags
541 */
542 inline uint32_t *u32Flags_addr(void) const
543 {
544 return (uint32_t *)(u32LAPIC_addr() + 1);
545 }
546
547 /**
548 * Address of per-CPU LAPIC descriptions
549 */
550 inline ACPITBLLAPIC *LApics_addr(void) const
551 {
552 return (ACPITBLLAPIC *)(u32Flags_addr() + 1);
553 }
554
555 /**
556 * Address of IO APIC description
557 */
558 inline ACPITBLIOAPIC *IOApic_addr(void) const
559 {
560 return (ACPITBLIOAPIC *)(LApics_addr() + m_cCpus);
561 }
562
563 /**
564 * Size of MADT.
565 * Note that this function assumes IOApic to be the last field in structure.
566 */
567 inline uint32_t size(void) const
568 {
569 return (uint8_t *)(IOApic_addr() + 1) - (uint8_t *)header_addr();
570 }
571
572 /**
573 * Raw data of MADT.
574 */
575 inline const uint8_t *data(void) const
576 {
577 return m_pbData;
578 }
579
580 /**
581 * Size of MADT for given ACPI config, useful to compute layout.
582 */
583 static uint32_t sizeFor(ACPIState *s)
584 {
585 return AcpiTableMADT(s->cCpus).size();
586 }
587
588 /*
589 * Constructor, only works in Ring 3, doesn't look like a big deal.
590 */
591 AcpiTableMADT(uint32_t cCpus)
592 {
593 m_cCpus = cCpus;
594 m_pbData = NULL; /* size() uses this and gcc will complain if not initilized. */
595 uint32_t cb = size();
596 m_pbData = (uint8_t *)RTMemAllocZ(cb);
597 }
598
599 ~AcpiTableMADT()
600 {
601 RTMemFree(m_pbData);
602 }
603};
604# endif /* IN_RING3 */
605
606#pragma pack()
607
608
609#ifndef VBOX_DEVICE_STRUCT_TESTCASE /* exclude the rest of the file */
610/*******************************************************************************
611* Internal Functions *
612*******************************************************************************/
613RT_C_DECLS_BEGIN
614PDMBOTHCBDECL(int) acpiPMTmrRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
615#ifdef IN_RING3
616PDMBOTHCBDECL(int) acpiPm1aEnRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
617PDMBOTHCBDECL(int) acpiPM1aEnWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
618PDMBOTHCBDECL(int) acpiPm1aStsRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
619PDMBOTHCBDECL(int) acpiPM1aStsWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
620PDMBOTHCBDECL(int) acpiPm1aCtlRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
621PDMBOTHCBDECL(int) acpiPM1aCtlWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
622PDMBOTHCBDECL(int) acpiSmiWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
623PDMBOTHCBDECL(int) acpiBatIndexWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
624PDMBOTHCBDECL(int) acpiBatDataRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
625PDMBOTHCBDECL(int) acpiSysInfoDataRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
626PDMBOTHCBDECL(int) acpiSysInfoDataWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
627PDMBOTHCBDECL(int) acpiGpe0EnRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
628PDMBOTHCBDECL(int) acpiGpe0EnWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
629PDMBOTHCBDECL(int) acpiGpe0StsRead( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
630PDMBOTHCBDECL(int) acpiGpe0StsWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
631PDMBOTHCBDECL(int) acpiResetWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
632# ifdef DEBUG_ACPI
633PDMBOTHCBDECL(int) acpiDhexWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
634PDMBOTHCBDECL(int) acpiDchrWrite( PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
635# endif
636#endif /* IN_RING3 */
637RT_C_DECLS_END
638
639
640#ifdef IN_RING3
641
642static RTIOPORT acpiPmPort(ACPIState* pAcpi, int32_t offset)
643{
644 Assert(pAcpi->uPmIoPortBase != 0);
645
646 if (offset == -1)
647 return 0;
648
649 return RTIOPORT(pAcpi->uPmIoPortBase + offset);
650}
651
652/* Simple acpiChecksum: all the bytes must add up to 0. */
653static uint8_t acpiChecksum(const uint8_t * const data, size_t len)
654{
655 uint8_t sum = 0;
656 for (size_t i = 0; i < len; ++i)
657 sum += data[i];
658 return -sum;
659}
660
661static void acpiPrepareHeader(ACPITBLHEADER *header, const char au8Signature[4],
662 uint32_t u32Length, uint8_t u8Revision)
663{
664 memcpy(header->au8Signature, au8Signature, 4);
665 header->u32Length = RT_H2LE_U32(u32Length);
666 header->u8Revision = u8Revision;
667 memcpy(header->au8OemId, "VBOX ", 6);
668 memcpy(header->au8OemTabId, "VBOX", 4);
669 memcpy(header->au8OemTabId+4, au8Signature, 4);
670 header->u32OemRevision = RT_H2LE_U32(1);
671 memcpy(header->au8CreatorId, "ASL ", 4);
672 header->u32CreatorRev = RT_H2LE_U32(0x61);
673}
674
675static void acpiWriteGenericAddr(ACPIGENADDR *g, uint8_t u8AddressSpaceId,
676 uint8_t u8RegisterBitWidth, uint8_t u8RegisterBitOffset,
677 uint8_t u8AccessSize, uint64_t u64Address)
678{
679 g->u8AddressSpaceId = u8AddressSpaceId;
680 g->u8RegisterBitWidth = u8RegisterBitWidth;
681 g->u8RegisterBitOffset = u8RegisterBitOffset;
682 g->u8AccessSize = u8AccessSize;
683 g->u64Address = RT_H2LE_U64(u64Address);
684}
685
686static void acpiPhyscpy(ACPIState *s, RTGCPHYS32 dst, const void * const src, size_t size)
687{
688 PDMDevHlpPhysWrite(s->pDevIns, dst, src, size);
689}
690
691/** Differentiated System Description Table (DSDT) */
692
693static void acpiSetupDSDT(ACPIState *s, RTGCPHYS32 addr,
694 void* pPtr, size_t uDsdtLen)
695{
696 acpiPhyscpy(s, addr, pPtr, uDsdtLen);
697}
698
699/** Secondary System Description Table (SSDT) */
700
701static void acpiSetupSSDT(ACPIState *s, RTGCPHYS32 addr,
702 void* pPtr, size_t uSsdtLen)
703{
704 acpiPhyscpy(s, addr, pPtr, uSsdtLen);
705}
706
707/** Firmware ACPI Control Structure (FACS) */
708static void acpiSetupFACS(ACPIState *s, RTGCPHYS32 addr)
709{
710 ACPITBLFACS facs;
711
712 memset(&facs, 0, sizeof(facs));
713 memcpy(facs.au8Signature, "FACS", 4);
714 facs.u32Length = RT_H2LE_U32(sizeof(ACPITBLFACS));
715 facs.u32HWSignature = RT_H2LE_U32(0);
716 facs.u32FWVector = RT_H2LE_U32(0);
717 facs.u32GlobalLock = RT_H2LE_U32(0);
718 facs.u32Flags = RT_H2LE_U32(0);
719 facs.u64X_FWVector = RT_H2LE_U64(0);
720 facs.u8Version = 1;
721
722 acpiPhyscpy(s, addr, (const uint8_t *)&facs, sizeof(facs));
723}
724
725/** Fixed ACPI Description Table (FADT aka FACP) */
726static void acpiSetupFADT(ACPIState *s, RTGCPHYS32 GCPhysAcpi1, RTGCPHYS32 GCPhysAcpi2, RTGCPHYS32 GCPhysFacs, RTGCPHYS GCPhysDsdt)
727{
728 ACPITBLFADT fadt;
729
730 /* First the ACPI version 2+ version of the structure. */
731 memset(&fadt, 0, sizeof(fadt));
732 acpiPrepareHeader(&fadt.header, "FACP", sizeof(fadt), 4);
733 fadt.u32FACS = RT_H2LE_U32(GCPhysFacs);
734 fadt.u32DSDT = RT_H2LE_U32(GCPhysDsdt);
735 fadt.u8IntModel = 0; /* dropped from the ACPI 2.0 spec. */
736 fadt.u8PreferredPMProfile = 0; /* unspecified */
737 fadt.u16SCIInt = RT_H2LE_U16(SCI_INT);
738 fadt.u32SMICmd = RT_H2LE_U32(SMI_CMD);
739 fadt.u8AcpiEnable = ACPI_ENABLE;
740 fadt.u8AcpiDisable = ACPI_DISABLE;
741 fadt.u8S4BIOSReq = 0;
742 fadt.u8PStateCnt = 0;
743 fadt.u32PM1aEVTBLK = RT_H2LE_U32(acpiPmPort(s, PM1a_EVT_OFFSET));
744 fadt.u32PM1bEVTBLK = RT_H2LE_U32(acpiPmPort(s, PM1b_EVT_OFFSET));
745 fadt.u32PM1aCTLBLK = RT_H2LE_U32(acpiPmPort(s, PM1a_CTL_OFFSET));
746 fadt.u32PM1bCTLBLK = RT_H2LE_U32(acpiPmPort(s, PM1b_CTL_OFFSET));
747 fadt.u32PM2CTLBLK = RT_H2LE_U32(acpiPmPort(s, PM2_CTL_OFFSET));
748 fadt.u32PMTMRBLK = RT_H2LE_U32(acpiPmPort(s, PM_TMR_OFFSET));
749 fadt.u32GPE0BLK = RT_H2LE_U32(acpiPmPort(s, GPE0_OFFSET));
750 fadt.u32GPE1BLK = RT_H2LE_U32(acpiPmPort(s, GPE1_OFFSET));
751 fadt.u8PM1EVTLEN = 4;
752 fadt.u8PM1CTLLEN = 2;
753 fadt.u8PM2CTLLEN = 0;
754 fadt.u8PMTMLEN = 4;
755 fadt.u8GPE0BLKLEN = GPE0_BLK_LEN;
756 fadt.u8GPE1BLKLEN = GPE1_BLK_LEN;
757 fadt.u8GPE1BASE = GPE1_BASE;
758 fadt.u8CSTCNT = 0;
759 fadt.u16PLVL2LAT = RT_H2LE_U16(P_LVL2_LAT);
760 fadt.u16PLVL3LAT = RT_H2LE_U16(P_LVL3_LAT);
761 fadt.u16FlushSize = RT_H2LE_U16(FLUSH_SIZE);
762 fadt.u16FlushStride = RT_H2LE_U16(FLUSH_STRIDE);
763 fadt.u8DutyOffset = 0;
764 fadt.u8DutyWidth = 0;
765 fadt.u8DayAlarm = 0;
766 fadt.u8MonAlarm = 0;
767 fadt.u8Century = 0;
768 fadt.u16IAPCBOOTARCH = RT_H2LE_U16(IAPC_BOOT_ARCH_LEGACY_DEV | IAPC_BOOT_ARCH_8042);
769 /** @note WBINVD is required for ACPI versions newer than 1.0 */
770 fadt.u32Flags = RT_H2LE_U32( FADT_FL_WBINVD
771 | FADT_FL_FIX_RTC
772 | FADT_FL_TMR_VAL_EXT);
773
774 /* We have to force physical APIC mode or Linux can't use more than 8 CPUs */
775 if (s->fCpuHotPlug)
776 fadt.u32Flags |= RT_H2LE_U32(FADT_FL_FORCE_APIC_PHYS_DEST_MODE);
777
778 acpiWriteGenericAddr(&fadt.ResetReg, 1, 8, 0, 1, ACPI_RESET_BLK);
779 fadt.u8ResetVal = ACPI_RESET_REG_VAL;
780 fadt.u64XFACS = RT_H2LE_U64((uint64_t)GCPhysFacs);
781 fadt.u64XDSDT = RT_H2LE_U64((uint64_t)GCPhysDsdt);
782 acpiWriteGenericAddr(&fadt.X_PM1aEVTBLK, 1, 32, 0, 2, acpiPmPort(s, PM1a_EVT_OFFSET));
783 acpiWriteGenericAddr(&fadt.X_PM1bEVTBLK, 0, 0, 0, 0, acpiPmPort(s, PM1b_EVT_OFFSET));
784 acpiWriteGenericAddr(&fadt.X_PM1aCTLBLK, 1, 16, 0, 2, acpiPmPort(s, PM1a_CTL_OFFSET));
785 acpiWriteGenericAddr(&fadt.X_PM1bCTLBLK, 0, 0, 0, 0, acpiPmPort(s, PM1b_CTL_OFFSET));
786 acpiWriteGenericAddr(&fadt.X_PM2CTLBLK, 0, 0, 0, 0, acpiPmPort(s, PM2_CTL_OFFSET));
787 acpiWriteGenericAddr(&fadt.X_PMTMRBLK, 1, 32, 0, 3, acpiPmPort(s, PM_TMR_OFFSET));
788 acpiWriteGenericAddr(&fadt.X_GPE0BLK, 1, 16, 0, 1, acpiPmPort(s, GPE0_OFFSET));
789 acpiWriteGenericAddr(&fadt.X_GPE1BLK, 0, 0, 0, 0, acpiPmPort(s, GPE1_OFFSET));
790 fadt.header.u8Checksum = acpiChecksum((uint8_t *)&fadt, sizeof(fadt));
791 acpiPhyscpy(s, GCPhysAcpi2, &fadt, sizeof(fadt));
792
793 /* Now the ACPI 1.0 version. */
794 fadt.header.u32Length = ACPITBLFADT_VERSION1_SIZE;
795 fadt.u8IntModel = INT_MODEL_DUAL_PIC;
796 fadt.header.u8Checksum = 0; /* Must be zeroed before recalculating checksum! */
797 fadt.header.u8Checksum = acpiChecksum((uint8_t *)&fadt, ACPITBLFADT_VERSION1_SIZE);
798 acpiPhyscpy(s, GCPhysAcpi1, &fadt, ACPITBLFADT_VERSION1_SIZE);
799}
800
801/**
802 * Root System Description Table.
803 * The RSDT and XSDT tables are basically identical. The only difference is 32 vs 64 bits
804 * addresses for description headers. RSDT is for ACPI 1.0. XSDT for ACPI 2.0 and up.
805 */
806static int acpiSetupRSDT(ACPIState *s, RTGCPHYS32 addr, unsigned int nb_entries, uint32_t *addrs)
807{
808 ACPITBLRSDT *rsdt;
809 const size_t size = sizeof(ACPITBLHEADER) + nb_entries * sizeof(rsdt->u32Entry[0]);
810
811 rsdt = (ACPITBLRSDT*)RTMemAllocZ(size);
812 if (!rsdt)
813 return PDMDEV_SET_ERROR(s->pDevIns, VERR_NO_TMP_MEMORY, N_("Cannot allocate RSDT"));
814
815 acpiPrepareHeader(&rsdt->header, "RSDT", (uint32_t)size, 1);
816 for (unsigned int i = 0; i < nb_entries; ++i)
817 {
818 rsdt->u32Entry[i] = RT_H2LE_U32(addrs[i]);
819 Log(("Setup RSDT: [%d] = %x\n", i, rsdt->u32Entry[i]));
820 }
821 rsdt->header.u8Checksum = acpiChecksum((uint8_t*)rsdt, size);
822 acpiPhyscpy(s, addr, rsdt, size);
823 RTMemFree(rsdt);
824 return VINF_SUCCESS;
825}
826
827/** Extended System Description Table. */
828static int acpiSetupXSDT(ACPIState *s, RTGCPHYS32 addr, unsigned int nb_entries, uint32_t *addrs)
829{
830 ACPITBLXSDT *xsdt;
831 const size_t size = sizeof(ACPITBLHEADER) + nb_entries * sizeof(xsdt->u64Entry[0]);
832
833 xsdt = (ACPITBLXSDT*)RTMemAllocZ(size);
834 if (!xsdt)
835 return VERR_NO_TMP_MEMORY;
836
837 acpiPrepareHeader(&xsdt->header, "XSDT", (uint32_t)size, 1 /* according to ACPI 3.0 specs */);
838 for (unsigned int i = 0; i < nb_entries; ++i)
839 {
840 xsdt->u64Entry[i] = RT_H2LE_U64((uint64_t)addrs[i]);
841 Log(("Setup XSDT: [%d] = %RX64\n", i, xsdt->u64Entry[i]));
842 }
843 xsdt->header.u8Checksum = acpiChecksum((uint8_t*)xsdt, size);
844 acpiPhyscpy(s, addr, xsdt, size);
845 RTMemFree(xsdt);
846 return VINF_SUCCESS;
847}
848
849/** Root System Description Pointer (RSDP) */
850static void acpiSetupRSDP(ACPITBLRSDP *rsdp, RTGCPHYS32 GCPhysRsdt, RTGCPHYS GCPhysXsdt)
851{
852 memset(rsdp, 0, sizeof(*rsdp));
853
854 /* ACPI 1.0 part (RSDT */
855 memcpy(rsdp->au8Signature, "RSD PTR ", 8);
856 memcpy(rsdp->au8OemId, "VBOX ", 6);
857 rsdp->u8Revision = ACPI_REVISION;
858 rsdp->u32RSDT = RT_H2LE_U32(GCPhysRsdt);
859 rsdp->u8Checksum = acpiChecksum((uint8_t*)rsdp, RT_OFFSETOF(ACPITBLRSDP, u32Length));
860
861 /* ACPI 2.0 part (XSDT) */
862 rsdp->u32Length = RT_H2LE_U32(sizeof(ACPITBLRSDP));
863 rsdp->u64XSDT = RT_H2LE_U64(GCPhysXsdt);
864 rsdp->u8ExtChecksum = acpiChecksum((uint8_t*)rsdp, sizeof(ACPITBLRSDP));
865}
866
867/**
868 * Multiple APIC Description Table.
869 *
870 * @note APIC without IO-APIC hangs Windows Vista therefore we setup both
871 *
872 * @todo All hardcoded, should set this up based on the actual VM config!!!!!
873 */
874static void acpiSetupMADT(ACPIState *s, RTGCPHYS32 addr)
875{
876 uint16_t cpus = s->cCpus;
877 AcpiTableMADT madt(cpus);
878
879 acpiPrepareHeader(madt.header_addr(), "APIC", madt.size(), 2);
880
881 *madt.u32LAPIC_addr() = RT_H2LE_U32(0xfee00000);
882 *madt.u32Flags_addr() = RT_H2LE_U32(PCAT_COMPAT);
883
884 ACPITBLLAPIC* lapic = madt.LApics_addr();
885 for (uint16_t i = 0; i < cpus; i++)
886 {
887 lapic->u8Type = 0;
888 lapic->u8Length = sizeof(ACPITBLLAPIC);
889 lapic->u8ProcId = i;
890 lapic->u8ApicId = i;
891 lapic->u32Flags = VMCPUSET_IS_PRESENT(&s->CpuSetAttached, i) ? RT_H2LE_U32(LAPIC_ENABLED) : 0;
892 lapic++;
893 }
894
895 ACPITBLIOAPIC* ioapic = madt.IOApic_addr();
896
897 ioapic->u8Type = 1;
898 ioapic->u8Length = sizeof(ACPITBLIOAPIC);
899 /** @todo is this the right id? */
900 ioapic->u8IOApicId = cpus;
901 ioapic->u8Reserved = 0;
902 ioapic->u32Address = RT_H2LE_U32(0xfec00000);
903 ioapic->u32GSIB = RT_H2LE_U32(0);
904
905 madt.header_addr()->u8Checksum = acpiChecksum(madt.data(), madt.size());
906 acpiPhyscpy(s, addr, madt.data(), madt.size());
907}
908
909
910/** High Performance Event Timer (HPET) descriptor */
911static void acpiSetupHPET(ACPIState *s, RTGCPHYS32 addr)
912{
913 ACPITBLHPET hpet;
914
915 memset(&hpet, 0, sizeof(hpet));
916
917 acpiPrepareHeader(&hpet.aHeader, "HPET", sizeof(hpet), 1);
918 /* Keep base address consistent with appropriate DSDT entry (vbox.dsl) */
919 acpiWriteGenericAddr(&hpet.HpetAddr,
920 0 /* Memory address space */,
921 64 /* Register bit width */,
922 0 /* Bit offset */,
923 0, /* Register access size, is it correct? */
924 0xfed00000 /* Address */);
925
926 hpet.u32Id = 0x8086a201; /* must match what HPET ID returns, is it correct ? */
927 hpet.u32Number = 0;
928 hpet.u32MinTick = 4096;
929 hpet.u8Attributes = 0;
930
931 hpet.aHeader.u8Checksum = acpiChecksum((uint8_t *)&hpet, sizeof(hpet));
932
933 acpiPhyscpy(s, addr, (const uint8_t *)&hpet, sizeof(hpet));
934}
935
936/* SCI IRQ */
937DECLINLINE(void) acpiSetIrq(ACPIState *s, int level)
938{
939 if (s->pm1a_ctl & SCI_EN)
940 PDMDevHlpPCISetIrq(s->pDevIns, -1, level);
941}
942
943DECLINLINE(uint32_t) pm1a_pure_en(uint32_t en)
944{
945 return en & ~(RSR_EN | IGN_EN);
946}
947
948DECLINLINE(uint32_t) pm1a_pure_sts(uint32_t sts)
949{
950 return sts & ~(RSR_STS | IGN_STS);
951}
952
953DECLINLINE(int) pm1a_level(ACPIState *s)
954{
955 return (pm1a_pure_en(s->pm1a_en) & pm1a_pure_sts(s->pm1a_sts)) != 0;
956}
957
958DECLINLINE(int) gpe0_level(ACPIState *s)
959{
960 return (s->gpe0_en & s->gpe0_sts) != 0;
961}
962
963static void update_pm1a(ACPIState *s, uint32_t sts, uint32_t en)
964{
965 int old_level, new_level;
966
967 if (gpe0_level(s))
968 return;
969
970 old_level = pm1a_level(s);
971 new_level = (pm1a_pure_en(en) & pm1a_pure_sts(sts)) != 0;
972
973 s->pm1a_en = en;
974 s->pm1a_sts = sts;
975
976 if (new_level != old_level)
977 acpiSetIrq(s, new_level);
978}
979
980static void update_gpe0(ACPIState *s, uint32_t sts, uint32_t en)
981{
982 int old_level, new_level;
983
984 if (pm1a_level(s))
985 return;
986
987 old_level = (s->gpe0_en & s->gpe0_sts) != 0;
988 new_level = (en & sts) != 0;
989
990 s->gpe0_en = en;
991 s->gpe0_sts = sts;
992
993 if (new_level != old_level)
994 acpiSetIrq(s, new_level);
995}
996
997static int acpiPowerDown(ACPIState *s)
998{
999 int rc = PDMDevHlpVMPowerOff(s->pDevIns);
1000 if (RT_FAILURE(rc))
1001 AssertMsgFailed(("Could not power down the VM. rc = %Rrc\n", rc));
1002 return rc;
1003}
1004
1005/** Converts a ACPI port interface pointer to an ACPI state pointer. */
1006#define IACPIPORT_2_ACPISTATE(pInterface) ( (ACPIState*)((uintptr_t)pInterface - RT_OFFSETOF(ACPIState, IACPIPort)) )
1007
1008/**
1009 * Send an ACPI power off event.
1010 *
1011 * @returns VBox status code
1012 * @param pInterface Pointer to the interface structure containing the called function pointer.
1013 */
1014static DECLCALLBACK(int) acpiPowerButtonPress(PPDMIACPIPORT pInterface)
1015{
1016 ACPIState *s = IACPIPORT_2_ACPISTATE(pInterface);
1017 s->fPowerButtonHandled = false;
1018 update_pm1a(s, s->pm1a_sts | PWRBTN_STS, s->pm1a_en);
1019 return VINF_SUCCESS;
1020}
1021
1022/**
1023 * Check if the ACPI power button event was handled.
1024 *
1025 * @returns VBox status code
1026 * @param pInterface Pointer to the interface structure containing the called function pointer.
1027 * @param pfHandled Return true if the power button event was handled by the guest.
1028 */
1029static DECLCALLBACK(int) acpiGetPowerButtonHandled(PPDMIACPIPORT pInterface, bool *pfHandled)
1030{
1031 ACPIState *s = IACPIPORT_2_ACPISTATE(pInterface);
1032 *pfHandled = s->fPowerButtonHandled;
1033 return VINF_SUCCESS;
1034}
1035
1036/**
1037 * Check if the Guest entered into G0 (working) or G1 (sleeping).
1038 *
1039 * @returns VBox status code
1040 * @param pInterface Pointer to the interface structure containing the called function pointer.
1041 * @param pfEntered Return true if the guest entered the ACPI mode.
1042 */
1043static DECLCALLBACK(int) acpiGetGuestEnteredACPIMode(PPDMIACPIPORT pInterface, bool *pfEntered)
1044{
1045 ACPIState *s = IACPIPORT_2_ACPISTATE(pInterface);
1046 *pfEntered = (s->pm1a_ctl & SCI_EN) != 0;
1047 return VINF_SUCCESS;
1048}
1049
1050static DECLCALLBACK(int) acpiGetCpuStatus(PPDMIACPIPORT pInterface, unsigned uCpu, bool *pfLocked)
1051{
1052 ACPIState *s = IACPIPORT_2_ACPISTATE(pInterface);
1053 *pfLocked = VMCPUSET_IS_PRESENT(&s->CpuSetLocked, uCpu);
1054 return VINF_SUCCESS;
1055}
1056
1057/**
1058 * Send an ACPI sleep button event.
1059 *
1060 * @returns VBox status code
1061 * @param pInterface Pointer to the interface structure containing the called function pointer.
1062 */
1063static DECLCALLBACK(int) acpiSleepButtonPress(PPDMIACPIPORT pInterface)
1064{
1065 ACPIState *s = IACPIPORT_2_ACPISTATE(pInterface);
1066 update_pm1a(s, s->pm1a_sts | SLPBTN_STS, s->pm1a_en);
1067 return VINF_SUCCESS;
1068}
1069
1070/* PM1a_EVT_BLK enable */
1071static uint32_t acpiPm1aEnReadw(ACPIState *s, uint32_t addr)
1072{
1073 uint16_t val = s->pm1a_en;
1074 Log(("acpi: acpiPm1aEnReadw -> %#x\n", val));
1075 return val;
1076}
1077
1078static void acpiPM1aEnWritew(ACPIState *s, uint32_t addr, uint32_t val)
1079{
1080 Log(("acpi: acpiPM1aEnWritew <- %#x (%#x)\n", val, val & ~(RSR_EN | IGN_EN)));
1081 val &= ~(RSR_EN | IGN_EN);
1082 update_pm1a(s, s->pm1a_sts, val);
1083}
1084
1085/* PM1a_EVT_BLK status */
1086static uint32_t acpiPm1aStsReadw(ACPIState *s, uint32_t addr)
1087{
1088 uint16_t val = s->pm1a_sts;
1089 Log(("acpi: acpiPm1aStsReadw -> %#x\n", val));
1090 return val;
1091}
1092
1093static void acpiPM1aStsWritew(ACPIState *s, uint32_t addr, uint32_t val)
1094{
1095 Log(("acpi: acpiPM1aStsWritew <- %#x (%#x)\n", val, val & ~(RSR_STS | IGN_STS)));
1096 if (val & PWRBTN_STS)
1097 s->fPowerButtonHandled = true; /* Remember that the guest handled the last power button event */
1098 val = s->pm1a_sts & ~(val & ~(RSR_STS | IGN_STS));
1099 update_pm1a(s, val, s->pm1a_en);
1100}
1101
1102/* PM1a_CTL_BLK */
1103static uint32_t acpiPm1aCtlReadw(ACPIState *s, uint32_t addr)
1104{
1105 uint16_t val = s->pm1a_ctl;
1106 Log(("acpi: acpiPm1aCtlReadw -> %#x\n", val));
1107 return val;
1108}
1109
1110static int acpiPM1aCtlWritew(ACPIState *s, uint32_t addr, uint32_t val)
1111{
1112 uint32_t uSleepState;
1113
1114 Log(("acpi: acpiPM1aCtlWritew <- %#x (%#x)\n", val, val & ~(RSR_CNT | IGN_CNT)));
1115 s->pm1a_ctl = val & ~(RSR_CNT | IGN_CNT);
1116
1117 uSleepState = (s->pm1a_ctl >> SLP_TYPx_SHIFT) & SLP_TYPx_MASK;
1118 if (uSleepState != s->uSleepState)
1119 {
1120 s->uSleepState = uSleepState;
1121 switch (uSleepState)
1122 {
1123 case 0x00: /* S0 */
1124 break;
1125 case 0x05: /* S5 */
1126 LogRel(("Entering S5 (power down)\n"));
1127 return acpiPowerDown(s);
1128 default:
1129 AssertMsgFailed(("Unknown sleep state %#x\n", uSleepState));
1130 break;
1131 }
1132 }
1133 return VINF_SUCCESS;
1134}
1135
1136/* GPE0_BLK */
1137static uint32_t acpiGpe0EnReadb(ACPIState *s, uint32_t addr)
1138{
1139 uint8_t val = s->gpe0_en;
1140 Log(("acpi: acpiGpe0EnReadl -> %#x\n", val));
1141 return val;
1142}
1143
1144static void acpiGpe0EnWriteb(ACPIState *s, uint32_t addr, uint32_t val)
1145{
1146 Log(("acpi: acpiGpe0EnWritel <- %#x\n", val));
1147 update_gpe0(s, s->gpe0_sts, val);
1148}
1149
1150static uint32_t acpiGpe0StsReadb(ACPIState *s, uint32_t addr)
1151{
1152 uint8_t val = s->gpe0_sts;
1153 Log(("acpi: acpiGpe0StsReadl -> %#x\n", val));
1154 return val;
1155}
1156
1157static void acpiGpe0StsWriteb(ACPIState *s, uint32_t addr, uint32_t val)
1158{
1159 val = s->gpe0_sts & ~val;
1160 update_gpe0(s, val, s->gpe0_en);
1161 Log(("acpi: acpiGpe0StsWritel <- %#x\n", val));
1162}
1163
1164static int acpiResetWriteU8(ACPIState *s, uint32_t addr, uint32_t val)
1165{
1166 int rc = VINF_SUCCESS;
1167
1168 Log(("ACPI: acpiResetWriteU8: %x %x\n", addr, val));
1169 if (val == ACPI_RESET_REG_VAL)
1170 {
1171# ifndef IN_RING3
1172 rc = VINF_IOM_HC_IOPORT_WRITE;
1173# else /* IN_RING3 */
1174 rc = PDMDevHlpVMReset(s->pDevIns);
1175# endif /* !IN_RING3 */
1176 }
1177 return rc;
1178}
1179
1180/* SMI */
1181static void acpiSmiWriteU8(ACPIState *s, uint32_t addr, uint32_t val)
1182{
1183 Log(("acpi: acpiSmiWriteU8 %#x\n", val));
1184 if (val == ACPI_ENABLE)
1185 s->pm1a_ctl |= SCI_EN;
1186 else if (val == ACPI_DISABLE)
1187 s->pm1a_ctl &= ~SCI_EN;
1188 else
1189 Log(("acpi: acpiSmiWriteU8 %#x <- unknown value\n", val));
1190}
1191
1192static uint32_t find_rsdp_space(void)
1193{
1194 return 0xe0000;
1195}
1196
1197static int acpiPMTimerReset(ACPIState *s)
1198{
1199 uint64_t interval, freq;
1200
1201 freq = TMTimerGetFreq(s->CTX_SUFF(ts));
1202 interval = ASMMultU64ByU32DivByU32(0xffffffff, freq, PM_TMR_FREQ);
1203 Log(("interval = %RU64\n", interval));
1204 TMTimerSet(s->CTX_SUFF(ts), TMTimerGet(s->CTX_SUFF(ts)) + interval);
1205
1206 return VINF_SUCCESS;
1207}
1208
1209static DECLCALLBACK(void) acpiTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1210{
1211 ACPIState *s = (ACPIState *)pvUser;
1212
1213 Log(("acpi: pm timer sts %#x (%d), en %#x (%d)\n",
1214 s->pm1a_sts, (s->pm1a_sts & TMR_STS) != 0,
1215 s->pm1a_en, (s->pm1a_en & TMR_EN) != 0));
1216
1217 update_pm1a(s, s->pm1a_sts | TMR_STS, s->pm1a_en);
1218 acpiPMTimerReset(s);
1219}
1220
1221/**
1222 * _BST method.
1223 */
1224static int acpiFetchBatteryStatus(ACPIState *s)
1225{
1226 uint32_t *p = s->au8BatteryInfo;
1227 bool fPresent; /* battery present? */
1228 PDMACPIBATCAPACITY hostRemainingCapacity; /* 0..100 */
1229 PDMACPIBATSTATE hostBatteryState; /* bitfield */
1230 uint32_t hostPresentRate; /* 0..1000 */
1231 int rc;
1232
1233 if (!s->pDrv)
1234 return VINF_SUCCESS;
1235 rc = s->pDrv->pfnQueryBatteryStatus(s->pDrv, &fPresent, &hostRemainingCapacity,
1236 &hostBatteryState, &hostPresentRate);
1237 AssertRC(rc);
1238
1239 /* default values */
1240 p[BAT_STATUS_STATE] = hostBatteryState;
1241 p[BAT_STATUS_PRESENT_RATE] = hostPresentRate == ~0U ? 0xFFFFFFFF
1242 : hostPresentRate * 50; /* mW */
1243 p[BAT_STATUS_REMAINING_CAPACITY] = 50000; /* mWh */
1244 p[BAT_STATUS_PRESENT_VOLTAGE] = 10000; /* mV */
1245
1246 /* did we get a valid battery state? */
1247 if (hostRemainingCapacity != PDM_ACPI_BAT_CAPACITY_UNKNOWN)
1248 p[BAT_STATUS_REMAINING_CAPACITY] = hostRemainingCapacity * 500; /* mWh */
1249 if (hostBatteryState == PDM_ACPI_BAT_STATE_CHARGED)
1250 p[BAT_STATUS_PRESENT_RATE] = 0; /* mV */
1251
1252 return VINF_SUCCESS;
1253}
1254
1255/**
1256 * _BIF method.
1257 */
1258static int acpiFetchBatteryInfo(ACPIState *s)
1259{
1260 uint32_t *p = s->au8BatteryInfo;
1261
1262 p[BAT_INFO_UNITS] = 0; /* mWh */
1263 p[BAT_INFO_DESIGN_CAPACITY] = 50000; /* mWh */
1264 p[BAT_INFO_LAST_FULL_CHARGE_CAPACITY] = 50000; /* mWh */
1265 p[BAT_INFO_TECHNOLOGY] = BAT_TECH_PRIMARY;
1266 p[BAT_INFO_DESIGN_VOLTAGE] = 10000; /* mV */
1267 p[BAT_INFO_DESIGN_CAPACITY_OF_WARNING] = 100; /* mWh */
1268 p[BAT_INFO_DESIGN_CAPACITY_OF_LOW] = 50; /* mWh */
1269 p[BAT_INFO_CAPACITY_GRANULARITY_1] = 1; /* mWh */
1270 p[BAT_INFO_CAPACITY_GRANULARITY_2] = 1; /* mWh */
1271
1272 return VINF_SUCCESS;
1273}
1274
1275/**
1276 * _STA method.
1277 */
1278static uint32_t acpiGetBatteryDeviceStatus(ACPIState *s)
1279{
1280 bool fPresent; /* battery present? */
1281 PDMACPIBATCAPACITY hostRemainingCapacity; /* 0..100 */
1282 PDMACPIBATSTATE hostBatteryState; /* bitfield */
1283 uint32_t hostPresentRate; /* 0..1000 */
1284 int rc;
1285
1286 if (!s->pDrv)
1287 return 0;
1288 rc = s->pDrv->pfnQueryBatteryStatus(s->pDrv, &fPresent, &hostRemainingCapacity,
1289 &hostBatteryState, &hostPresentRate);
1290 AssertRC(rc);
1291
1292 return fPresent
1293 ? STA_DEVICE_PRESENT_MASK /* present */
1294 | STA_DEVICE_ENABLED_MASK /* enabled and decodes its resources */
1295 | STA_DEVICE_SHOW_IN_UI_MASK /* should be shown in UI */
1296 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK /* functioning properly */
1297 | STA_BATTERY_PRESENT_MASK /* battery is present */
1298 : 0; /* device not present */
1299}
1300
1301static uint32_t acpiGetPowerSource(ACPIState *s)
1302{
1303 PDMACPIPOWERSOURCE ps;
1304
1305 /* query the current power source from the host driver */
1306 if (!s->pDrv)
1307 return AC_ONLINE;
1308 int rc = s->pDrv->pfnQueryPowerSource(s->pDrv, &ps);
1309 AssertRC(rc);
1310 return ps == PDM_ACPI_POWER_SOURCE_BATTERY ? AC_OFFLINE : AC_ONLINE;
1311}
1312
1313PDMBOTHCBDECL(int) acpiBatIndexWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1314{
1315 ACPIState *s = (ACPIState *)pvUser;
1316
1317 switch (cb)
1318 {
1319 case 4:
1320 u32 >>= s->u8IndexShift;
1321 /* see comment at the declaration of u8IndexShift */
1322 if (s->u8IndexShift == 0 && u32 == (BAT_DEVICE_STATUS << 2))
1323 {
1324 s->u8IndexShift = 2;
1325 u32 >>= 2;
1326 }
1327 Assert(u32 < BAT_INDEX_LAST);
1328 s->uBatteryIndex = u32;
1329 break;
1330 default:
1331 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1332 break;
1333 }
1334 return VINF_SUCCESS;
1335}
1336
1337PDMBOTHCBDECL(int) acpiBatDataRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1338{
1339 ACPIState *s = (ACPIState *)pvUser;
1340
1341 switch (cb)
1342 {
1343 case 4:
1344 switch (s->uBatteryIndex)
1345 {
1346 case BAT_STATUS_STATE:
1347 acpiFetchBatteryStatus(s);
1348 case BAT_STATUS_PRESENT_RATE:
1349 case BAT_STATUS_REMAINING_CAPACITY:
1350 case BAT_STATUS_PRESENT_VOLTAGE:
1351 *pu32 = s->au8BatteryInfo[s->uBatteryIndex];
1352 break;
1353
1354 case BAT_INFO_UNITS:
1355 acpiFetchBatteryInfo(s);
1356 case BAT_INFO_DESIGN_CAPACITY:
1357 case BAT_INFO_LAST_FULL_CHARGE_CAPACITY:
1358 case BAT_INFO_TECHNOLOGY:
1359 case BAT_INFO_DESIGN_VOLTAGE:
1360 case BAT_INFO_DESIGN_CAPACITY_OF_WARNING:
1361 case BAT_INFO_DESIGN_CAPACITY_OF_LOW:
1362 case BAT_INFO_CAPACITY_GRANULARITY_1:
1363 case BAT_INFO_CAPACITY_GRANULARITY_2:
1364 *pu32 = s->au8BatteryInfo[s->uBatteryIndex];
1365 break;
1366
1367 case BAT_DEVICE_STATUS:
1368 *pu32 = acpiGetBatteryDeviceStatus(s);
1369 break;
1370
1371 case BAT_POWER_SOURCE:
1372 *pu32 = acpiGetPowerSource(s);
1373 break;
1374
1375 default:
1376 AssertMsgFailed(("Invalid battery index %d\n", s->uBatteryIndex));
1377 break;
1378 }
1379 break;
1380 default:
1381 return VERR_IOM_IOPORT_UNUSED;
1382 }
1383 return VINF_SUCCESS;
1384}
1385
1386PDMBOTHCBDECL(int) acpiSysInfoIndexWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1387{
1388 ACPIState *s = (ACPIState *)pvUser;
1389
1390 Log(("system_index = %d, %d\n", u32, u32 >> 2));
1391 switch (cb)
1392 {
1393 case 4:
1394 if (u32 == SYSTEM_INFO_INDEX_VALID || u32 == SYSTEM_INFO_INDEX_INVALID)
1395 s->uSystemInfoIndex = u32;
1396 else
1397 {
1398 /* see comment at the declaration of u8IndexShift */
1399 if (s->u8IndexShift == 0)
1400 {
1401 if (((u32 >> 2) < SYSTEM_INFO_INDEX_END) && ((u32 & 0x3)) == 0)
1402 {
1403 s->u8IndexShift = 2;
1404 }
1405 }
1406
1407 u32 >>= s->u8IndexShift;
1408 Assert(u32 < SYSTEM_INFO_INDEX_END);
1409 s->uSystemInfoIndex = u32;
1410 }
1411 break;
1412
1413 default:
1414 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1415 break;
1416 }
1417 return VINF_SUCCESS;
1418}
1419
1420PDMBOTHCBDECL(int) acpiSysInfoDataRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1421{
1422 ACPIState *s = (ACPIState *)pvUser;
1423
1424 switch (cb)
1425 {
1426 case 4:
1427 switch (s->uSystemInfoIndex)
1428 {
1429 case SYSTEM_INFO_INDEX_LOW_MEMORY_LENGTH:
1430 *pu32 = s->cbRamLow;
1431 break;
1432
1433 case SYSTEM_INFO_INDEX_HIGH_MEMORY_LENGTH:
1434 *pu32 = s->cbRamHigh >> 16; /* 64KB units */
1435 Assert(((uint64_t)*pu32 << 16) == s->cbRamHigh);
1436 break;
1437
1438 case SYSTEM_INFO_INDEX_USE_IOAPIC:
1439 *pu32 = s->u8UseIOApic;
1440 break;
1441
1442 case SYSTEM_INFO_INDEX_HPET_STATUS:
1443 *pu32 = s->fUseHpet ? ( STA_DEVICE_PRESENT_MASK
1444 | STA_DEVICE_ENABLED_MASK
1445 | STA_DEVICE_SHOW_IN_UI_MASK
1446 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK)
1447 : 0;
1448 break;
1449
1450 case SYSTEM_INFO_INDEX_SMC_STATUS:
1451 *pu32 = s->fUseSmc ? ( STA_DEVICE_PRESENT_MASK
1452 | STA_DEVICE_ENABLED_MASK
1453 /* no need to show this device in the UI */
1454 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK)
1455 : 0;
1456 break;
1457
1458 case SYSTEM_INFO_INDEX_FDC_STATUS:
1459 *pu32 = s->fUseFdc ? ( STA_DEVICE_PRESENT_MASK
1460 | STA_DEVICE_ENABLED_MASK
1461 | STA_DEVICE_SHOW_IN_UI_MASK
1462 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK)
1463 : 0;
1464 break;
1465
1466
1467 case SYSTEM_INFO_INDEX_CPU0_STATUS:
1468 case SYSTEM_INFO_INDEX_CPU1_STATUS:
1469 case SYSTEM_INFO_INDEX_CPU2_STATUS:
1470 case SYSTEM_INFO_INDEX_CPU3_STATUS:
1471 *pu32 = s->fShowCpu
1472 && s->uSystemInfoIndex - SYSTEM_INFO_INDEX_CPU0_STATUS < s->cCpus
1473 && VMCPUSET_IS_PRESENT(&s->CpuSetAttached, s->uSystemInfoIndex - SYSTEM_INFO_INDEX_CPU0_STATUS)
1474 ?
1475 STA_DEVICE_PRESENT_MASK
1476 | STA_DEVICE_ENABLED_MASK
1477 | STA_DEVICE_SHOW_IN_UI_MASK
1478 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK
1479 : 0;
1480
1481 case SYSTEM_INFO_INDEX_RTC_STATUS:
1482 *pu32 = s->fShowRtc ? ( STA_DEVICE_PRESENT_MASK
1483 | STA_DEVICE_ENABLED_MASK
1484 | STA_DEVICE_SHOW_IN_UI_MASK
1485 | STA_DEVICE_FUNCTIONING_PROPERLY_MASK)
1486 : 0;
1487 break;
1488
1489 case SYSTEM_INFO_INDEX_CPU_LOCKED:
1490 {
1491 if (s->idCpuLockCheck < VMM_MAX_CPU_COUNT)
1492 {
1493 *pu32 = VMCPUSET_IS_PRESENT(&s->CpuSetLocked, s->idCpuLockCheck);
1494 s->idCpuLockCheck = UINT32_C(0xffffffff); /* Make the entry invalid */
1495 }
1496 else
1497 {
1498 AssertMsgFailed(("ACPI: CPU lock check protocol violation\n"));
1499 /* Always return locked status just to be safe */
1500 *pu32 = 1;
1501 }
1502 break;
1503 }
1504
1505 case SYSTEM_INFO_INDEX_CPU_EVENT_TYPE:
1506 {
1507 *pu32 = s->u32CpuEventType;
1508 break;
1509 }
1510
1511 case SYSTEM_INFO_INDEX_CPU_EVENT:
1512 {
1513 *pu32 = s->u32CpuEvent;
1514 break;
1515 }
1516
1517 /* Solaris 9 tries to read from this index */
1518 case SYSTEM_INFO_INDEX_INVALID:
1519 *pu32 = 0;
1520 break;
1521
1522 default:
1523 AssertMsgFailed(("Invalid system info index %d\n", s->uSystemInfoIndex));
1524 break;
1525 }
1526 break;
1527
1528 default:
1529 return VERR_IOM_IOPORT_UNUSED;
1530 }
1531
1532 Log(("index %d val %d\n", s->uSystemInfoIndex, *pu32));
1533 return VINF_SUCCESS;
1534}
1535
1536PDMBOTHCBDECL(int) acpiSysInfoDataWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1537{
1538 ACPIState *s = (ACPIState *)pvUser;
1539
1540 Log(("addr=%#x cb=%d u32=%#x si=%#x\n", Port, cb, u32, s->uSystemInfoIndex));
1541
1542 if (cb == 4)
1543 {
1544 switch (s->uSystemInfoIndex)
1545 {
1546 case SYSTEM_INFO_INDEX_INVALID:
1547 AssertMsg(u32 == 0xbadc0de, ("u32=%u\n", u32));
1548 s->u8IndexShift = 0;
1549 break;
1550
1551 case SYSTEM_INFO_INDEX_VALID:
1552 AssertMsg(u32 == 0xbadc0de, ("u32=%u\n", u32));
1553 s->u8IndexShift = 2;
1554 break;
1555
1556 case SYSTEM_INFO_INDEX_CPU_LOCK_CHECK:
1557 s->idCpuLockCheck = u32;
1558 break;
1559
1560 case SYSTEM_INFO_INDEX_CPU_LOCKED:
1561 if (u32 < s->cCpus)
1562 VMCPUSET_DEL(&s->CpuSetLocked, u32); /* Unlock the CPU */
1563 else
1564 LogRel(("ACPI: CPU %u does not exist\n", u32));
1565 break;
1566
1567 default:
1568 AssertMsgFailed(("Port=%#x cb=%d u32=%#x system_index=%#x\n",
1569 Port, cb, u32, s->uSystemInfoIndex));
1570 break;
1571 }
1572 }
1573 else
1574 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1575 return VINF_SUCCESS;
1576}
1577
1578/** @todo Don't call functions, but do the job in the read/write handlers
1579 * here! */
1580
1581/* IO Helpers */
1582PDMBOTHCBDECL(int) acpiPm1aEnRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1583{
1584 switch (cb)
1585 {
1586 case 2:
1587 *pu32 = acpiPm1aEnReadw((ACPIState*)pvUser, Port);
1588 break;
1589 default:
1590 return VERR_IOM_IOPORT_UNUSED;
1591 }
1592 return VINF_SUCCESS;
1593}
1594
1595PDMBOTHCBDECL(int) acpiPm1aStsRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1596{
1597 switch (cb)
1598 {
1599 case 2:
1600 *pu32 = acpiPm1aStsReadw((ACPIState*)pvUser, Port);
1601 break;
1602 default:
1603 return VERR_IOM_IOPORT_UNUSED;
1604 }
1605 return VINF_SUCCESS;
1606}
1607
1608PDMBOTHCBDECL(int) acpiPm1aCtlRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1609{
1610 switch (cb)
1611 {
1612 case 2:
1613 *pu32 = acpiPm1aCtlReadw((ACPIState*)pvUser, Port);
1614 break;
1615 default:
1616 return VERR_IOM_IOPORT_UNUSED;
1617 }
1618 return VINF_SUCCESS;
1619}
1620
1621PDMBOTHCBDECL(int) acpiPM1aEnWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1622{
1623 switch (cb)
1624 {
1625 case 2:
1626 acpiPM1aEnWritew((ACPIState*)pvUser, Port, u32);
1627 break;
1628 default:
1629 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1630 break;
1631 }
1632 return VINF_SUCCESS;
1633}
1634
1635PDMBOTHCBDECL(int) acpiPM1aStsWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1636{
1637 switch (cb)
1638 {
1639 case 2:
1640 acpiPM1aStsWritew((ACPIState*)pvUser, Port, u32);
1641 break;
1642 default:
1643 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1644 break;
1645 }
1646 return VINF_SUCCESS;
1647}
1648
1649PDMBOTHCBDECL(int) acpiPM1aCtlWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1650{
1651 switch (cb)
1652 {
1653 case 2:
1654 return acpiPM1aCtlWritew((ACPIState*)pvUser, Port, u32);
1655 default:
1656 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1657 break;
1658 }
1659 return VINF_SUCCESS;
1660}
1661
1662#endif /* IN_RING3 */
1663
1664/**
1665 * PMTMR readable from host/guest.
1666 */
1667PDMBOTHCBDECL(int) acpiPMTmrRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1668{
1669 if (cb == 4)
1670 {
1671 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
1672 int64_t now = TMTimerGet(s->CTX_SUFF(ts));
1673 int64_t elapsed = now - s->pm_timer_initial;
1674
1675 *pu32 = ASMMultU64ByU32DivByU32(elapsed, PM_TMR_FREQ, TMTimerGetFreq(s->CTX_SUFF(ts)));
1676 Log(("acpi: acpiPMTmrRead -> %#x\n", *pu32));
1677 return VINF_SUCCESS;
1678 }
1679 return VERR_IOM_IOPORT_UNUSED;
1680}
1681
1682#ifdef IN_RING3
1683
1684PDMBOTHCBDECL(int) acpiGpe0StsRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1685{
1686 switch (cb)
1687 {
1688 case 1:
1689 *pu32 = acpiGpe0StsReadb((ACPIState*)pvUser, Port);
1690 break;
1691 default:
1692 return VERR_IOM_IOPORT_UNUSED;
1693 }
1694 return VINF_SUCCESS;
1695}
1696
1697PDMBOTHCBDECL(int) acpiGpe0EnRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1698{
1699 switch (cb)
1700 {
1701 case 1:
1702 *pu32 = acpiGpe0EnReadb((ACPIState*)pvUser, Port);
1703 break;
1704 default:
1705 return VERR_IOM_IOPORT_UNUSED;
1706 }
1707 return VINF_SUCCESS;
1708}
1709
1710PDMBOTHCBDECL(int) acpiGpe0StsWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1711{
1712 switch (cb)
1713 {
1714 case 1:
1715 acpiGpe0StsWriteb((ACPIState*)pvUser, Port, u32);
1716 break;
1717 default:
1718 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1719 break;
1720 }
1721 return VINF_SUCCESS;
1722}
1723
1724PDMBOTHCBDECL(int) acpiGpe0EnWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1725{
1726 switch (cb)
1727 {
1728 case 1:
1729 acpiGpe0EnWriteb((ACPIState*)pvUser, Port, u32);
1730 break;
1731 default:
1732 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1733 break;
1734 }
1735 return VINF_SUCCESS;
1736}
1737
1738PDMBOTHCBDECL(int) acpiSmiWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1739{
1740 switch (cb)
1741 {
1742 case 1:
1743 acpiSmiWriteU8((ACPIState*)pvUser, Port, u32);
1744 break;
1745 default:
1746 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1747 break;
1748 }
1749 return VINF_SUCCESS;
1750}
1751
1752PDMBOTHCBDECL(int) acpiResetWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1753{
1754 switch (cb)
1755 {
1756 case 1:
1757 return acpiResetWriteU8((ACPIState*)pvUser, Port, u32);
1758 default:
1759 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1760 break;
1761 }
1762 return VINF_SUCCESS;
1763}
1764
1765#ifdef DEBUG_ACPI
1766
1767PDMBOTHCBDECL(int) acpiDhexWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1768{
1769 switch (cb)
1770 {
1771 case 1:
1772 Log(("%#x\n", u32 & 0xff));
1773 break;
1774 case 2:
1775 Log(("%#6x\n", u32 & 0xffff));
1776 case 4:
1777 Log(("%#10x\n", u32));
1778 break;
1779 default:
1780 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1781 break;
1782 }
1783 return VINF_SUCCESS;
1784}
1785
1786PDMBOTHCBDECL(int) acpiDchrWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1787{
1788 switch (cb)
1789 {
1790 case 1:
1791 Log(("%c", u32 & 0xff));
1792 break;
1793 default:
1794 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1795 break;
1796 }
1797 return VINF_SUCCESS;
1798}
1799
1800#endif /* DEBUG_ACPI */
1801
1802static int acpiRegisterPmHandlers(ACPIState* pThis)
1803{
1804 int rc = VINF_SUCCESS;
1805
1806#define R(offset, cnt, writer, reader, description) \
1807 do { \
1808 rc = PDMDevHlpIOPortRegister(pThis->pDevIns, acpiPmPort(pThis, offset), cnt, pThis, writer, reader, \
1809 NULL, NULL, description); \
1810 if (RT_FAILURE(rc)) \
1811 return rc; \
1812 } while (0)
1813#define L (GPE0_BLK_LEN / 2)
1814
1815 R(PM1a_EVT_OFFSET+2, 1, acpiPM1aEnWrite, acpiPm1aEnRead, "ACPI PM1a Enable");
1816 R(PM1a_EVT_OFFSET, 1, acpiPM1aStsWrite, acpiPm1aStsRead, "ACPI PM1a Status");
1817 R(PM1a_CTL_OFFSET, 1, acpiPM1aCtlWrite, acpiPm1aCtlRead, "ACPI PM1a Control");
1818 R(PM_TMR_OFFSET, 1, NULL, acpiPMTmrRead, "ACPI PM Timer");
1819 R(GPE0_OFFSET + L, L, acpiGpe0EnWrite, acpiGpe0EnRead, "ACPI GPE0 Enable");
1820 R(GPE0_OFFSET, L, acpiGpe0StsWrite, acpiGpe0StsRead, "ACPI GPE0 Status");
1821#undef L
1822#undef R
1823
1824 /* register RC stuff */
1825 if (pThis->fGCEnabled)
1826 {
1827 rc = PDMDevHlpIOPortRegisterRC(pThis->pDevIns, acpiPmPort(pThis, PM_TMR_OFFSET),
1828 1, 0, NULL, "acpiPMTmrRead",
1829 NULL, NULL, "ACPI PM Timer");
1830 AssertRCReturn(rc, rc);
1831 }
1832
1833 /* register R0 stuff */
1834 if (pThis->fR0Enabled)
1835 {
1836 rc = PDMDevHlpIOPortRegisterR0(pThis->pDevIns, acpiPmPort(pThis, PM_TMR_OFFSET),
1837 1, 0, NULL, "acpiPMTmrRead",
1838 NULL, NULL, "ACPI PM Timer");
1839 AssertRCReturn(rc, rc);
1840 }
1841
1842 return rc;
1843}
1844
1845static int acpiUnregisterPmHandlers(ACPIState *pThis)
1846{
1847#define U(offset, cnt) \
1848 do { \
1849 int rc = PDMDevHlpIOPortDeregister(pThis->pDevIns, acpiPmPort(pThis, offset), cnt); \
1850 AssertRCReturn(rc, rc); \
1851 } while (0)
1852#define L (GPE0_BLK_LEN / 2)
1853
1854 U(PM1a_EVT_OFFSET+2, 1);
1855 U(PM1a_EVT_OFFSET, 1);
1856 U(PM1a_CTL_OFFSET, 1);
1857 U(PM_TMR_OFFSET, 1);
1858 U(GPE0_OFFSET + L, L);
1859 U(GPE0_OFFSET, L);
1860#undef L
1861#undef U
1862
1863 return VINF_SUCCESS;
1864}
1865
1866/**
1867 * Saved state structure description, version 4.
1868 */
1869static const SSMFIELD g_AcpiSavedStateFields4[] =
1870{
1871 SSMFIELD_ENTRY(ACPIState, pm1a_en),
1872 SSMFIELD_ENTRY(ACPIState, pm1a_sts),
1873 SSMFIELD_ENTRY(ACPIState, pm1a_ctl),
1874 SSMFIELD_ENTRY(ACPIState, pm_timer_initial),
1875 SSMFIELD_ENTRY(ACPIState, gpe0_en),
1876 SSMFIELD_ENTRY(ACPIState, gpe0_sts),
1877 SSMFIELD_ENTRY(ACPIState, uBatteryIndex),
1878 SSMFIELD_ENTRY(ACPIState, uSystemInfoIndex),
1879 SSMFIELD_ENTRY(ACPIState, u64RamSize),
1880 SSMFIELD_ENTRY(ACPIState, u8IndexShift),
1881 SSMFIELD_ENTRY(ACPIState, u8UseIOApic),
1882 SSMFIELD_ENTRY(ACPIState, uSleepState),
1883 SSMFIELD_ENTRY_TERM()
1884};
1885
1886/**
1887 * Saved state structure description, version 5.
1888 */
1889static const SSMFIELD g_AcpiSavedStateFields5[] =
1890{
1891 SSMFIELD_ENTRY(ACPIState, pm1a_en),
1892 SSMFIELD_ENTRY(ACPIState, pm1a_sts),
1893 SSMFIELD_ENTRY(ACPIState, pm1a_ctl),
1894 SSMFIELD_ENTRY(ACPIState, pm_timer_initial),
1895 SSMFIELD_ENTRY(ACPIState, gpe0_en),
1896 SSMFIELD_ENTRY(ACPIState, gpe0_sts),
1897 SSMFIELD_ENTRY(ACPIState, uBatteryIndex),
1898 SSMFIELD_ENTRY(ACPIState, uSystemInfoIndex),
1899 SSMFIELD_ENTRY(ACPIState, uSleepState),
1900 SSMFIELD_ENTRY(ACPIState, u8IndexShift),
1901 SSMFIELD_ENTRY(ACPIState, uPmIoPortBase),
1902 SSMFIELD_ENTRY_TERM()
1903};
1904
1905
1906static DECLCALLBACK(int) acpi_save_state(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle)
1907{
1908 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
1909 return SSMR3PutStruct(pSSMHandle, s, &g_AcpiSavedStateFields5[0]);
1910}
1911
1912static DECLCALLBACK(int) acpi_load_state(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle,
1913 uint32_t uVersion, uint32_t uPass)
1914{
1915 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
1916
1917 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
1918 /*
1919 * Unregister PM handlers, will register with actual base
1920 * after state successfully loaded.
1921 */
1922 int rc = acpiUnregisterPmHandlers(s);
1923 if (RT_FAILURE(rc))
1924 return rc;
1925
1926 switch (uVersion)
1927 {
1928 case 4:
1929 rc = SSMR3GetStruct(pSSMHandle, s, &g_AcpiSavedStateFields4[0]);
1930 /** @todo Provide saner defaults for fields not found in saved state. */
1931 break;
1932 case 5:
1933 rc = SSMR3GetStruct(pSSMHandle, s, &g_AcpiSavedStateFields5[0]);
1934 break;
1935 default:
1936 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1937 }
1938 if (RT_SUCCESS(rc))
1939 {
1940 rc = acpiRegisterPmHandlers(s);
1941 if (RT_FAILURE(rc))
1942 return rc;
1943 rc = acpiFetchBatteryStatus(s);
1944 if (RT_FAILURE(rc))
1945 return rc;
1946 rc = acpiFetchBatteryInfo(s);
1947 if (RT_FAILURE(rc))
1948 return rc;
1949 rc = acpiPMTimerReset(s);
1950 if (RT_FAILURE(rc))
1951 return rc;
1952 }
1953 return rc;
1954}
1955
1956/**
1957 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1958 */
1959static DECLCALLBACK(void *) acpiQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1960{
1961 ACPIState *pThis = RT_FROM_MEMBER(pInterface, ACPIState, IBase);
1962 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
1963 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIACPIPORT, &pThis->IACPIPort);
1964 return NULL;
1965}
1966
1967/**
1968 * Create the ACPI tables.
1969 */
1970static int acpiPlantTables(ACPIState *s)
1971{
1972 int rc;
1973 RTGCPHYS32 GCPhysCur, GCPhysRsdt, GCPhysXsdt, GCPhysFadtAcpi1, GCPhysFadtAcpi2, GCPhysFacs, GCPhysDsdt;
1974 RTGCPHYS32 GCPhysHpet = 0, GCPhysApic = 0, GCPhysSsdt = 0;
1975 uint32_t addend = 0;
1976 RTGCPHYS32 aGCPhysRsdt[4];
1977 RTGCPHYS32 aGCPhysXsdt[4];
1978 uint32_t cAddr, iMadt = 0, iHpet = 0, iSsdt = 0;
1979 size_t cbRsdt = sizeof(ACPITBLHEADER);
1980 size_t cbXsdt = sizeof(ACPITBLHEADER);
1981
1982 cAddr = 1; /* FADT */
1983 if (s->u8UseIOApic)
1984 iMadt = cAddr++; /* MADT */
1985
1986 if (s->fUseHpet)
1987 iHpet = cAddr++; /* HPET */
1988
1989 iSsdt = cAddr++; /* SSDT */
1990
1991 cbRsdt += cAddr*sizeof(uint32_t); /* each entry: 32 bits phys. address. */
1992 cbXsdt += cAddr*sizeof(uint64_t); /* each entry: 64 bits phys. address. */
1993
1994 rc = CFGMR3QueryU64(s->pDevIns->pCfg, "RamSize", &s->u64RamSize);
1995 if (RT_FAILURE(rc))
1996 return PDMDEV_SET_ERROR(s->pDevIns, rc,
1997 N_("Configuration error: Querying \"RamSize\" as integer failed"));
1998
1999 uint32_t cbRamHole;
2000 rc = CFGMR3QueryU32Def(s->pDevIns->pCfg, "RamHoleSize", &cbRamHole, MM_RAM_HOLE_SIZE_DEFAULT);
2001 if (RT_FAILURE(rc))
2002 return PDMDEV_SET_ERROR(s->pDevIns, rc,
2003 N_("Configuration error: Querying \"RamHoleSize\" as integer failed"));
2004
2005 /*
2006 * Calculate the sizes for the high and low regions.
2007 */
2008 const uint64_t offRamHole = _4G - cbRamHole;
2009 s->cbRamHigh = offRamHole < s->u64RamSize ? s->u64RamSize - offRamHole : 0;
2010 uint64_t cbRamLow = offRamHole < s->u64RamSize ? offRamHole : s->u64RamSize;
2011 if (cbRamLow > UINT32_C(0xffe00000)) /* See MEM3. */
2012 {
2013 /* Note: This is also enforced by DevPcBios.cpp. */
2014 LogRel(("DevACPI: Clipping cbRamLow=%#RX64 down to 0xffe00000.\n", cbRamLow));
2015 cbRamLow = UINT32_C(0xffe00000);
2016 }
2017 s->cbRamLow = (uint32_t)cbRamLow;
2018
2019 GCPhysCur = 0;
2020 GCPhysRsdt = GCPhysCur;
2021
2022 GCPhysCur = RT_ALIGN_32(GCPhysCur + cbRsdt, 16);
2023 GCPhysXsdt = GCPhysCur;
2024
2025 GCPhysCur = RT_ALIGN_32(GCPhysCur + cbXsdt, 16);
2026 GCPhysFadtAcpi1 = GCPhysCur;
2027
2028 GCPhysCur = RT_ALIGN_32(GCPhysCur + ACPITBLFADT_VERSION1_SIZE, 16);
2029 GCPhysFadtAcpi2 = GCPhysCur;
2030
2031 GCPhysCur = RT_ALIGN_32(GCPhysCur + sizeof(ACPITBLFADT), 64);
2032 GCPhysFacs = GCPhysCur;
2033
2034 GCPhysCur = RT_ALIGN_32(GCPhysCur + sizeof(ACPITBLFACS), 16);
2035 if (s->u8UseIOApic)
2036 {
2037 GCPhysApic = GCPhysCur;
2038 GCPhysCur = RT_ALIGN_32(GCPhysCur + AcpiTableMADT::sizeFor(s), 16);
2039 }
2040 if (s->fUseHpet)
2041 {
2042 GCPhysHpet = GCPhysCur;
2043 GCPhysCur = RT_ALIGN_32(GCPhysCur + sizeof(ACPITBLHPET), 16);
2044 }
2045
2046 void* pSsdtCode = NULL;
2047 size_t cbSsdtSize = 0;
2048 rc = acpiPrepareSsdt(s->pDevIns, &pSsdtCode, &cbSsdtSize);
2049 if (RT_FAILURE(rc))
2050 return rc;
2051
2052 GCPhysSsdt = GCPhysCur;
2053 GCPhysCur = RT_ALIGN_32(GCPhysCur + cbSsdtSize, 16);
2054
2055 GCPhysDsdt = GCPhysCur;
2056
2057 void* pDsdtCode = NULL;
2058 size_t cbDsdtSize = 0;
2059 rc = acpiPrepareDsdt(s->pDevIns, &pDsdtCode, &cbDsdtSize);
2060 if (RT_FAILURE(rc))
2061 return rc;
2062
2063 GCPhysCur = RT_ALIGN_32(GCPhysCur + cbDsdtSize, 16);
2064
2065 if (GCPhysCur > 0x10000)
2066 return PDMDEV_SET_ERROR(s->pDevIns, VERR_TOO_MUCH_DATA,
2067 N_("Error: ACPI tables bigger than 64KB"));
2068
2069 Log(("RSDP 0x%08X\n", find_rsdp_space()));
2070 addend = s->cbRamLow - 0x10000;
2071 Log(("RSDT 0x%08X XSDT 0x%08X\n", GCPhysRsdt + addend, GCPhysXsdt + addend));
2072 Log(("FACS 0x%08X FADT (1.0) 0x%08X, FADT (2+) 0x%08X\n", GCPhysFacs + addend, GCPhysFadtAcpi1 + addend, GCPhysFadtAcpi2 + addend));
2073 Log(("DSDT 0x%08X", GCPhysDsdt + addend));
2074 if (s->u8UseIOApic)
2075 Log((" MADT 0x%08X", GCPhysApic + addend));
2076 if (s->fUseHpet)
2077 Log((" HPET 0x%08X", GCPhysHpet + addend));
2078 Log((" SSDT 0x%08X", GCPhysSsdt + addend));
2079 Log(("\n"));
2080
2081 acpiSetupRSDP((ACPITBLRSDP*)s->au8RSDPPage, GCPhysRsdt + addend, GCPhysXsdt + addend);
2082 acpiSetupDSDT(s, GCPhysDsdt + addend, pDsdtCode, cbDsdtSize);
2083 acpiCleanupDsdt(s->pDevIns, pDsdtCode);
2084 acpiSetupFACS(s, GCPhysFacs + addend);
2085 acpiSetupFADT(s, GCPhysFadtAcpi1 + addend, GCPhysFadtAcpi2 + addend, GCPhysFacs + addend, GCPhysDsdt + addend);
2086
2087 aGCPhysRsdt[0] = GCPhysFadtAcpi1 + addend;
2088 aGCPhysXsdt[0] = GCPhysFadtAcpi2 + addend;
2089 if (s->u8UseIOApic)
2090 {
2091 acpiSetupMADT(s, GCPhysApic + addend);
2092 aGCPhysRsdt[iMadt] = GCPhysApic + addend;
2093 aGCPhysXsdt[iMadt] = GCPhysApic + addend;
2094 }
2095 if (s->fUseHpet)
2096 {
2097 acpiSetupHPET(s, GCPhysHpet + addend);
2098 aGCPhysRsdt[iHpet] = GCPhysHpet + addend;
2099 aGCPhysXsdt[iHpet] = GCPhysHpet + addend;
2100 }
2101 acpiSetupSSDT(s, GCPhysSsdt + addend, pSsdtCode, cbSsdtSize);
2102 acpiCleanupSsdt(s->pDevIns, pSsdtCode);
2103 aGCPhysRsdt[iSsdt] = GCPhysSsdt + addend;
2104 aGCPhysXsdt[iSsdt] = GCPhysSsdt + addend;
2105
2106 rc = acpiSetupRSDT(s, GCPhysRsdt + addend, cAddr, aGCPhysRsdt);
2107 if (RT_FAILURE(rc))
2108 return rc;
2109 return acpiSetupXSDT(s, GCPhysXsdt + addend, cAddr, aGCPhysXsdt);
2110}
2111
2112static int acpiUpdatePmHandlers(ACPIState *pThis, RTIOPORT uNewBase)
2113{
2114 Log(("acpi: rebasing PM 0x%x -> 0x%x\n", pThis->uPmIoPortBase, uNewBase));
2115 if (uNewBase != pThis->uPmIoPortBase)
2116 {
2117 int rc;
2118
2119 rc = acpiUnregisterPmHandlers(pThis);
2120 if (RT_FAILURE(rc))
2121 return rc;
2122
2123 pThis->uPmIoPortBase = uNewBase;
2124
2125 rc = acpiRegisterPmHandlers(pThis);
2126 if (RT_FAILURE(rc))
2127 return rc;
2128
2129 /* We have to update FADT table acccording to the new base */
2130 rc = acpiPlantTables(pThis);
2131 AssertRC(rc);
2132 if (RT_FAILURE(rc))
2133 return rc;
2134 }
2135
2136 return VINF_SUCCESS;
2137}
2138
2139static uint32_t acpiPciConfigRead(PPCIDEVICE pPciDev, uint32_t Address, unsigned cb)
2140{
2141 PPDMDEVINS pDevIns = pPciDev->pDevIns;
2142 ACPIState* pThis = PDMINS_2_DATA(pDevIns, ACPIState *);
2143
2144 Log2(("acpi: PCI config read: 0x%x (%d)\n", Address, cb));
2145
2146 return pThis->pfnAcpiPciConfigRead(pPciDev, Address, cb);
2147}
2148
2149static void acpiPciConfigWrite(PPCIDEVICE pPciDev, uint32_t Address, uint32_t u32Value, unsigned cb)
2150{
2151 PPDMDEVINS pDevIns = pPciDev->pDevIns;
2152 ACPIState *pThis = PDMINS_2_DATA(pDevIns, ACPIState *);
2153
2154 Log2(("acpi: PCI config write: 0x%x -> 0x%x (%d)\n", u32Value, Address, cb));
2155 pThis->pfnAcpiPciConfigWrite(pPciDev, Address, u32Value, cb);
2156
2157 /* PMREGMISC written */
2158 if (Address == 0x80)
2159 {
2160 /* Check Power Management IO Space Enable (PMIOSE) bit */
2161 if (pPciDev->config[0x80] & 0x1)
2162 {
2163 int rc;
2164
2165 RTIOPORT uNewBase =
2166 RTIOPORT(RT_LE2H_U32(*(uint32_t*)&pPciDev->config[0x40]));
2167 uNewBase &= 0xffc0;
2168
2169 rc = acpiUpdatePmHandlers(pThis, uNewBase);
2170 AssertRC(rc);
2171 }
2172 }
2173}
2174
2175/**
2176 * Attach a new CPU.
2177 *
2178 * @returns VBox status code.
2179 * @param pDevIns The device instance.
2180 * @param iLUN The logical unit which is being attached.
2181 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
2182 *
2183 * @remarks This code path is not used during construction.
2184 */
2185static DECLCALLBACK(int) acpiAttach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
2186{
2187 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
2188
2189 LogFlow(("acpiAttach: pDevIns=%p iLUN=%u fFlags=%#x\n", pDevIns, iLUN, fFlags));
2190
2191 AssertMsgReturn(!(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG),
2192 ("Hot-plug flag is not set\n"),
2193 VERR_NOT_SUPPORTED);
2194 AssertReturn(iLUN < VMM_MAX_CPU_COUNT, VERR_PDM_NO_SUCH_LUN);
2195
2196 /* Check if it was already attached */
2197 if (VMCPUSET_IS_PRESENT(&s->CpuSetAttached, iLUN))
2198 return VINF_SUCCESS;
2199
2200 PPDMIBASE IBaseTmp;
2201 int rc = PDMDevHlpDriverAttach(pDevIns, iLUN, &s->IBase, &IBaseTmp, "ACPI CPU");
2202
2203 if (RT_SUCCESS(rc))
2204 {
2205 /* Enable the CPU */
2206 VMCPUSET_ADD(&s->CpuSetAttached, iLUN);
2207
2208 /*
2209 * Lock the CPU because we don't know if the guest will use it or not.
2210 * Prevents ejection while the CPU is still used
2211 */
2212 VMCPUSET_ADD(&s->CpuSetLocked, iLUN);
2213 s->u32CpuEventType = CPU_EVENT_TYPE_ADD;
2214 s->u32CpuEvent = iLUN;
2215 /* Notify the guest */
2216 update_gpe0(s, s->gpe0_sts | 0x2, s->gpe0_en);
2217 }
2218 return rc;
2219}
2220
2221/**
2222 * Detach notification.
2223 *
2224 * @param pDevIns The device instance.
2225 * @param iLUN The logical unit which is being detached.
2226 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
2227 */
2228static DECLCALLBACK(void) acpiDetach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
2229{
2230 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
2231
2232 LogFlow(("acpiDetach: pDevIns=%p iLUN=%u fFlags=%#x\n", pDevIns, iLUN, fFlags));
2233
2234 AssertMsgReturnVoid(!(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG),
2235 ("Hot-plug flag is not set\n"));
2236
2237 /* Check if it was already detached */
2238 if (VMCPUSET_IS_PRESENT(&s->CpuSetAttached, iLUN))
2239 {
2240 AssertMsgReturnVoid(!(VMCPUSET_IS_PRESENT(&s->CpuSetLocked, iLUN)), ("CPU is still locked by the guest\n"));
2241
2242 /* Disable the CPU */
2243 VMCPUSET_DEL(&s->CpuSetAttached, iLUN);
2244 s->u32CpuEventType = CPU_EVENT_TYPE_REMOVE;
2245 s->u32CpuEvent = iLUN;
2246 /* Notify the guest */
2247 update_gpe0(s, s->gpe0_sts | 0x2, s->gpe0_en);
2248 }
2249}
2250
2251static DECLCALLBACK(void) acpiReset(PPDMDEVINS pDevIns)
2252{
2253 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
2254
2255 s->pm1a_en = 0;
2256 s->pm1a_sts = 0;
2257 s->pm1a_ctl = 0;
2258 s->pm_timer_initial = TMTimerGet(s->CTX_SUFF(ts));
2259 acpiPMTimerReset(s);
2260 s->uBatteryIndex = 0;
2261 s->uSystemInfoIndex = 0;
2262 s->gpe0_en = 0;
2263 s->gpe0_sts = 0;
2264 s->uSleepState = 0;
2265
2266 /** @todo Should we really reset PM base? */
2267 acpiUpdatePmHandlers(s, PM_PORT_BASE);
2268
2269 acpiPlantTables(s);
2270}
2271
2272/**
2273 * Relocates the GC pointer members.
2274 */
2275static DECLCALLBACK(void) acpiRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
2276{
2277 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
2278 s->tsRC = TMTimerRCPtr(s->CTX_SUFF(ts));
2279}
2280
2281/**
2282 * @interface_method_impl{PDMDEVREG,pfnConstruct}
2283 */
2284static DECLCALLBACK(int) acpiConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
2285{
2286 ACPIState *s = PDMINS_2_DATA(pDevIns, ACPIState *);
2287 PCIDevice *dev = &s->dev;
2288 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
2289
2290 /* Validate and read the configuration. */
2291 if (!CFGMR3AreValuesValid(pCfg,
2292 "RamSize\0"
2293 "RamHoleSize\0"
2294 "IOAPIC\0"
2295 "NumCPUs\0"
2296 "GCEnabled\0"
2297 "R0Enabled\0"
2298 "HpetEnabled\0"
2299 "SmcEnabled\0"
2300 "FdcEnabled\0"
2301 "ShowRtc\0"
2302 "ShowCpu\0"
2303 "CpuHotPlug\0"
2304 "AmlFilePath\0"
2305 ))
2306 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
2307 N_("Configuration error: Invalid config key for ACPI device"));
2308
2309 s->pDevIns = pDevIns;
2310
2311 /* query whether we are supposed to present an IOAPIC */
2312 int rc = CFGMR3QueryU8Def(pCfg, "IOAPIC", &s->u8UseIOApic, 1);
2313 if (RT_FAILURE(rc))
2314 return PDMDEV_SET_ERROR(pDevIns, rc,
2315 N_("Configuration error: Failed to read \"IOAPIC\""));
2316
2317 rc = CFGMR3QueryU16Def(pCfg, "NumCPUs", &s->cCpus, 1);
2318 if (RT_FAILURE(rc))
2319 return PDMDEV_SET_ERROR(pDevIns, rc,
2320 N_("Configuration error: Querying \"NumCPUs\" as integer failed"));
2321
2322 /* query whether we are supposed to present an FDC controller */
2323 rc = CFGMR3QueryBoolDef(pCfg, "FdcEnabled", &s->fUseFdc, true);
2324 if (RT_FAILURE(rc))
2325 return PDMDEV_SET_ERROR(pDevIns, rc,
2326 N_("Configuration error: Failed to read \"FdcEnabled\""));
2327
2328 /* query whether we are supposed to present HPET */
2329 rc = CFGMR3QueryBoolDef(pCfg, "HpetEnabled", &s->fUseHpet, false);
2330 if (RT_FAILURE(rc))
2331 return PDMDEV_SET_ERROR(pDevIns, rc,
2332 N_("Configuration error: Failed to read \"HpetEnabled\""));
2333 /* query whether we are supposed to present SMC */
2334 rc = CFGMR3QueryBoolDef(pCfg, "SmcEnabled", &s->fUseSmc, false);
2335 if (RT_FAILURE(rc))
2336 return PDMDEV_SET_ERROR(pDevIns, rc,
2337 N_("Configuration error: Failed to read \"SmcEnabled\""));
2338
2339 /* query whether we are supposed to present RTC object */
2340 rc = CFGMR3QueryBoolDef(pCfg, "ShowRtc", &s->fShowRtc, false);
2341 if (RT_FAILURE(rc))
2342 return PDMDEV_SET_ERROR(pDevIns, rc,
2343 N_("Configuration error: Failed to read \"ShowRtc\""));
2344
2345 /* query whether we are supposed to present CPU objects */
2346 rc = CFGMR3QueryBoolDef(pCfg, "ShowCpu", &s->fShowCpu, false);
2347 if (RT_FAILURE(rc))
2348 return PDMDEV_SET_ERROR(pDevIns, rc,
2349 N_("Configuration error: Failed to read \"ShowCpu\""));
2350
2351 /* query whether we are allow CPU hot plugging */
2352 rc = CFGMR3QueryBoolDef(pCfg, "CpuHotPlug", &s->fCpuHotPlug, false);
2353 if (RT_FAILURE(rc))
2354 return PDMDEV_SET_ERROR(pDevIns, rc,
2355 N_("Configuration error: Failed to read \"CpuHotPlug\""));
2356
2357 rc = CFGMR3QueryBool(pCfg, "GCEnabled", &s->fGCEnabled);
2358 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
2359 s->fGCEnabled = true;
2360 else if (RT_FAILURE(rc))
2361 return PDMDEV_SET_ERROR(pDevIns, rc,
2362 N_("Configuration error: Failed to read \"GCEnabled\""));
2363
2364 rc = CFGMR3QueryBool(pCfg, "R0Enabled", &s->fR0Enabled);
2365 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
2366 s->fR0Enabled = true;
2367 else if (RT_FAILURE(rc))
2368 return PDMDEV_SET_ERROR(pDevIns, rc,
2369 N_("configuration error: failed to read R0Enabled as boolean"));
2370
2371 /*
2372 * Interfaces
2373 */
2374 /* IBase */
2375 s->IBase.pfnQueryInterface = acpiQueryInterface;
2376 /* IACPIPort */
2377 s->IACPIPort.pfnSleepButtonPress = acpiSleepButtonPress;
2378 s->IACPIPort.pfnPowerButtonPress = acpiPowerButtonPress;
2379 s->IACPIPort.pfnGetPowerButtonHandled = acpiGetPowerButtonHandled;
2380 s->IACPIPort.pfnGetGuestEnteredACPIMode = acpiGetGuestEnteredACPIMode;
2381 s->IACPIPort.pfnGetCpuStatus = acpiGetCpuStatus;
2382
2383 VMCPUSET_EMPTY(&s->CpuSetAttached);
2384 VMCPUSET_EMPTY(&s->CpuSetLocked);
2385 s->idCpuLockCheck = UINT32_C(0xffffffff);
2386 s->u32CpuEventType = 0;
2387 s->u32CpuEvent = UINT32_C(0xffffffff);
2388
2389 /* The first CPU can't be attached/detached */
2390 VMCPUSET_ADD(&s->CpuSetAttached, 0);
2391 VMCPUSET_ADD(&s->CpuSetLocked, 0);
2392
2393 /* Try to attach the other CPUs */
2394 for (unsigned i = 1; i < s->cCpus; i++)
2395 {
2396 if (s->fCpuHotPlug)
2397 {
2398 PPDMIBASE IBaseTmp;
2399 rc = PDMDevHlpDriverAttach(pDevIns, i, &s->IBase, &IBaseTmp, "ACPI CPU");
2400
2401 if (RT_SUCCESS(rc))
2402 {
2403 VMCPUSET_ADD(&s->CpuSetAttached, i);
2404 VMCPUSET_ADD(&s->CpuSetLocked, i);
2405 Log(("acpi: Attached CPU %u\n", i));
2406 }
2407 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
2408 Log(("acpi: CPU %u not attached yet\n", i));
2409 else
2410 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach CPU object\n"));
2411 }
2412 else
2413 {
2414 /* CPU is always attached if hot-plug is not enabled. */
2415 VMCPUSET_ADD(&s->CpuSetAttached, i);
2416 VMCPUSET_ADD(&s->CpuSetLocked, i);
2417 }
2418 }
2419
2420
2421 /* Set default port base */
2422 s->uPmIoPortBase = PM_PORT_BASE;
2423
2424 /*
2425 * FDC and SMC try to use the same non-shareable interrupt (6),
2426 * enable only one device.
2427 */
2428 if (s->fUseSmc)
2429 s->fUseFdc = false;
2430
2431 /* */
2432 RTGCPHYS32 GCPhysRsdp = find_rsdp_space();
2433 if (!GCPhysRsdp)
2434 return PDMDEV_SET_ERROR(pDevIns, VERR_NO_MEMORY,
2435 N_("Can not find space for RSDP. ACPI is disabled"));
2436
2437 rc = acpiPlantTables(s);
2438 if (RT_FAILURE(rc))
2439 return rc;
2440
2441 rc = PDMDevHlpROMRegister(pDevIns, GCPhysRsdp, 0x1000, s->au8RSDPPage,
2442 PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "ACPI RSDP");
2443 if (RT_FAILURE(rc))
2444 return rc;
2445
2446 rc = acpiRegisterPmHandlers(s);
2447 if (RT_FAILURE(rc))
2448 return rc;
2449
2450#define R(addr, cnt, writer, reader, description) \
2451 do { \
2452 rc = PDMDevHlpIOPortRegister(pDevIns, addr, cnt, s, writer, reader, \
2453 NULL, NULL, description); \
2454 if (RT_FAILURE(rc)) \
2455 return rc; \
2456 } while (0)
2457 R(SMI_CMD, 1, acpiSmiWrite, NULL, "ACPI SMI");
2458#ifdef DEBUG_ACPI
2459 R(DEBUG_HEX, 1, acpiDhexWrite, NULL, "ACPI Debug hex");
2460 R(DEBUG_CHR, 1, acpiDchrWrite, NULL, "ACPI Debug char");
2461#endif
2462 R(BAT_INDEX, 1, acpiBatIndexWrite, NULL, "ACPI Battery status index");
2463 R(BAT_DATA, 1, NULL, acpiBatDataRead, "ACPI Battery status data");
2464 R(SYSI_INDEX, 1, acpiSysInfoIndexWrite, NULL, "ACPI system info index");
2465 R(SYSI_DATA, 1, acpiSysInfoDataWrite, acpiSysInfoDataRead, "ACPI system info data");
2466 R(ACPI_RESET_BLK, 1, acpiResetWrite, NULL, "ACPI Reset");
2467#undef R
2468
2469 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL_SYNC, acpiTimer, dev,
2470 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "ACPI Timer", &s->tsR3);
2471 if (RT_FAILURE(rc))
2472 {
2473 AssertMsgFailed(("pfnTMTimerCreate -> %Rrc\n", rc));
2474 return rc;
2475 }
2476
2477 s->tsR0 = TMTimerR0Ptr(s->tsR3);
2478 s->tsRC = TMTimerRCPtr(s->tsR3);
2479 s->pm_timer_initial = TMTimerGet(s->tsR3);
2480 acpiPMTimerReset(s);
2481
2482 PCIDevSetVendorId(dev, 0x8086); /* Intel */
2483 PCIDevSetDeviceId(dev, 0x7113); /* 82371AB */
2484
2485 /* See p. 50 of PIIX4 manual */
2486 dev->config[0x04] = 0x01; /* command */
2487 dev->config[0x05] = 0x00;
2488
2489 dev->config[0x06] = 0x80; /* status */
2490 dev->config[0x07] = 0x02;
2491
2492 dev->config[0x08] = 0x08; /* revision number */
2493
2494 dev->config[0x09] = 0x00; /* class code */
2495 dev->config[0x0a] = 0x80;
2496 dev->config[0x0b] = 0x06;
2497
2498 dev->config[0x0e] = 0x80; /* header type */
2499
2500 dev->config[0x0f] = 0x00; /* reserved */
2501
2502 dev->config[0x3c] = SCI_INT; /* interrupt line */
2503
2504#if 0
2505 dev->config[0x3d] = 0x01; /* interrupt pin */
2506#endif
2507
2508 dev->config[0x40] = 0x01; /* PM base address, this bit marks it as IO range, not PA */
2509
2510#if 0
2511 int smb_io_base = 0xb100;
2512 dev->config[0x90] = smb_io_base | 1; /* SMBus base address */
2513 dev->config[0x90] = smb_io_base >> 8;
2514#endif
2515
2516 rc = PDMDevHlpPCIRegister(pDevIns, dev);
2517 if (RT_FAILURE(rc))
2518 return rc;
2519
2520 PDMDevHlpPCISetConfigCallbacks(pDevIns, dev,
2521 acpiPciConfigRead, &s->pfnAcpiPciConfigRead,
2522 acpiPciConfigWrite, &s->pfnAcpiPciConfigWrite);
2523
2524 rc = PDMDevHlpSSMRegister(pDevIns, 5, sizeof(*s), acpi_save_state, acpi_load_state);
2525 if (RT_FAILURE(rc))
2526 return rc;
2527
2528 /*
2529 * Get the corresponding connector interface
2530 */
2531 rc = PDMDevHlpDriverAttach(pDevIns, 0, &s->IBase, &s->pDrvBase, "ACPI Driver Port");
2532 if (RT_SUCCESS(rc))
2533 {
2534 s->pDrv = PDMIBASE_QUERY_INTERFACE(s->pDrvBase, PDMIACPICONNECTOR);
2535 if (!s->pDrv)
2536 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_MISSING_INTERFACE,
2537 N_("LUN #0 doesn't have an ACPI connector interface"));
2538 }
2539 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
2540 {
2541 Log(("acpi: %s/%d: warning: no driver attached to LUN #0!\n",
2542 pDevIns->pReg->szName, pDevIns->iInstance));
2543 rc = VINF_SUCCESS;
2544 }
2545 else
2546 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach LUN #0"));
2547
2548 return rc;
2549}
2550
2551/**
2552 * The device registration structure.
2553 */
2554const PDMDEVREG g_DeviceACPI =
2555{
2556 /* u32Version */
2557 PDM_DEVREG_VERSION,
2558 /* szName */
2559 "acpi",
2560 /* szRCMod */
2561 "VBoxDDGC.gc",
2562 /* szR0Mod */
2563 "VBoxDDR0.r0",
2564 /* pszDescription */
2565 "Advanced Configuration and Power Interface",
2566 /* fFlags */
2567 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
2568 /* fClass */
2569 PDM_DEVREG_CLASS_ACPI,
2570 /* cMaxInstances */
2571 ~0,
2572 /* cbInstance */
2573 sizeof(ACPIState),
2574 /* pfnConstruct */
2575 acpiConstruct,
2576 /* pfnDestruct */
2577 NULL,
2578 /* pfnRelocate */
2579 acpiRelocate,
2580 /* pfnIOCtl */
2581 NULL,
2582 /* pfnPowerOn */
2583 NULL,
2584 /* pfnReset */
2585 acpiReset,
2586 /* pfnSuspend */
2587 NULL,
2588 /* pfnResume */
2589 NULL,
2590 /* pfnAttach */
2591 acpiAttach,
2592 /* pfnDetach */
2593 acpiDetach,
2594 /* pfnQueryInterface. */
2595 NULL,
2596 /* pfnInitComplete */
2597 NULL,
2598 /* pfnPowerOff */
2599 NULL,
2600 /* pfnSoftReset */
2601 NULL,
2602 /* u32VersionEnd */
2603 PDM_DEVREG_VERSION
2604};
2605
2606#endif /* IN_RING3 */
2607#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
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