DynamicTablesPkg: AmlLib APIs
AmlLib library implements an AML parser, AML tree interface, serialiser, code generator and other interfaces to generate Definition Block tables. The AmlLib APIs are a collection of interfaces that enable parsing, iterating, modifying, adding, and serialising AML data to generate a Definition Block table. The AmlLib APIs are declared in Include\AmlLib\AmlLib.h Signed-off-by: Pierre Gondois <pierre.gondois@arm.com> Signed-off-by: Sami Mujawar <sami.mujawar@arm.com> Reviewed-by: Alexei Fedorov <Alexei.Fedorov@arm.com>
This commit is contained in:
committed by
mergify[bot]
parent
3196253710
commit
c85ac5245c
219
DynamicTablesPkg/Library/Common/AmlLib/Api/AmlApiHelper.c
Normal file
219
DynamicTablesPkg/Library/Common/AmlLib/Api/AmlApiHelper.c
Normal file
@@ -0,0 +1,219 @@
|
||||
/** @file
|
||||
AML Helper.
|
||||
|
||||
Copyright (c) 2020, Arm Limited. All rights reserved.<BR>
|
||||
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
**/
|
||||
|
||||
/* Even though this file has access to the internal Node definition,
|
||||
i.e. AML_ROOT_NODE, AML_OBJECT_NODE, etc. Only the external node
|
||||
handle types should be used, i.e. AML_NODE_HANDLE, AML_ROOT_NODE_HANDLE,
|
||||
etc.
|
||||
Indeed, the functions in the "Api" folder should be implemented only
|
||||
using the "safe" functions available in the "Include" folder. This
|
||||
makes the functions available in the "Api" folder easy to export.
|
||||
*/
|
||||
#include <Api/AmlApiHelper.h>
|
||||
|
||||
#include <AmlCoreInterface.h>
|
||||
#include <AmlInclude.h>
|
||||
#include <String/AmlString.h>
|
||||
|
||||
/** Compare the NameString defined by the "Name ()" ASL function,
|
||||
and stored in the NameOpNode, with the input NameString.
|
||||
|
||||
An ASL NameString is expected to be NULL terminated, and can be composed
|
||||
of NameSegs that have less that 4 chars, like "DEV". "DEV" will be expanded
|
||||
as "DEV_".
|
||||
|
||||
An AML NameString is not NULL terminated and is is only composed of
|
||||
4 chars long NameSegs.
|
||||
|
||||
@param [in] NameOpNode NameOp object node defining a variable.
|
||||
Must have an AML_NAME_OP/0 OpCode/SubOpCode.
|
||||
NameOp object nodes are defined in ASL
|
||||
using the "Name ()" function.
|
||||
@param [in] AslName ASL NameString to compare the NameOp's name with.
|
||||
Must be NULL terminated.
|
||||
|
||||
@retval TRUE If the AslName and the AmlName defined by the NameOp node
|
||||
are similar.
|
||||
@retval FALSE Otherwise.
|
||||
**/
|
||||
BOOLEAN
|
||||
EFIAPI
|
||||
AmlNameOpCompareName (
|
||||
IN AML_OBJECT_NODE_HANDLE NameOpNode,
|
||||
IN CHAR8 * AslName
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
AML_DATA_NODE_HANDLE NameDataNode;
|
||||
|
||||
CHAR8 * AmlName;
|
||||
UINT32 AmlNameSize;
|
||||
|
||||
BOOLEAN RetVal;
|
||||
|
||||
if ((NameOpNode == NULL) ||
|
||||
(AmlGetNodeType ((AML_NODE_HANDLE)NameOpNode) != EAmlNodeObject) ||
|
||||
(!AmlNodeHasOpCode (NameOpNode, AML_NAME_OP, 0)) ||
|
||||
(AslName == NULL)) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Get the NameOp name, being in a data node
|
||||
// which is the first fixed argument (i.e. index 0).
|
||||
NameDataNode = (AML_DATA_NODE_HANDLE)AmlGetFixedArgument (
|
||||
NameOpNode,
|
||||
EAmlParseIndexTerm0
|
||||
);
|
||||
if ((NameDataNode == NULL) ||
|
||||
(AmlGetNodeType ((AML_NODE_HANDLE)NameDataNode) != EAmlNodeData) ||
|
||||
(!AmlNodeHasDataType (NameDataNode, EAmlNodeDataTypeNameString))) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Get the size of the name.
|
||||
Status = AmlGetDataNodeBuffer (NameDataNode, NULL, &AmlNameSize);
|
||||
if (EFI_ERROR (Status)) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Allocate memory to fetch the name.
|
||||
AmlName = AllocateZeroPool (AmlNameSize);
|
||||
if (AmlName == NULL) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Fetch the name.
|
||||
Status = AmlGetDataNodeBuffer (NameDataNode, (UINT8*)AmlName, &AmlNameSize);
|
||||
if (EFI_ERROR (Status)) {
|
||||
FreePool (AmlName);
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Compare the input AslName and the AmlName stored in the NameOp node.
|
||||
RetVal = CompareAmlWithAslNameString (AmlName, AslName);
|
||||
|
||||
// Free the string buffer.
|
||||
FreePool (AmlName);
|
||||
return RetVal;
|
||||
}
|
||||
|
||||
/** Check whether ObjectNode has the input OpCode/SubOpcode couple.
|
||||
|
||||
@param [in] ObjectNode Pointer to an object node.
|
||||
@param [in] OpCode OpCode to check
|
||||
@param [in] SubOpCode SubOpCode to check
|
||||
|
||||
@retval TRUE The node is an object node and
|
||||
the Opcode and SubOpCode match.
|
||||
@retval FALSE Otherwise.
|
||||
**/
|
||||
BOOLEAN
|
||||
EFIAPI
|
||||
AmlNodeHasOpCode (
|
||||
IN AML_OBJECT_NODE_HANDLE ObjectNode,
|
||||
IN UINT8 OpCode,
|
||||
IN UINT8 SubOpCode
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
UINT8 NodeOpCode;
|
||||
UINT8 NodeSubOpCode;
|
||||
|
||||
// Get the Node information.
|
||||
Status = AmlGetObjectNodeInfo (
|
||||
ObjectNode,
|
||||
&NodeOpCode,
|
||||
&NodeSubOpCode,
|
||||
NULL,
|
||||
NULL
|
||||
);
|
||||
if (EFI_ERROR (Status)) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Check the OpCode and SubOpCode.
|
||||
if ((OpCode != NodeOpCode) ||
|
||||
(SubOpCode != NodeSubOpCode)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/** Check whether DataNode has the input DataType.
|
||||
|
||||
@param [in] DataNode Pointer to a data node.
|
||||
@param [in] DataType DataType to check.
|
||||
|
||||
@retval TRUE The node is a data node and
|
||||
the DataType match.
|
||||
@retval FALSE Otherwise.
|
||||
**/
|
||||
BOOLEAN
|
||||
EFIAPI
|
||||
AmlNodeHasDataType (
|
||||
IN AML_DATA_NODE_HANDLE DataNode,
|
||||
IN EAML_NODE_DATA_TYPE DataType
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
EAML_NODE_DATA_TYPE NodeDataType;
|
||||
|
||||
// Get the data type.
|
||||
Status = AmlGetNodeDataType (DataNode, &NodeDataType);
|
||||
if (EFI_ERROR (Status)) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Check the data type.
|
||||
if (NodeDataType != DataType) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/** Check whether RdNode has the input RdDataType.
|
||||
|
||||
@param [in] RdNode Pointer to a data node.
|
||||
@param [in] RdDataType DataType to check.
|
||||
|
||||
@retval TRUE The node is a Resource Data node and
|
||||
the RdDataType match.
|
||||
@retval FALSE Otherwise.
|
||||
**/
|
||||
BOOLEAN
|
||||
EFIAPI
|
||||
AmlNodeHasRdDataType (
|
||||
IN AML_DATA_NODE_HANDLE RdNode,
|
||||
IN AML_RD_HEADER RdDataType
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
AML_RD_HEADER NodeRdDataType;
|
||||
|
||||
// Get the resource data type.
|
||||
Status = AmlGetResourceDataType (
|
||||
RdNode,
|
||||
&NodeRdDataType
|
||||
);
|
||||
if (EFI_ERROR (Status)) {
|
||||
ASSERT (0);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Check the RdDataType.
|
||||
return AmlRdCompareDescId (&NodeRdDataType, RdDataType);
|
||||
}
|
Reference in New Issue
Block a user