/** @file AML Node. Copyright (c) 2019 - 2020, Arm Limited. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #include #include #include /** Initialize an AML_NODE_HEADER structure. @param [in] Node Pointer to a node header. @param [in] NodeType NodeType to initialize the Node with. Must be an EAML_NODE_TYPE. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. **/ STATIC EFI_STATUS EFIAPI AmlInitializeNodeHeader ( IN AML_NODE_HEADER * Node, IN EAML_NODE_TYPE NodeType ) { if (Node == NULL) { ASSERT (0); return EFI_INVALID_PARAMETER; } InitializeListHead (&Node->Link); Node->Parent = NULL; Node->NodeType = NodeType; return EFI_SUCCESS; } /** Delete a root node and its ACPI DSDT/SSDT header. It is the caller's responsibility to check the RootNode has been removed from the tree and is not referencing any other node in the tree. @param [in] RootNode Pointer to a root node. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. **/ STATIC EFI_STATUS EFIAPI AmlDeleteRootNode ( IN AML_ROOT_NODE * RootNode ) { if (!IS_AML_ROOT_NODE (RootNode)) { ASSERT (0); return EFI_INVALID_PARAMETER; } if ((RootNode->SdtHeader != NULL)) { FreePool (RootNode->SdtHeader); } else { ASSERT (0); return EFI_INVALID_PARAMETER; } FreePool (RootNode); return EFI_SUCCESS; } /** Create an AML_ROOT_NODE. This node will be the root of the tree. @param [in] SdtHeader Pointer to an ACPI DSDT/SSDT header to copy the data from. @param [out] NewRootNodePtr The created AML_ROOT_NODE. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. @retval EFI_OUT_OF_RESOURCES Could not allocate memory. **/ EFI_STATUS EFIAPI AmlCreateRootNode ( IN CONST EFI_ACPI_DESCRIPTION_HEADER * SdtHeader, OUT AML_ROOT_NODE ** NewRootNodePtr ) { EFI_STATUS Status; AML_ROOT_NODE * RootNode; if ((SdtHeader == NULL) || (NewRootNodePtr == NULL)) { ASSERT (0); return EFI_INVALID_PARAMETER; } RootNode = AllocateZeroPool (sizeof (AML_ROOT_NODE)); if (RootNode == NULL) { ASSERT (0); return EFI_OUT_OF_RESOURCES; } Status = AmlInitializeNodeHeader (&RootNode->NodeHeader, EAmlNodeRoot); if (EFI_ERROR (Status)) { FreePool (RootNode); ASSERT (0); return Status; } InitializeListHead (&RootNode->VariableArgs); RootNode->SdtHeader = AllocateCopyPool ( sizeof (EFI_ACPI_DESCRIPTION_HEADER), SdtHeader ); if (RootNode->SdtHeader == NULL) { ASSERT (0); AmlDeleteRootNode (RootNode); return EFI_OUT_OF_RESOURCES; } *NewRootNodePtr = RootNode; return EFI_SUCCESS; } /** Delete an object node. It is the caller's responsibility to check the ObjectNode has been removed from the tree and is not referencing any other node in the tree. @param [in] ObjectNode Pointer to an object node. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. **/ STATIC EFI_STATUS EFIAPI AmlDeleteObjectNode ( IN AML_OBJECT_NODE * ObjectNode ) { if (!IS_AML_OBJECT_NODE (ObjectNode)) { ASSERT (0); return EFI_INVALID_PARAMETER; } FreePool (ObjectNode); return EFI_SUCCESS; } /** Create an AML_OBJECT_NODE. @param [in] AmlByteEncoding Byte encoding entry. @param [in] PkgLength PkgLength of the node if the AmlByteEncoding has the PkgLen attribute. 0 otherwise. @param [out] NewObjectNodePtr The created AML_OBJECT_NODE. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. @retval EFI_OUT_OF_RESOURCES Could not allocate memory. **/ EFI_STATUS EFIAPI AmlCreateObjectNode ( IN CONST AML_BYTE_ENCODING * AmlByteEncoding, IN UINT32 PkgLength, OUT AML_OBJECT_NODE ** NewObjectNodePtr ) { EFI_STATUS Status; AML_OBJECT_NODE * ObjectNode; if ((AmlByteEncoding == NULL) || (NewObjectNodePtr == NULL)) { ASSERT (0); return EFI_INVALID_PARAMETER; } ObjectNode = AllocateZeroPool (sizeof (AML_OBJECT_NODE)); if (ObjectNode == NULL) { ASSERT (0); return EFI_OUT_OF_RESOURCES; } Status = AmlInitializeNodeHeader (&ObjectNode->NodeHeader, EAmlNodeObject); if (EFI_ERROR (Status)) { FreePool (ObjectNode); ASSERT (0); return Status; } InitializeListHead (&ObjectNode->VariableArgs); // ObjectNode->FixedArgs[...] is already initialised to NULL as the // ObjectNode is Zero allocated. ObjectNode->AmlByteEncoding = AmlByteEncoding; ObjectNode->PkgLen = PkgLength; *NewObjectNodePtr = ObjectNode; return EFI_SUCCESS; } /** Delete a data node and its buffer. It is the caller's responsibility to check the DataNode has been removed from the tree and is not referencing any other node in the tree. @param [in] DataNode Pointer to a data node. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. @retval EFI_OUT_OF_RESOURCES Could not allocate memory. **/ STATIC EFI_STATUS EFIAPI AmlDeleteDataNode ( IN AML_DATA_NODE * DataNode ) { if (!IS_AML_DATA_NODE (DataNode)) { ASSERT (0); return EFI_INVALID_PARAMETER; } if (DataNode->Buffer != NULL) { FreePool (DataNode->Buffer); } else { ASSERT (0); return EFI_INVALID_PARAMETER; } FreePool (DataNode); return EFI_SUCCESS; } /** Create an AML_DATA_NODE. @param [in] DataType DataType of the node. @param [in] Data Pointer to the AML bytecode corresponding to this node. Data is copied from there. @param [in] DataSize Number of bytes to consider at the address pointed by Data. @param [out] NewDataNodePtr The created AML_DATA_NODE. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. @retval EFI_OUT_OF_RESOURCES Could not allocate memory. **/ EFI_STATUS EFIAPI AmlCreateDataNode ( IN EAML_NODE_DATA_TYPE DataType, IN CONST UINT8 * Data, IN UINT32 DataSize, OUT AML_DATA_NODE ** NewDataNodePtr ) { EFI_STATUS Status; AML_DATA_NODE * DataNode; // A data node must not be created for certain data types. if ((DataType == EAmlNodeDataTypeNone) || (DataType == EAmlNodeDataTypeReserved1) || (DataType == EAmlNodeDataTypeReserved2) || (DataType == EAmlNodeDataTypeReserved3) || (DataType == EAmlNodeDataTypeReserved4) || (DataType == EAmlNodeDataTypeReserved5) || (Data == NULL) || (DataSize == 0) || (NewDataNodePtr == NULL)) { ASSERT (0); return EFI_INVALID_PARAMETER; } DataNode = AllocateZeroPool (sizeof (AML_DATA_NODE)); if (DataNode == NULL) { ASSERT (0); return EFI_OUT_OF_RESOURCES; } Status = AmlInitializeNodeHeader (&DataNode->NodeHeader, EAmlNodeData); if (EFI_ERROR (Status)) { FreePool (DataNode); ASSERT (0); return Status; } DataNode->Buffer = AllocateCopyPool (DataSize, Data); if (DataNode->Buffer == NULL) { AmlDeleteDataNode (DataNode); ASSERT (0); return EFI_OUT_OF_RESOURCES; } DataNode->DataType = DataType; DataNode->Size = DataSize; *NewDataNodePtr = DataNode; return EFI_SUCCESS; } /** Delete a Node. @param [in] Node Pointer to a Node. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. **/ EFI_STATUS EFIAPI AmlDeleteNode ( IN AML_NODE_HEADER * Node ) { EFI_STATUS Status; EAML_PARSE_INDEX Index; // Check that the node being deleted is unlinked. // When removing the node, its parent and list are reset // with InitializeListHead. Thus it must be empty. if (!IS_AML_NODE_VALID (Node) || !AML_NODE_IS_DETACHED (Node)) { ASSERT (0); return EFI_INVALID_PARAMETER; } switch (Node->NodeType) { case EAmlNodeRoot: { // Check the variable list of arguments has been cleaned. if (!IsListEmpty (AmlNodeGetVariableArgList (Node))) { ASSERT (0); return EFI_INVALID_PARAMETER; } Status = AmlDeleteRootNode ((AML_ROOT_NODE*)Node); if (EFI_ERROR (Status)) { ASSERT (0); } break; } case EAmlNodeObject: { // Check the variable list of arguments has been cleaned. if (!IsListEmpty (AmlNodeGetVariableArgList (Node))) { ASSERT (0); return EFI_INVALID_PARAMETER; } // Check the fixed argument list has been cleaned. for (Index = EAmlParseIndexTerm0; Index < EAmlParseIndexMax; Index++) { if (((AML_OBJECT_NODE*)Node)->FixedArgs[Index] != NULL) { ASSERT (0); return EFI_INVALID_PARAMETER; } } Status = AmlDeleteObjectNode ((AML_OBJECT_NODE*)Node); if (EFI_ERROR (Status)) { ASSERT (0); } break; } case EAmlNodeData: { Status = AmlDeleteDataNode ((AML_DATA_NODE*)Node); if (EFI_ERROR (Status)) { ASSERT (0); } break; } default: { ASSERT (0); Status = EFI_INVALID_PARAMETER; break; } } // switch return Status; } /** Check whether ObjectNode has the input attribute. This function can be used to check ObjectNode is an object node at the same time. @param [in] ObjectNode Pointer to an object node. @param [in] Attribute Attribute to check for. @retval TRUE The node is an AML object and the attribute is present. @retval FALSE Otherwise. **/ BOOLEAN EFIAPI AmlNodeHasAttribute ( IN CONST AML_OBJECT_NODE * ObjectNode, IN AML_OP_ATTRIBUTE Attribute ) { if (!IS_AML_OBJECT_NODE (ObjectNode) || (ObjectNode->AmlByteEncoding == NULL)) { return FALSE; } return ((ObjectNode->AmlByteEncoding->Attribute & Attribute) == 0 ? FALSE : TRUE); } /** 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 AML object and the Opcode and the SubOpCode match. @retval FALSE Otherwise. **/ BOOLEAN EFIAPI AmlNodeCompareOpCode ( IN CONST AML_OBJECT_NODE * ObjectNode, IN UINT8 OpCode, IN UINT8 SubOpCode ) { if (!IS_AML_OBJECT_NODE (ObjectNode) || (ObjectNode->AmlByteEncoding == NULL)) { return FALSE; } ASSERT (AmlIsOpCodeValid (OpCode, SubOpCode)); return ((ObjectNode->AmlByteEncoding->OpCode == OpCode) && (ObjectNode->AmlByteEncoding->SubOpCode == SubOpCode)) ? TRUE : FALSE; } /** Check whether a Node is an integer node. By integer node we mean an object node having one of the following opcode: - AML_BYTE_PREFIX; - AML_WORD_PREFIX; - AML_DWORD_PREFIX; - AML_QWORD_PREFIX. @param [in] Node The node to check. @retval TRUE The Node is an integer node. @retval FALSE Otherwise. **/ BOOLEAN EFIAPI IsIntegerNode ( IN AML_OBJECT_NODE * Node ) { UINT8 OpCode; if (!IS_AML_OBJECT_NODE (Node) || (Node->AmlByteEncoding == NULL)) { return FALSE; } // Check Node is an integer node. OpCode = Node->AmlByteEncoding->OpCode; if ((OpCode != AML_BYTE_PREFIX) && (OpCode != AML_WORD_PREFIX) && (OpCode != AML_DWORD_PREFIX) && (OpCode != AML_QWORD_PREFIX)) { return FALSE; } return TRUE; } /** Check whether a Node is a ZeroOp, a OneOp or a OnesOp. These two objects don't have a data node holding a value. This require special handling. @param [in] Node The node to check. @retval TRUE The Node is a ZeroOp or OneOp. @retval FALSE Otherwise. **/ BOOLEAN EFIAPI IsSpecialIntegerNode ( IN AML_OBJECT_NODE * Node ) { UINT8 OpCode; if (!IS_AML_OBJECT_NODE (Node) || (Node->AmlByteEncoding == NULL)) { return FALSE; } OpCode = Node->AmlByteEncoding->OpCode; if ((OpCode != AML_ZERO_OP) && (OpCode != AML_ONE_OP) && (OpCode != AML_ONES_OP)) { return FALSE; } return TRUE; } /** Check whether Node corresponds to a method definition. A method definition can be introduced: - By a method object, having an AML_METHOD_OP OpCode; - By an external definition of a method, having an AML_EXTERNAL_OP OpCode and an ObjectType byte set to the MethodObj. Note: An alias node, having an AML_ALIAS_OP, can be resolved to a method definition. This function doesn't handle this case. @param [in] Node Node to check whether it is a method definition. @retval TRUE The Node is a method definition. @retval FALSE Otherwise. **/ BOOLEAN EFIAPI AmlIsMethodDefinitionNode ( IN CONST AML_OBJECT_NODE * Node ) { AML_DATA_NODE * ObjectType; // Node is checked to be an object node aswell. if (AmlNodeCompareOpCode (Node, AML_METHOD_OP, 0)) { return TRUE; } else if (AmlNodeCompareOpCode (Node, AML_EXTERNAL_OP, 0)) { // If the node is an external definition, check this is a method. // DefExternal := ExternalOp NameString ObjectType ArgumentCount // ExternalOp := 0x15 // ObjectType := ByteData // ArgumentCount := ByteData (0 - 7) ObjectType = (AML_DATA_NODE*)AmlGetFixedArgument ( (AML_OBJECT_NODE*)Node, EAmlParseIndexTerm1 ); if (IS_AML_DATA_NODE (ObjectType) && (ObjectType->DataType == EAmlNodeDataTypeUInt) && ((ObjectType->Size == 1))) { if (*((UINT8*)ObjectType->Buffer) == (UINT8)EAmlObjTypeMethodObj) { // The external definition is a method. return TRUE; } else { // The external definition is not a method. return FALSE; } } else { // The tree is inconsistent. ASSERT (0); return FALSE; } } // This is not a method definition. return FALSE; } /** Get the index at which the name of the node is stored. @param [in] ObjectNode Pointer to an object node. Must have the AML_IN_NAMESPACE attribute. @param [out] Index Index of the name in the fixed list of arguments. @retval EFI_SUCCESS The function completed successfully. @retval EFI_INVALID_PARAMETER Invalid parameter. **/ EFI_STATUS AmlNodeGetNameIndex ( IN CONST AML_OBJECT_NODE * ObjectNode, OUT EAML_PARSE_INDEX * Index ) { EAML_PARSE_INDEX NameIndex; if (!AmlNodeHasAttribute (ObjectNode, AML_IN_NAMESPACE) || (ObjectNode->AmlByteEncoding == NULL) || (Index == NULL)) { ASSERT (0); return EFI_INVALID_PARAMETER; } NameIndex = ObjectNode->AmlByteEncoding->NameIndex; if ((NameIndex > ObjectNode->AmlByteEncoding->MaxIndex) || (ObjectNode->AmlByteEncoding->Format[NameIndex] != EAmlName)) { ASSERT (0); return EFI_INVALID_PARAMETER; } *Index = NameIndex; return EFI_SUCCESS; } /** Get the name of the Node. Node must be part of the namespace. @param [in] ObjectNode Pointer to an object node, which is part of the namespace. @return A pointer to the name. NULL otherwise. Return NULL for the root node. **/ CHAR8 * EFIAPI AmlNodeGetName ( IN CONST AML_OBJECT_NODE * ObjectNode ) { EFI_STATUS Status; EAML_PARSE_INDEX NameIndex; AML_DATA_NODE * DataNode; if (!AmlNodeHasAttribute (ObjectNode, AML_IN_NAMESPACE)) { ASSERT (0); return NULL; } // Get the index at which the name is stored in the fixed arguments list. Status = AmlNodeGetNameIndex (ObjectNode, &NameIndex); if (EFI_ERROR (Status)) { ASSERT (0); return NULL; } // The name is stored in a Data node. DataNode = (AML_DATA_NODE*)ObjectNode->FixedArgs[NameIndex]; if (IS_AML_DATA_NODE (DataNode) && (DataNode->DataType == EAmlNodeDataTypeNameString)) { return (CHAR8*)DataNode->Buffer; } /* Return NULL if no name is found. This can occur if the name of a node is defined as a further fixed argument. E.g.: CreateField (BD03, 0x28, Add (ID03 + 0x08), BF33) ^ The parser is here. The parent of the Add statement is the CreateField statement. This statement defines a name in the AML namespace. This name defined as the fourth fixed argument. It hasn't been parsed yet. */ return NULL; }