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>
		
			
				
	
	
		
			220 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			220 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @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);
 | |
| }
 |