/*++ Copyright (c) 2000 Microsoft Corporation Module Name: acpi.c Abstract: WinDbg Extension Api for interpretting ACPI data structures Supports rsdt, fadt, facs, mapic, gbl and inf Author: Ported to 64 bit by Graham Laverty (t-gralav) 10-Mar-2000 Based on Code by: Stephane Plante (splante) 21-Mar-1997 Peter Wieland (peterwie) 16-Oct-1995 Ken Reneris (kenr) 06-June-1994 Environment: User Mode. Revision History: Ported to 64 bit by Graham Laverty (t-gralav) 10-Mar-2000 --*/ #include "precomp.h" #pragma hdrstop // Needed ? (what does it do?) // // Verbose flags (for device extensions) // #define VERBOSE_1 0x01 #define VERBOSE_2 0x02 // // BUG BUG // These need to be converted to enums in the ACPI Driver // #define DATAF_BUFF_ALIAS 0x00000001 #define DATAF_GLOBAL_LOCK 0x00000002 #define OBJTYPE_UNKNOWN 0x00 #define OBJTYPE_INTDATA 0x01 #define OBJTYPE_STRDATA 0x02 #define OBJTYPE_BUFFDATA 0x03 #define OBJTYPE_PKGDATA 0x04 #define OBJTYPE_FIELDUNIT 0x05 #define OBJTYPE_DEVICE 0x06 #define OBJTYPE_EVENT 0x07 #define OBJTYPE_METHOD 0x08 #define OBJTYPE_MUTEX 0x09 #define OBJTYPE_OPREGION 0x0a #define OBJTYPE_POWERRES 0x0b #define OBJTYPE_PROCESSOR 0x0c #define OBJTYPE_THERMALZONE 0x0d #define OBJTYPE_BUFFFIELD 0x0e #define OBJTYPE_DDBHANDLE 0x0f #define OBJTYPE_DEBUG 0x10 #define OBJTYPE_INTERNAL 0x80 #define OBJTYPE_OBJALIAS (OBJTYPE_INTERNAL + 0x00) #define OBJTYPE_DATAALIAS (OBJTYPE_INTERNAL + 0x01) #define OBJTYPE_BANKFIELD (OBJTYPE_INTERNAL + 0x02) #define OBJTYPE_FIELD (OBJTYPE_INTERNAL + 0x03) #define OBJTYPE_INDEXFIELD (OBJTYPE_INTERNAL + 0x04) #define OBJTYPE_DATA (OBJTYPE_INTERNAL + 0x05) #define OBJTYPE_DATAFIELD (OBJTYPE_INTERNAL + 0x06) #define OBJTYPE_DATAOBJ (OBJTYPE_INTERNAL + 0x07) // definition of FADT.flags bits // this one bit flag indicates whether or not the WBINVD instruction works properly,if this bit is not set we can not use S2, S3 states, or // C3 on MP machines #define WRITEBACKINVALIDATE_WORKS_BIT 0 #define WRITEBACKINVALIDATE_WORKS (1 << WRITEBACKINVALIDATE_WORKS_BIT) // this flag indicates if wbinvd works EXCEPT that it does not invalidate the cache #define WRITEBACKINVALIDATE_DOESNT_INVALIDATE_BIT 1 #define WRITEBACKINVALIDATE_DOESNT_INVALIDATE (1 << WRITEBACKINVALIDATE_DOESNT_INVALIDATE_BIT) // this flag indicates that the C1 state is supported on all processors. #define SYSTEM_SUPPORTS_C1_BIT 2 #define SYSTEM_SUPPORTS_C1 (1 << SYSTEM_SUPPORTS_C1_BIT) // this one bit flag indicates whether support for the C2 state is restricted to uniprocessor machines #define P_LVL2_UP_ONLY_BIT 3 #define P_LVL2_UP_ONLY (1 << P_LVL2_UP_ONLY_BIT) // this bit indicates whether the PWR button is treated as a fix feature (0) or a generic feature (1) #define PWR_BUTTON_GENERIC_BIT 4 #define PWR_BUTTON_GENERIC (1 << PWR_BUTTON_GENERIC_BIT) #define SLEEP_BUTTON_GENERIC_BIT 5 #define SLEEP_BUTTON_GENERIC (1 << SLEEP_BUTTON_GENERIC_BIT) // this bit indicates whether the RTC wakeup status is reported in fix register space (0) or not (1) #define RTC_WAKE_GENERIC_BIT 6 #define RTC_WAKE_GENERIC (1 << RTC_WAKE_GENERIC_BIT) #define RTC_WAKE_FROM_S4_BIT 7 #define RTC_WAKE_FROM_S4 (1 << RTC_WAKE_FROM_S4_BIT) // This bit indicates whether the machine implements a 24 or 32 bit timer. #define TMR_VAL_EXT_BIT 8 #define TMR_VAL_EXT (1 << TMR_VAL_EXT_BIT) // This bit indicates whether the machine supports docking #define DCK_CAP_BIT 9 #define DCK_CAP (1 << DCK_CAP_BIT) // This bit indicates whether the machine supports reset #define RESET_CAP_BIT 10 #define RESET_CAP (1 << RESET_CAP_BIT) #define SEALED_CASE_CAP_BIT 11 #define SEALED_CASE_CAP (1 << SEALED_CASE_CAP_BIT) #define HEADLESS_CAP_BIT 12 #define HEADLESS_CAP (1 << HEADLESS_CAP_BIT) #define CPU_SW_SLP_BIT 13 #define CPU_SW_SLP (1 << CPU_SW_SLP_BIT) // // Definition of FADT.boot_arch flags // #define LEGACY_DEVICES 1 #define I8042 2 // // Verbose flags (for contexts) // #define VERBOSE_CONTEXT 0x01 #define VERBOSE_CALL 0x02 #define VERBOSE_HEAP 0x04 #define VERBOSE_OBJECT 0x08 #define VERBOSE_NSOBJ 0x10 #define VERBOSE_RECURSE 0x20 UCHAR Buffer[2048]; #define RSDP_SIGNATURE 0x2052545020445352 // "RSD PTR " #define RSDT_SIGNATURE 0x54445352 // "RSDT" #define FADT_SIGNATURE 0x50434146 // "FACP" #define FACS_SIGNATURE 0x53434146 // "FACS" #define APIC_SIGNATURE 0x43495041 // "APIC" #define SRAT_SIGNATURE 0x54415253 // "SRAT" #ifndef NEC_98 #define RSDP_SEARCH_RANGE_BEGIN 0xE0000 // physical address where we begin searching for the RSDP #else // NEC_98 #define RSDP_SEARCH_RANGE_BEGIN 0xE8000 // physical address where we begin searching for the RSDP #endif // NEC_98 #define RSDP_SEARCH_RANGE_END 0xFFFFF #define RSDP_SEARCH_RANGE_LENGTH (RSDP_SEARCH_RANGE_END-RSDP_SEARCH_RANGE_BEGIN+1) #define RSDP_SEARCH_INTERVAL 16 // search on 16 byte boundaries // FACS Stuff ************************************************************************************ // FACS Flags definitions #define FACS_S4BIOS_SUPPORTED_BIT 0 // flag indicates whether or not the BIOS will save/restore memory around S4 #define FACS_S4BIOS_SUPPORTED (1 << FACS_S4BIOS_SUPPORTED_BIT) // FACS.GlobalLock bit field definitions #define GL_PENDING_BIT 0x00 #define GL_PENDING (1 << GL_PENDING_BIT) #define GL_OWNER_BIT 0x01 #define GL_OWNER (1 << GL_OWNER_BIT) //#define GL_NON_RESERVED_BITS_MASK (GL_PENDING+GL_OWNED) // MAPIC Stuff ************************************************************************************ // Multiple APIC description table // Multiple APIC structure flags #define PCAT_COMPAT_BIT 0 // indicates that the system also has a dual 8259 pic setup. #define PCAT_COMPAT (1 << PCAT_COMPAT_BIT) // APIC Structure Types #define PROCESSOR_LOCAL_APIC 0 #define IO_APIC 1 #define ISA_VECTOR_OVERRIDE 2 #define IO_NMI_SOURCE 3 #define LOCAL_NMI_SOURCE 4 #define ADDRESS_EXTENSION_STRUCTURE 5 #define IO_SAPIC 6 #define LOCAL_SAPIC 7 #define PLATFORM_INTERRUPT_SOURCE 8 #define PROCESSOR_LOCAL_APIC_LENGTH 8 #define IO_APIC_LENGTH 12 #define ISA_VECTOR_OVERRIDE_LENGTH 10 #define IO_NMI_SOURCE_LENGTH 8 #define LOCAL_NMI_SOURCE_LENGTH 6 #define PLATFORM_INTERRUPT_SOURCE_LENGTH 16 #define IO_SAPIC_LENGTH 16 #define PROCESSOR_LOCAL_SAPIC_LENGTH 12 // Platform Interrupt Types #define PLATFORM_INT_PMI 1 #define PLATFORM_INT_INIT 2 #define PLATFORM_INT_CPE 3 // Processor Local APIC Flags #define PLAF_ENABLED_BIT 0 #define PLAF_ENABLED (1 << PLAF_ENABLED_BIT) // These defines come from the MPS 1.4 spec, section 4.3.4 and they are referenced as // such in the ACPI spec. #define PO_BITS 3 #define POLARITY_HIGH 1 #define POLARITY_LOW 3 #define POLARITY_CONFORMS_WITH_BUS 0 #define EL_BITS 0xc #define EL_BIT_SHIFT 2 #define EL_EDGE_TRIGGERED 4 #define EL_LEVEL_TRIGGERED 0xc #define EL_CONFORMS_WITH_BUS 0 #define FADT_REV_1_SIZE 116 #define FADT_REV_2_SIZE 129 #define FADT_REV_3_SIZE 244 // // SRAT Stuff // #define SRAT_ENTRY_TYPE_PROCESSOR 0 #define SRAT_ENTRY_TYPE_MEMORY 1 // GBL Stuff ************************************************************************************ // // This structure lets us know the state of one entry in the RSDT // // INF Stuff ************************************************************************************ // // descriptions of bits in ACPIInformation.ACPI_Flags // #define C2_SUPPORTED_BIT 3 #define C2_SUPPORTED (1 << C2_SUPPORTED_BIT) #define C3_SUPPORTED_BIT 4 #define C3_SUPPORTED (1 << C3_SUPPORTED_BIT) #define C3_PREFERRED_BIT 5 #define C3_PREFERRED (1 << C3_PREFERRED_BIT) // // descriptions of bits in ACPIInformation.ACPI_Capabilities // #define CSTATE_C1_BIT 4 #define CSTATE_C1 (1 << CSTATE_C1_BIT) #define CSTATE_C2_BIT 5 #define CSTATE_C2 (1 << CSTATE_C2_BIT) #define CSTATE_C3_BIT 6 #define CSTATE_C3 (1 << CSTATE_C3_BIT) #define DUMP_FLAG_NO_INDENT 0x000001 #define DUMP_FLAG_NO_EOL 0x000002 #define DUMP_FLAG_SINGLE_LINE 0x000004 #define DUMP_FLAG_TABLE 0x000008 #define DUMP_FLAG_LONG_NAME 0x000010 #define DUMP_FLAG_SHORT_NAME 0x000020 #define DUMP_FLAG_SHOW_BIT 0x000040 #define DUMP_FLAG_ALREADY_INDENTED 0x000080 typedef struct _FLAG_RECORD { ULONGLONG Bit; PCCHAR ShortName; PCCHAR LongName; PCCHAR NotShortName; PCCHAR NotLongName; } FLAG_RECORD, *PFLAG_RECORD; FLAG_RECORD PM1ControlFlags[] = { { 0x0001, "", "SCI_EN" , NULL, NULL }, { 0x0002, "", "BM_RLD" , NULL, NULL }, { 0x0004, "", "GBL_RLS" , NULL, NULL }, { 0x0400, "", "SLP_TYP0" , NULL, NULL }, { 0x0800, "", "SLP_TYP1" , NULL, NULL }, { 0x1000, "", "SLP_TYP2" , NULL, NULL }, { 0x2000, "", "SLP_EN" , NULL, NULL }, }; FLAG_RECORD PM1StatusFlags[] = { { 0x0001, "", "TMR_STS" , NULL, NULL }, { 0x0010, "", "BM_STS" , NULL, NULL }, { 0x0020, "", "GBL_STS" , NULL, NULL }, { 0x0100, "", "PWRBTN_STS" , NULL, NULL }, { 0x0200, "", "SLPBTN_STS" , NULL, NULL }, { 0x0400, "", "RTC_STS" , NULL, NULL }, { 0x8000, "", "WAK_STS" , NULL, NULL }, }; FLAG_RECORD PM1EnableFlags[] = { { 0x0001, "", "TMR_EN" , NULL, NULL }, { 0x0020, "", "GBL_EN" , NULL, NULL }, { 0x0100, "", "PWRBTN_EN" , NULL, NULL }, { 0x0200, "", "SLPBTN_EN" , NULL, NULL }, { 0x0400, "", "RTC_EN" , NULL, NULL }, }; #define RSDTELEMENT_MAPPED 0x1 ULONG64 AcpiRsdtAddress = 0; ULONG64 AcpiFadtAddress = 0; ULONG64 AcpiFacsAddress = 0; ULONG64 AcpiMapicAddress = 0; // // Local Function Prototypes // VOID dumpNSObject(IN ULONG64 Address, IN ULONG Verbose, IN ULONG IndentLevel); // // Actual code // BOOL ReadPhysicalOrVirtual( IN ULONG64 Address, IN PVOID Buffer, IN ULONG Size, IN OUT PULONG ReturnLength, IN BOOL Virtual ) /*++ Routine Description: This is a way to abstract out the differences between ROM images and mapped memory Arguments: Address - Where (either physical, or virtual) the buffer is located Buffer - Address of where to copy the memory to Size - How many bytes to copy (maximum) ReturnLength - How many bytes where copied Virtual - False if this is physical memory --*/ { BOOL status = TRUE; PHYSICAL_ADDRESS physicalAddress = { 0L, 0L }; if (Virtual) { status = ReadMemory( Address, Buffer, Size, ReturnLength ); } else { physicalAddress.QuadPart = Address; ReadPhysical( physicalAddress.QuadPart, Buffer, Size, ReturnLength ); } if (ReturnLength && *ReturnLength != Size) { // // Didn't get enough memory // status = FALSE; } return status; } ULONG64 GetPointerFromAddressPhysicalOrVirtual( IN ULONG64 Address, IN BOOL Virtual ) { ULONG answer32 = 0; ULONG64 answer64 = 0; ULONG size; ULONG returnLength; BOOL status; size = GetTypeSize("hal!ULONG_PTR"); if (size == 4) { status = ReadPhysicalOrVirtual(Address, &answer32, size, &returnLength, Virtual); answer64 = (ULONG64) answer32; } else { status = ReadPhysicalOrVirtual(Address, &answer64, size, &returnLength, Virtual); } if ((status == FALSE) || (returnLength != size)) { return 0; } return answer64; } BOOLEAN findRSDT( IN PULONG64 Address ) /*++ Routine Description: This searchs the memory on the target system for the RSDT pointer Arguments: Address - Where to store the result Return Value: TRUE - If we found the RSDT --*/ { PHYSICAL_ADDRESS address = { 0L, 0L }; UCHAR index; UCHAR sum; ULONG64 limit; ULONG returnLength = 0; ULONG64 start, initAddress; ULONGLONG compSignature; ULONG addr; int siz; // // Calculate the start and end of the search range // start = RSDP_SEARCH_RANGE_BEGIN; limit = start + RSDP_SEARCH_RANGE_LENGTH - RSDP_SEARCH_INTERVAL; dprintf( "Searching for RSDP."); // // Loop for a while // for (; start <= limit; start += RSDP_SEARCH_INTERVAL) { if (start % (RSDP_SEARCH_INTERVAL * 100 ) == 0) { dprintf("."); if (CheckControlC()) { return FALSE; } } // // Read the data from the target // address.LowPart = (ULONG) start; memset( Buffer, 0, GetTypeSize("hal!_RSDT_32") ); ReadPhysical( address.QuadPart, &Buffer, GetTypeSize("hal!_RSDP"), &returnLength); if (returnLength != GetTypeSize("hal!_RSDP")) { dprintf( "%#08lx: Read %#08lx of %#08lx bytes\n", start, returnLength, GetTypeSize("hal!_RSDP") ); return FALSE; } // // Is this a match? // // INIT TYPE READ PHYSICAL TAKES MAYBE 15 TIME LONGER! initAddress = InitTypeReadPhysical( address.QuadPart, hal!_RSDP ); if ( ReadField(Signature) != RSDP_SIGNATURE) { continue; } // // Check the checksum out // for (index = 0, sum = 0; index < GetTypeSize("hal!_RSDP"); index++) { sum = (UCHAR) (sum + *( (UCHAR *) ( (ULONG64) &Buffer + index ) ) ); } if (sum != 0) { continue; } // // Found RSDP // dprintf("\nRSDP - %016I64x\n", start ); initAddress = InitTypeReadPhysical( address.QuadPart, hal!_RSDP ); // The following error message has been remarked out because the FIRST call to // a InitTypeReadPhysical does NOT access the memory (and returns error 0x01: // MEMORY_READ_ERROR. This is done when ReadField happens, so IT STILL WORKS. // The false error message is a kd bug, and will be fixed in a later build. // Once this has been done, feel free to unremark it. // if (initAddress) { // dprintf("Failed to initialize hal!_RSDP. Error code: %d.", initAddress); // } initAddress = ReadField(Signature); memset( Buffer, 0, 2048 ); memcpy( Buffer, &initAddress, GetTypeSize("ULONGLONG") ); dprintf(" Signature: %s\n", Buffer ); dprintf(" Checksum: %#03x\n", (UCHAR) ReadField(Checksum) ); initAddress = ReadField(OEMID); GetFieldOffset( "hal!_RSDP", "OEMID", &addr); memset( Buffer, 0, GetTypeSize("ULONGLONG") ); ReadPhysical( (address.QuadPart + (ULONG64) addr), &Buffer, 6, &returnLength); if (returnLength != 6) { // 6 is hard-coded in the specs dprintf( "%#08lx: Read %#08lx of 6 bytes in OEMID\n", (address.QuadPart + (ULONG64)addr), returnLength ); return FALSE; } dprintf(" OEMID: %s\n", Buffer ); dprintf(" Reserved: %#02x\n", ReadField(Reserved) ); dprintf(" RsdtAddress: %016I64x\n", ReadField(RsdtAddress) ); // // Done // *Address = ReadField(RsdtAddress);//rsdp.RsdtAddress; return TRUE; } return FALSE; } PUCHAR ReadPhysVirField( IN ULONG64 Address, IN PUCHAR StructName, IN PUCHAR FieldName, IN ULONG Length, IN BOOLEAN Physical ) /*++ Routine Description: This function returns a text string field from physical or virtual memory into Buffer, then returns Buffer Arugments: Address - Where the table is located StructName - Structure name FieldName - Field name Length - Length (number of characters) in field Physical - Read from Physical (TRUE) or Virtual Memory Return Value: String containing contents --*/ { ULONG addr; ULONG returnLength; memset( Buffer, 0, Length + 1); GetFieldOffset( StructName, FieldName, &addr); if (Physical) { ReadPhysical( (Address + (ULONG64) addr), &Buffer, Length, &returnLength); } else { ReadMemory( (Address + (ULONG64) addr), &Buffer, Length, &returnLength); } return Buffer; } VOID dumpHeader( IN ULONG64 Address, IN BOOLEAN Verbose, IN BOOLEAN Physical ) /*++ Routine Description: This function dumps out a table header Arugments: Address - Where the table is located Header - The table header Verbose - How much information to give Return Value: NULL --*/ { if (Physical) { InitTypeReadPhysical( Address, hal!_DESCRIPTION_HEADER); } else { InitTypeRead( Address, hal!_DESCRIPTION_HEADER); } if (Verbose) { dprintf( "HEADER - %016I64x\n" " Signature: %s\n" " Length: 0x%08lx\n" " Revision: 0x%02x\n" " Checksum: 0x%02x\n", Address, ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "Signature", sizeof(ULONG), Physical), (ULONG) ReadField(Length), (UCHAR) ReadField(Revision), (UCHAR) ReadField(Checksum) ); dprintf(" OEMID: %s\n", ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "OEMID", 6, Physical) ); dprintf(" OEMTableID: %s\n", ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "OEMTableID", 8, Physical) ); dprintf(" OEMRevision: 0x%08lx\n", ReadField(OEMRevision) ); dprintf(" CreatorID: %s\n", ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "CreatorID", 4, Physical) ); dprintf(" CreatorRev: 0x%08lx\n", ReadField(CreatorRev) ); } else { dprintf( " %s @(%016I64x) Rev: %#03x Len: %#08lx", ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "Signature", sizeof(ULONG64), Physical), Address, (UCHAR) ReadField(Revision), (ULONG) ReadField(Length) ); dprintf(" TableID: %s\n", ReadPhysVirField(Address, "hal!_DESCRIPTION_HEADER", "OEMTableID", 8, Physical) ); } return; } VOID dumpRSDT( IN ULONG64 Address, IN BOOLEAN Physical ) /*++ Routine Description: This search the dumps the RSDT table Arguments: Pointer to the table Return Value: NONE --*/ { BOOL status; ULONG64 index; ULONG64 numEntries; ULONG addr; ULONG returnLength = 0; ULONG64 a; dprintf("RSDT - "); if (Physical) { // The following do NOT have their status read as a bug in the return value would give us errors when none exist. The signature check would catch them, anyway. InitTypeReadPhysical( Address, hal!_DESCRIPTION_HEADER); } else { InitTypeRead( Address, hal!_DESCRIPTION_HEADER); } if (ReadField(Signature) != RSDT_SIGNATURE) { dprintf( "dumpRSDT: Invalid Signature 0x%08lx != RSDT_SIGNATURE\n", ReadField(Signature) ); dumpHeader( Address, TRUE, Physical ); return; } dumpHeader( Address, TRUE, Physical ); dprintf("RSDT - BODY - %016I64x\n", Address + GetTypeSize("hal!_DESCRIPTION_HEADER") ); numEntries = ( ReadField(Length) - GetTypeSize("hal!_DESCRIPTION_HEADER") ) / sizeof(ULONG); GetFieldOffset( "hal!_RSDT_32", "Tables", &addr); for (index = 0; index < numEntries; index++) { // // Note: unless things radically change, the pointers in the // rsdt will always point to bios memory! // if (Physical) { ReadPhysical(Address + index + (ULONG64) addr, &a, 4, &returnLength); } else { ReadPointer(Address + index + (ULONG64) addr, &a); } dumpHeader( a, FALSE, TRUE ); } return; } VOID dumpFADT( IN ULONG64 Address ) /*++ Routine Description: This dumps the FADT at the specified address Arguments: The address where the FADT is located at Return Value: NONE --*/ { ULONG fadtLength; ULONG addr; ULONG flags; UCHAR Revision; UCHAR AddressSpaceID; ULONG64 reset_reg_addr; PCHAR addressSpace; BOOLEAN Physical = FALSE; // // First check to see if we find the correct things // dprintf("FADT -- %p", Address); if (Physical) { InitTypeReadPhysical( Address, hal!_DESCRIPTION_HEADER); } else { InitTypeRead( Address, hal!_DESCRIPTION_HEADER); } if (ReadField(Signature) != FADT_SIGNATURE) { dprintf( "dumpRSDT: Invalid Signature 0x%08lx != FADT_SIGNATURE\n", ReadField(Signature) ); dumpHeader( Address, TRUE, Physical ); return; } Revision = (UCHAR)ReadField(Revision); if (Revision == 1) { fadtLength = FADT_REV_1_SIZE; } else if (Revision == 2) { fadtLength = FADT_REV_2_SIZE; } else if (Revision == 3) { fadtLength = FADT_REV_3_SIZE; } else { dprintf("FADT revision is %d, which is not understood by this debugger\n", Revision); fadtLength = FADT_REV_3_SIZE; } // // Do we have a correctly sized data structure // if ((ULONG) ReadField(Length) < fadtLength) { dprintf( "dumpFADT: (%016I64x) Length (%#08lx) is not the size of the FADT (%#08lx)\n", Address, (ULONG) ReadField(Length), fadtLength ); dumpHeader( Address, TRUE, Physical ); return; } // // Dump the table // dumpHeader( Address, TRUE, Physical ); if (Physical) { // Physical/Virtual should have been established above InitTypeReadPhysical( Address, hal!_FADT); } else { InitTypeRead( Address, hal!_FADT); } dprintf( "FADT - BODY - %016I64x\n" " FACS: 0x%08lx\n" " DSDT: 0x%08lx\n" " Int Model: %s\n" " SCI Vector: 0x%03x\n" " SMI Port: 0x%08lx\n" " ACPI On Value: 0x%03x\n" " ACPI Off Value: 0x%03x\n" " SMI CMD For S4 State: 0x%03x\n" " PM1A Event Block: 0x%08lx\n" " PM1B Event Block: 0x%08lx\n" " PM1 Event Length: 0x%03x\n" " PM1A Control Block: 0x%08lx\n" " PM1B Control Block: 0x%08lx\n" " PM1 Control Length: 0x%03x\n" " PM2 Control Block: 0x%08lx\n" " PM2 Control Length: 0x%03x\n" " PM Timer Block: 0x%08lx\n" " PM Timer Length: 0x%03x\n" " GP0 Block: 0x%08lx\n" " GP0 Length: 0x%03x\n" " GP1 Block: 0x%08lx\n" " GP1 Length: 0x%08lx\n" " GP1 Base: 0x%08lx\n" " C2 Latency: 0x%05lx\n" " C3 Latency: 0x%05lx\n" " Memory Flush Size: 0x%05lx\n" " Memory Flush Stride: 0x%05lx\n" " Duty Cycle Index: 0x%03x\n" " Duty Cycle Index Width: 0x%03x\n" " Day Alarm Index: 0x%03x\n" " Month Alarm Index: 0x%03x\n" " Century byte (CMOS): 0x%03x\n" " Boot Architecture: 0x%04x\n" " Flags: 0x%08lx\n", Address + GetTypeSize("hal!_DESCRIPTION_HEADER"), (ULONG) ReadField(facs), (ULONG) ReadField(dsdt), (ReadField(int_model) == 0 ? "Dual PIC" : "Multiple APIC" ), (USHORT) ReadField(sci_int_vector), (ULONG) ReadField(smi_cmd_io_port), (UCHAR) ReadField(acpi_on_value), (UCHAR) ReadField(acpi_off_value), (UCHAR) ReadField(s4bios_req), (ULONG) ReadField(pm1a_evt_blk_io_port), (ULONG) ReadField(pm1b_evt_blk_io_port), (UCHAR) ReadField(pm1_evt_len), (ULONG) ReadField(pm1a_ctrl_blk_io_port), (ULONG) ReadField(pm1b_ctrl_blk_io_port), (UCHAR) ReadField(pm1_ctrl_len), (ULONG) ReadField(pm2_ctrl_blk_io_port), (UCHAR) ReadField(pm2_ctrl_len), (ULONG) ReadField(pm_tmr_blk_io_port), (UCHAR) ReadField(pm_tmr_len), (ULONG) ReadField(gp0_blk_io_port), (UCHAR) ReadField(gp0_blk_len), (ULONG) ReadField(gp1_blk_io_port), (UCHAR) ReadField(gp1_blk_len), (UCHAR) ReadField(gp1_base), (USHORT) ReadField(lvl2_latency), (USHORT) ReadField(lvl3_latency), #ifndef _IA64_ // XXTF (USHORT) ReadField(flush_size), (USHORT) ReadField(flush_stride), (UCHAR) ReadField(duty_offset), (UCHAR) ReadField(duty_width), #endif (UCHAR) ReadField(day_alarm_index), (UCHAR) ReadField(month_alarm_index), (UCHAR) ReadField(century_alarm_index), (USHORT) ReadField(boot_arch), (ULONG) ReadField(flags) ); flags = (ULONG) ReadField(flags); if (flags & WRITEBACKINVALIDATE_WORKS) { dprintf(" Write Back Invalidate is supported\n"); } if (flags & WRITEBACKINVALIDATE_DOESNT_INVALIDATE) { dprintf(" Write Back Invalidate doesn't invalidate the caches\n"); } if (flags & SYSTEM_SUPPORTS_C1) { dprintf(" System supports C1 Power state on all processors\n"); } if (flags & P_LVL2_UP_ONLY) { dprintf(" System supports C2 in MP and UP configurations\n"); } if (flags & PWR_BUTTON_GENERIC) { dprintf(" Power Button is treated as a generic feature\n"); } if (flags & SLEEP_BUTTON_GENERIC) { dprintf(" Sleep Button is treated as a generic feature\n"); } if (flags & RTC_WAKE_GENERIC) { dprintf(" RTC Wake is not supported in fixed register space\n"); } if (flags & RTC_WAKE_FROM_S4) { dprintf(" RTC Wake can work from an S4 state\n"); } if (flags & TMR_VAL_EXT) { dprintf(" TMR_VAL implemented as 32-bit value\n"); } if (Revision > 1) { if (!(ReadField(boot_arch) & LEGACY_DEVICES)) { dprintf(" The machine does not contain legacy ISA devices\n"); } if (!(ReadField(boot_arch) & I8042)) { dprintf(" The machine does not contain a legacy i8042\n"); } if (flags & RESET_CAP) { dprintf(" The reset register is supported\n"); dprintf(" Reset Val: %x\n", ReadField(reset_val)); GetFieldOffset("hal!_FADT", "reset_reg", &addr); GetFieldValue(Address + (ULONG64)addr, "hal!_GEN_ADDR", "AddressSpaceID", AddressSpaceID); switch (AddressSpaceID) { case 0: addressSpace = "Memory"; break; case 1: addressSpace = "I/O"; break; case 2: addressSpace = "PCIConfig"; break; default: addressSpace = "undefined"; } GetFieldOffset("hal!_GEN_ADDR", "Address", &addr); GetFieldValue(Address + (ULONG64)addr, "hal!_LARGE_INTEGER", "QuadPart", reset_reg_addr); dprintf(" Reset register: %s - %016I64x\n", addressSpace, reset_reg_addr ); } if (flags & DCK_CAP_BIT) { dprintf(" The machine has the docking capable attribute.\n"); } if (flags & SEALED_CASE_CAP_BIT) { dprintf(" The machine has the sealed case system attribute.\n"); } if (flags & HEADLESS_CAP_BIT) { dprintf(" The machine is reported has the sealed case system attribute.\n"); } if (flags & CPU_SW_SLP_BIT) { dprintf(" The machine has the CPU_SW_SLP_BIT set.\n"); } } return; } BOOL GetUlongPtr ( IN PCHAR String, IN PULONG64 Address ) { ULONG64 Location; Location = GetExpression( String ); if (!Location) { dprintf("Sorry: Unable to get %s.\n",String); return FALSE; } return ReadPointer(Location, Address); } DECLARE_API( rsdt ) { BOOLEAN Physical = FALSE; if (args != NULL) { AcpiRsdtAddress = GetExpression( args ); // Should work } if (AcpiRsdtAddress == 0) { UINT64 status; // formerly BOOL ULONG64 address; status = GetUlongPtr( "ACPI!AcpiInformation", &address ); if (status == TRUE) { status = GetFieldValue(address,"ACPI!_ACPIInformation","RootSystemDescTable",AcpiRsdtAddress); } } if (AcpiRsdtAddress == 0) { if (!findRSDT( &AcpiRsdtAddress) ) { dprintf("Could not locate the RSDT pointer\n"); return E_INVALIDARG; } Physical = TRUE; } dumpRSDT( AcpiRsdtAddress, Physical ); return S_OK; } DECLARE_API( fadt ) { if (args != NULL && *args != '\0') { AcpiFadtAddress = GetExpression( args ); } if (AcpiFadtAddress == 0) { AcpiFadtAddress = GetExpression( "HAL!HalpFixedAcpiDescTable" ); } if (AcpiFadtAddress == 0) { dprintf("fadt
\n"); return E_INVALIDARG; } dumpFADT( AcpiFadtAddress ); return S_OK; } VOID dumpFACS( IN ULONG64 Address ) /*++ Routine Description: This dumps the FADT at the specified address Arguments: The address where the FADT is located at Return Value: NONE --*/ { BOOLEAN Physical = FALSE; // // Read the data // dprintf("FACS - %016I64x\n", Address); if (Physical) { InitTypeReadPhysical( Address, hal!_FACS); } else { InitTypeRead( Address, hal!_FACS); } if (ReadField(Signature) != FACS_SIGNATURE) { dprintf( "dumpFACS: Invalid Signature 0x%08lx != FACS_SIGNATURE\n", (ULONG) ReadField(Signature) ); return; } // // Dump the table // dprintf( " Signature: %s\n" " Length: %#08lx\n" " Hardware Signature: %#08lx\n" " Firmware Wake Vector: %#08lx\n" " Global Lock : %#08lx\n", ReadPhysVirField(Address, "hal!_FACS", "Signature", sizeof(ULONG), Physical), ReadField(Length), ReadField(HardwareSignature), ReadField(pFirmwareWakingVector), ReadField(GlobalLock) ); if ( (ReadField(GlobalLock) & GL_PENDING) ) { dprintf(" Request for Ownership Pending\n"); } if ( (ReadField(GlobalLock) & GL_OWNER) ) { dprintf(" Global Lock is Owned\n"); } dprintf(" Flags: %#08lx\n", (ULONG) ReadField(Flags) ); if ( (ReadField(Flags) & FACS_S4BIOS_SUPPORTED) ) { dprintf(" S4BIOS_REQ Supported\n"); } return; } DECLARE_API( facs ) { if (args != NULL) { AcpiFacsAddress = GetExpression( args ); } if (AcpiFacsAddress == 0) { BOOL status; UINT64 address; status = GetUlongPtr( "ACPI!AcpiInformation", &address ); if (status == TRUE) { status = GetFieldValue(address,"ACPI!_ACPIInformation","FirmwareACPIControlStructure",AcpiFacsAddress); } } if (AcpiFacsAddress == 0) { dprintf("facs \n"); return E_INVALIDARG; } dumpFACS( AcpiFacsAddress ); return S_OK; } // ReturnXxx Functions - these are just a few functions I wrote that simplify // dealing with certain types of Symbols UCHAR ReturnUCHAR( IN ULONG64 Address, IN PUCHAR StructName, IN PUCHAR FieldName ) /*++ Routine Description: Return char using GetFieldValue --*/ { UCHAR returnChar; if (GetFieldValue(Address, StructName, FieldName, returnChar)){ // // Failed. try just the base symbols name before giving up // PUCHAR symName=NULL; ULONG i; for(i=strlen(StructName); i > 0 && StructName[i] != '!'; i--); i++; symName = StructName + i; // // Try again // GetFieldValue(Address, symName, FieldName, returnChar); } return returnChar; } ULONG ReturnUSHORT( IN ULONG64 Address, IN PUCHAR StructName, IN PUCHAR FieldName ) /*++ Routine Description: Return USHORT using GetFieldValue --*/ { USHORT returnUSHORT; if (GetFieldValue(Address, StructName, FieldName, returnUSHORT)){ // // Failed. try just the base symbols name before giving up // PUCHAR symName=NULL; ULONG i; for(i=strlen(StructName); i > 0 && StructName[i] != '!'; i--); i++; symName = StructName + i; // // Try again // GetFieldValue(Address, symName, FieldName, returnUSHORT); } return returnUSHORT; } ULONG ReturnULONG( IN ULONG64 Address, IN PUCHAR StructName, IN PUCHAR FieldName ) /*++ Routine Description: Return ULONG using GetFieldValue --*/ { ULONG returnULONG; if (GetFieldValue(Address, StructName, FieldName, returnULONG)){ // // Failed. try just the base symbols name before giving up // PUCHAR symName=NULL; ULONG i; for(i=strlen(StructName); i > 0 && StructName[i] != '!'; i--); i++; symName = StructName + i; // // Try again // GetFieldValue(Address, symName, FieldName, returnULONG); } return returnULONG; } ULONG64 ReturnULONG64( IN ULONG64 Address, IN PUCHAR StructName, IN PUCHAR FieldName ) /*++ Routine Description: Return ULONG64 using GetFieldValue --*/ { ULONG64 returnULONG64; if (GetFieldValue(Address, StructName, FieldName, returnULONG64)){ // // Failed. try just the base symbols name before giving up // PUCHAR symName=NULL; ULONG i; for(i=strlen(StructName); i > 0 && StructName[i] != '!'; i--); i++; symName = StructName + i; // // Try again // GetFieldValue(Address, symName, FieldName, returnULONG64); } return returnULONG64; } VOID dumpMAPIC( IN ULONG64 Address ) /*++ Routine Description: This dumps the multiple apic table Arguments: Address of the table Return Value: None --*/ { BOOL hasMPSFlags; BOOL status; BOOL virtualMemory; ULONG mapicLength; ULONG64 iso; // interruptSourceOverride USHORT isoFlags; ULONG64 buffer; ULONG64 limit; ULONG index; ULONG returnLength; ULONG flags; ULONG get_value; BOOLEAN Physical = FALSE; // // First check to see if we find the correct things // dprintf("MAPIC - "); if (Physical) { InitTypeReadPhysical( Address, hal!_DESCRIPTION_HEADER); } else { InitTypeRead( Address, hal!_DESCRIPTION_HEADER); } if (ReadField(Signature) != APIC_SIGNATURE) { dprintf( "dumpFACS: Invalid Signature 0x%08lx != APIC_SIGNATURE (%x)\n", (ULONG) ReadField(Signature), APIC_SIGNATURE ); return; } mapicLength = (ULONG)ReadField(Length); dumpHeader( Address, TRUE, FALSE ); dprintf("MAPIC - BODY - %016I64x\n", Address + GetTypeSize("hal!_DESCRIPTION_HEADER") ); dprintf(" Local APIC Address: %#08lx\n", ReturnULONG(Address, "hal!_MAPIC","LocalAPICAddress")); GetFieldValue(Address,"hal!_MAPIC","Flags",get_value); dprintf(" Flags: %#08lx\n", get_value ); if (get_value & PCAT_COMPAT) { // Check the flags dprintf(" PC-AT dual 8259 compatible setup\n"); } //gsig2 GetFieldOffset( "hal!_MAPIC", "APICTables", &get_value); buffer = Address + get_value; limit = ( Address + ReadField(Length) ); while (buffer < limit) { if (CheckControlC()) { break; } // // Assume that no flags are set // hasMPSFlags = FALSE; // // Lets see what kind of table we have? // iso = (ULONG64) buffer; // // Is it a localApic? // if (ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "Type") == PROCESSOR_LOCAL_APIC) { buffer += ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "Length"); dprintf( " Processor Local Apic\n" " ACPI Processor ID: 0x%02x\n" " APIC ID: 0x%02x\n" " Flags: 0x%08lx\n", ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "ACPIProcessorID"), ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "APICID"), ReturnULONG(iso, "acpi!_PROCLOCALAPIC", "Flags") ); if (ReturnULONG(iso, "acpi!_PROCLOCALAPIC", "Flags") & PLAF_ENABLED) { dprintf(" Processor is Enabled\n"); } if (ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "Length") != PROCESSOR_LOCAL_APIC_LENGTH) { dprintf( " Local Apic has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "acpi!_PROCLOCALAPIC", "Length"), PROCESSOR_LOCAL_APIC_LENGTH ); break; } } else if (ReturnUCHAR(iso, "hal!_IOAPIC", "Type") == IO_APIC) { buffer += ReturnUCHAR(iso, "hal!_IOAPIC", "Length"); dprintf( " IO Apic\n" " IO APIC ID: 0x%02x\n" " IO APIC ADDRESS: 0x%08lx\n" " System Vector Base: 0x%08lx\n", ReturnUCHAR(iso, "hal!_IOAPIC", "IOAPICID"), ReturnULONG(iso, "hal!_IOAPIC", "IOAPICAddress"), ReturnULONG(iso, "hal!_IOAPIC", "SystemVectorBase") ); if (ReturnUCHAR(iso, "hal!_IOAPIC", "Length") != IO_APIC_LENGTH) { dprintf( " IO Apic has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_IOAPIC", "Length"), IO_APIC_LENGTH ); break; } } else if (ReturnUCHAR(iso,"hal!_ISA_VECTOR","Type") == ISA_VECTOR_OVERRIDE) { buffer += ReturnUCHAR(iso, "hal!_ISA_VECTOR", "Length"); GetFieldValue(iso, "hal!_ISA_VECTOR", "Flags", isoFlags); dprintf( " Interrupt Source Override\n" " Bus: 0x%02x\n" " Source: 0x%02x\n" " Global Interrupt: 0x%08lx\n" " Flags: 0x%04x\n", ReturnUCHAR(iso, "hal!_ISA_VECTOR", "Bus"), ReturnUCHAR(iso, "hal!_ISA_VECTOR", "Source"), ReturnULONG(iso, "hal!_ISA_VECTOR", "GlobalSystemInterruptVector"), isoFlags ); if (ReturnUCHAR(iso,"hal!_ISA_VECTOR","Length") != ISA_VECTOR_OVERRIDE_LENGTH) { dprintf( " Interrupt Source Override has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_ISA_VECTOR", "Length"), ISA_VECTOR_OVERRIDE_LENGTH ); break; } hasMPSFlags = TRUE; flags = isoFlags; } else if (ReturnUCHAR(iso,"acpi!_IO_NMISOURCE","Type") == IO_NMI_SOURCE) { buffer += ReturnUCHAR(iso, "acpi!_IO_NMISOURCE", "Length"); GetFieldValue(iso, "acpi!_IO_NMISOURCE", "Flags", isoFlags); dprintf( " Non Maskable Interrupt Source - on I/O APIC\n" " Flags: 0x%02x\n" " Global Interrupt: 0x%08lx\n", isoFlags, ReturnULONG(iso, "acpi!_IO_NMISOURCE", "GlobalSystemInterruptVector") ); if (ReturnUCHAR(iso,"acpi!_IO_NMISOURCE","Length") != IO_NMI_SOURCE_LENGTH) { dprintf( " Non Maskable Interrupt source has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "acpi!_IO_NMISOURCE", "Length"), IO_NMI_SOURCE_LENGTH ); break; } hasMPSFlags = TRUE; flags = isoFlags; } else if (ReturnUCHAR(iso,"hal!_LOCAL_NMISOURCE","Type") == LOCAL_NMI_SOURCE) { buffer += ReturnUCHAR(iso, "hal!_LOCAL_NMISOURCE", "Length"); GetFieldValue(iso, "hal!_LOCAL_NMISOURCE", "Flags", isoFlags); dprintf( " Non Maskable Interrupt Source - local to processor\n" " Flags: 0x%04x\n" " Processor: 0x%02x %s\n" " LINTIN: 0x%02x\n", isoFlags, ReturnUCHAR(iso, "hal!_LOCAL_NMISOURCE", "ProcessorID"), ReturnUCHAR(iso,"hal!_LOCAL_NMISOURCE","ProcessorID") == 0xff ? "(all)" : "", ReturnUCHAR(iso, "hal!_LOCAL_NMISOURCE", "LINTIN") ); if (ReturnUCHAR(iso,"hal!_LOCAL_NMISOURCE","Length") != LOCAL_NMI_SOURCE_LENGTH) { dprintf( " Non Maskable Interrupt source has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_LOCAL_NMISOURCE", "Length"), IO_NMI_SOURCE_LENGTH ); break; } hasMPSFlags = TRUE; flags = isoFlags; } else if (ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "Type") == LOCAL_SAPIC) { buffer += ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "Length"); dprintf( " Processor Local SAPIC\n" " ACPI Processor ID: 0x%02x\n" " APIC ID: 0x%02x\n" " APIC EID: 0x%02x\n" " Flags: 0x%08lx\n", ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "ACPIProcessorID"), ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "APICID"), ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "APICEID"), ReturnULONG(iso, "hal!_PROCLOCALSAPIC", "Flags") ); if (ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "Length") != PROCESSOR_LOCAL_SAPIC_LENGTH) { dprintf( " Processor Local SAPIC has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_PROCLOCALSAPIC", "Length"), PROCESSOR_LOCAL_SAPIC_LENGTH ); break; } } else if (ReturnUCHAR(iso, "hal!_IOSAPIC", "Type") == IO_SAPIC) { buffer += ReturnUCHAR(iso, "hal!_IOSAPIC", "Length"); dprintf( " IO SApic\n" " IO SAPIC ADDRESS: 0x%016I64x\n" " System Vector Base: 0x%08lx\n", ReturnULONG64(iso, "hal!_IOSAPIC", "IOSAPICAddress"), ReturnULONG(iso, "hal!_IOSAPIC", "SystemVectorBase") ); if (ReturnUCHAR(iso, "hal!_IOSAPIC", "Length") != IO_SAPIC_LENGTH) { dprintf( " IO SApic has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_IOSAPIC", "Length"), IO_SAPIC_LENGTH ); break; } } else if (ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "Type") == PLATFORM_INTERRUPT_SOURCE) { UCHAR InterruptType = ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "InterruptType"); buffer += ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "Length"); dprintf( " Platform Interrupt Source\n" " Flags: 0x%04x\n" " Interrupt Type: %s\n" " APICID: 0x%02x\n" " APICEID: 0x%02x\n" " IOSAPICVector: 0x%02x\n" " GlobalVector: 0x%08x\n", ReturnUSHORT(iso, "hal!_PLATFORM_INTERRUPT", "Flags"), InterruptType == PLATFORM_INT_PMI ? "PMI" : (InterruptType == PLATFORM_INT_INIT ? "INIT" : (InterruptType == PLATFORM_INT_CPE ? "CPE" : "UNKNOWN")), ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "APICID"), ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "APICEID"), ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "IOSAPICVector"), ReturnULONG(iso, "hal!_PLATFORM_INTERRUPT", "GlobalVector") ); if (ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "Length") != PLATFORM_INTERRUPT_SOURCE_LENGTH) { dprintf( " Platform Interrupt Source has length 0x%x instead of 0x%x\n", ReturnUCHAR(iso, "hal!_PLATFORM_INTERRUPT", "Length"), PLATFORM_INTERRUPT_SOURCE_LENGTH ); break; } } else { dprintf(" UNKNOWN RECORD (%p)\n", iso); dprintf(" Type: 0x%08x\n", ReturnUCHAR(iso,"hal!_IOAPIC","Type")); dprintf(" Length: 0x%08x\n", ReturnUCHAR(iso,"hal!_IOAPIC","Length")); // // Dont spin forever if we encounter an known with zero length // if ((ReturnUCHAR(iso,"hal!_IOAPIC","Length")) == 0) { break; } buffer += ReturnUCHAR(iso,"hal!_IOAPIC","Length"); } // // Do we have any flags to dump out? // if (hasMPSFlags) { switch (flags & PO_BITS) { case POLARITY_HIGH: dprintf(" POLARITY_HIGH\n"); break; case POLARITY_LOW: dprintf(" POLARITY_LOW\n"); break; case POLARITY_CONFORMS_WITH_BUS: dprintf(" POLARITY_CONFORMS_WITH_BUS\n"); break; default: dprintf(" POLARITY_UNKNOWN\n"); break; } switch (flags & EL_BITS) { case EL_EDGE_TRIGGERED: dprintf(" EL_EDGE_TRIGGERED\n"); break; case EL_LEVEL_TRIGGERED: dprintf(" EL_LEVEL_TRIGGERED\n"); break; case EL_CONFORMS_WITH_BUS: dprintf(" EL_CONFORMS_WITH_BUS\n"); break; default: dprintf(" EL_UNKNOWN\n"); break; } } } return; } DECLARE_API( mapic ) { if (args != NULL) { AcpiMapicAddress = GetExpression( args ); } if (AcpiMapicAddress == 0) { BOOL status; ULONG64 address; status = GetUlongPtr( "ACPI!AcpiInformation", &address ); if (status == TRUE) { status = GetFieldValue(address,"ACPI!_ACPIInformation","MultipleApicTable",AcpiMapicAddress); } } if (AcpiMapicAddress == 0) { dprintf("mapic \n"); return E_INVALIDARG; } dumpMAPIC( AcpiMapicAddress ); return S_OK; } VOID dumpSRAT( IN ULONG64 Address, IN BOOLEAN Physical ) /*++ Routine Description: This dumps the static resource affinity table Arguments: Address -- Address of the table Physical -- TRUE indicates a physical address, FALSE indicates a virtual address Return Value: None --*/ { BOOL status; ULONG sratLength; ULONG64 sratEnd; ULONG64 current; // // First check to see if we find the correct things // dprintf("SRAT - "); if (Physical) { InitTypeReadPhysical( Address, hal!_ACPI_SRAT); } else { InitTypeRead( Address, hal!_ACPI_SRAT); } if (ReadField(Header.Signature) != SRAT_SIGNATURE) { dprintf( "dumpSRAT: Invalid Signature 0x%08lx != SRAT_SIGNATURE (%x)\n", (ULONG) ReadField(Signature), SRAT_SIGNATURE ); return; } dumpHeader( Address, TRUE, Physical ); current = Address + GetTypeSize("hal!_ACPI_SRAT"); dprintf("SRAT - BODY - %016I64x\n", current); dprintf(" Table Revision: %d\n", (ULONG) ReadField(TableRevision)); sratEnd = Address + (ULONG)ReadField(Length); while (current < sratEnd) { if (CheckControlC()) { break; } if (Physical) { InitTypeReadPhysical( current, hal!_ACPI_SRAT_ENTRY); } else { InitTypeRead( current, hal!_ACPI_SRAT_ENTRY); } dprintf("ENTRY:\n\tType: 0x%02X\n\tLength: 0x%02X\n\tProximityId: 0x%02X\n", (ULONG) ReadField(Type), (ULONG) ReadField(Length), (ULONG) ReadField(ProximityDomain)); switch (ReadField(Type)) { case SRAT_ENTRY_TYPE_PROCESSOR: dprintf("\tProcessor:\n"); dprintf("\t\tEnabled: %s\n", ReadField(ApicAffinity.Flags.Enabled) ? "TRUE" : "FALSE"); dprintf("\t\tAPIC ID: 0x%02X", (ULONG) ReadField(ApicAffinity.ApicId)); if (TargetMachine == IMAGE_FILE_MACHINE_IA64) { dprintf(" SAPIC EID 0x%02X\n", (ULONG) ReadField(ApicAffinity.SApicEid)); } else { dprintf("\n"); } break; case SRAT_ENTRY_TYPE_MEMORY: dprintf("\tMemory:\n"); dprintf("\t\tEnabled: %s\n\t\tHotPlug: %s\n", ReadField(MemoryAffinity.Flags.Enabled) ? "TRUE" : "FALSE", ReadField(MemoryAffinity.Flags.HotPlug) ? "TRUE" : "FALSE"); dprintf("\t\tBase: %016I64X\tLength: %016I64X\n", ReadField(MemoryAffinity.Base), ReadField(MemoryAffinity.Length)); dprintf("\t\t (%016I64X - %016I64X)\n", ReadField(MemoryAffinity.Base), ReadField(MemoryAffinity.Base) + ReadField(MemoryAffinity.Length) - 1); break; default: dprintf("\tUNKNOWN SRAT ENTRY TYPE\n"); } current += ReadField(Length); } } DECLARE_API( srat ) { ULONG64 acpiSRATAddress = 0; BOOLEAN Physical = FALSE; if (args != NULL) { acpiSRATAddress = GetExpression( args ); Physical = TRUE; } if (acpiSRATAddress == 0) { BOOL status; ULONG64 address; status = GetUlongPtr("hal!HalpAcpiSrat", &acpiSRATAddress ); if ((status == TRUE) && acpiSRATAddress) { dprintf("Retrieved srat address from HAL NUMA code\n"); Physical = FALSE; } } if (acpiSRATAddress == 0) { dprintf("srat [