2015-01-02 13:04:15 +01:00
|
|
|
/** @file
|
|
|
|
|
|
|
|
Copyright (C) 2013 - 2014, Red Hat, Inc.
|
|
|
|
Copyright (c) 2011 - 2013, Intel Corporation. All rights reserved.<BR>
|
2021-10-11 15:35:06 +02:00
|
|
|
(C) Copyright 2021 Hewlett Packard Enterprise Development LP<BR>
|
2024-04-30 09:48:09 +02:00
|
|
|
Copyright (c) 2024 Loongson Technology Corporation Limited. All rights reserved.<BR>
|
2015-01-02 13:04:15 +01:00
|
|
|
|
2019-04-04 01:03:23 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2015-01-02 13:04:15 +01:00
|
|
|
**/
|
|
|
|
|
2024-04-30 09:48:09 +02:00
|
|
|
#include <Base.h>
|
2016-04-08 11:44:58 +02:00
|
|
|
#include <Uefi.h>
|
|
|
|
|
2024-04-30 09:48:09 +02:00
|
|
|
#include <Pi/PiBootMode.h>
|
|
|
|
#include <Pi/PiHob.h>
|
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
#include <Library/BaseLib.h>
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
2015-09-24 23:40:41 +02:00
|
|
|
#include <Library/DebugLib.h>
|
2024-04-30 09:48:09 +02:00
|
|
|
#include <Library/HobLib.h>
|
2015-01-02 13:04:15 +01:00
|
|
|
#include <Library/IoLib.h>
|
|
|
|
#include <Library/QemuFwCfgLib.h>
|
2016-04-08 11:44:58 +02:00
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
|
|
|
|
#include <Protocol/FdtClient.h>
|
2015-01-02 13:04:15 +01:00
|
|
|
|
2024-04-30 09:48:02 +02:00
|
|
|
#include "QemuFwCfgLibMmioInternal.h"
|
2015-09-24 23:40:41 +02:00
|
|
|
|
|
|
|
//
|
2017-01-27 07:29:12 +01:00
|
|
|
// These correspond to the implementation we detect at runtime.
|
2015-09-24 23:40:41 +02:00
|
|
|
//
|
2024-04-30 09:48:02 +02:00
|
|
|
READ_BYTES_FUNCTION *InternalQemuFwCfgReadBytes = MmioReadBytes;
|
|
|
|
WRITE_BYTES_FUNCTION *InternalQemuFwCfgWriteBytes = MmioWriteBytes;
|
|
|
|
SKIP_BYTES_FUNCTION *InternalQemuFwCfgSkipBytes = MmioSkipBytes;
|
2015-01-02 13:04:15 +01:00
|
|
|
|
2024-04-30 09:48:09 +02:00
|
|
|
/**
|
|
|
|
Build firmware configure resource HOB.
|
|
|
|
|
|
|
|
@param[in] FwCfgResource A pointer to firmware configure resource.
|
|
|
|
|
|
|
|
@retval VOID
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
QemuBuildFwCfgResourceHob (
|
|
|
|
IN QEMU_FW_CFG_RESOURCE *FwCfgResource
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BuildGuidDataHob (
|
|
|
|
&gQemuFirmwareResourceHobGuid,
|
|
|
|
(VOID *)FwCfgResource,
|
|
|
|
sizeof (QEMU_FW_CFG_RESOURCE)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get firmware configure resource in HOB.
|
|
|
|
|
|
|
|
@param VOID
|
|
|
|
|
|
|
|
@retval non-NULL The firmware configure resource in HOB.
|
|
|
|
NULL The firmware configure resource not found.
|
|
|
|
**/
|
|
|
|
QEMU_FW_CFG_RESOURCE *
|
|
|
|
QemuGetFwCfgResourceHob (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_HOB_GUID_TYPE *GuidHob;
|
|
|
|
|
|
|
|
GuidHob = NULL;
|
|
|
|
|
|
|
|
GuidHob = GetFirstGuidHob (&gQemuFirmwareResourceHobGuid);
|
|
|
|
if (GuidHob == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (QEMU_FW_CFG_RESOURCE *)GET_GUID_HOB_DATA (GuidHob);
|
|
|
|
}
|
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
/**
|
|
|
|
Returns a boolean indicating if the firmware configuration interface
|
|
|
|
is available or not.
|
|
|
|
|
|
|
|
This function may change fw_cfg state.
|
|
|
|
|
|
|
|
@retval TRUE The interface is available
|
|
|
|
@retval FALSE The interface is not available
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgIsAvailable (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2024-04-30 09:48:09 +02:00
|
|
|
return (BOOLEAN)(QemuGetFwCfgSelectorAddress () != 0 && QemuGetFwCfgDataAddress () != 0);
|
2015-01-02 13:04:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Selects a firmware configuration item for reading.
|
|
|
|
|
|
|
|
Following this call, any data read from this item will start from the
|
|
|
|
beginning of the configuration item's data.
|
|
|
|
|
|
|
|
@param[in] QemuFwCfgItem Firmware Configuration item to read
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgSelectItem (
|
|
|
|
IN FIRMWARE_CONFIG_ITEM QemuFwCfgItem
|
|
|
|
)
|
|
|
|
{
|
2016-11-30 10:07:06 +01:00
|
|
|
if (QemuFwCfgIsAvailable ()) {
|
2024-04-30 09:48:09 +02:00
|
|
|
MmioWrite16 (QemuGetFwCfgSelectorAddress (), SwapBytes16 ((UINT16)QemuFwCfgItem));
|
2015-01-02 13:04:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-09-24 23:40:41 +02:00
|
|
|
Slow READ_BYTES_FUNCTION.
|
2015-01-02 13:04:15 +01:00
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
2015-09-24 23:40:41 +02:00
|
|
|
MmioReadBytes (
|
2015-01-02 13:04:15 +01:00
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer OPTIONAL
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Left;
|
|
|
|
UINT8 *Ptr;
|
|
|
|
UINT8 *End;
|
|
|
|
|
2024-04-30 09:48:02 +02:00
|
|
|
#if defined (MDE_CPU_AARCH64) || defined (MDE_CPU_RISCV64) || defined (MDE_CPU_LOONGARCH64)
|
2015-01-02 13:04:15 +01:00
|
|
|
Left = Size & 7;
|
|
|
|
#else
|
|
|
|
Left = Size & 3;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Size -= Left;
|
|
|
|
Ptr = Buffer;
|
|
|
|
End = Ptr + Size;
|
|
|
|
|
2024-04-30 09:48:02 +02:00
|
|
|
#if defined (MDE_CPU_AARCH64) || defined (MDE_CPU_RISCV64) || defined (MDE_CPU_LOONGARCH64)
|
2015-01-02 13:04:15 +01:00
|
|
|
while (Ptr < End) {
|
2024-04-30 09:48:09 +02:00
|
|
|
*(UINT64 *)Ptr = MmioRead64 (QemuGetFwCfgDataAddress ());
|
2015-01-02 13:04:15 +01:00
|
|
|
Ptr += 8;
|
|
|
|
}
|
2021-12-05 23:54:09 +01:00
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
if (Left & 4) {
|
2024-04-30 09:48:09 +02:00
|
|
|
*(UINT32 *)Ptr = MmioRead32 (QemuGetFwCfgDataAddress ());
|
2015-01-02 13:04:15 +01:00
|
|
|
Ptr += 4;
|
|
|
|
}
|
2021-12-05 23:54:09 +01:00
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
#else
|
|
|
|
while (Ptr < End) {
|
2024-04-30 09:48:09 +02:00
|
|
|
*(UINT32 *)Ptr = MmioRead32 (QemuGetFwCfgDataAddress ());
|
2015-01-02 13:04:15 +01:00
|
|
|
Ptr += 4;
|
|
|
|
}
|
2021-12-05 23:54:09 +01:00
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
if (Left & 2) {
|
2024-04-30 09:48:09 +02:00
|
|
|
*(UINT16 *)Ptr = MmioRead16 (QemuGetFwCfgDataAddress ());
|
2015-01-02 13:04:15 +01:00
|
|
|
Ptr += 2;
|
|
|
|
}
|
2021-12-05 23:54:09 +01:00
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
if (Left & 1) {
|
2024-04-30 09:48:09 +02:00
|
|
|
*Ptr = MmioRead8 (QemuGetFwCfgDataAddress ());
|
2015-01-02 13:04:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-24 23:40:41 +02:00
|
|
|
/**
|
2017-01-27 06:56:19 +01:00
|
|
|
Transfer an array of bytes, or skip a number of bytes, using the DMA
|
|
|
|
interface.
|
|
|
|
|
|
|
|
@param[in] Size Size in bytes to transfer or skip.
|
|
|
|
|
|
|
|
@param[in,out] Buffer Buffer to read data into or write data from. Ignored,
|
|
|
|
and may be NULL, if Size is zero, or Control is
|
|
|
|
FW_CFG_DMA_CTL_SKIP.
|
|
|
|
|
|
|
|
@param[in] Control One of the following:
|
|
|
|
FW_CFG_DMA_CTL_WRITE - write to fw_cfg from Buffer.
|
|
|
|
FW_CFG_DMA_CTL_READ - read from fw_cfg into Buffer.
|
|
|
|
FW_CFG_DMA_CTL_SKIP - skip bytes in fw_cfg.
|
2015-09-24 23:40:41 +02:00
|
|
|
**/
|
|
|
|
VOID
|
2017-01-27 06:56:19 +01:00
|
|
|
DmaTransferBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN OUT VOID *Buffer OPTIONAL,
|
|
|
|
IN UINT32 Control
|
2015-09-24 23:40:41 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
volatile FW_CFG_DMA_ACCESS Access;
|
|
|
|
UINT32 Status;
|
|
|
|
|
2017-01-27 06:56:19 +01:00
|
|
|
ASSERT (
|
|
|
|
Control == FW_CFG_DMA_CTL_WRITE || Control == FW_CFG_DMA_CTL_READ ||
|
|
|
|
Control == FW_CFG_DMA_CTL_SKIP
|
|
|
|
);
|
|
|
|
|
2015-09-24 23:40:41 +02:00
|
|
|
if (Size == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT (Size <= MAX_UINT32);
|
|
|
|
|
2017-01-27 06:56:19 +01:00
|
|
|
Access.Control = SwapBytes32 (Control);
|
2015-09-24 23:40:41 +02:00
|
|
|
Access.Length = SwapBytes32 ((UINT32)Size);
|
|
|
|
Access.Address = SwapBytes64 ((UINT64)(UINTN)Buffer);
|
|
|
|
|
|
|
|
//
|
|
|
|
// We shouldn't start the transfer before setting up Access.
|
|
|
|
//
|
|
|
|
MemoryFence ();
|
|
|
|
|
|
|
|
//
|
|
|
|
// This will fire off the transfer.
|
|
|
|
//
|
2024-04-30 09:48:02 +02:00
|
|
|
#if defined (MDE_CPU_AARCH64) || defined (MDE_CPU_RISCV64) || defined (MDE_CPU_LOONGARCH64)
|
2024-04-30 09:48:09 +02:00
|
|
|
MmioWrite64 (QemuGetFwCfgDmaAddress (), SwapBytes64 ((UINT64)&Access));
|
2015-09-24 23:40:41 +02:00
|
|
|
#else
|
2024-04-30 09:48:09 +02:00
|
|
|
MmioWrite32 ((UINT32)(QemuGetFwCfgDmaAddress () + 4), SwapBytes32 ((UINT32)&Access));
|
2015-09-24 23:40:41 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//
|
|
|
|
// We shouldn't look at Access.Control before starting the transfer.
|
|
|
|
//
|
|
|
|
MemoryFence ();
|
|
|
|
|
|
|
|
do {
|
|
|
|
Status = SwapBytes32 (Access.Control);
|
|
|
|
ASSERT ((Status & FW_CFG_DMA_CTL_ERROR) == 0);
|
|
|
|
} while (Status != 0);
|
|
|
|
|
|
|
|
//
|
|
|
|
// The caller will want to access the transferred data.
|
|
|
|
//
|
|
|
|
MemoryFence ();
|
|
|
|
}
|
|
|
|
|
2017-01-27 06:56:19 +01:00
|
|
|
/**
|
|
|
|
Fast READ_BYTES_FUNCTION.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
DmaReadBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer OPTIONAL
|
|
|
|
)
|
|
|
|
{
|
|
|
|
DmaTransferBytes (Size, Buffer, FW_CFG_DMA_CTL_READ);
|
|
|
|
}
|
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
/**
|
|
|
|
Reads firmware configuration bytes into a buffer
|
|
|
|
|
|
|
|
If called multiple times, then the data read will continue at the offset of
|
|
|
|
the firmware configuration item where the previous read ended.
|
|
|
|
|
|
|
|
@param[in] Size Size in bytes to read
|
|
|
|
@param[in] Buffer Buffer to store data into
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgReadBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer
|
|
|
|
)
|
|
|
|
{
|
2016-11-30 10:07:06 +01:00
|
|
|
if (QemuFwCfgIsAvailable ()) {
|
2015-01-02 13:04:15 +01:00
|
|
|
InternalQemuFwCfgReadBytes (Size, Buffer);
|
|
|
|
} else {
|
|
|
|
ZeroMem (Buffer, Size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-27 07:29:12 +01:00
|
|
|
/**
|
|
|
|
Slow WRITE_BYTES_FUNCTION.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
MmioWriteBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer OPTIONAL
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Idx;
|
|
|
|
|
|
|
|
for (Idx = 0; Idx < Size; ++Idx) {
|
2024-04-30 09:48:09 +02:00
|
|
|
MmioWrite8 (QemuGetFwCfgDataAddress (), ((UINT8 *)Buffer)[Idx]);
|
2017-01-27 07:29:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Fast WRITE_BYTES_FUNCTION.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
DmaWriteBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer OPTIONAL
|
|
|
|
)
|
|
|
|
{
|
|
|
|
DmaTransferBytes (Size, Buffer, FW_CFG_DMA_CTL_WRITE);
|
|
|
|
}
|
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
/**
|
|
|
|
Write firmware configuration bytes from a buffer
|
|
|
|
|
|
|
|
If called multiple times, then the data written will continue at the offset
|
|
|
|
of the firmware configuration item where the previous write ended.
|
|
|
|
|
|
|
|
@param[in] Size Size in bytes to write
|
|
|
|
@param[in] Buffer Buffer to read data from
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgWriteBytes (
|
|
|
|
IN UINTN Size,
|
|
|
|
IN VOID *Buffer
|
|
|
|
)
|
|
|
|
{
|
2016-11-30 10:07:06 +01:00
|
|
|
if (QemuFwCfgIsAvailable ()) {
|
2017-01-27 07:29:12 +01:00
|
|
|
InternalQemuFwCfgWriteBytes (Size, Buffer);
|
2015-01-02 13:04:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-27 08:10:15 +01:00
|
|
|
/**
|
|
|
|
Slow SKIP_BYTES_FUNCTION.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
MmioSkipBytes (
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN ChunkSize;
|
|
|
|
UINT8 SkipBuffer[256];
|
|
|
|
|
|
|
|
//
|
|
|
|
// Emulate the skip by reading data in chunks, and throwing it away. The
|
|
|
|
// implementation below doesn't affect the static data footprint for client
|
|
|
|
// modules. Large skips are not expected, therefore this fallback is not
|
|
|
|
// performance critical. The size of SkipBuffer is thought not to exert a
|
|
|
|
// large pressure on the stack.
|
|
|
|
//
|
|
|
|
while (Size > 0) {
|
|
|
|
ChunkSize = MIN (Size, sizeof SkipBuffer);
|
|
|
|
MmioReadBytes (ChunkSize, SkipBuffer);
|
|
|
|
Size -= ChunkSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Fast SKIP_BYTES_FUNCTION.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
DmaSkipBytes (
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
DmaTransferBytes (Size, NULL, FW_CFG_DMA_CTL_SKIP);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Skip bytes in the firmware configuration item.
|
|
|
|
|
|
|
|
Increase the offset of the firmware configuration item without transferring
|
|
|
|
bytes between the item and a caller-provided buffer. Subsequent read, write
|
|
|
|
or skip operations will commence at the increased offset.
|
|
|
|
|
|
|
|
@param[in] Size Number of bytes to skip.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgSkipBytes (
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (QemuFwCfgIsAvailable ()) {
|
|
|
|
InternalQemuFwCfgSkipBytes (Size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-02 13:04:15 +01:00
|
|
|
/**
|
|
|
|
Reads a UINT8 firmware configuration value
|
|
|
|
|
|
|
|
@return Value of Firmware Configuration item read
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT8
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgRead8 (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Result;
|
|
|
|
|
|
|
|
QemuFwCfgReadBytes (sizeof Result, &Result);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads a UINT16 firmware configuration value
|
|
|
|
|
|
|
|
@return Value of Firmware Configuration item read
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT16
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgRead16 (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT16 Result;
|
|
|
|
|
|
|
|
QemuFwCfgReadBytes (sizeof Result, &Result);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads a UINT32 firmware configuration value
|
|
|
|
|
|
|
|
@return Value of Firmware Configuration item read
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT32
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgRead32 (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 Result;
|
|
|
|
|
|
|
|
QemuFwCfgReadBytes (sizeof Result, &Result);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads a UINT64 firmware configuration value
|
|
|
|
|
|
|
|
@return Value of Firmware Configuration item read
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgRead64 (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT64 Result;
|
|
|
|
|
|
|
|
QemuFwCfgReadBytes (sizeof Result, &Result);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Find the configuration item corresponding to the firmware configuration file.
|
|
|
|
|
|
|
|
@param[in] Name Name of file to look up.
|
|
|
|
@param[out] Item Configuration item corresponding to the file, to be passed
|
|
|
|
to QemuFwCfgSelectItem ().
|
|
|
|
@param[out] Size Number of bytes in the file.
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS If file is found.
|
|
|
|
@retval RETURN_NOT_FOUND If file is not found.
|
|
|
|
@retval RETURN_UNSUPPORTED If firmware configuration is unavailable.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
QemuFwCfgFindFile (
|
|
|
|
IN CONST CHAR8 *Name,
|
|
|
|
OUT FIRMWARE_CONFIG_ITEM *Item,
|
|
|
|
OUT UINTN *Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 Count;
|
|
|
|
UINT32 Idx;
|
|
|
|
|
2016-11-30 10:07:06 +01:00
|
|
|
if (!QemuFwCfgIsAvailable ()) {
|
2015-01-02 13:04:15 +01:00
|
|
|
return RETURN_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
QemuFwCfgSelectItem (QemuFwCfgItemFileDir);
|
|
|
|
Count = SwapBytes32 (QemuFwCfgRead32 ());
|
|
|
|
|
|
|
|
for (Idx = 0; Idx < Count; ++Idx) {
|
|
|
|
UINT32 FileSize;
|
|
|
|
UINT16 FileSelect;
|
|
|
|
CHAR8 FName[QEMU_FW_CFG_FNAME_SIZE];
|
|
|
|
|
|
|
|
FileSize = QemuFwCfgRead32 ();
|
|
|
|
FileSelect = QemuFwCfgRead16 ();
|
|
|
|
QemuFwCfgRead16 (); // skip the field called "reserved"
|
|
|
|
InternalQemuFwCfgReadBytes (sizeof (FName), FName);
|
|
|
|
|
|
|
|
if (AsciiStrCmp (Name, FName) == 0) {
|
2015-02-25 19:47:12 +01:00
|
|
|
*Item = (FIRMWARE_CONFIG_ITEM)SwapBytes16 (FileSelect);
|
2015-01-02 13:04:15 +01:00
|
|
|
*Size = SwapBytes32 (FileSize);
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return RETURN_NOT_FOUND;
|
|
|
|
}
|