/** @file FADT table parser Copyright (c) 2016 - 2018, ARM Limited. All rights reserved. This program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License which accompanies this distribution. The full text of the license may be found at http://opensource.org/licenses/bsd-license.php THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. @par Reference(s): - ACPI 6.2 Specification - Errata A, September 2017 **/ #include #include #include "AcpiParser.h" #include "AcpiTableParser.h" // Local variables STATIC CONST UINT32* DsdtAddress; STATIC CONST UINT64* X_DsdtAddress; STATIC CONST UINT8* FadtMinorRevision; STATIC ACPI_DESCRIPTION_HEADER_INFO AcpiHdrInfo; /** A macro defining the Hardware reduced ACPI flag **/ #define HW_REDUCED_ACPI BIT20 /** Get the ACPI XSDT header info. **/ CONST ACPI_DESCRIPTION_HEADER_INFO* CONST EFIAPI GetAcpiXsdtHeaderInfo ( VOID ); /** This function validates the Firmware Control Field. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateFirmwareCtrl ( IN UINT8* Ptr, IN VOID* Context ); /** This function validates the X_Firmware Control Field. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateXFirmwareCtrl ( IN UINT8* Ptr, IN VOID* Context ); /** This function validates the flags. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateFlags ( IN UINT8* Ptr, IN VOID* Context ); /** An ACPI_PARSER array describing the ACPI FADT Table. **/ STATIC CONST ACPI_PARSER FadtParser[] = { PARSE_ACPI_HEADER (&AcpiHdrInfo), {L"FIRMWARE_CTRL", 4, 36, L"0x%x", NULL, NULL, ValidateFirmwareCtrl, NULL}, {L"DSDT", 4, 40, L"0x%x", NULL, (VOID**)&DsdtAddress, NULL, NULL}, {L"Reserved", 1, 44, L"%x", NULL, NULL, NULL, NULL}, {L"Preferred_PM_Profile", 1, 45, L"0x%x", NULL, NULL, NULL, NULL}, {L"SCI_INT", 2, 46, L"0x%x", NULL, NULL, NULL, NULL}, {L"SMI_CMD", 4, 48, L"0x%x", NULL, NULL, NULL, NULL}, {L"ACPI_ENABLE", 1, 52, L"0x%x", NULL, NULL, NULL, NULL}, {L"ACPI_DISABLE", 1, 53, L"0x%x", NULL, NULL, NULL, NULL}, {L"S4BIOS_REQ", 1, 54, L"0x%x", NULL, NULL, NULL, NULL}, {L"PSTATE_CNT", 1, 55, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1a_EVT_BLK", 4, 56, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1b_EVT_BLK", 4, 60, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1a_CNT_BLK", 4, 64, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1b_CNT_BLK", 4, 68, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM2_CNT_BLK", 4, 72, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM_TMR_BLK", 4, 76, L"0x%x", NULL, NULL, NULL, NULL}, {L"GPE0_BLK", 4, 80, L"0x%x", NULL, NULL, NULL, NULL}, {L"GPE1_BLK", 4, 84, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1_EVT_LEN", 1, 88, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM1_CNT_LEN", 1, 89, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM2_CNT_LEN", 1, 90, L"0x%x", NULL, NULL, NULL, NULL}, {L"PM_TMR_LEN", 1, 91, L"0x%x", NULL, NULL, NULL, NULL}, {L"GPE0_BLK_LEN", 1, 92, L"0x%x", NULL, NULL, NULL, NULL}, {L"GPE1_BLK_LEN", 1, 93, L"0x%x", NULL, NULL, NULL, NULL}, {L"GPE1_BASE", 1, 94, L"0x%x", NULL, NULL, NULL, NULL}, {L"CST_CNT", 1, 95, L"0x%x", NULL, NULL, NULL, NULL}, {L"P_LVL2_LAT", 2, 96, L"0x%x", NULL, NULL, NULL, NULL}, {L"P_LVL3_LAT", 2, 98, L"0x%x", NULL, NULL, NULL, NULL}, {L"FLUSH_SIZE", 2, 100, L"0x%x", NULL, NULL, NULL, NULL}, {L"FLUSH_STRIDE", 2, 102, L"0x%x", NULL, NULL, NULL, NULL}, {L"DUTY_OFFSET", 1, 104, L"0x%x", NULL, NULL, NULL, NULL}, {L"DUTY_WIDTH", 1, 105, L"0x%x", NULL, NULL, NULL, NULL}, {L"DAY_ALRM", 1, 106, L"0x%x", NULL, NULL, NULL, NULL}, {L"MON_ALRM", 1, 107, L"0x%x", NULL, NULL, NULL, NULL}, {L"CENTURY", 1, 108, L"0x%x", NULL, NULL, NULL, NULL}, {L"IAPC_BOOT_ARCH", 2, 109, L"0x%x", NULL, NULL, NULL, NULL}, {L"Reserved", 1, 111, L"0x%x", NULL, NULL, NULL, NULL}, {L"Flags", 4, 112, L"0x%x", NULL, NULL, ValidateFlags, NULL}, {L"RESET_REG", 12, 116, NULL, DumpGas, NULL, NULL, NULL}, {L"RESET_VALUE", 1, 128, L"0x%x", NULL, NULL, NULL, NULL}, {L"ARM_BOOT_ARCH", 2, 129, L"0x%x", NULL, NULL, NULL, NULL}, {L"FADT Minor Version", 1, 131, L"0x%x", NULL, (VOID**)&FadtMinorRevision, NULL, NULL}, {L"X_FIRMWARE_CTRL", 8, 132, L"0x%lx", NULL, NULL, ValidateXFirmwareCtrl, NULL}, {L"X_DSDT", 8, 140, L"0x%lx", NULL, (VOID**)&X_DsdtAddress, NULL, NULL}, {L"X_PM1a_EVT_BLK", 12, 148, NULL, DumpGas, NULL, NULL, NULL}, {L"X_PM1b_EVT_BLK", 12, 160, NULL, DumpGas, NULL, NULL, NULL}, {L"X_PM1a_CNT_BLK", 12, 172, NULL, DumpGas, NULL, NULL, NULL}, {L"X_PM1b_CNT_BLK", 12, 184, NULL, DumpGas, NULL, NULL, NULL}, {L"X_PM2_CNT_BLK", 12, 196, NULL, DumpGas, NULL, NULL, NULL}, {L"X_PM_TMR_BLK", 12, 208, NULL, DumpGas, NULL, NULL, NULL}, {L"X_GPE0_BLK", 12, 220, NULL, DumpGas, NULL, NULL, NULL}, {L"X_GPE1_BLK", 12, 232, NULL, DumpGas, NULL, NULL, NULL}, {L"SLEEP_CONTROL_REG", 12, 244, NULL, DumpGas, NULL, NULL, NULL}, {L"SLEEP_STATUS_REG", 12, 256, NULL, DumpGas, NULL, NULL, NULL}, {L"Hypervisor VendorIdentity", 8, 268, L"%lx", NULL, NULL, NULL, NULL} }; /** This function validates the Firmware Control Field. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateFirmwareCtrl ( IN UINT8* Ptr, IN VOID* Context ) { #if defined (MDE_CPU_ARM) || defined (MDE_CPU_AARCH64) if (*(UINT32*)Ptr != 0) { IncrementErrorCount (); Print ( L"\nERROR: Firmware Control must be zero for ARM platforms." ); } #endif } /** This function validates the X_Firmware Control Field. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateXFirmwareCtrl ( IN UINT8* Ptr, IN VOID* Context ) { #if defined (MDE_CPU_ARM) || defined (MDE_CPU_AARCH64) if (*(UINT64*)Ptr != 0) { IncrementErrorCount (); Print ( L"\nERROR: X Firmware Control must be zero for ARM platforms." ); } #endif } /** This function validates the flags. @param [in] Ptr Pointer to the start of the field data. @param [in] Context Pointer to context specific information e.g. this could be a pointer to the ACPI table header. **/ STATIC VOID EFIAPI ValidateFlags ( IN UINT8* Ptr, IN VOID* Context ) { #if defined (MDE_CPU_ARM) || defined (MDE_CPU_AARCH64) if (((*(UINT32*)Ptr) & HW_REDUCED_ACPI) == 0) { IncrementErrorCount (); Print ( L"\nERROR: HW_REDUCED_ACPI flag must be set for ARM platforms." ); } #endif } /** This function parses the ACPI FADT table. This function parses the FADT table and optionally traces the ACPI table fields. This function also performs validation of the ACPI table fields. @param [in] Trace If TRUE, trace the ACPI fields. @param [in] Ptr Pointer to the start of the buffer. @param [in] AcpiTableLength Length of the ACPI table. @param [in] AcpiTableRevision Revision of the ACPI table. **/ VOID EFIAPI ParseAcpiFadt ( IN BOOLEAN Trace, IN UINT8* Ptr, IN UINT32 AcpiTableLength, IN UINT8 AcpiTableRevision ) { UINT8* DsdtPtr; ParseAcpi ( Trace, 0, "FADT", Ptr, AcpiTableLength, PARSER_PARAMS (FadtParser) ); if (Trace) { Print (L"\nSummary:\n"); PrintFieldName (2, L"FADT Version"); Print (L"%d.%d\n", *AcpiHdrInfo.Revision, *FadtMinorRevision); if (*GetAcpiXsdtHeaderInfo ()->OemTableId != *AcpiHdrInfo.OemTableId) { IncrementErrorCount (); Print (L"ERROR: OEM Table Id does not match with RSDT/XSDT.\n"); } } // If X_DSDT is not zero then use X_DSDT and ignore DSDT, // else use DSDT. if (*X_DsdtAddress != 0) { DsdtPtr = (UINT8*)(UINTN)(*X_DsdtAddress); } else if (*DsdtAddress != 0) { DsdtPtr = (UINT8*)(UINTN)(*DsdtAddress); } else { // Both DSDT and X_DSDT cannot be zero. #if defined (MDE_CPU_ARM) || defined (MDE_CPU_AARCH64) if (Trace) { // The DSDT Table is mandatory for ARM systems // as the CPU information MUST be presented in // the DSDT. IncrementErrorCount (); Print (L"ERROR: Both X_DSDT and DSDT are NULL.\n"); } #endif return; } ProcessAcpiTable (DsdtPtr); }