Ovmf/HardwareInfoLib: Add Dxe lib to dynamically parse heterogenous data

Following the Hardware Info library, create the DxeHardwareInfoLib
which implements the whole API capable of parsing heterogeneous hardware
information. The list-like API grants callers a flexible and common
pattern to retrieve the data. Moreover, the initial source is a BLOB
which generalizes the host-to-guest transmission mechanism.

The Hardware Info library main objective is to provide a way to
describe non-discoverable hardware so that the host can share the
available resources with the guest in Ovmf platforms. This change
features and embraces the main idea behind the library by providing
an API that parses a BLOB into a linked list to retrieve hardware
data from any source. Additionally, list-like APIs are provided so
that the hardware info list can be traversed conveniently.
Similarly, the capability is provided to filter results by specific
hardware types. However, heterogeneous elements can be added to the
list, increasing the flexibility. This way, a single source, for
example a fw-cfg file, can be used to describe several instances of
multiple types of hardware.

This part of the Hardware Info library makes use of dynamic memory
and is intended for stages in which memory services are available.
A motivation example is the PciHostBridgeLib. This library, part
of the PCI driver populates the list of PCI root bridges during DXE
stage for future steps to discover the resources under them. The
hardware info library can be used to obtain the detailed description
of available host bridges, for instance in the form of a fw-cfg file,
and parse that information into a dynmaic list that allows, first to
verify consistency of the data, and second discover the resources
availabe for each root bridge.

Cc: Alexander Graf <graf@amazon.de>
Cc: Gerd Hoffmann <kraxel@redhat.com>
Acked-by: Gerd Hoffmann <kraxel@redhat.com>
Signed-off-by: Nicolas Ojeda Leon <ncoleon@amazon.com>
This commit is contained in:
Nicolas Ojeda Leon 2022-01-19 15:12:30 +01:00 committed by mergify[bot]
parent 2b1a5b8c61
commit a1bd79c514
11 changed files with 302 additions and 0 deletions

View File

@ -82,6 +82,7 @@
PciHostBridgeLib|OvmfPkg/Fdt/FdtPciHostBridgeLib/FdtPciHostBridgeLib.inf
PciHostBridgeUtilityLib|OvmfPkg/Library/PciHostBridgeUtilityLib/PciHostBridgeUtilityLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(TPM2_ENABLE) == TRUE
Tpm2CommandLib|SecurityPkg/Library/Tpm2CommandLib/Tpm2CommandLib.inf

View File

@ -169,6 +169,7 @@
BlobVerifierLib|OvmfPkg/AmdSev/BlobVerifierLibSevHashes/BlobVerifierLibSevHashes.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SOURCE_DEBUG_ENABLE) == TRUE
PeCoffExtraActionLib|SourceLevelDebugPkg/Library/PeCoffExtraActionLibDebug/PeCoffExtraActionLibDebug.inf

View File

@ -171,6 +171,7 @@
LockBoxLib|OvmfPkg/Library/LockBoxLib/LockBoxBaseLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
CustomizedDisplayLib|MdeModulePkg/Library/CustomizedDisplayLib/CustomizedDisplayLib.inf
FrameBufferBltLib|MdeModulePkg/Library/FrameBufferBltLib/FrameBufferBltLib.inf

View File

@ -181,6 +181,7 @@
LoadLinuxLib|OvmfPkg/Library/LoadLinuxLib/LoadLinuxLib.inf
MemEncryptSevLib|OvmfPkg/Library/BaseMemEncryptSevLib/DxeMemEncryptSevLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SMM_REQUIRE) == FALSE
LockBoxLib|OvmfPkg/Library/LockBoxLib/LockBoxBaseLib.inf
!endif

View File

@ -0,0 +1,39 @@
## @file
# Hardware information library to describe non-discoverable hardware resources
#
# Copyright 2021 - 2022 Amazon.com, Inc. or its affiliates. All Rights Reserved.
#
# SPDX-License-Identifier: BSD-2-Clause-Patent
#
#
##
[Defines]
INF_VERSION = 0x00010005
BASE_NAME = DxeHardwareInfoLib
FILE_GUID = F60B206A-5C56-11EC-AEAC-67CB080BCFF2
MODULE_TYPE = BASE
VERSION_STRING = 1.0
LIBRARY_CLASS = DxeHardwareInfoLib
#
# The following information is for reference only and not required by the build
# tools.
#
# VALID_ARCHITECTURES = X64
#
[Sources]
HardwareInfoDxe.c
HardwareInfoPciHostBridgeLib.c
QemuFwCfgHardwareInfoLib.c
[Packages]
MdeModulePkg/MdeModulePkg.dec
MdePkg/MdePkg.dec
OvmfPkg/OvmfPkg.dec
[LibraryClasses]
BaseMemoryLib
DebugLib
MemoryAllocationLib

View File

@ -0,0 +1,254 @@
/*/@file
Hardware info parsing functions.
Binary data is expected as a consecutive series of header - object pairs.
Complete library providing list-like interface to dynamically manipulate
hardware info objects and parsing from a generic blob.
Copyright 2021 - 2022 Amazon.com, Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent
**/
#include <Uefi/UefiBaseType.h>
#include <Uefi/UefiSpec.h>
#include <Library/DebugLib.h>
#include <Library/MemoryAllocationLib.h>
#include <Library/BaseMemoryLib.h>
#include <Library/BaseLib.h>
#include <Library/UefiLib.h>
#include <Library/HardwareInfoLib.h>
EFI_STATUS
CreateHardwareInfoList (
IN UINT8 *Blob,
IN UINTN BlobSize,
IN HARDWARE_INFO_TYPE TypeFilter,
OUT LIST_ENTRY *ListHead
)
{
UINT8 *Index;
UINT8 *BlobEnd;
HARDWARE_INFO *HwComponent;
if ((Blob == NULL) || (BlobSize <= 0) ||
(ListHead == NULL))
{
return EFI_INVALID_PARAMETER;
}
Index = Blob;
BlobEnd = Blob + BlobSize;
while (Index < BlobEnd) {
HwComponent = AllocateZeroPool (sizeof (HARDWARE_INFO));
if (HwComponent == NULL) {
goto FailedAllocate;
}
HwComponent->Header.Type.Uint64 = *((UINT64 *)Index);
Index += sizeof (HwComponent->Header.Type);
HwComponent->Header.Size = *((UINT64 *)(Index));
Index += sizeof (HwComponent->Header.Size);
if ((HwComponent->Header.Size > MAX_UINTN) || (Index < Blob) || ((Index + HwComponent->Header.Size) > BlobEnd)) {
goto FreeResources;
}
//
// Check if optional TypeFilter is set, skip if the current
// object is of a different type and release the partially
// allocated object
//
if ((TypeFilter != HardwareInfoTypeUndefined) &&
(HwComponent->Header.Type.Value != TypeFilter))
{
FreePool (HwComponent);
Index += HwComponent->Header.Size;
continue;
}
HwComponent->Data.Raw = AllocateZeroPool ((UINTN)HwComponent->Header.Size);
if (HwComponent->Data.Raw == NULL) {
goto FreeResources;
}
CopyMem (HwComponent->Data.Raw, Index, (UINTN)HwComponent->Header.Size);
Index += HwComponent->Header.Size;
InsertTailList (ListHead, &HwComponent->Link);
}
return EFI_SUCCESS;
FreeResources:
//
// Clean the resources allocated in the incomplete cycle
//
FreePool (HwComponent);
FailedAllocate:
DEBUG ((
EFI_D_ERROR,
"%a: Failed to allocate memory for hardware info\n",
__FUNCTION__
));
return EFI_OUT_OF_RESOURCES;
}
VOID
FreeHardwareInfoList (
IN OUT LIST_ENTRY *ListHead
)
{
LIST_ENTRY *CurrentLink;
HARDWARE_INFO *HwComponent;
if (IsListEmpty (ListHead)) {
return;
}
CurrentLink = ListHead->ForwardLink;
while (CurrentLink != NULL && CurrentLink != ListHead) {
HwComponent = HARDWARE_INFO_FROM_LINK (CurrentLink);
//
// Remove item from list before invalidating the pointers
//
CurrentLink = RemoveEntryList (CurrentLink);
FreePool (HwComponent->Data.Raw);
FreePool (HwComponent);
}
}
/**
Validates if the specified Node has a valid data size and is of
specified type.
The data size can be less or equal to the provided type size to be
regarded as valid and thus accessible with the typed pointer.
For future compatibility the size is allowed to be smaller so that
different versions interpret fields differently and, particularly,
have smaller data structures. However, it cannot be larger than the
type size to avoid accessing memory out of bounds.
@param[in] Node Hardware Info node to be validated
@param[in] TypeSize Size (in bytes) of the data type intended to be
used to dereference the data.
@retval TRUE Node is valid and can be accessed
@retval FALSE Node is not valid
/*/
STATIC
BOOLEAN
IsHardwareInfoNodeValidByType (
IN LIST_ENTRY *ListHead,
IN LIST_ENTRY *Link,
IN HARDWARE_INFO_TYPE Type,
IN UINTN TypeSize
)
{
HARDWARE_INFO *HwComponent;
if (IsNull (ListHead, Link)) {
return FALSE;
}
HwComponent = HARDWARE_INFO_FROM_LINK (Link);
//
// Verify if the node type is the specified one and the size of
// the data allocated to the node is greater than the size of
// the type intended to dereference it in order to avoid access
// to memory out of bondaries.
//
if ((HwComponent->Header.Type.Value == Type) &&
(HwComponent->Header.Size >= TypeSize))
{
return TRUE;
}
return FALSE;
}
UINTN
GetHardwareInfoCountByType (
IN LIST_ENTRY *ListHead,
IN HARDWARE_INFO_TYPE Type,
IN UINTN TypeSize
)
{
UINTN Count;
LIST_ENTRY *Link;
Count = 0;
for (Link = GetFirstHardwareInfoByType (ListHead, Type, TypeSize);
!IsNull (ListHead, Link);
Link = GetNextHardwareInfoByType (ListHead, Link, Type, TypeSize))
{
if (IsHardwareInfoNodeValidByType (ListHead, Link, Type, TypeSize)) {
Count++;
}
}
return Count;
}
LIST_ENTRY *
GetFirstHardwareInfoByType (
IN LIST_ENTRY *ListHead,
IN HARDWARE_INFO_TYPE Type,
IN UINTN TypeSize
)
{
LIST_ENTRY *Link;
if (IsListEmpty (ListHead)) {
return ListHead;
}
Link = GetFirstNode (ListHead);
if (IsHardwareInfoNodeValidByType (ListHead, Link, Type, TypeSize)) {
return Link;
}
return GetNextHardwareInfoByType (ListHead, Link, Type, TypeSize);
}
LIST_ENTRY *
GetNextHardwareInfoByType (
IN LIST_ENTRY *ListHead,
IN LIST_ENTRY *Node,
IN HARDWARE_INFO_TYPE Type,
IN UINTN TypeSize
)
{
LIST_ENTRY *Link;
Link = GetNextNode (ListHead, Node);
while (!IsNull (ListHead, Link)) {
if (IsHardwareInfoNodeValidByType (ListHead, Link, Type, TypeSize)) {
//
// Found a node of specified type and with valid size. Break and
// return the found node.
//
break;
}
Link = GetNextNode (ListHead, Link);
}
return Link;
}
BOOLEAN
EndOfHardwareInfoList (
IN LIST_ENTRY *ListHead,
IN LIST_ENTRY *Node
)
{
return IsNull (ListHead, Node);
}

View File

@ -184,6 +184,7 @@
FrameBufferBltLib|MdeModulePkg/Library/FrameBufferBltLib/FrameBufferBltLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SOURCE_DEBUG_ENABLE) == TRUE
PeCoffExtraActionLib|SourceLevelDebugPkg/Library/PeCoffExtraActionLibDebug/PeCoffExtraActionLibDebug.inf

View File

@ -179,6 +179,7 @@
MemEncryptSevLib|OvmfPkg/Library/BaseMemEncryptSevLib/DxeMemEncryptSevLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLibNull.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SMM_REQUIRE) == FALSE
LockBoxLib|OvmfPkg/Library/LockBoxLib/LockBoxBaseLib.inf
!endif

View File

@ -183,6 +183,7 @@
MemEncryptSevLib|OvmfPkg/Library/BaseMemEncryptSevLib/DxeMemEncryptSevLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLibNull.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SMM_REQUIRE) == FALSE
LockBoxLib|OvmfPkg/Library/LockBoxLib/LockBoxBaseLib.inf
!endif

View File

@ -195,6 +195,7 @@
MemEncryptSevLib|OvmfPkg/Library/BaseMemEncryptSevLib/DxeMemEncryptSevLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SMM_REQUIRE) == FALSE
LockBoxLib|OvmfPkg/Library/LockBoxLib/LockBoxBaseLib.inf

View File

@ -173,6 +173,7 @@
FrameBufferBltLib|MdeModulePkg/Library/FrameBufferBltLib/FrameBufferBltLib.inf
MemEncryptTdxLib|OvmfPkg/Library/BaseMemEncryptTdxLib/BaseMemEncryptTdxLib.inf
PeiHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/PeiHardwareInfoLib.inf
DxeHardwareInfoLib|OvmfPkg/Library/HardwareInfoLib/DxeHardwareInfoLib.inf
!if $(SOURCE_DEBUG_ENABLE) == TRUE
PeCoffExtraActionLib|SourceLevelDebugPkg/Library/PeCoffExtraActionLibDebug/PeCoffExtraActionLibDebug.inf