2018-06-04 03:14:51 +02:00
|
|
|
/** @file
|
2018-04-20 10:08:22 +02:00
|
|
|
|
2020-03-25 10:39:22 +01:00
|
|
|
Copyright (c) 2016 - 2020, ARM Limited. All rights reserved.
|
2019-04-04 01:07:06 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2020-03-25 10:39:24 +01:00
|
|
|
|
|
|
|
@par Glossary:
|
|
|
|
- Sbbr or SBBR - Server Base Boot Requirements
|
|
|
|
|
|
|
|
@par Reference(s):
|
|
|
|
- Arm Server Base Boot Requirements 1.2, September 2019
|
2018-04-20 10:08:22 +02:00
|
|
|
**/
|
|
|
|
|
|
|
|
#include <Library/PrintLib.h>
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include <Library/ShellLib.h>
|
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include "AcpiParser.h"
|
|
|
|
#include "AcpiTableParser.h"
|
|
|
|
#include "AcpiView.h"
|
2020-06-19 13:59:54 +02:00
|
|
|
#include "AcpiViewConfig.h"
|
2018-04-20 10:08:22 +02:00
|
|
|
#include "UefiShellAcpiViewCommandLib.h"
|
|
|
|
|
2020-03-25 10:39:24 +01:00
|
|
|
#if defined(MDE_CPU_ARM) || defined (MDE_CPU_AARCH64)
|
|
|
|
#include "Arm/SbbrValidator.h"
|
|
|
|
#endif
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC UINT32 mTableCount;
|
|
|
|
STATIC UINT32 mBinTableCount;
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function dumps the ACPI table to a file.
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
@param [in] Ptr Pointer to the ACPI table data.
|
|
|
|
@param [in] Length The length of the ACPI table.
|
|
|
|
|
|
|
|
@retval TRUE Success.
|
|
|
|
@retval FALSE Failure.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC
|
|
|
|
BOOLEAN
|
|
|
|
DumpAcpiTableToFile (
|
|
|
|
IN CONST UINT8* Ptr,
|
|
|
|
IN CONST UINTN Length
|
|
|
|
)
|
|
|
|
{
|
2020-06-19 13:59:54 +02:00
|
|
|
CHAR16 FileNameBuffer[MAX_FILE_NAME_LEN];
|
|
|
|
UINTN TransferBytes;
|
|
|
|
SELECTED_ACPI_TABLE *SelectedTable;
|
2018-06-05 03:20:05 +02:00
|
|
|
|
2020-06-19 13:59:54 +02:00
|
|
|
GetSelectedAcpiTable (&SelectedTable);
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
UnicodeSPrint (
|
|
|
|
FileNameBuffer,
|
|
|
|
sizeof (FileNameBuffer),
|
|
|
|
L".\\%s%04d.bin",
|
2020-06-19 13:59:54 +02:00
|
|
|
SelectedTable->Name,
|
2018-04-20 10:08:22 +02:00
|
|
|
mBinTableCount++
|
|
|
|
);
|
|
|
|
|
|
|
|
Print (L"Dumping ACPI table to : %s ... ", FileNameBuffer);
|
|
|
|
|
2020-06-19 13:59:54 +02:00
|
|
|
TransferBytes = ShellDumpBufferToFile (FileNameBuffer, Ptr, Length);
|
2018-04-20 10:08:22 +02:00
|
|
|
return (Length == TransferBytes);
|
|
|
|
}
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function processes the table reporting options for the ACPI table.
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@param [in] Signature The ACPI table Signature.
|
|
|
|
@param [in] TablePtr Pointer to the ACPI table data.
|
|
|
|
@param [in] Length The length fo the ACPI table.
|
|
|
|
|
|
|
|
@retval Returns TRUE if the ACPI table should be traced.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
BOOLEAN
|
|
|
|
ProcessTableReportOptions (
|
|
|
|
IN CONST UINT32 Signature,
|
|
|
|
IN CONST UINT8* TablePtr,
|
|
|
|
IN CONST UINT32 Length
|
|
|
|
)
|
|
|
|
{
|
2020-06-19 13:59:54 +02:00
|
|
|
UINTN OriginalAttribute;
|
|
|
|
UINT8 *SignaturePtr;
|
|
|
|
BOOLEAN Log;
|
|
|
|
BOOLEAN HighLight;
|
|
|
|
SELECTED_ACPI_TABLE *SelectedTable;
|
2018-06-05 03:20:05 +02:00
|
|
|
|
2019-08-15 07:35:42 +02:00
|
|
|
//
|
|
|
|
// set local variables to suppress incorrect compiler/analyzer warnings
|
|
|
|
//
|
|
|
|
OriginalAttribute = 0;
|
2018-06-05 03:20:05 +02:00
|
|
|
SignaturePtr = (UINT8*)(UINTN)&Signature;
|
|
|
|
Log = FALSE;
|
|
|
|
HighLight = GetColourHighlighting ();
|
2020-06-19 13:59:54 +02:00
|
|
|
GetSelectedAcpiTable (&SelectedTable);
|
2018-06-05 03:20:05 +02:00
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
switch (GetReportOption ()) {
|
2018-06-05 03:20:05 +02:00
|
|
|
case ReportAll:
|
2018-04-20 10:08:22 +02:00
|
|
|
Log = TRUE;
|
|
|
|
break;
|
2018-06-05 03:20:05 +02:00
|
|
|
case ReportSelected:
|
2020-06-19 13:59:54 +02:00
|
|
|
if (Signature == SelectedTable->Type) {
|
2018-04-20 10:08:22 +02:00
|
|
|
Log = TRUE;
|
2020-06-19 13:59:54 +02:00
|
|
|
SelectedTable->Found = TRUE;
|
2018-04-20 10:08:22 +02:00
|
|
|
}
|
|
|
|
break;
|
2018-06-05 03:20:05 +02:00
|
|
|
case ReportTableList:
|
2018-04-20 10:08:22 +02:00
|
|
|
if (mTableCount == 0) {
|
|
|
|
if (HighLight) {
|
|
|
|
OriginalAttribute = gST->ConOut->Mode->Attribute;
|
|
|
|
gST->ConOut->SetAttribute (
|
|
|
|
gST->ConOut,
|
|
|
|
EFI_TEXT_ATTR(EFI_CYAN,
|
|
|
|
((OriginalAttribute&(BIT4|BIT5|BIT6))>>4))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Print (L"\nInstalled Table(s):\n");
|
|
|
|
if (HighLight) {
|
|
|
|
gST->ConOut->SetAttribute (gST->ConOut, OriginalAttribute);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Print (
|
|
|
|
L"\t%4d. %c%c%c%c\n",
|
|
|
|
++mTableCount,
|
|
|
|
SignaturePtr[0],
|
|
|
|
SignaturePtr[1],
|
|
|
|
SignaturePtr[2],
|
|
|
|
SignaturePtr[3]
|
|
|
|
);
|
|
|
|
break;
|
2018-06-05 03:20:05 +02:00
|
|
|
case ReportDumpBinFile:
|
2020-06-19 13:59:54 +02:00
|
|
|
if (Signature == SelectedTable->Type) {
|
|
|
|
SelectedTable->Found = TRUE;
|
2018-04-20 10:08:22 +02:00
|
|
|
DumpAcpiTableToFile (TablePtr, Length);
|
|
|
|
}
|
|
|
|
break;
|
2018-06-05 03:20:05 +02:00
|
|
|
case ReportMax:
|
2018-04-20 10:08:22 +02:00
|
|
|
// We should never be here.
|
|
|
|
// This case is only present to prevent compiler warning.
|
|
|
|
break;
|
|
|
|
} // switch
|
|
|
|
|
|
|
|
if (Log) {
|
|
|
|
if (HighLight) {
|
|
|
|
OriginalAttribute = gST->ConOut->Mode->Attribute;
|
|
|
|
gST->ConOut->SetAttribute (
|
|
|
|
gST->ConOut,
|
|
|
|
EFI_TEXT_ATTR(EFI_LIGHTBLUE,
|
|
|
|
((OriginalAttribute&(BIT4|BIT5|BIT6))>>4))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Print (
|
|
|
|
L"\n\n --------------- %c%c%c%c Table --------------- \n\n",
|
|
|
|
SignaturePtr[0],
|
|
|
|
SignaturePtr[1],
|
|
|
|
SignaturePtr[2],
|
|
|
|
SignaturePtr[3]
|
|
|
|
);
|
|
|
|
if (HighLight) {
|
|
|
|
gST->ConOut->SetAttribute (gST->ConOut, OriginalAttribute);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Log;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function iterates the configuration table entries in the
|
|
|
|
system table, retrieves the RSDP pointer and starts parsing the ACPI tables.
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@param [in] SystemTable Pointer to the EFI system table.
|
|
|
|
|
|
|
|
@retval Returns EFI_NOT_FOUND if the RSDP pointer is not found.
|
|
|
|
Returns EFI_UNSUPPORTED if the RSDP version is less than 2.
|
|
|
|
Returns EFI_SUCCESS if successful.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
AcpiView (
|
|
|
|
IN EFI_SYSTEM_TABLE* SystemTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Index;
|
|
|
|
EFI_CONFIGURATION_TABLE* EfiConfigurationTable;
|
|
|
|
BOOLEAN FoundAcpiTable;
|
|
|
|
UINTN OriginalAttribute;
|
|
|
|
UINTN PrintAttribute;
|
|
|
|
EREPORT_OPTION ReportOption;
|
|
|
|
UINT8* RsdpPtr;
|
|
|
|
UINT32 RsdpLength;
|
|
|
|
UINT8 RsdpRevision;
|
|
|
|
PARSE_ACPI_TABLE_PROC RsdpParserProc;
|
|
|
|
BOOLEAN Trace;
|
2020-06-19 13:59:54 +02:00
|
|
|
SELECTED_ACPI_TABLE *SelectedTable;
|
2018-04-20 10:08:22 +02:00
|
|
|
|
2019-08-15 07:35:42 +02:00
|
|
|
//
|
|
|
|
// set local variables to suppress incorrect compiler/analyzer warnings
|
|
|
|
//
|
|
|
|
EfiConfigurationTable = NULL;
|
|
|
|
OriginalAttribute = 0;
|
|
|
|
|
2020-06-19 13:59:54 +02:00
|
|
|
// Reset Table counts
|
|
|
|
mTableCount = 0;
|
|
|
|
mBinTableCount = 0;
|
|
|
|
|
2020-06-19 13:59:54 +02:00
|
|
|
// Reset The error/warning counters
|
|
|
|
ResetErrorCount ();
|
|
|
|
ResetWarningCount ();
|
|
|
|
|
2020-06-19 13:59:54 +02:00
|
|
|
// Retrieve the user selection of ACPI table to process
|
|
|
|
GetSelectedAcpiTable (&SelectedTable);
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
// Search the table for an entry that matches the ACPI Table Guid
|
|
|
|
FoundAcpiTable = FALSE;
|
|
|
|
for (Index = 0; Index < SystemTable->NumberOfTableEntries; Index++) {
|
|
|
|
if (CompareGuid (&gEfiAcpiTableGuid,
|
|
|
|
&(SystemTable->ConfigurationTable[Index].VendorGuid))) {
|
|
|
|
EfiConfigurationTable = &SystemTable->ConfigurationTable[Index];
|
|
|
|
FoundAcpiTable = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FoundAcpiTable) {
|
|
|
|
RsdpPtr = (UINT8*)EfiConfigurationTable->VendorTable;
|
|
|
|
|
|
|
|
// The RSDP revision is 1 byte starting at offset 15
|
|
|
|
RsdpRevision = *(RsdpPtr + RSDP_REVISION_OFFSET);
|
|
|
|
|
|
|
|
if (RsdpRevision < 2) {
|
|
|
|
Print (
|
|
|
|
L"ERROR: RSDP version less than 2 is not supported.\n"
|
|
|
|
);
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
2020-03-25 10:39:24 +01:00
|
|
|
#if defined(MDE_CPU_ARM) || defined (MDE_CPU_AARCH64)
|
|
|
|
if (GetMandatoryTableValidate ()) {
|
|
|
|
ArmSbbrResetTableCounts ();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
// The RSDP length is 4 bytes starting at offset 20
|
|
|
|
RsdpLength = *(UINT32*)(RsdpPtr + RSDP_LENGTH_OFFSET);
|
|
|
|
|
|
|
|
Trace = ProcessTableReportOptions (RSDP_TABLE_INFO, RsdpPtr, RsdpLength);
|
|
|
|
|
|
|
|
Status = GetParser (RSDP_TABLE_INFO, &RsdpParserProc);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
Print (
|
|
|
|
L"ERROR: No registered parser found for RSDP.\n"
|
|
|
|
);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsdpParserProc (
|
|
|
|
Trace,
|
|
|
|
RsdpPtr,
|
|
|
|
RsdpLength,
|
|
|
|
RsdpRevision
|
|
|
|
);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Failed to find ACPI Table Guid in System Configuration Table.\n"
|
|
|
|
);
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
2020-03-25 10:39:24 +01:00
|
|
|
#if defined(MDE_CPU_ARM) || defined (MDE_CPU_AARCH64)
|
|
|
|
if (GetMandatoryTableValidate ()) {
|
|
|
|
ArmSbbrReqsValidate ((ARM_SBBR_VERSION)GetMandatoryTableSpec ());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-04-20 10:08:22 +02:00
|
|
|
ReportOption = GetReportOption ();
|
2018-06-05 03:20:05 +02:00
|
|
|
if (ReportTableList != ReportOption) {
|
|
|
|
if (((ReportSelected == ReportOption) ||
|
|
|
|
(ReportDumpBinFile == ReportOption)) &&
|
2020-06-19 13:59:54 +02:00
|
|
|
(!SelectedTable->Found)) {
|
2018-04-20 10:08:22 +02:00
|
|
|
Print (L"\nRequested ACPI Table not found.\n");
|
2019-06-28 10:56:58 +02:00
|
|
|
} else if (GetConsistencyChecking () &&
|
|
|
|
(ReportDumpBinFile != ReportOption)) {
|
2018-04-20 10:08:22 +02:00
|
|
|
OriginalAttribute = gST->ConOut->Mode->Attribute;
|
|
|
|
|
|
|
|
Print (L"\nTable Statistics:\n");
|
|
|
|
|
|
|
|
if (GetColourHighlighting ()) {
|
|
|
|
PrintAttribute = (GetErrorCount () > 0) ?
|
|
|
|
EFI_TEXT_ATTR (
|
|
|
|
EFI_RED,
|
|
|
|
((OriginalAttribute&(BIT4|BIT5|BIT6))>>4)
|
|
|
|
) :
|
|
|
|
OriginalAttribute;
|
|
|
|
gST->ConOut->SetAttribute (gST->ConOut, PrintAttribute);
|
|
|
|
}
|
|
|
|
Print (L"\t%d Error(s)\n", GetErrorCount ());
|
|
|
|
|
|
|
|
if (GetColourHighlighting ()) {
|
|
|
|
PrintAttribute = (GetWarningCount () > 0) ?
|
|
|
|
EFI_TEXT_ATTR (
|
|
|
|
EFI_RED,
|
|
|
|
((OriginalAttribute&(BIT4|BIT5|BIT6))>>4)
|
|
|
|
) :
|
|
|
|
OriginalAttribute;
|
|
|
|
|
|
|
|
gST->ConOut->SetAttribute (gST->ConOut, PrintAttribute);
|
|
|
|
}
|
|
|
|
Print (L"\t%d Warning(s)\n", GetWarningCount ());
|
|
|
|
|
|
|
|
if (GetColourHighlighting ()) {
|
|
|
|
gST->ConOut->SetAttribute (gST->ConOut, OriginalAttribute);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|