2018-06-04 03:14:51 +02:00
|
|
|
/** @file
|
2018-04-20 10:08:22 +02:00
|
|
|
RSDP table parser
|
|
|
|
|
2019-07-19 03:04:55 +02:00
|
|
|
Copyright (c) 2016 - 2019, ARM Limited. All rights reserved.
|
2019-04-04 01:07:06 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@par Reference(s):
|
|
|
|
- ACPI 6.2 Specification - Errata A, September 2017
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include "AcpiParser.h"
|
|
|
|
#include "AcpiTableParser.h"
|
|
|
|
|
|
|
|
// Local Variables
|
|
|
|
STATIC CONST UINT64* XsdtAddress;
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function validates the RSDT Address.
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@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.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ValidateRsdtAddress (
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN VOID* Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
#if defined(MDE_CPU_ARM) || defined (MDE_CPU_AARCH64)
|
|
|
|
// Reference: Server Base Boot Requirements System Software on ARM Platforms
|
|
|
|
// Section: 4.2.1.1 RSDP
|
|
|
|
// Root System Description Pointer (RSDP), ACPI ? 5.2.5.
|
|
|
|
// - Within the RSDP, the RsdtAddress field must be null (zero) and the
|
|
|
|
// XsdtAddresss MUST be a valid, non-null, 64-bit value.
|
2018-06-05 03:20:05 +02:00
|
|
|
UINT32 RsdtAddr;
|
|
|
|
|
|
|
|
RsdtAddr = *(UINT32*)Ptr;
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
if (RsdtAddr != 0) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Rsdt Address = 0x%p. This must be NULL on ARM Platforms.",
|
|
|
|
RsdtAddr
|
|
|
|
);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function validates the XSDT Address.
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@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.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ValidateXsdtAddress (
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN VOID* Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
#if defined(MDE_CPU_ARM) || defined (MDE_CPU_AARCH64)
|
|
|
|
// Reference: Server Base Boot Requirements System Software on ARM Platforms
|
|
|
|
// Section: 4.2.1.1 RSDP
|
|
|
|
// Root System Description Pointer (RSDP), ACPI ? 5.2.5.
|
|
|
|
// - Within the RSDP, the RsdtAddress field must be null (zero) and the
|
|
|
|
// XsdtAddresss MUST be a valid, non-null, 64-bit value.
|
2018-06-05 03:20:05 +02:00
|
|
|
UINT64 XsdtAddr;
|
|
|
|
|
|
|
|
XsdtAddr = *(UINT64*)Ptr;
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
if (XsdtAddr == 0) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Xsdt Address = 0x%p. This must not be NULL on ARM Platforms.",
|
|
|
|
XsdtAddr
|
|
|
|
);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2019-07-19 03:04:55 +02:00
|
|
|
/**
|
|
|
|
An array describing the ACPI RSDP Table.
|
|
|
|
**/
|
|
|
|
STATIC CONST ACPI_PARSER RsdpParser[] = {
|
|
|
|
{L"Signature", 8, 0, NULL, Dump8Chars, NULL, NULL, NULL},
|
|
|
|
{L"Checksum", 1, 8, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Oem ID", 6, 9, NULL, Dump6Chars, NULL, NULL, NULL},
|
|
|
|
{L"Revision", 1, 15, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"RSDT Address", 4, 16, L"0x%x", NULL, NULL, ValidateRsdtAddress, NULL},
|
|
|
|
{L"Length", 4, 20, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"XSDT Address", 8, 24, L"0x%lx", NULL, (VOID**)&XsdtAddress,
|
|
|
|
ValidateXsdtAddress, NULL},
|
|
|
|
{L"Extended Checksum", 1, 32, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 3, 33, L"%x %x %x", Dump3Chars, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function parses the ACPI RSDP table.
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
This function invokes the parser for the XSDT table.
|
|
|
|
* Note - This function does not support parsing of RSDT table.
|
|
|
|
|
|
|
|
This function also performs a RAW dump of the ACPI table and
|
|
|
|
validates the checksum.
|
|
|
|
|
|
|
|
@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.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ParseAcpiRsdp (
|
|
|
|
IN BOOLEAN Trace,
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN UINT32 AcpiTableLength,
|
|
|
|
IN UINT8 AcpiTableRevision
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (Trace) {
|
|
|
|
DumpRaw (Ptr, AcpiTableLength);
|
|
|
|
VerifyChecksum (TRUE, Ptr, AcpiTableLength);
|
|
|
|
}
|
|
|
|
|
2019-07-23 00:50:23 +02:00
|
|
|
ParseAcpi (
|
|
|
|
Trace,
|
|
|
|
0,
|
|
|
|
"RSDP",
|
|
|
|
Ptr,
|
|
|
|
AcpiTableLength,
|
|
|
|
PARSER_PARAMS (RsdpParser)
|
|
|
|
);
|
2018-04-20 10:08:22 +02:00
|
|
|
|
2020-01-20 12:13:42 +01:00
|
|
|
// Check if the values used to control the parsing logic have been
|
|
|
|
// successfully read.
|
|
|
|
if (XsdtAddress == NULL) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Insufficient table length. AcpiTableLength = %d." \
|
|
|
|
L"RSDP parsing aborted.\n",
|
|
|
|
AcpiTableLength
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
// This code currently supports parsing of XSDT table only
|
|
|
|
// and does not parse the RSDT table. Platforms provide the
|
|
|
|
// RSDT to enable compatibility with ACPI 1.0 operating systems.
|
|
|
|
// Therefore the RSDT should not be used on ARM platforms.
|
|
|
|
if ((*XsdtAddress) == 0) {
|
|
|
|
IncrementErrorCount ();
|
2019-07-23 00:50:23 +02:00
|
|
|
Print (L"ERROR: XSDT Pointer is not set. RSDP parsing aborted.\n");
|
2018-04-20 10:08:22 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcessAcpiTable ((UINT8*)(UINTN)(*XsdtAddress));
|
|
|
|
}
|