mirror of https://github.com/acidanthera/audk.git
BaseTools/FMMT: Add a tool FMMT
FMMT is a tool to enable removal, addition and replacement of FFS files in FD image binaries. https://bugzilla.tianocore.org/show_bug.cgi?id=1847 Cc: Bob Feng <bob.c.feng@intel.com> Cc: Liming Gao <liming.gao@intel.com> Signed-off-by: Shenglei Zhang <shenglei.zhang@intel.com> Reviewed-by: Bob Feng <bob.c.feng@intel.com>
This commit is contained in:
parent
3c59d94637
commit
080981d72d
|
@ -0,0 +1,29 @@
|
||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
full_cmd=${BASH_SOURCE:-$0} # see http://mywiki.wooledge.org/BashFAQ/028 for a discussion of why $0 is not a good choice here
|
||||||
|
dir=$(dirname "$full_cmd")
|
||||||
|
cmd=${full_cmd##*/}
|
||||||
|
|
||||||
|
if [ -n "$WORKSPACE" ] && [ -e "$WORKSPACE/Conf/BaseToolsCBinaries" ]
|
||||||
|
then
|
||||||
|
exec "$WORKSPACE/Conf/BaseToolsCBinaries/$cmd"
|
||||||
|
elif [ -n "$WORKSPACE" ] && [ -e "$EDK_TOOLS_PATH/Source/C" ]
|
||||||
|
then
|
||||||
|
if [ ! -e "$EDK_TOOLS_PATH/Source/C/bin/$cmd" ]
|
||||||
|
then
|
||||||
|
echo "BaseTools C Tool binary was not found ($cmd)"
|
||||||
|
echo "You may need to run:"
|
||||||
|
echo " make -C $EDK_TOOLS_PATH/Source/C"
|
||||||
|
else
|
||||||
|
exec "$EDK_TOOLS_PATH/Source/C/bin/$cmd" "$@"
|
||||||
|
fi
|
||||||
|
elif [ -e "$dir/../../Source/C/bin/$cmd" ]
|
||||||
|
then
|
||||||
|
exec "$dir/../../Source/C/bin/$cmd" "$@"
|
||||||
|
else
|
||||||
|
echo "Unable to find the real '$cmd' to run"
|
||||||
|
echo "This message was printed by"
|
||||||
|
echo " $0"
|
||||||
|
exit 127
|
||||||
|
fi
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,479 @@
|
||||||
|
/** @file
|
||||||
|
|
||||||
|
Structures and functions declaration.
|
||||||
|
|
||||||
|
Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
||||||
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
#ifndef _BIN_FILE_MANAGER_
|
||||||
|
#define _BIN_FILE_MANAGER_
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#include <unistd.h>
|
||||||
|
#else
|
||||||
|
#include <io.h>
|
||||||
|
#include <direct.h>
|
||||||
|
#endif
|
||||||
|
#include <FvLib.h>
|
||||||
|
#include <Common/UefiBaseTypes.h>
|
||||||
|
#include <Common/PiFirmwareVolume.h>
|
||||||
|
#include <Common/PiFirmwareFile.h>
|
||||||
|
#include <Protocol/GuidedSectionExtraction.h>
|
||||||
|
|
||||||
|
#include "CommonLib.h"
|
||||||
|
#include "EfiUtilityMsgs.h"
|
||||||
|
#include "ParseInf.h"
|
||||||
|
#include "ParseGuidedSectionTools.h"
|
||||||
|
#include "StringFuncs.h"
|
||||||
|
#include "Compress.h"
|
||||||
|
#include "Decompress.h"
|
||||||
|
|
||||||
|
#ifndef _MAX_PATH
|
||||||
|
#define _MAX_PATH 500
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(_MSC_VER) && _MSC_VER < 1900
|
||||||
|
#define snprintf _snprintf
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#define OS_SEP '/'
|
||||||
|
#define OS_SEP_STR "/"
|
||||||
|
#define COPY_STR "cp \"%s\" \"%s\" > /dev/null"
|
||||||
|
#define RMDIR_STR "rm -r \"%s\" > /dev/null"
|
||||||
|
#define DEL_STR "rm \"%s\" > /dev/null"
|
||||||
|
#else
|
||||||
|
#define OS_SEP '\\'
|
||||||
|
#define OS_SEP_STR "\\"
|
||||||
|
#define COPY_STR "copy \"%s\" \"%s\" > NUL"
|
||||||
|
#define RMDIR_STR "rmdir /S /Q \"%s\" > NUL"
|
||||||
|
#define DEL_STR "del \"%s\" > NUL"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define UTILITY_NAME "Firmware Module Management Tool(FMMT)"
|
||||||
|
#define UTILITY_SHORT_NAME "FMMT"
|
||||||
|
#define UTILITY_MAJOR_VERSION 0
|
||||||
|
#define UTILITY_MINOR_VERSION 23
|
||||||
|
#define MAX_BASENAME_LEN 60 // not good to HardCode, but let's be reasonable
|
||||||
|
#define EFI_SECTION_ERROR EFIERR (100)
|
||||||
|
//
|
||||||
|
// The maximum number of Pad file guid entries.
|
||||||
|
//
|
||||||
|
#define MAX_NUMBER_OF_PAD_FILE_GUIDS 1024
|
||||||
|
|
||||||
|
//
|
||||||
|
// The maximum number of block map entries supported by the library
|
||||||
|
//
|
||||||
|
#define MAX_NUMBER_OF_FV_BLOCKS 100
|
||||||
|
|
||||||
|
|
||||||
|
//
|
||||||
|
// The maximum number of sections in an FFS file.
|
||||||
|
//
|
||||||
|
#define MAX_NUMBER_OF_SECTION_IN_FFS 100
|
||||||
|
|
||||||
|
//
|
||||||
|
// The maximum number of files in the FV supported by the library
|
||||||
|
//
|
||||||
|
#define MAX_NUMBER_OF_FILES_IN_FV 1000
|
||||||
|
#define MAX_NUMBER_OF_FILES_IN_CAP 1000
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
///
|
||||||
|
/// If present, this must be the first and only opcode,
|
||||||
|
/// EFI_DEP_BEFORE is only used by DXE driver.
|
||||||
|
///
|
||||||
|
#define EFI_DEP_BEFORE 0x00
|
||||||
|
|
||||||
|
///
|
||||||
|
/// If present, this must be the first and only opcode,
|
||||||
|
/// EFI_DEP_AFTER is only used by DXE driver.
|
||||||
|
///
|
||||||
|
#define EFI_DEP_AFTER 0x01
|
||||||
|
|
||||||
|
#define EFI_DEP_PUSH 0x02
|
||||||
|
#define EFI_DEP_AND 0x03
|
||||||
|
#define EFI_DEP_OR 0x04
|
||||||
|
#define EFI_DEP_NOT 0x05
|
||||||
|
#define EFI_DEP_TRUE 0x06
|
||||||
|
#define EFI_DEP_FALSE 0x07
|
||||||
|
#define EFI_DEP_END 0x08
|
||||||
|
|
||||||
|
|
||||||
|
///
|
||||||
|
/// If present, this must be the first opcode,
|
||||||
|
/// EFI_DEP_SOR is only used by DXE driver.
|
||||||
|
///
|
||||||
|
#define EFI_DEP_SOR 0x09
|
||||||
|
|
||||||
|
//
|
||||||
|
// INF file strings
|
||||||
|
//
|
||||||
|
#define OPTIONS_SECTION_STRING "[options]"
|
||||||
|
#define ATTRIBUTES_SECTION_STRING "[attributes]"
|
||||||
|
#define FILES_SECTION_STRING "[files]"
|
||||||
|
#define FV_BASE_ADDRESS_STRING "[FV_BASE_ADDRESS]"
|
||||||
|
|
||||||
|
//
|
||||||
|
// Options section
|
||||||
|
//
|
||||||
|
#define EFI_FV_BASE_ADDRESS_STRING "EFI_BASE_ADDRESS"
|
||||||
|
#define EFI_FV_FILE_NAME_STRING "EFI_FILE_NAME"
|
||||||
|
#define EFI_NUM_BLOCKS_STRING "EFI_NUM_BLOCKS"
|
||||||
|
#define EFI_BLOCK_SIZE_STRING "EFI_BLOCK_SIZE"
|
||||||
|
#define EFI_GUID_STRING "EFI_GUID"
|
||||||
|
#define EFI_FV_FILESYSTEMGUID_STRING "EFI_FV_GUID"
|
||||||
|
#define EFI_FV_NAMEGUID_STRING "EFI_FVNAME_GUID"
|
||||||
|
#define EFI_CAPSULE_GUID_STRING "EFI_CAPSULE_GUID"
|
||||||
|
#define EFI_CAPSULE_HEADER_SIZE_STRING "EFI_CAPSULE_HEADER_SIZE"
|
||||||
|
#define EFI_CAPSULE_FLAGS_STRING "EFI_CAPSULE_FLAGS"
|
||||||
|
#define EFI_CAPSULE_VERSION_STRING "EFI_CAPSULE_VERSION"
|
||||||
|
|
||||||
|
#define EFI_FV_TOTAL_SIZE_STRING "EFI_FV_TOTAL_SIZE"
|
||||||
|
#define EFI_FV_TAKEN_SIZE_STRING "EFI_FV_TAKEN_SIZE"
|
||||||
|
#define EFI_FV_SPACE_SIZE_STRING "EFI_FV_SPACE_SIZE"
|
||||||
|
|
||||||
|
|
||||||
|
typedef UINT32 FMMT_ENCAP_TYPE;
|
||||||
|
|
||||||
|
#define MAX_LEVEL_IN_FV_FILE 32
|
||||||
|
|
||||||
|
//
|
||||||
|
// Types of FMMT_ENCAP_TREENODE_TYPE
|
||||||
|
//
|
||||||
|
#define FMMT_ENCAP_TREE_FV 0x1
|
||||||
|
#define FMMT_ENCAP_TREE_FFS 0x2
|
||||||
|
#define FMMT_ENCAP_TREE_GUIDED_SECTION 0x3
|
||||||
|
#define FMMT_ENCAP_TREE_COMPRESS_SECTION 0x4
|
||||||
|
#define FMMT_ENCAP_TREE_FV_SECTION 0x5
|
||||||
|
|
||||||
|
extern EFI_HANDLE mParsedGuidedSectionTools;
|
||||||
|
|
||||||
|
|
||||||
|
#define TEMP_DIR_NAME "FmmtTemp"
|
||||||
|
|
||||||
|
//
|
||||||
|
// Structure to keep a list of GUID-To-BaseNames
|
||||||
|
//
|
||||||
|
typedef struct _GUID_TO_BASENAME {
|
||||||
|
struct _GUID_TO_BASENAME *Next;
|
||||||
|
INT8 Guid[PRINTED_GUID_BUFFER_SIZE];
|
||||||
|
INT8 BaseName[MAX_BASENAME_LEN];
|
||||||
|
} GUID_TO_BASENAME;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct _GUID_SEC_TOOL_ENTRY {
|
||||||
|
EFI_GUID Guid;
|
||||||
|
CHAR8* Name;
|
||||||
|
CHAR8* Path;
|
||||||
|
struct _GUID_SEC_TOOL_ENTRY *Next;
|
||||||
|
} GUID_SEC_TOOL_ENTRY;
|
||||||
|
|
||||||
|
|
||||||
|
//
|
||||||
|
// Private data types
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Component information
|
||||||
|
//
|
||||||
|
typedef struct {
|
||||||
|
UINTN Size;
|
||||||
|
CHAR8 ComponentName[_MAX_PATH];
|
||||||
|
} COMPONENT_INFO;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
CHAR8 FfsName[_MAX_PATH];
|
||||||
|
|
||||||
|
//
|
||||||
|
// UI Name for this FFS file, if has.
|
||||||
|
//
|
||||||
|
CHAR16 UiName[_MAX_PATH];
|
||||||
|
UINT32 UiNameSize;
|
||||||
|
//
|
||||||
|
// Total section number in this FFS.
|
||||||
|
//
|
||||||
|
UINT32 TotalSectionNum;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Describe the position of the FFS file.
|
||||||
|
//
|
||||||
|
UINT8 Level;
|
||||||
|
//
|
||||||
|
// If this FFS has no encapsulate section, this flag will set to True.
|
||||||
|
//
|
||||||
|
BOOLEAN IsLeaf;
|
||||||
|
//
|
||||||
|
// Section type for each section in FFS.
|
||||||
|
//
|
||||||
|
EFI_SECTION_TYPE SectionType[MAX_NUMBER_OF_SECTION_IN_FFS];
|
||||||
|
//
|
||||||
|
// Offset relative to current FV
|
||||||
|
//
|
||||||
|
UINT32 Offset;
|
||||||
|
UINT8 FvLevel;
|
||||||
|
EFI_GUID GuidName;
|
||||||
|
UINT8 *Depex;
|
||||||
|
UINT32 DepexLen;
|
||||||
|
BOOLEAN IsHandle;
|
||||||
|
BOOLEAN IsFvStart;
|
||||||
|
BOOLEAN IsFvEnd;
|
||||||
|
}FFS_ATTRIBUTES;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct __ENCAP_INFO_DATA{
|
||||||
|
//
|
||||||
|
// Now Level
|
||||||
|
//
|
||||||
|
UINT8 Level;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Encapsulate type.
|
||||||
|
//
|
||||||
|
FMMT_ENCAP_TYPE Type;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Data, if it's FV, should be FV header.
|
||||||
|
//
|
||||||
|
VOID *Data;
|
||||||
|
|
||||||
|
//
|
||||||
|
//FvId, match FvId with FvGuidName.
|
||||||
|
//
|
||||||
|
UINT8 FvId;
|
||||||
|
|
||||||
|
//
|
||||||
|
// if FV ExtHeaderOffset not to zero, should also have FvExtHeader information
|
||||||
|
//
|
||||||
|
EFI_FIRMWARE_VOLUME_EXT_HEADER *FvExtHeader;
|
||||||
|
|
||||||
|
CHAR16 UiName[_MAX_PATH];
|
||||||
|
UINT32 UiNameSize;
|
||||||
|
UINT8 *Depex;
|
||||||
|
UINT32 DepexLen;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Next node.
|
||||||
|
//
|
||||||
|
struct __ENCAP_INFO_DATA *NextNode;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Right node.
|
||||||
|
//
|
||||||
|
struct __ENCAP_INFO_DATA *RightNode;
|
||||||
|
} ENCAP_INFO_DATA;
|
||||||
|
|
||||||
|
typedef struct _FFS_INFOMATION{
|
||||||
|
CHAR8 *FFSName;
|
||||||
|
UINT32 InFvId;
|
||||||
|
UINT8 ParentLevel;
|
||||||
|
BOOLEAN IsFFS;
|
||||||
|
CHAR16 UiName[_MAX_PATH];
|
||||||
|
UINT32 UiNameSize;
|
||||||
|
UINT8 *Depex;
|
||||||
|
UINT32 DepexLen;
|
||||||
|
BOOLEAN FfsFoundFlag;
|
||||||
|
struct _FFS_INFOMATION *Next;
|
||||||
|
} FFS_INFORMATION;
|
||||||
|
|
||||||
|
//
|
||||||
|
// FV and capsule information holder
|
||||||
|
//
|
||||||
|
typedef struct _FV_INFOMATION{
|
||||||
|
EFI_FIRMWARE_VOLUME_HEADER *FvHeader;
|
||||||
|
EFI_FIRMWARE_VOLUME_EXT_HEADER *FvExtHeader;
|
||||||
|
UINT32 ImageAddress;
|
||||||
|
UINT32 FfsNumbers;
|
||||||
|
CHAR8 FvName[_MAX_PATH];
|
||||||
|
EFI_FV_BLOCK_MAP_ENTRY FvBlocks[MAX_NUMBER_OF_FV_BLOCKS];
|
||||||
|
FFS_ATTRIBUTES FfsAttuibutes[MAX_NUMBER_OF_FILES_IN_FV];
|
||||||
|
EFI_FFS_FILE_HEADER2 FfsHeader[MAX_NUMBER_OF_FILES_IN_FV];
|
||||||
|
struct _FV_INFOMATION *FvNext;
|
||||||
|
ENCAP_INFO_DATA *EncapData;
|
||||||
|
UINT8 FvLevel;
|
||||||
|
CHAR8 *FvUiName;
|
||||||
|
UINT8 MulFvLevel;
|
||||||
|
CHAR8 AlignmentStr[16];
|
||||||
|
FFS_INFORMATION *ChildFvFFS;
|
||||||
|
} FV_INFORMATION;
|
||||||
|
|
||||||
|
typedef struct _FIRMWARE_DEVICE {
|
||||||
|
///
|
||||||
|
/// Size of FD file
|
||||||
|
///
|
||||||
|
UINT32 Size;
|
||||||
|
FV_INFORMATION *Fv;
|
||||||
|
} FIRMWARE_DEVICE;
|
||||||
|
|
||||||
|
typedef struct _FILENode {
|
||||||
|
CHAR8 *FileName;
|
||||||
|
UINT8 SubLevel;
|
||||||
|
struct _FILENode *Next;
|
||||||
|
} FILENode;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
CHAR8 *FvId;
|
||||||
|
FILENode *NewFile;
|
||||||
|
FILENode *OldFile;
|
||||||
|
FIRMWARE_DEVICE *FdData;
|
||||||
|
UINT8 FvLevel;
|
||||||
|
FV_INFORMATION *FvInFd;
|
||||||
|
} Data;
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
LibFindFvInFd (
|
||||||
|
IN FILE *InputFile,
|
||||||
|
IN OUT FIRMWARE_DEVICE **FdData
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
TODO: Add function description
|
||||||
|
|
||||||
|
@param[in] Fv - Firmware Volume to get information from
|
||||||
|
|
||||||
|
@return EFI_STATUS
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
LibGetFvInfo (
|
||||||
|
IN VOID *Fv,
|
||||||
|
IN OUT FV_INFORMATION *CurrentFv,
|
||||||
|
IN CHAR8 *FvName,
|
||||||
|
IN UINT8 Level,
|
||||||
|
IN ENCAP_INFO_DATA **CurrentFvEncapData,
|
||||||
|
IN UINT32 *FfsCount,
|
||||||
|
IN OUT UINT8 *FvCount,
|
||||||
|
IN BOOLEAN ViewFlag,
|
||||||
|
IN BOOLEAN IsChildFv
|
||||||
|
);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Get size info from FV file.
|
||||||
|
|
||||||
|
@param[in]
|
||||||
|
@param[out]
|
||||||
|
|
||||||
|
@retval
|
||||||
|
|
||||||
|
*/
|
||||||
|
EFI_STATUS
|
||||||
|
LibGetFvSize (
|
||||||
|
IN FILE *InputFile,
|
||||||
|
OUT UINT32 *FvSize
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
This function returns the next larger size that meets the alignment
|
||||||
|
requirement specified.
|
||||||
|
|
||||||
|
@param[in] ActualSize The size.
|
||||||
|
@param[in] Alignment The desired alignment.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Function completed successfully.
|
||||||
|
@retval EFI_ABORTED The function encountered an error.
|
||||||
|
|
||||||
|
**/
|
||||||
|
UINT32
|
||||||
|
GetOccupiedSize (
|
||||||
|
IN UINT32 ActualSize,
|
||||||
|
IN UINT32 Alignment
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Converts ASCII characters to Unicode.
|
||||||
|
Assumes that the Unicode characters are only these defined in the ASCII set.
|
||||||
|
|
||||||
|
String - Pointer to string that is written to FILE.
|
||||||
|
UniString - Pointer to unicode string
|
||||||
|
|
||||||
|
The address to the ASCII string - same as AsciiStr.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
LibAscii2Unicode (
|
||||||
|
IN CHAR8 *String,
|
||||||
|
OUT CHAR16 *UniString
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Delete a directory and files in it.
|
||||||
|
|
||||||
|
@param[in] DirName Name of the directory need to be deleted.
|
||||||
|
|
||||||
|
@return EFI_INVALID_PARAMETER
|
||||||
|
@return EFI_SUCCESS
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
LibRmDir (
|
||||||
|
IN CHAR8* DirName
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Delete a file.
|
||||||
|
|
||||||
|
@param[in] FileName Name of the file need to be deleted.
|
||||||
|
|
||||||
|
@return EFI_INVALID_PARAMETER
|
||||||
|
@return EFI_SUCCESS
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
LibFmmtDeleteFile(
|
||||||
|
IN CHAR8 *FileName
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
Free the whole Fd data structure.
|
||||||
|
|
||||||
|
@param[in] Fd The pointer point to the Fd data structure.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
LibFmmtFreeFd (
|
||||||
|
FIRMWARE_DEVICE *Fd
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
LibEncapNewFvFile(
|
||||||
|
IN FV_INFORMATION *FvInFd,
|
||||||
|
IN CHAR8 *TemDir,
|
||||||
|
IN ENCAP_INFO_DATA *CurrentEncapData,
|
||||||
|
IN UINT32 Level_Break,
|
||||||
|
OUT FFS_INFORMATION **OutputFile
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
LibLocateFvViaFvId (
|
||||||
|
IN FIRMWARE_DEVICE *FdData,
|
||||||
|
IN CHAR8 *FvId,
|
||||||
|
IN OUT FV_INFORMATION **FvInFd
|
||||||
|
);
|
||||||
|
|
||||||
|
EFI_HANDLE
|
||||||
|
LibPreDefinedGuidedTools (
|
||||||
|
VOID
|
||||||
|
);
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
FvBufGetSize(
|
||||||
|
IN VOID *Fv,
|
||||||
|
OUT UINTN *Size
|
||||||
|
);
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
FvBufFindNextFile(
|
||||||
|
IN VOID *Fv,
|
||||||
|
IN OUT UINTN *Key,
|
||||||
|
OUT VOID **File
|
||||||
|
);
|
||||||
|
#endif
|
|
@ -0,0 +1,6 @@
|
||||||
|
a31280ad-481e-41b6-95e8-127f4c984779 TIANO TianoCompress
|
||||||
|
ee4e5898-3914-4259-9d6e-dc7bd79403cf LZMA LzmaCompress
|
||||||
|
fc1bcdb0-7d31-49aa-936a-a4600d9dd083 CRC32 GenCrc32
|
||||||
|
d42ae6bd-1352-4bfb-909a-ca72a6eae889 LZMAF86 LzmaF86Compress
|
||||||
|
3d532050-5cda-4fd0-879e-0f7f630d5afb BROTLI BrotliCompress
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,16 @@
|
||||||
|
## @file
|
||||||
|
# GNU/Linux makefile for 'FMMT' module build.
|
||||||
|
#
|
||||||
|
# Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
||||||
|
# SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
#
|
||||||
|
MAKEROOT ?= ..
|
||||||
|
|
||||||
|
APPNAME = FMMT
|
||||||
|
|
||||||
|
LIBS = -lCommon
|
||||||
|
|
||||||
|
OBJECTS = FmmtLib.o Rebase.o FirmwareModuleManagement.o
|
||||||
|
|
||||||
|
include $(MAKEROOT)/Makefiles/app.makefile
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
## @file
|
||||||
|
# Windows makefile for 'FMMT' module build.
|
||||||
|
#
|
||||||
|
# Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
||||||
|
# SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
#
|
||||||
|
|
||||||
|
!INCLUDE ..\Makefiles\ms.common
|
||||||
|
|
||||||
|
APPNAME = FMMT
|
||||||
|
|
||||||
|
LIBS = $(LIB_PATH)\Common.lib
|
||||||
|
|
||||||
|
OBJECTS = FirmwareModuleManagement.obj FmmtLib.obj Rebase.obj
|
||||||
|
|
||||||
|
!INCLUDE ..\Makefiles\ms.app
|
||||||
|
|
|
@ -0,0 +1,846 @@
|
||||||
|
/** @file
|
||||||
|
|
||||||
|
Library to rebase PE image.
|
||||||
|
|
||||||
|
Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
||||||
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
#include "Rebase.h"
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#include <unistd.h>
|
||||||
|
#else
|
||||||
|
#include <io.h>
|
||||||
|
#include <direct.h>
|
||||||
|
#endif
|
||||||
|
#include <PeCoffLib.h>
|
||||||
|
#include <CommonLib.h>
|
||||||
|
#include <IndustryStandard/PeImage.h>
|
||||||
|
#include <FvLib.h>
|
||||||
|
#include "EfiUtilityMsgs.h"
|
||||||
|
|
||||||
|
static
|
||||||
|
EFI_STATUS
|
||||||
|
FfsRebaseImageRead(
|
||||||
|
IN VOID *FileHandle,
|
||||||
|
IN UINTN FileOffset,
|
||||||
|
IN OUT UINT32 *ReadSize,
|
||||||
|
OUT VOID *Buffer
|
||||||
|
);
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
RebaseFfs(
|
||||||
|
IN OUT UINT64 BaseAddress,
|
||||||
|
IN CHAR8 *FileName,
|
||||||
|
IN OUT EFI_FFS_FILE_HEADER *FfsFile,
|
||||||
|
IN UINTN XipOffset
|
||||||
|
)
|
||||||
|
/*++
|
||||||
|
|
||||||
|
Routine Description:
|
||||||
|
|
||||||
|
This function determines if a file is XIP and should be rebased. It will
|
||||||
|
rebase any PE32 sections found in the file using the base address.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
|
||||||
|
FvInfo A pointer to FV_INFO struture.
|
||||||
|
FileName Ffs File PathName
|
||||||
|
FfsFile A pointer to Ffs file image.
|
||||||
|
XipOffset The offset address to use for rebasing the XIP file image.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
EFI_SUCCESS The image was properly rebased.
|
||||||
|
EFI_INVALID_PARAMETER An input parameter is invalid.
|
||||||
|
EFI_ABORTED An error occurred while rebasing the input file image.
|
||||||
|
EFI_OUT_OF_RESOURCES Could not allocate a required resource.
|
||||||
|
EFI_NOT_FOUND No compressed sections could be found.
|
||||||
|
|
||||||
|
--*/
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
PE_COFF_LOADER_IMAGE_CONTEXT ImageContext;
|
||||||
|
PE_COFF_LOADER_IMAGE_CONTEXT OrigImageContext;
|
||||||
|
EFI_PHYSICAL_ADDRESS XipBase;
|
||||||
|
EFI_PHYSICAL_ADDRESS NewPe32BaseAddress;
|
||||||
|
UINTN Index;
|
||||||
|
EFI_FILE_SECTION_POINTER CurrentPe32Section;
|
||||||
|
EFI_FFS_FILE_STATE SavedState;
|
||||||
|
EFI_IMAGE_OPTIONAL_HEADER_UNION *ImgHdr;
|
||||||
|
EFI_TE_IMAGE_HEADER *TEImageHeader;
|
||||||
|
UINT8 *MemoryImagePointer;
|
||||||
|
EFI_IMAGE_SECTION_HEADER *SectionHeader;
|
||||||
|
CHAR8 PeFileName[MAX_LONG_FILE_PATH];
|
||||||
|
CHAR8 *Cptr;
|
||||||
|
FILE *PeFile;
|
||||||
|
UINT8 *PeFileBuffer;
|
||||||
|
UINT32 PeFileSize;
|
||||||
|
CHAR8 *PdbPointer;
|
||||||
|
UINT32 FfsHeaderSize;
|
||||||
|
UINT32 CurSecHdrSize;
|
||||||
|
CHAR8 *LongFilePathName;
|
||||||
|
|
||||||
|
Index = 0;
|
||||||
|
MemoryImagePointer = NULL;
|
||||||
|
TEImageHeader = NULL;
|
||||||
|
ImgHdr = NULL;
|
||||||
|
SectionHeader = NULL;
|
||||||
|
Cptr = NULL;
|
||||||
|
PeFile = NULL;
|
||||||
|
PeFileBuffer = NULL;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Don't need to relocate image when BaseAddress is zero and no ForceRebase Flag specified.
|
||||||
|
//
|
||||||
|
if (BaseAddress == 0) {
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
XipBase = BaseAddress + XipOffset;
|
||||||
|
|
||||||
|
//
|
||||||
|
// We only process files potentially containing PE32 sections.
|
||||||
|
//
|
||||||
|
switch (FfsFile->Type) {
|
||||||
|
case EFI_FV_FILETYPE_SECURITY_CORE:
|
||||||
|
case EFI_FV_FILETYPE_PEI_CORE:
|
||||||
|
case EFI_FV_FILETYPE_PEIM:
|
||||||
|
case EFI_FV_FILETYPE_COMBINED_PEIM_DRIVER:
|
||||||
|
case EFI_FV_FILETYPE_DRIVER:
|
||||||
|
case EFI_FV_FILETYPE_DXE_CORE:
|
||||||
|
break;
|
||||||
|
case EFI_FV_FILETYPE_FIRMWARE_VOLUME_IMAGE:
|
||||||
|
//
|
||||||
|
// Rebase the inside FvImage.
|
||||||
|
//
|
||||||
|
GetChildFvFromFfs (BaseAddress, FfsFile, XipOffset);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Search PE/TE section in FV sectin.
|
||||||
|
//
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
FfsHeaderSize = GetFfsHeaderLength(FfsFile);
|
||||||
|
//
|
||||||
|
// Rebase each PE32 section
|
||||||
|
//
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
for (Index = 1;; Index++) {
|
||||||
|
//
|
||||||
|
// Init Value
|
||||||
|
//
|
||||||
|
NewPe32BaseAddress = 0;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Find Pe Image
|
||||||
|
//
|
||||||
|
Status = GetSectionByType(FfsFile, EFI_SECTION_PE32, Index, &CurrentPe32Section);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
CurSecHdrSize = GetSectionHeaderLength(CurrentPe32Section.CommonHeader);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Initialize context
|
||||||
|
//
|
||||||
|
memset(&ImageContext, 0, sizeof (ImageContext));
|
||||||
|
ImageContext.Handle = (VOID *)((UINTN)CurrentPe32Section.Pe32Section + CurSecHdrSize);
|
||||||
|
ImageContext.ImageRead = (PE_COFF_LOADER_READ_FILE)FfsRebaseImageRead;
|
||||||
|
Status = PeCoffLoaderGetImageInfo(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid PeImage", "The input file is %s and the return status is %x", FileName, (int)Status);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
//if ((ImageContext.Machine == EFI_IMAGE_MACHINE_ARMT) ||
|
||||||
|
// (ImageContext.Machine == EFI_IMAGE_MACHINE_AARCH64)) {
|
||||||
|
// mArm = TRUE;
|
||||||
|
//}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Keep Image Context for PE image in FV
|
||||||
|
//
|
||||||
|
memcpy(&OrigImageContext, &ImageContext, sizeof (ImageContext));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get File PdbPointer
|
||||||
|
//
|
||||||
|
PdbPointer = PeCoffLoaderGetPdbPointer(ImageContext.Handle);
|
||||||
|
if (PdbPointer == NULL) {
|
||||||
|
PdbPointer = FileName;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get PeHeader pointer
|
||||||
|
//
|
||||||
|
ImgHdr = (EFI_IMAGE_OPTIONAL_HEADER_UNION *)((UINTN)CurrentPe32Section.Pe32Section + CurSecHdrSize + ImageContext.PeCoffHeaderOffset);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Calculate the PE32 base address, based on file type
|
||||||
|
//
|
||||||
|
switch (FfsFile->Type) {
|
||||||
|
case EFI_FV_FILETYPE_SECURITY_CORE:
|
||||||
|
case EFI_FV_FILETYPE_PEI_CORE:
|
||||||
|
case EFI_FV_FILETYPE_PEIM:
|
||||||
|
case EFI_FV_FILETYPE_COMBINED_PEIM_DRIVER:
|
||||||
|
//
|
||||||
|
// Check if section-alignment and file-alignment match or not
|
||||||
|
//
|
||||||
|
if ((ImgHdr->Pe32.OptionalHeader.SectionAlignment != ImgHdr->Pe32.OptionalHeader.FileAlignment)) {
|
||||||
|
//
|
||||||
|
// Xip module has the same section alignment and file alignment.
|
||||||
|
//
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "Section-Alignment and File-Alignment do not match : %s.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// PeImage has no reloc section. It will try to get reloc data from the original EFI image.
|
||||||
|
//
|
||||||
|
if (ImageContext.RelocationsStripped) {
|
||||||
|
//
|
||||||
|
// Construct the original efi file Name
|
||||||
|
//
|
||||||
|
if (strlen (FileName) > MAX_LONG_FILE_PATH - 1) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "The file name for %s is too long.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
strncpy(PeFileName, FileName, MAX_LONG_FILE_PATH - 1);
|
||||||
|
PeFileName[MAX_LONG_FILE_PATH - 1] = 0;
|
||||||
|
Cptr = PeFileName + strlen(PeFileName);
|
||||||
|
while (*Cptr != '.') {
|
||||||
|
Cptr--;
|
||||||
|
}
|
||||||
|
if (*Cptr != '.') {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
*(Cptr + 1) = 'e';
|
||||||
|
*(Cptr + 2) = 'f';
|
||||||
|
*(Cptr + 3) = 'i';
|
||||||
|
*(Cptr + 4) = '\0';
|
||||||
|
}
|
||||||
|
LongFilePathName = LongFilePath(PeFileName);
|
||||||
|
if (LongFilePathName == NULL) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "Fail to get long file path for file %s.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
PeFile = fopen(LongFilePathName, "rb");
|
||||||
|
if (PeFile == NULL) {
|
||||||
|
Warning(NULL, 0, 0, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
//Error (NULL, 0, 3000, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
//return EFI_ABORTED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Get the file size
|
||||||
|
//
|
||||||
|
PeFileSize = _filelength(fileno(PeFile));
|
||||||
|
PeFileBuffer = (UINT8 *)malloc(PeFileSize);
|
||||||
|
if (PeFileBuffer == NULL) {
|
||||||
|
Error(NULL, 0, 4001, "Resource", "memory cannot be allocated on rebase of %s", FileName);
|
||||||
|
fclose(PeFile);
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Read Pe File
|
||||||
|
//
|
||||||
|
fread(PeFileBuffer, sizeof (UINT8), PeFileSize, PeFile);
|
||||||
|
//
|
||||||
|
// close file
|
||||||
|
//
|
||||||
|
fclose(PeFile);
|
||||||
|
//
|
||||||
|
// Handle pointer to the original efi image.
|
||||||
|
//
|
||||||
|
ImageContext.Handle = PeFileBuffer;
|
||||||
|
Status = PeCoffLoaderGetImageInfo(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid PeImage", "The input file is %s and the return status is %x", FileName, (int)Status);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
ImageContext.RelocationsStripped = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
NewPe32BaseAddress = XipBase + (UINTN)CurrentPe32Section.Pe32Section + CurSecHdrSize - (UINTN)FfsFile;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EFI_FV_FILETYPE_DRIVER:
|
||||||
|
case EFI_FV_FILETYPE_DXE_CORE:
|
||||||
|
//
|
||||||
|
// Check if section-alignment and file-alignment match or not
|
||||||
|
//
|
||||||
|
if ((ImgHdr->Pe32.OptionalHeader.SectionAlignment != ImgHdr->Pe32.OptionalHeader.FileAlignment)) {
|
||||||
|
//
|
||||||
|
// Xip module has the same section alignment and file alignment.
|
||||||
|
//
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "Section-Alignment and File-Alignment do not match : %s.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
NewPe32BaseAddress = XipBase + (UINTN)CurrentPe32Section.Pe32Section + CurSecHdrSize - (UINTN)FfsFile;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
//
|
||||||
|
// Not supported file type
|
||||||
|
//
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Relocation doesn't exist
|
||||||
|
//
|
||||||
|
if (ImageContext.RelocationsStripped) {
|
||||||
|
Warning(NULL, 0, 0, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Relocation exist and rebase
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Load and Relocate Image Data
|
||||||
|
//
|
||||||
|
MemoryImagePointer = (UINT8 *)malloc((UINTN)ImageContext.ImageSize + ImageContext.SectionAlignment);
|
||||||
|
if (MemoryImagePointer == NULL) {
|
||||||
|
Error(NULL, 0, 4001, "Resource", "memory cannot be allocated on rebase of %s", FileName);
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
memset((VOID *)MemoryImagePointer, 0, (UINTN)ImageContext.ImageSize + ImageContext.SectionAlignment);
|
||||||
|
ImageContext.ImageAddress = ((UINTN)MemoryImagePointer + ImageContext.SectionAlignment - 1) & (~((UINTN)ImageContext.SectionAlignment - 1));
|
||||||
|
|
||||||
|
Status = PeCoffLoaderLoadImage(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "LocateImage() call failed on rebase of %s", FileName);
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageContext.DestinationAddress = NewPe32BaseAddress;
|
||||||
|
Status = PeCoffLoaderRelocateImage(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "RelocateImage() call failed on rebase of %s", FileName);
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Copy Relocated data to raw image file.
|
||||||
|
//
|
||||||
|
SectionHeader = (EFI_IMAGE_SECTION_HEADER *)(
|
||||||
|
(UINTN)ImgHdr +
|
||||||
|
sizeof (UINT32)+
|
||||||
|
sizeof (EFI_IMAGE_FILE_HEADER)+
|
||||||
|
ImgHdr->Pe32.FileHeader.SizeOfOptionalHeader
|
||||||
|
);
|
||||||
|
|
||||||
|
for (Index = 0; Index < ImgHdr->Pe32.FileHeader.NumberOfSections; Index++, SectionHeader++) {
|
||||||
|
CopyMem(
|
||||||
|
(UINT8 *)CurrentPe32Section.Pe32Section + CurSecHdrSize + SectionHeader->PointerToRawData,
|
||||||
|
(VOID*)(UINTN)(ImageContext.ImageAddress + SectionHeader->VirtualAddress),
|
||||||
|
SectionHeader->SizeOfRawData
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
MemoryImagePointer = NULL;
|
||||||
|
if (PeFileBuffer != NULL) {
|
||||||
|
free(PeFileBuffer);
|
||||||
|
PeFileBuffer = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Update Image Base Address
|
||||||
|
//
|
||||||
|
if (ImgHdr->Pe32.OptionalHeader.Magic == EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC) {
|
||||||
|
ImgHdr->Pe32.OptionalHeader.ImageBase = (UINT32)NewPe32BaseAddress;
|
||||||
|
}
|
||||||
|
else if (ImgHdr->Pe32Plus.OptionalHeader.Magic == EFI_IMAGE_NT_OPTIONAL_HDR64_MAGIC) {
|
||||||
|
ImgHdr->Pe32Plus.OptionalHeader.ImageBase = NewPe32BaseAddress;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "unknown PE magic signature %X in PE32 image %s",
|
||||||
|
ImgHdr->Pe32.OptionalHeader.Magic,
|
||||||
|
FileName
|
||||||
|
);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Now update file checksum
|
||||||
|
//
|
||||||
|
if (FfsFile->Attributes & FFS_ATTRIB_CHECKSUM) {
|
||||||
|
SavedState = FfsFile->State;
|
||||||
|
FfsFile->IntegrityCheck.Checksum.File = 0;
|
||||||
|
FfsFile->State = 0;
|
||||||
|
FfsFile->IntegrityCheck.Checksum.File = CalculateChecksum8(
|
||||||
|
(UINT8 *)((UINT8 *)FfsFile + FfsHeaderSize),
|
||||||
|
GetFfsFileLength(FfsFile) - FfsHeaderSize
|
||||||
|
);
|
||||||
|
FfsFile->State = SavedState;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (FfsFile->Type != EFI_FV_FILETYPE_SECURITY_CORE &&
|
||||||
|
FfsFile->Type != EFI_FV_FILETYPE_PEI_CORE &&
|
||||||
|
FfsFile->Type != EFI_FV_FILETYPE_PEIM &&
|
||||||
|
FfsFile->Type != EFI_FV_FILETYPE_COMBINED_PEIM_DRIVER &&
|
||||||
|
FfsFile->Type != EFI_FV_FILETYPE_FIRMWARE_VOLUME_IMAGE
|
||||||
|
) {
|
||||||
|
//
|
||||||
|
// Only Peim code may have a TE section
|
||||||
|
//
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Now process TE sections
|
||||||
|
//
|
||||||
|
for (Index = 1;; Index++) {
|
||||||
|
NewPe32BaseAddress = 0;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Find Te Image
|
||||||
|
//
|
||||||
|
Status = GetSectionByType(FfsFile, EFI_SECTION_TE, Index, &CurrentPe32Section);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
CurSecHdrSize = GetSectionHeaderLength(CurrentPe32Section.CommonHeader);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Calculate the TE base address, the FFS file base plus the offset of the TE section less the size stripped off
|
||||||
|
// by GenTEImage
|
||||||
|
//
|
||||||
|
TEImageHeader = (EFI_TE_IMAGE_HEADER *)((UINT8 *)CurrentPe32Section.Pe32Section + CurSecHdrSize);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Initialize context, load image info.
|
||||||
|
//
|
||||||
|
memset(&ImageContext, 0, sizeof (ImageContext));
|
||||||
|
ImageContext.Handle = (VOID *)TEImageHeader;
|
||||||
|
ImageContext.ImageRead = (PE_COFF_LOADER_READ_FILE)FfsRebaseImageRead;
|
||||||
|
Status = PeCoffLoaderGetImageInfo(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid TeImage", "The input file is %s and the return status is %x", FileName, (int)Status);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
//if ((ImageContext.Machine == EFI_IMAGE_MACHINE_ARMT) ||
|
||||||
|
// (ImageContext.Machine == EFI_IMAGE_MACHINE_AARCH64)) {
|
||||||
|
// mArm = TRUE;
|
||||||
|
//}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Keep Image Context for TE image in FV
|
||||||
|
//
|
||||||
|
memcpy(&OrigImageContext, &ImageContext, sizeof (ImageContext));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get File PdbPointer
|
||||||
|
//
|
||||||
|
PdbPointer = PeCoffLoaderGetPdbPointer(ImageContext.Handle);
|
||||||
|
if (PdbPointer == NULL) {
|
||||||
|
PdbPointer = FileName;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Set new rebased address.
|
||||||
|
//
|
||||||
|
NewPe32BaseAddress = XipBase + (UINTN)TEImageHeader + sizeof (EFI_TE_IMAGE_HEADER) \
|
||||||
|
- TEImageHeader->StrippedSize - (UINTN)FfsFile;
|
||||||
|
|
||||||
|
//
|
||||||
|
// if reloc is stripped, try to get the original efi image to get reloc info.
|
||||||
|
//
|
||||||
|
if (ImageContext.RelocationsStripped) {
|
||||||
|
//
|
||||||
|
// Construct the original efi file name
|
||||||
|
//
|
||||||
|
if (strlen (FileName) > MAX_LONG_FILE_PATH - 1) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "The file name for %s is too long.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
strncpy(PeFileName, FileName, MAX_LONG_FILE_PATH - 1);
|
||||||
|
PeFileName[MAX_LONG_FILE_PATH - 1] = 0;
|
||||||
|
Cptr = PeFileName + strlen(PeFileName);
|
||||||
|
while (*Cptr != '.') {
|
||||||
|
Cptr--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*Cptr != '.') {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
*(Cptr + 1) = 'e';
|
||||||
|
*(Cptr + 2) = 'f';
|
||||||
|
*(Cptr + 3) = 'i';
|
||||||
|
*(Cptr + 4) = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
LongFilePathName = LongFilePath(PeFileName);
|
||||||
|
if (LongFilePathName == NULL) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "Fail to get long file path for file %s.", FileName);
|
||||||
|
return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
PeFile = fopen(LongFilePathName, "rb");
|
||||||
|
if (PeFile == NULL) {
|
||||||
|
Warning(NULL, 0, 0, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
//Error (NULL, 0, 3000, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
//return EFI_ABORTED;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//
|
||||||
|
// Get the file size
|
||||||
|
//
|
||||||
|
PeFileSize = _filelength(fileno(PeFile));
|
||||||
|
PeFileBuffer = (UINT8 *)malloc(PeFileSize);
|
||||||
|
if (PeFileBuffer == NULL) {
|
||||||
|
Error(NULL, 0, 4001, "Resource", "memory cannot be allocated on rebase of %s", FileName);
|
||||||
|
fclose(PeFile);
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Read Pe File
|
||||||
|
//
|
||||||
|
fread(PeFileBuffer, sizeof (UINT8), PeFileSize, PeFile);
|
||||||
|
//
|
||||||
|
// close file
|
||||||
|
//
|
||||||
|
fclose(PeFile);
|
||||||
|
//
|
||||||
|
// Append reloc section into TeImage
|
||||||
|
//
|
||||||
|
ImageContext.Handle = PeFileBuffer;
|
||||||
|
Status = PeCoffLoaderGetImageInfo(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid TeImage", "The input file is %s and the return status is %x", FileName, (int)Status);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
ImageContext.RelocationsStripped = FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Relocation doesn't exist
|
||||||
|
//
|
||||||
|
if (ImageContext.RelocationsStripped) {
|
||||||
|
Warning(NULL, 0, 0, "Invalid", "The file %s has no .reloc section.", FileName);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Relocation exist and rebase
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Load and Relocate Image Data
|
||||||
|
//
|
||||||
|
MemoryImagePointer = (UINT8 *)malloc((UINTN)ImageContext.ImageSize + ImageContext.SectionAlignment);
|
||||||
|
if (MemoryImagePointer == NULL) {
|
||||||
|
Error(NULL, 0, 4001, "Resource", "memory cannot be allocated on rebase of %s", FileName);
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
memset((VOID *)MemoryImagePointer, 0, (UINTN)ImageContext.ImageSize + ImageContext.SectionAlignment);
|
||||||
|
ImageContext.ImageAddress = ((UINTN)MemoryImagePointer + ImageContext.SectionAlignment - 1) & (~((UINTN)ImageContext.SectionAlignment - 1));
|
||||||
|
|
||||||
|
Status = PeCoffLoaderLoadImage(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "LocateImage() call failed on rebase of %s", FileName);
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Reloacate TeImage
|
||||||
|
//
|
||||||
|
ImageContext.DestinationAddress = NewPe32BaseAddress;
|
||||||
|
Status = PeCoffLoaderRelocateImage(&ImageContext);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
Error(NULL, 0, 3000, "Invalid", "RelocateImage() call failed on rebase of TE image %s", FileName);
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Copy the relocated image into raw image file.
|
||||||
|
//
|
||||||
|
SectionHeader = (EFI_IMAGE_SECTION_HEADER *)(TEImageHeader + 1);
|
||||||
|
for (Index = 0; Index < TEImageHeader->NumberOfSections; Index++, SectionHeader++) {
|
||||||
|
if (!ImageContext.IsTeImage) {
|
||||||
|
CopyMem(
|
||||||
|
(UINT8 *)TEImageHeader + sizeof (EFI_TE_IMAGE_HEADER)-TEImageHeader->StrippedSize + SectionHeader->PointerToRawData,
|
||||||
|
(VOID*)(UINTN)(ImageContext.ImageAddress + SectionHeader->VirtualAddress),
|
||||||
|
SectionHeader->SizeOfRawData
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
CopyMem(
|
||||||
|
(UINT8 *)TEImageHeader + sizeof (EFI_TE_IMAGE_HEADER)-TEImageHeader->StrippedSize + SectionHeader->PointerToRawData,
|
||||||
|
(VOID*)(UINTN)(ImageContext.ImageAddress + sizeof (EFI_TE_IMAGE_HEADER)-TEImageHeader->StrippedSize + SectionHeader->VirtualAddress),
|
||||||
|
SectionHeader->SizeOfRawData
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Free the allocated memory resource
|
||||||
|
//
|
||||||
|
free((VOID *)MemoryImagePointer);
|
||||||
|
MemoryImagePointer = NULL;
|
||||||
|
if (PeFileBuffer != NULL) {
|
||||||
|
free(PeFileBuffer);
|
||||||
|
PeFileBuffer = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Update Image Base Address
|
||||||
|
//
|
||||||
|
TEImageHeader->ImageBase = NewPe32BaseAddress;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Now update file checksum
|
||||||
|
//
|
||||||
|
if (FfsFile->Attributes & FFS_ATTRIB_CHECKSUM) {
|
||||||
|
SavedState = FfsFile->State;
|
||||||
|
FfsFile->IntegrityCheck.Checksum.File = 0;
|
||||||
|
FfsFile->State = 0;
|
||||||
|
FfsFile->IntegrityCheck.Checksum.File = CalculateChecksum8(
|
||||||
|
(UINT8 *)((UINT8 *)FfsFile + FfsHeaderSize),
|
||||||
|
GetFfsFileLength(FfsFile) - FfsHeaderSize
|
||||||
|
);
|
||||||
|
FfsFile->State = SavedState;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
FfsRebaseImageRead(
|
||||||
|
IN VOID *FileHandle,
|
||||||
|
IN UINTN FileOffset,
|
||||||
|
IN OUT UINT32 *ReadSize,
|
||||||
|
OUT VOID *Buffer
|
||||||
|
)
|
||||||
|
/*++
|
||||||
|
|
||||||
|
Routine Description:
|
||||||
|
|
||||||
|
Support routine for the PE/COFF Loader that reads a buffer from a PE/COFF file
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
|
||||||
|
FileHandle - The handle to the PE/COFF file
|
||||||
|
|
||||||
|
FileOffset - The offset, in bytes, into the file to read
|
||||||
|
|
||||||
|
ReadSize - The number of bytes to read from the file starting at FileOffset
|
||||||
|
|
||||||
|
Buffer - A pointer to the buffer to read the data into.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
EFI_SUCCESS - ReadSize bytes of data were read into Buffer from the PE/COFF file starting at FileOffset
|
||||||
|
|
||||||
|
--*/
|
||||||
|
{
|
||||||
|
CHAR8 *Destination8;
|
||||||
|
CHAR8 *Source8;
|
||||||
|
UINT32 Length;
|
||||||
|
|
||||||
|
Destination8 = Buffer;
|
||||||
|
Source8 = (CHAR8 *)((UINTN)FileHandle + FileOffset);
|
||||||
|
Length = *ReadSize;
|
||||||
|
while (Length--) {
|
||||||
|
*(Destination8++) = *(Source8++);
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
GetChildFvFromFfs (
|
||||||
|
IN UINT64 BaseAddress,
|
||||||
|
IN EFI_FFS_FILE_HEADER *FfsFile,
|
||||||
|
IN UINTN XipOffset
|
||||||
|
)
|
||||||
|
/*++
|
||||||
|
|
||||||
|
Routine Description:
|
||||||
|
|
||||||
|
This function gets all child FvImages in the input FfsFile, and records
|
||||||
|
their base address to the parent image.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
FvInfo A pointer to FV_INFO struture.
|
||||||
|
FfsFile A pointer to Ffs file image that may contain FvImage.
|
||||||
|
XipOffset The offset address to the parent FvImage base.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
EFI_SUCCESS Base address of child Fv image is recorded.
|
||||||
|
--*/
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
UINTN Index;
|
||||||
|
EFI_FILE_SECTION_POINTER SubFvSection;
|
||||||
|
EFI_FIRMWARE_VOLUME_HEADER *SubFvImageHeader;
|
||||||
|
EFI_PHYSICAL_ADDRESS SubFvBaseAddress;
|
||||||
|
EFI_FIRMWARE_VOLUME_HEADER *OrigFvHeader;
|
||||||
|
UINT32 OrigFvLength;
|
||||||
|
EFI_PHYSICAL_ADDRESS OrigFvBaseAddress;
|
||||||
|
EFI_FFS_FILE_HEADER *CurrentFile;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Initialize FV library, saving previous values
|
||||||
|
//
|
||||||
|
OrigFvHeader = NULL;
|
||||||
|
GetFvHeader (&OrigFvHeader, &OrigFvLength);
|
||||||
|
OrigFvBaseAddress = BaseAddress;
|
||||||
|
for (Index = 1;; Index++) {
|
||||||
|
//
|
||||||
|
// Find FV section
|
||||||
|
//
|
||||||
|
Status = GetSectionByType (FfsFile, EFI_SECTION_FIRMWARE_VOLUME_IMAGE, Index, &SubFvSection);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
SubFvImageHeader = (EFI_FIRMWARE_VOLUME_HEADER *) ((UINT8 *) SubFvSection.FVImageSection + GetSectionHeaderLength(SubFvSection.FVImageSection));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Rebase on Flash
|
||||||
|
//
|
||||||
|
SubFvBaseAddress = OrigFvBaseAddress + (UINTN) SubFvImageHeader - (UINTN) FfsFile + XipOffset;
|
||||||
|
//mFvBaseAddress[mFvBaseAddressNumber ++ ] = SubFvBaseAddress;
|
||||||
|
BaseAddress = SubFvBaseAddress;
|
||||||
|
InitializeFvLib(SubFvImageHeader, (UINT32) SubFvImageHeader->FvLength);
|
||||||
|
|
||||||
|
Status = GetNextFile (NULL, &CurrentFile);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
Error (NULL, 0, 0003, "error parsing FV image", "FFS file can't be found");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
while (CurrentFile) {
|
||||||
|
RebaseFfs (BaseAddress, "", CurrentFile, (UINTN) CurrentFile - (UINTN) SubFvImageHeader);
|
||||||
|
Status = GetNextFile (CurrentFile, &CurrentFile);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BaseAddress = OrigFvBaseAddress;
|
||||||
|
if (OrigFvHeader != NULL) {
|
||||||
|
InitializeFvLib(OrigFvHeader, OrigFvLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
GetPe32Info (
|
||||||
|
IN UINT8 *Pe32,
|
||||||
|
OUT UINT32 *EntryPoint,
|
||||||
|
OUT UINT32 *BaseOfCode,
|
||||||
|
OUT UINT16 *MachineType
|
||||||
|
)
|
||||||
|
/*++
|
||||||
|
|
||||||
|
Routine Description:
|
||||||
|
|
||||||
|
Retrieves the PE32 entry point offset and machine type from PE image or TeImage.
|
||||||
|
See EfiImage.h for machine types. The entry point offset is from the beginning
|
||||||
|
of the PE32 buffer passed in.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
|
||||||
|
Pe32 Beginning of the PE32.
|
||||||
|
EntryPoint Offset from the beginning of the PE32 to the image entry point.
|
||||||
|
BaseOfCode Base address of code.
|
||||||
|
MachineType Magic number for the machine type.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
EFI_SUCCESS Function completed successfully.
|
||||||
|
EFI_ABORTED Error encountered.
|
||||||
|
EFI_INVALID_PARAMETER A required parameter was NULL.
|
||||||
|
EFI_UNSUPPORTED The operation is unsupported.
|
||||||
|
|
||||||
|
--*/
|
||||||
|
{
|
||||||
|
EFI_IMAGE_DOS_HEADER *DosHeader;
|
||||||
|
EFI_IMAGE_OPTIONAL_HEADER_UNION *ImgHdr;
|
||||||
|
EFI_TE_IMAGE_HEADER *TeHeader;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Verify input parameters
|
||||||
|
//
|
||||||
|
if (Pe32 == NULL) {
|
||||||
|
return EFI_INVALID_PARAMETER;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// First check whether it is one TE Image.
|
||||||
|
//
|
||||||
|
TeHeader = (EFI_TE_IMAGE_HEADER *) Pe32;
|
||||||
|
if (TeHeader->Signature == EFI_TE_IMAGE_HEADER_SIGNATURE) {
|
||||||
|
//
|
||||||
|
// By TeImage Header to get output
|
||||||
|
//
|
||||||
|
*EntryPoint = TeHeader->AddressOfEntryPoint + sizeof (EFI_TE_IMAGE_HEADER) - TeHeader->StrippedSize;
|
||||||
|
*BaseOfCode = TeHeader->BaseOfCode + sizeof (EFI_TE_IMAGE_HEADER) - TeHeader->StrippedSize;
|
||||||
|
*MachineType = TeHeader->Machine;
|
||||||
|
} else {
|
||||||
|
|
||||||
|
//
|
||||||
|
// Then check whether
|
||||||
|
// First is the DOS header
|
||||||
|
//
|
||||||
|
DosHeader = (EFI_IMAGE_DOS_HEADER *) Pe32;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Verify DOS header is expected
|
||||||
|
//
|
||||||
|
if (DosHeader->e_magic != EFI_IMAGE_DOS_SIGNATURE) {
|
||||||
|
Error (NULL, 0, 3000, "Invalid", "Unknown magic number in the DOS header, 0x%04X.", DosHeader->e_magic);
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Immediately following is the NT header.
|
||||||
|
//
|
||||||
|
ImgHdr = (EFI_IMAGE_OPTIONAL_HEADER_UNION *) ((UINTN) Pe32 + DosHeader->e_lfanew);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Verify NT header is expected
|
||||||
|
//
|
||||||
|
if (ImgHdr->Pe32.Signature != EFI_IMAGE_NT_SIGNATURE) {
|
||||||
|
Error (NULL, 0, 3000, "Invalid", "Unrecognized image signature 0x%08X.", (unsigned) ImgHdr->Pe32.Signature);
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Get output
|
||||||
|
//
|
||||||
|
*EntryPoint = ImgHdr->Pe32.OptionalHeader.AddressOfEntryPoint;
|
||||||
|
*BaseOfCode = ImgHdr->Pe32.OptionalHeader.BaseOfCode;
|
||||||
|
*MachineType = ImgHdr->Pe32.FileHeader.Machine;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Verify machine type is supported
|
||||||
|
//
|
||||||
|
if ((*MachineType != EFI_IMAGE_MACHINE_IA32) && (*MachineType != EFI_IMAGE_MACHINE_X64) && (*MachineType != EFI_IMAGE_MACHINE_EBC) &&
|
||||||
|
(*MachineType != EFI_IMAGE_MACHINE_ARMT) && (*MachineType != EFI_IMAGE_MACHINE_AARCH64)) {
|
||||||
|
Error (NULL, 0, 3000, "Invalid", "Unrecognized machine type in the PE32 file.");
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
/** @file Rebase.h
|
||||||
|
|
||||||
|
Library to rebase PE image.
|
||||||
|
|
||||||
|
Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
||||||
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
#ifndef _FMMT_REBASE_H
|
||||||
|
#define _FMMT_REBASE_H
|
||||||
|
|
||||||
|
#include <Common/UefiBaseTypes.h>
|
||||||
|
#include <Common/PiFirmwareFile.h>
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
RebaseFfs(
|
||||||
|
IN OUT UINT64 BaseAddress,
|
||||||
|
IN CHAR8 *FileName,
|
||||||
|
IN OUT EFI_FFS_FILE_HEADER *FfsFile,
|
||||||
|
IN UINTN XipOffset
|
||||||
|
);
|
||||||
|
|
||||||
|
EFI_STATUS
|
||||||
|
GetChildFvFromFfs (
|
||||||
|
IN UINT64 BaseAddress,
|
||||||
|
IN EFI_FFS_FILE_HEADER *FfsFile,
|
||||||
|
IN UINTN XipOffset
|
||||||
|
);
|
||||||
|
|
||||||
|
#endif
|
|
@ -47,6 +47,7 @@ VFRAUTOGEN = VfrCompile/VfrLexer.h
|
||||||
APPLICATIONS = \
|
APPLICATIONS = \
|
||||||
BrotliCompress \
|
BrotliCompress \
|
||||||
VfrCompile \
|
VfrCompile \
|
||||||
|
FMMT \
|
||||||
BfmLib \
|
BfmLib \
|
||||||
EfiRom \
|
EfiRom \
|
||||||
FCE \
|
FCE \
|
||||||
|
|
|
@ -12,6 +12,7 @@ LIBRARIES = Common
|
||||||
APPLICATIONS = \
|
APPLICATIONS = \
|
||||||
VfrCompile \
|
VfrCompile \
|
||||||
BrotliCompress \
|
BrotliCompress \
|
||||||
|
FMMT \
|
||||||
BfmLib \
|
BfmLib \
|
||||||
EfiRom \
|
EfiRom \
|
||||||
FCE \
|
FCE \
|
||||||
|
|
Loading…
Reference in New Issue