2015-01-23 16:59:27 +01:00
|
|
|
/** @file
|
|
|
|
*
|
2015-05-29 16:39:41 +02:00
|
|
|
* Copyright (c) 2014-2015, ARM Limited. All rights reserved.
|
2021-09-09 10:24:26 +02:00
|
|
|
* Copyright (c) 2021, Ampere Computing LLC. All rights reserved.
|
2015-01-23 16:59:27 +01:00
|
|
|
*
|
2019-04-04 01:03:38 +02:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause-Patent
|
2015-01-23 16:59:27 +01:00
|
|
|
*
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <Uefi.h>
|
|
|
|
|
|
|
|
#include <Library/AcpiLib.h>
|
2021-09-09 10:24:26 +02:00
|
|
|
#include <Library/BaseLib.h>
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
2015-01-23 16:59:27 +01:00
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
|
2021-09-09 10:24:26 +02:00
|
|
|
#include <Protocol/AcpiSystemDescriptionTable.h>
|
2015-01-23 16:59:27 +01:00
|
|
|
#include <Protocol/AcpiTable.h>
|
|
|
|
#include <Protocol/FirmwareVolume2.h>
|
|
|
|
|
|
|
|
#include <IndustryStandard/Acpi.h>
|
|
|
|
|
|
|
|
/**
|
2015-05-29 16:39:41 +02:00
|
|
|
Locate and Install the ACPI tables from the Firmware Volume if it verifies
|
|
|
|
the function condition.
|
2015-01-23 16:59:27 +01:00
|
|
|
|
2015-05-29 16:39:41 +02:00
|
|
|
@param AcpiFile Guid of the ACPI file into the Firmware Volume
|
|
|
|
@param CheckAcpiTableFunction Function that checks if the ACPI table should be installed
|
2015-01-23 16:59:27 +01:00
|
|
|
|
2015-05-29 16:39:41 +02:00
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_NOT_FOUND The protocol could not be located.
|
|
|
|
@return EFI_OUT_OF_RESOURCES There are not enough resources to find the protocol.
|
2015-01-23 16:59:27 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2015-05-29 16:39:41 +02:00
|
|
|
LocateAndInstallAcpiFromFvConditional (
|
2021-12-05 23:53:56 +01:00
|
|
|
IN CONST EFI_GUID *AcpiFile,
|
2015-05-29 16:39:41 +02:00
|
|
|
IN EFI_LOCATE_ACPI_CHECK CheckAcpiTableFunction
|
2015-01-23 16:59:27 +01:00
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:53:56 +01:00
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_ACPI_TABLE_PROTOCOL *AcpiProtocol;
|
|
|
|
EFI_HANDLE *HandleBuffer;
|
|
|
|
UINTN NumberOfHandles;
|
|
|
|
UINT32 FvStatus;
|
|
|
|
UINTN Index;
|
|
|
|
EFI_FIRMWARE_VOLUME2_PROTOCOL *FvInstance;
|
|
|
|
INTN SectionInstance;
|
|
|
|
UINTN SectionSize;
|
|
|
|
EFI_ACPI_COMMON_HEADER *AcpiTable;
|
|
|
|
UINTN AcpiTableSize;
|
|
|
|
UINTN AcpiTableKey;
|
|
|
|
BOOLEAN Valid;
|
2015-01-23 16:59:27 +01:00
|
|
|
|
|
|
|
// Ensure the ACPI Table is present
|
|
|
|
Status = gBS->LocateProtocol (
|
|
|
|
&gEfiAcpiTableProtocolGuid,
|
|
|
|
NULL,
|
2021-12-05 23:53:56 +01:00
|
|
|
(VOID **)&AcpiProtocol
|
2015-01-23 16:59:27 +01:00
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
FvStatus = 0;
|
|
|
|
SectionInstance = 0;
|
|
|
|
|
|
|
|
// Locate all the Firmware Volume protocols.
|
|
|
|
Status = gBS->LocateHandleBuffer (
|
2021-12-05 23:53:56 +01:00
|
|
|
ByProtocol,
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
NULL,
|
|
|
|
&NumberOfHandles,
|
|
|
|
&HandleBuffer
|
|
|
|
);
|
2015-01-23 16:59:27 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Looking for FV with ACPI storage file
|
|
|
|
for (Index = 0; Index < NumberOfHandles; Index++) {
|
|
|
|
//
|
|
|
|
// Get the protocol on this handle
|
|
|
|
// This should not fail because of LocateHandleBuffer
|
|
|
|
//
|
|
|
|
Status = gBS->HandleProtocol (
|
2021-12-05 23:53:56 +01:00
|
|
|
HandleBuffer[Index],
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
(VOID **)&FvInstance
|
|
|
|
);
|
2015-01-23 16:59:27 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FREE_HANDLE_BUFFER;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (Status == EFI_SUCCESS) {
|
|
|
|
// AcpiTable must be allocated by ReadSection (ie: AcpiTable == NULL)
|
|
|
|
AcpiTable = NULL;
|
|
|
|
|
|
|
|
// See if it has the ACPI storage file
|
|
|
|
Status = FvInstance->ReadSection (
|
2021-12-05 23:53:56 +01:00
|
|
|
FvInstance,
|
|
|
|
AcpiFile,
|
|
|
|
EFI_SECTION_RAW,
|
|
|
|
SectionInstance,
|
|
|
|
(VOID **)&AcpiTable,
|
|
|
|
&SectionSize,
|
|
|
|
&FvStatus
|
|
|
|
);
|
2015-01-23 16:59:27 +01:00
|
|
|
if (!EFI_ERROR (Status)) {
|
2021-12-05 23:53:56 +01:00
|
|
|
AcpiTableKey = 0;
|
|
|
|
AcpiTableSize = ((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable)->Length;
|
2015-01-23 16:59:27 +01:00
|
|
|
ASSERT (SectionSize >= AcpiTableSize);
|
|
|
|
|
2021-12-05 23:53:56 +01:00
|
|
|
DEBUG ((
|
|
|
|
DEBUG_ERROR,
|
|
|
|
"- Found '%c%c%c%c' ACPI Table\n",
|
|
|
|
(((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable)->Signature & 0xFF),
|
|
|
|
((((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable)->Signature >> 8) & 0xFF),
|
|
|
|
((((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable)->Signature >> 16) & 0xFF),
|
|
|
|
((((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable)->Signature >> 24) & 0xFF)
|
|
|
|
));
|
2015-01-23 16:59:27 +01:00
|
|
|
|
2015-05-29 16:39:41 +02:00
|
|
|
// Is the ACPI table valid?
|
|
|
|
if (CheckAcpiTableFunction) {
|
|
|
|
Valid = CheckAcpiTableFunction ((EFI_ACPI_DESCRIPTION_HEADER *)AcpiTable);
|
|
|
|
} else {
|
|
|
|
Valid = TRUE;
|
|
|
|
}
|
|
|
|
|
2015-01-23 16:59:27 +01:00
|
|
|
// Install the ACPI Table
|
2015-05-29 16:39:41 +02:00
|
|
|
if (Valid) {
|
|
|
|
Status = AcpiProtocol->InstallAcpiTable (
|
2021-12-05 23:53:56 +01:00
|
|
|
AcpiProtocol,
|
|
|
|
AcpiTable,
|
|
|
|
AcpiTableSize,
|
|
|
|
&AcpiTableKey
|
|
|
|
);
|
2015-05-29 16:39:41 +02:00
|
|
|
}
|
|
|
|
|
2015-01-23 16:59:27 +01:00
|
|
|
// Free memory allocated by ReadSection
|
|
|
|
gBS->FreePool (AcpiTable);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Increment the section instance
|
|
|
|
SectionInstance++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FREE_HANDLE_BUFFER:
|
|
|
|
//
|
|
|
|
// Free any allocated buffers
|
|
|
|
//
|
|
|
|
gBS->FreePool (HandleBuffer);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
2015-05-29 16:39:41 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
Locate and Install the ACPI tables from the Firmware Volume
|
|
|
|
|
|
|
|
@param AcpiFile Guid of the ACPI file into the Firmware Volume
|
|
|
|
|
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_NOT_FOUND The protocol could not be located.
|
|
|
|
@return EFI_OUT_OF_RESOURCES There are not enough resources to find the protocol.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
LocateAndInstallAcpiFromFv (
|
2021-12-05 23:53:56 +01:00
|
|
|
IN CONST EFI_GUID *AcpiFile
|
2015-05-29 16:39:41 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return LocateAndInstallAcpiFromFvConditional (AcpiFile, NULL);
|
|
|
|
}
|
2021-09-09 10:24:26 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
This function calculates and updates a UINT8 checksum
|
|
|
|
in an ACPI description table header.
|
|
|
|
|
|
|
|
@param Buffer Pointer to buffer to checksum
|
|
|
|
@param Size Number of bytes to checksum
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function completed successfully.
|
|
|
|
@retval EFI_INVALID_PARAMETER Invalid parameter.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
AcpiUpdateChecksum (
|
|
|
|
IN OUT UINT8 *Buffer,
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN ChecksumOffset;
|
|
|
|
|
|
|
|
if (Buffer == NULL || Size == 0) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
ChecksumOffset = OFFSET_OF (EFI_ACPI_DESCRIPTION_HEADER, Checksum);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Set checksum to 0 first
|
|
|
|
//
|
|
|
|
Buffer[ChecksumOffset] = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Update checksum value
|
|
|
|
//
|
|
|
|
Buffer[ChecksumOffset] = CalculateCheckSum8 (Buffer, Size);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function uses the ACPI SDT protocol to search an ACPI table
|
|
|
|
with a given signature.
|
|
|
|
|
|
|
|
@param AcpiTableSdtProtocol Pointer to ACPI SDT protocol.
|
|
|
|
@param TableSignature ACPI table signature.
|
|
|
|
@param Index The zero-based index of the table where to search the table.
|
|
|
|
The index will be updated to the next instance if the table
|
|
|
|
is found with the matched TableSignature.
|
|
|
|
@param Table Pointer to the table.
|
|
|
|
@param TableKey Pointer to the table key.
|
|
|
|
|
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_INVALID_PARAMETER At least one of parameters is invalid.
|
|
|
|
@retval EFI_NOT_FOUND The requested index is too large and a table was not found.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
AcpiLocateTableBySignature (
|
|
|
|
IN EFI_ACPI_SDT_PROTOCOL *AcpiSdtProtocol,
|
|
|
|
IN UINT32 TableSignature,
|
|
|
|
IN OUT UINTN *Index,
|
|
|
|
OUT EFI_ACPI_DESCRIPTION_HEADER **Table,
|
|
|
|
OUT UINTN *TableKey
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_ACPI_SDT_HEADER *TempTable;
|
|
|
|
EFI_ACPI_TABLE_VERSION TableVersion;
|
|
|
|
UINTN TableIndex;
|
|
|
|
|
|
|
|
if (AcpiSdtProtocol == NULL
|
|
|
|
|| Table == NULL
|
|
|
|
|| TableKey == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Search for ACPI Table with matching signature
|
|
|
|
//
|
|
|
|
TableVersion = 0;
|
|
|
|
TableIndex = *Index;
|
|
|
|
while (!EFI_ERROR (Status)) {
|
|
|
|
Status = AcpiSdtProtocol->GetAcpiTable (
|
|
|
|
TableIndex,
|
|
|
|
&TempTable,
|
|
|
|
&TableVersion,
|
|
|
|
TableKey
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
TableIndex++;
|
|
|
|
|
|
|
|
if (((EFI_ACPI_DESCRIPTION_HEADER *)TempTable)->Signature == TableSignature) {
|
|
|
|
*Table = (EFI_ACPI_DESCRIPTION_HEADER *)TempTable;
|
|
|
|
*Index = TableIndex;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function updates the integer value of an AML Object.
|
|
|
|
|
|
|
|
@param AcpiTableSdtProtocol Pointer to ACPI SDT protocol.
|
|
|
|
@param TableHandle Points to the table representing the starting point
|
|
|
|
for the object path search.
|
|
|
|
@param AsciiObjectPath Pointer to the ACPI path of the object being updated.
|
|
|
|
@param Value New value to write to the object.
|
|
|
|
|
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_INVALID_PARAMETER At least one of parameters is invalid or the data type
|
|
|
|
of the ACPI object is not an integer value.
|
|
|
|
@retval EFI_NOT_FOUND The object is not found with the given path.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
AcpiAmlObjectUpdateInteger (
|
|
|
|
IN EFI_ACPI_SDT_PROTOCOL *AcpiSdtProtocol,
|
|
|
|
IN EFI_ACPI_HANDLE TableHandle,
|
|
|
|
IN CHAR8 *AsciiObjectPath,
|
|
|
|
IN UINTN Value
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_ACPI_HANDLE ObjectHandle;
|
|
|
|
EFI_ACPI_HANDLE DataHandle;
|
|
|
|
EFI_ACPI_DATA_TYPE DataType;
|
|
|
|
UINT8 *Buffer;
|
|
|
|
UINTN BufferSize;
|
|
|
|
UINTN DataSize;
|
|
|
|
|
|
|
|
if (AcpiSdtProtocol == NULL || AsciiObjectPath == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjectHandle = NULL;
|
|
|
|
DataHandle = NULL;
|
|
|
|
|
|
|
|
Status = AcpiSdtProtocol->FindPath (TableHandle, AsciiObjectPath, &ObjectHandle);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = AcpiSdtProtocol->GetOption (ObjectHandle, 0, &DataType, (VOID *)&Buffer, &BufferSize);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
Status = EFI_NOT_FOUND;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
ASSERT (DataType == EFI_ACPI_DATA_TYPE_OPCODE);
|
|
|
|
ASSERT (Buffer != NULL);
|
|
|
|
|
|
|
|
if (Buffer[0] != AML_NAME_OP) {
|
|
|
|
Status = EFI_NOT_FOUND;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get handle of data object
|
|
|
|
//
|
|
|
|
Status = AcpiSdtProtocol->GetChild (ObjectHandle, &DataHandle);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
Status = AcpiSdtProtocol->GetOption (DataHandle, 0, &DataType, (VOID *)&Buffer, &BufferSize);
|
|
|
|
ASSERT (DataType == EFI_ACPI_DATA_TYPE_OPCODE);
|
|
|
|
ASSERT (Buffer != NULL);
|
|
|
|
|
|
|
|
if (Buffer[0] == AML_ZERO_OP || Buffer[0] == AML_ONE_OP) {
|
|
|
|
Status = AcpiSdtProtocol->SetOption (DataHandle, 0, (VOID *)&Value, sizeof (UINT8));
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Check the size of data object
|
|
|
|
//
|
|
|
|
switch (Buffer[0]) {
|
|
|
|
case AML_BYTE_PREFIX:
|
|
|
|
DataSize = sizeof (UINT8);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AML_WORD_PREFIX:
|
|
|
|
DataSize = sizeof (UINT16);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AML_DWORD_PREFIX:
|
|
|
|
DataSize = sizeof (UINT32);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AML_QWORD_PREFIX:
|
|
|
|
DataSize = sizeof (UINT64);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
// The data type of the ACPI object is not an integer
|
|
|
|
Status = EFI_INVALID_PARAMETER;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = AcpiSdtProtocol->SetOption (DataHandle, 1, (VOID *)&Value, DataSize);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
AcpiSdtProtocol->Close (DataHandle);
|
|
|
|
AcpiSdtProtocol->Close (ObjectHandle);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|