mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2025-01-07 18:48:04 +01:00
1393 lines
39 KiB
C
1393 lines
39 KiB
C
/** @file
|
|
UEFI image loader library implementation for UE images.
|
|
|
|
Copyright (c) 2021 - 2023, Marvin Häuser. All rights reserved.<BR>
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
**/
|
|
|
|
#include <Base.h>
|
|
|
|
#include <IndustryStandard/UeImage.h>
|
|
|
|
#include <Library/BaseLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/BaseOverflowLib.h>
|
|
#include <Library/DebugLib.h>
|
|
#include <Library/MemoryAllocationLib.h>
|
|
#include <Library/PcdLib.h>
|
|
#include <Library/PeCoffLib2.h>
|
|
#include <Library/UefiImageLib.h>
|
|
#include <Library/UeImageLib.h>
|
|
|
|
struct UE_LOADER_RUNTIME_CONTEXT_ {
|
|
UINT8 Machine;
|
|
UINT8 Reserved[7];
|
|
UINT32 FixupSize;
|
|
UINT64 *FixupData;
|
|
UINT32 UnchainedRelocsSize;
|
|
UINT8 *UnchainedRelocs;
|
|
};
|
|
|
|
typedef union {
|
|
UINT32 Value32;
|
|
UINT64 Value64;
|
|
} UE_RELOC_FIXUP_VALUE;
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalVerifySegments (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
CONST UE_SEGMENT *Segments;
|
|
UINT8 LastSegmentIndex;
|
|
UINT8 SegmentIndex;
|
|
UINT32 SegmentEndFileOffset;
|
|
UINT32 SegmentEndImageAddress;
|
|
BOOLEAN Overflow;
|
|
UINT32 SegmentImageSize;
|
|
|
|
Segments = Context->Segments;
|
|
LastSegmentIndex = Context->LastSegmentIndex;
|
|
//
|
|
// As it holds that SegmentEndFileOffset - Context->SegmentsFileOffset <= SegmentEndImageAddress,
|
|
// and it holds that SegmentEndImageAddress % 4 KiB = 0, SegmentEndFileOffset - Context->SegmentsFileOffset
|
|
// can be at most 0xFFFFF000. As it holds that Context->SegmentsFileOffset <= MAX_SIZE_OF_UE_HEADER <= 0xFF0,
|
|
// this cannot overflow.
|
|
//
|
|
STATIC_ASSERT (
|
|
MAX_SIZE_OF_UE_HEADER <= BASE_4KB - 8,
|
|
"The arithmetic below may overflow."
|
|
);
|
|
|
|
SegmentEndFileOffset = Context->SegmentsFileOffset;
|
|
//
|
|
// The first image segment must begin the image address space.
|
|
//
|
|
SegmentEndImageAddress = 0;
|
|
|
|
for (SegmentIndex = 0; SegmentIndex <= LastSegmentIndex; ++SegmentIndex) {
|
|
SegmentImageSize = UE_SEGMENT_SIZE (Segments[SegmentIndex].ImageInfo);
|
|
|
|
if (Segments[SegmentIndex].FileSize > SegmentImageSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
//
|
|
// Verify the image segments are aligned.
|
|
//
|
|
if (!IS_ALIGNED (SegmentImageSize, Context->SegmentAlignment)) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
//
|
|
// Determine the end of the current image segment.
|
|
//
|
|
Overflow = BaseOverflowAddU32 (
|
|
SegmentEndImageAddress,
|
|
SegmentImageSize,
|
|
&SegmentEndImageAddress
|
|
);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
//
|
|
// As it holds that SegmentFileSize <= SegmentImageSize and thus
|
|
// SegmentEndFileOffset - Context->SegmentsFileOffset <= SegmentEndImageAddress,
|
|
// this cannot overflow (see above).
|
|
//
|
|
SegmentEndFileOffset += Segments[SegmentIndex].FileSize;
|
|
}
|
|
//
|
|
// As it holds that SegmentEndFileOffset <= 0xFFFFFFF0, this cannot overflow.
|
|
//
|
|
SegmentEndFileOffset = ALIGN_VALUE (
|
|
SegmentEndFileOffset,
|
|
UE_LOAD_TABLE_ALIGNMENT
|
|
);
|
|
//
|
|
// Verify all image segment data are in bounds of the file buffer.
|
|
//
|
|
ASSERT (Context->SegmentsFileOffset <= SegmentEndFileOffset);
|
|
if (SegmentEndFileOffset > Context->UnsignedFileSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
Context->LoadTablesFileOffset = SegmentEndFileOffset;
|
|
Context->ImageSize = SegmentEndImageAddress;
|
|
|
|
if (Context->XIP) {
|
|
Context->ImageSize += Context->SegmentsFileOffset;
|
|
}
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalVerifyLoadTables (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
BOOLEAN Overflow;
|
|
CONST UE_LOAD_TABLE *LoadTable;
|
|
UINT8 LoadTableIndex;
|
|
INT16 PrevLoadTableId;
|
|
UINT32 LoadTableFileOffset;
|
|
UINT32 LoadTableFileSize;
|
|
UINT8 LoadTableId;
|
|
UINT32 LoadTableEndFileOffset;
|
|
UINT8 NumLoadTables;
|
|
|
|
Context->RelocTableSize = 0;
|
|
|
|
LoadTableEndFileOffset = Context->LoadTablesFileOffset;
|
|
NumLoadTables = Context->NumLoadTables;
|
|
PrevLoadTableId = -1;
|
|
|
|
if (0 < NumLoadTables) {
|
|
LoadTableIndex = 0;
|
|
do {
|
|
LoadTable = Context->LoadTables + LoadTableIndex;
|
|
LoadTableId = UE_LOAD_TABLE_ID (LoadTable->FileInfo);
|
|
|
|
if (PrevLoadTableId >= LoadTableId) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
LoadTableFileOffset = LoadTableEndFileOffset;
|
|
LoadTableFileSize = UE_LOAD_TABLE_SIZE (LoadTable->FileInfo);
|
|
|
|
Overflow = BaseOverflowAddU32 (
|
|
LoadTableFileOffset,
|
|
LoadTableFileSize,
|
|
&LoadTableEndFileOffset
|
|
);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
PrevLoadTableId = LoadTableId;
|
|
++LoadTableIndex;
|
|
} while (LoadTableIndex < NumLoadTables);
|
|
|
|
if (UE_LOAD_TABLE_ID (Context->LoadTables[0].FileInfo) == UeLoadTableIdReloc) {
|
|
if (Context->RelocsStripped) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
Context->RelocTableSize = UE_LOAD_TABLE_SIZE (
|
|
Context->LoadTables[0].FileInfo
|
|
);
|
|
}
|
|
}
|
|
|
|
if (LoadTableEndFileOffset != Context->UnsignedFileSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeInitializeContextPreHash (
|
|
OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
IN CONST VOID *FileBuffer,
|
|
IN UINT32 FileSize
|
|
)
|
|
{
|
|
//BOOLEAN Overflow;
|
|
CONST UE_HEADER *UeHdr;
|
|
//UINT32 UnsignedFileSize;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (FileBuffer != NULL || FileSize == 0);
|
|
|
|
if (MIN_SIZE_OF_UE_HEADER > FileSize) {
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
UeHdr = (CONST UE_HEADER *) FileBuffer;
|
|
|
|
if (UeHdr->Magic != UE_HEADER_MAGIC) {
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
ZeroMem (Context, sizeof (*Context));
|
|
|
|
/*UnsignedFileSize = UE_HEADER_FILE_SIZE (UeHdr->FileInfo);
|
|
|
|
Overflow = BaseOverflowSubU32 (
|
|
FileSize,
|
|
UnsignedFileSize,
|
|
&Context->CertTableSize
|
|
);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}*/
|
|
|
|
Context->FileBuffer = (CONST UINT8 *)UeHdr;
|
|
//Context->UnsignedFileSize = UnsignedFileSize;
|
|
Context->UnsignedFileSize = FileSize;
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
BOOLEAN
|
|
UeHashImageDefault (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
IN OUT VOID *HashContext,
|
|
IN UE_LOADER_HASH_UPDATE HashUpdate
|
|
)
|
|
{
|
|
BOOLEAN Result;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (HashContext != NULL);
|
|
ASSERT (HashUpdate != NULL);
|
|
|
|
Result = HashUpdate (
|
|
HashContext,
|
|
Context->FileBuffer,
|
|
Context->UnsignedFileSize
|
|
);
|
|
if (!Result) {
|
|
DEBUG_RAISE ();
|
|
}
|
|
|
|
return Result;
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalInitializeContextLate (
|
|
OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
if (Context->EntryPointAddress > Context->ImageSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
return InternalVerifyLoadTables (Context);
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeInitializeContextPostHash (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
CONST UE_HEADER *UeHdr;
|
|
UINT8 LastSegmentIndex;
|
|
UINT8 NumLoadTables;
|
|
UINT32 LoadTablesFileOffset;
|
|
UINT32 HeaderSize;
|
|
UINT64 BaseAddress;
|
|
RETURN_STATUS Status;
|
|
|
|
ASSERT (Context != NULL);
|
|
|
|
UeHdr = (CONST UE_HEADER *)Context->FileBuffer;
|
|
|
|
Context->FixedAddress = (UeHdr->ImageInfo & UE_HEADER_IMAGE_INFO_FIXED_ADDRESS) != 0;
|
|
Context->RelocsStripped = (UeHdr->ImageInfo & UE_HEADER_IMAGE_INFO_RELOCATION_FIXUPS_STRIPPED) != 0;
|
|
Context->XIP = (UeHdr->ImageInfo & UE_HEADER_IMAGE_INFO_XIP) != 0;
|
|
|
|
Context->Segments = UeHdr->Segments;
|
|
Context->SegmentImageInfoIterSize = sizeof (*UeHdr->Segments);
|
|
Context->SegmentAlignment = UE_HEADER_SEGMENT_ALIGNMENT (UeHdr->ImageInfo);
|
|
|
|
LastSegmentIndex = UE_HEADER_LAST_SEGMENT_INDEX (UeHdr->TableCounts);
|
|
NumLoadTables = UE_HEADER_NUM_LOAD_TABLES (UeHdr->TableCounts);
|
|
|
|
LoadTablesFileOffset = MIN_SIZE_OF_UE_HEADER
|
|
+ (UINT32) LastSegmentIndex * sizeof (UE_SEGMENT);
|
|
|
|
HeaderSize = LoadTablesFileOffset + (UINT32) NumLoadTables * sizeof (UE_LOAD_TABLE);
|
|
ASSERT (HeaderSize <= MAX_SIZE_OF_UE_HEADER);
|
|
if (Context->XIP) {
|
|
HeaderSize = ALIGN_VALUE (HeaderSize, Context->SegmentAlignment);
|
|
}
|
|
|
|
if (HeaderSize > Context->UnsignedFileSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
ASSERT (IS_ALIGNED (LoadTablesFileOffset, ALIGNOF (UE_LOAD_TABLE)));
|
|
|
|
Context->LoadTables = (CONST UE_LOAD_TABLE *)(
|
|
(CONST UINT8 *) UeHdr + LoadTablesFileOffset
|
|
);
|
|
|
|
Context->SegmentsFileOffset = HeaderSize;
|
|
Context->LastSegmentIndex = LastSegmentIndex;
|
|
Context->NumLoadTables = NumLoadTables;
|
|
|
|
BaseAddress = UE_HEADER_BASE_ADDRESS (UeHdr->ImageInfo);
|
|
|
|
if (!IS_ALIGNED (BaseAddress, Context->SegmentAlignment)) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
Context->FileBuffer = (CONST UINT8 *)UeHdr;
|
|
|
|
Status = InternalVerifySegments (Context);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
Context->BaseAddress = BaseAddress;
|
|
Context->EntryPointAddress = UeHdr->EntryPointAddress;
|
|
|
|
Context->Subsystem = UE_HEADER_SUBSYSTEM (UeHdr->Type);
|
|
Context->Machine = UE_HEADER_ARCH (UeHdr->Type);
|
|
|
|
return InternalInitializeContextLate (Context);
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeLoadImage (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT VOID *Destination,
|
|
IN UINT32 DestinationSize
|
|
)
|
|
{
|
|
UINT8 SegmentIndex;
|
|
UINT32 SegmentFileOffset;
|
|
UINT32 SegmentFileSize;
|
|
UINT32 SegmentImageAddress;
|
|
UINT32 SegmentImageSize;
|
|
UINT32 PrevSegmentDataEnd;
|
|
|
|
CONST UE_SEGMENT *Segments;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (Destination != NULL);
|
|
ASSERT (ADDRESS_IS_ALIGNED (Destination, Context->SegmentAlignment));
|
|
|
|
Context->ImageBuffer = Destination;
|
|
|
|
//
|
|
// Load all image load tables into the address space.
|
|
//
|
|
Segments = Context->Segments;
|
|
|
|
//
|
|
// Start zeroing from the start of the destination buffer.
|
|
//
|
|
PrevSegmentDataEnd = 0;
|
|
|
|
SegmentFileOffset = Context->SegmentsFileOffset;
|
|
SegmentImageAddress = 0;
|
|
|
|
SegmentIndex = 0;
|
|
do {
|
|
SegmentFileSize = Segments[SegmentIndex].FileSize;
|
|
SegmentImageSize = UE_SEGMENT_SIZE (Segments[SegmentIndex].ImageInfo);
|
|
//
|
|
// Zero from the end of the previous image segment to the start of this
|
|
// image segment.
|
|
//
|
|
ZeroMem (
|
|
(UINT8 *)Destination + PrevSegmentDataEnd,
|
|
SegmentImageAddress - PrevSegmentDataEnd
|
|
);
|
|
//
|
|
// Load the current Image segment into the address space.
|
|
//
|
|
ASSERT (SegmentFileSize <= SegmentImageSize);
|
|
CopyMem (
|
|
(UINT8 *)Destination + SegmentImageAddress,
|
|
Context->FileBuffer + SegmentFileOffset,
|
|
SegmentFileSize
|
|
);
|
|
PrevSegmentDataEnd = SegmentImageAddress + SegmentFileSize;
|
|
|
|
SegmentFileOffset += SegmentFileSize;
|
|
SegmentImageAddress += SegmentImageSize;
|
|
++SegmentIndex;
|
|
} while (SegmentIndex <= Context->LastSegmentIndex);
|
|
//
|
|
// Zero the trailing data after the last image segment.
|
|
//
|
|
ZeroMem (
|
|
(UINT8 *)Destination + PrevSegmentDataEnd,
|
|
DestinationSize - PrevSegmentDataEnd
|
|
);
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeLoaderGetRuntimeContextSize (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT UINT32 *Size
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
ASSERT (Size != NULL);
|
|
|
|
*Size = sizeof (UE_LOADER_RUNTIME_CONTEXT);
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
Apply an image relocation fixup.
|
|
|
|
Only a subset of the PE/COFF relocation fixup types are permited.
|
|
The relocation fixup target must be in bounds, aligned, and must not overlap
|
|
with the Relocation Directory.
|
|
|
|
@param[in] Context The context describing the image. Must have been
|
|
loaded by PeCoffLoadImage().
|
|
@param[in] RelocIndex The index of the relocation fixup to apply.
|
|
@param[in] Adjust The delta to add to the addresses.
|
|
|
|
@retval RETURN_SUCCESS The relocation fixup has been applied successfully.
|
|
@retval other The relocation fixup could not be applied successfully.
|
|
**/
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalApplyRelocation (
|
|
IN OUT VOID *Image,
|
|
IN UINT32 ImageSize,
|
|
IN UINT8 Machine,
|
|
IN UINT16 RelocType,
|
|
IN UINT32 *RelocTarget,
|
|
IN UINT64 Adjust,
|
|
IN OUT UINT64 *FixupData,
|
|
IN BOOLEAN IsRuntime
|
|
)
|
|
{
|
|
BOOLEAN Overflow;
|
|
|
|
UINT32 RemFixupTargetSize;
|
|
|
|
UINT32 FixupTarget;
|
|
VOID *Fixup;
|
|
UINT8 FixupSize;
|
|
UE_RELOC_FIXUP_VALUE FixupValue;
|
|
|
|
ASSERT (FixupData != NULL);
|
|
|
|
FixupTarget = *RelocTarget;
|
|
//
|
|
// Verify the relocation fixup target address is in bounds of the image buffer.
|
|
//
|
|
Overflow = BaseOverflowSubU32 (ImageSize, FixupTarget, &RemFixupTargetSize);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
Fixup = (UINT8 *)Image + FixupTarget;
|
|
//
|
|
// Apply the relocation fixup per type.
|
|
//
|
|
if (RelocType < UeRelocGenericMax) {
|
|
if (RelocType == UeReloc32) {
|
|
FixupSize = sizeof (UINT32);
|
|
//
|
|
// Verify the relocation fixup target is in bounds of the image buffer.
|
|
//
|
|
if (FixupSize > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate the target instruction.
|
|
//
|
|
FixupValue.Value32 = ReadUnaligned32 (Fixup);
|
|
//
|
|
// If the Image relocation target value mismatches, skip or abort.
|
|
//
|
|
if (IsRuntime && (FixupValue.Value32 != (UINT32)*FixupData)) {
|
|
if (PcdGetBool (PcdImageLoaderRtRelocAllowTargetMismatch)) {
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
FixupValue.Value32 += (UINT32) Adjust;
|
|
WriteUnaligned32 (Fixup, FixupValue.Value32);
|
|
} else if (RelocType == UeReloc64) {
|
|
FixupSize = sizeof (UINT64);
|
|
//
|
|
// Verify the image relocation fixup target is in bounds of the image
|
|
// buffer.
|
|
//
|
|
if (FixupSize > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate target the instruction.
|
|
//
|
|
FixupValue.Value64 = ReadUnaligned64 (Fixup);
|
|
//
|
|
// If the Image relocation target value mismatches, skip or abort.
|
|
//
|
|
if (IsRuntime && (FixupValue.Value64 != *FixupData)) {
|
|
if (PcdGetBool (PcdImageLoaderRtRelocAllowTargetMismatch)) {
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
FixupValue.Value64 += Adjust;
|
|
WriteUnaligned64 (Fixup, FixupValue.Value64);
|
|
} else if (RelocType == UeReloc32NoMeta) {
|
|
FixupSize = sizeof (UINT32);
|
|
//
|
|
// Verify the relocation fixup target is in bounds of the image buffer.
|
|
//
|
|
if (FixupSize > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate the target instruction.
|
|
//
|
|
FixupValue.Value32 = ReadUnaligned32 (Fixup);
|
|
//
|
|
// If the Image relocation target value mismatches, skip or abort.
|
|
//
|
|
if (IsRuntime && (FixupValue.Value32 != (UINT32)*FixupData)) {
|
|
if (PcdGetBool (PcdImageLoaderRtRelocAllowTargetMismatch)) {
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
FixupValue.Value32 += (UINT32) Adjust;
|
|
WriteUnaligned32 (Fixup, FixupValue.Value32);
|
|
|
|
if (!IsRuntime) {
|
|
*FixupData = FixupValue.Value32;
|
|
}
|
|
} else {
|
|
//
|
|
// The image relocation fixup type is unknown, disallow the image.
|
|
//
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
} else {
|
|
#if 0
|
|
if (Machine == UeMachineArmThumbMixed) {
|
|
switch (RelocType) {
|
|
// TODO: MOVW
|
|
case (UeMachineArmThumbMixed << 4U) | UeRelocArmMovt:
|
|
{
|
|
//
|
|
// Verify the relocation fixup target is in bounds of the image buffer.
|
|
//
|
|
if (sizeof (UINT64) > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Verify the relocation fixup target is sufficiently aligned.
|
|
// The ARM Thumb instruction pait must start on a 16-bit boundary.
|
|
//
|
|
if (!IS_ALIGNED (RelocTarget, ALIGNOF (UINT16))) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate the target instruction.
|
|
//
|
|
PeCoffThumbMovwMovtImmediateFixup (Fixup, Adjust);
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
//
|
|
// The image relocation fixup type is unknown, disallow the image.
|
|
//
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
//
|
|
// The image relocation fixup type is unknown, disallow the image.
|
|
//
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
*RelocTarget = FixupTarget + FixupSize;
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
UnchainReloc (
|
|
IN OUT UE_LOADER_RUNTIME_CONTEXT *RuntimeContext OPTIONAL,
|
|
IN CONST UINT8 *MetaSource OPTIONAL,
|
|
IN UINT32 MetaSize,
|
|
IN BOOLEAN IsRuntime,
|
|
IN UINT16 RelocOffset,
|
|
IN UINT64 *FixupData
|
|
)
|
|
{
|
|
UINT32 OldSize;
|
|
UINT16 FixupHdr;
|
|
UINT32 FixupIndex;
|
|
UINT8 Addend;
|
|
|
|
if ((RuntimeContext != NULL) && !IsRuntime) {
|
|
if (MetaSource == NULL) {
|
|
FixupIndex = RuntimeContext->UnchainedRelocsSize - sizeof (UINT16);
|
|
|
|
FixupHdr = *(UINT16 *)&RuntimeContext->UnchainedRelocs[FixupIndex];
|
|
FixupHdr = (RelocOffset << 4U) | UE_RELOC_FIXUP_TYPE(FixupHdr);
|
|
|
|
*(UINT16 *)&RuntimeContext->UnchainedRelocs[FixupIndex] = FixupHdr;
|
|
|
|
Addend = ALIGN_VALUE_ADDEND(RuntimeContext->UnchainedRelocsSize, ALIGNOF(UE_FIXUP_ROOT));
|
|
if ((RelocOffset == UE_HEAD_FIXUP_OFFSET_END) && (Addend != 0)) {
|
|
OldSize = RuntimeContext->UnchainedRelocsSize;
|
|
RuntimeContext->UnchainedRelocs = ReallocateRuntimePool (
|
|
OldSize,
|
|
OldSize + Addend,
|
|
RuntimeContext->UnchainedRelocs
|
|
);
|
|
if (RuntimeContext->UnchainedRelocs == NULL) {
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
ZeroMem (RuntimeContext->UnchainedRelocs + OldSize, Addend);
|
|
RuntimeContext->UnchainedRelocsSize += Addend;
|
|
}
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
OldSize = RuntimeContext->UnchainedRelocsSize;
|
|
RuntimeContext->UnchainedRelocs = ReallocateRuntimePool (
|
|
OldSize,
|
|
OldSize + MetaSize,
|
|
RuntimeContext->UnchainedRelocs
|
|
);
|
|
if (RuntimeContext->UnchainedRelocs == NULL) {
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
CopyMem (RuntimeContext->UnchainedRelocs + OldSize, MetaSource, MetaSize);
|
|
|
|
RuntimeContext->UnchainedRelocsSize += MetaSize;
|
|
|
|
if (FixupData != NULL) {
|
|
OldSize = RuntimeContext->FixupSize;
|
|
RuntimeContext->FixupData = ReallocateRuntimePool (
|
|
OldSize,
|
|
OldSize + sizeof (*FixupData),
|
|
RuntimeContext->FixupData
|
|
);
|
|
if (RuntimeContext->FixupData == NULL) {
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
CopyMem ((UINT8 *)RuntimeContext->FixupData + OldSize, FixupData, sizeof (*FixupData));
|
|
|
|
RuntimeContext->FixupSize += sizeof (*FixupData);
|
|
}
|
|
}
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalProcessRelocChain (
|
|
IN OUT VOID *Image,
|
|
IN UINT32 ImageSize,
|
|
IN UINT8 Machine,
|
|
IN UINT16 FirstRelocType,
|
|
IN UINT32 *ChainStart,
|
|
IN UINT64 Adjust,
|
|
IN OUT UE_LOADER_RUNTIME_CONTEXT *RuntimeContext OPTIONAL
|
|
)
|
|
{
|
|
RETURN_STATUS Status;
|
|
UINT16 RelocType;
|
|
UINT16 RelocOffset;
|
|
UINT32 RelocTarget;
|
|
|
|
BOOLEAN Overflow;
|
|
UINT32 RemFixupTargetSize;
|
|
|
|
UINT8 *Fixup;
|
|
UE_RELOC_FIXUP_VALUE FixupInfo;
|
|
UINT8 FixupSize;
|
|
UE_RELOC_FIXUP_VALUE FixupValue;
|
|
UINT16 FixupHdr;
|
|
UINT64 FixupData;
|
|
|
|
RelocType = FirstRelocType;
|
|
RelocTarget = *ChainStart;
|
|
|
|
while (TRUE) {
|
|
Overflow = BaseOverflowSubU32 (ImageSize, RelocTarget, &RemFixupTargetSize);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
Fixup = (UINT8 *)Image + RelocTarget;
|
|
|
|
if (RelocType < UeRelocGenericMax) {
|
|
if (RelocType == UeReloc64) {
|
|
FixupSize = sizeof (UINT64);
|
|
//
|
|
// Verify the relocation fixup target is in bounds of the image memory.
|
|
//
|
|
if (FixupSize > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate the target instruction.
|
|
//
|
|
FixupInfo.Value64 = ReadUnaligned64 ((CONST VOID *)Fixup);
|
|
FixupValue.Value64 = UE_CHAINED_RELOC_FIXUP_VALUE (FixupInfo.Value64);
|
|
FixupValue.Value64 += Adjust;
|
|
WriteUnaligned64 ((VOID *)Fixup, FixupValue.Value64);
|
|
|
|
FixupData = FixupValue.Value64;
|
|
} else if (RelocType == UeReloc32) {
|
|
FixupSize = sizeof (UINT32);
|
|
//
|
|
// Verify the image relocation fixup target is in bounds of the image
|
|
// buffer.
|
|
//
|
|
if (FixupSize > RemFixupTargetSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
//
|
|
// Relocate the target instruction.
|
|
//
|
|
FixupInfo.Value32 = ReadUnaligned32 ((CONST VOID *)Fixup);
|
|
FixupValue.Value32 = UE_CHAINED_RELOC_FIXUP_VALUE_32 (FixupInfo.Value32);
|
|
FixupValue.Value32 += (UINT32) Adjust;
|
|
WriteUnaligned32 ((VOID *)Fixup, FixupValue.Value32);
|
|
|
|
FixupData = FixupValue.Value32;
|
|
//
|
|
// Imitate the common header of UE chained relocation fixups,
|
|
// as for 32-bit files all relocs have the same type.
|
|
//
|
|
FixupInfo.Value32 = FixupInfo.Value32 << 4U;
|
|
FixupInfo.Value32 |= UeReloc32;
|
|
} else {
|
|
//
|
|
// The image relocation fixup type is unknown, disallow the image.
|
|
//
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
} else {
|
|
//
|
|
// The image relocation fixup type is unknown, disallow the image.
|
|
//
|
|
DEBUG_RAISE ();
|
|
return RETURN_UNSUPPORTED;
|
|
}
|
|
|
|
RelocTarget += FixupSize;
|
|
|
|
RelocOffset = UE_CHAINED_RELOC_FIXUP_NEXT_OFFSET (FixupInfo.Value32);
|
|
FixupHdr = (RelocOffset << 4U) | RelocType;
|
|
|
|
Status = UnchainReloc (
|
|
RuntimeContext,
|
|
(CONST UINT8 *)&FixupHdr,
|
|
sizeof (FixupHdr),
|
|
FALSE,
|
|
0,
|
|
&FixupData
|
|
);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
if (RelocOffset == UE_CHAINED_RELOC_FIXUP_OFFSET_END) {
|
|
*ChainStart = RelocTarget;
|
|
return RETURN_SUCCESS;
|
|
}
|
|
//
|
|
// It holds that ImageSize mod 4 KiB = 0, thus ImageSize <= 0xFFFFF000.
|
|
// Furthermore, it holds that RelocTarget <= ImageSize.
|
|
// Finally, it holds that RelocOffset <= 0xFFE.
|
|
// It follows that this cannot overflow.
|
|
//
|
|
RelocTarget += RelocOffset;
|
|
ASSERT (RelocOffset <= RelocTarget);
|
|
|
|
RelocType = UE_CHAINED_RELOC_FIXUP_NEXT_TYPE (FixupInfo.Value32);
|
|
}
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternaRelocateImage (
|
|
IN OUT VOID *Image,
|
|
IN UINT32 ImageSize,
|
|
IN UINT8 Machine,
|
|
IN UINT64 OldBaseAddress,
|
|
IN CONST VOID *RelocTable,
|
|
IN UINT32 RelocTableSize,
|
|
IN BOOLEAN Chaining,
|
|
IN UINT64 BaseAddress,
|
|
OUT UE_LOADER_RUNTIME_CONTEXT *RuntimeContext OPTIONAL,
|
|
IN BOOLEAN IsRuntime
|
|
)
|
|
{
|
|
RETURN_STATUS Status;
|
|
BOOLEAN Overflow;
|
|
|
|
UINT64 Adjust;
|
|
|
|
UINT32 RootOffsetMax;
|
|
UINT32 EntryOffsetMax;
|
|
|
|
UINT32 TableOffset;
|
|
CONST UE_FIXUP_ROOT *RelocRoot;
|
|
|
|
UINT16 FixupInfo;
|
|
UINT16 RelocType;
|
|
UINT16 RelocOffset;
|
|
UINT32 RelocTarget;
|
|
|
|
UINT32 OldTableOffset;
|
|
UINT64 FixupData;
|
|
UINT64 *FixupPointer;
|
|
|
|
ASSERT (Image != NULL);
|
|
ASSERT (RelocTable != NULL || RelocTableSize == 0);
|
|
//
|
|
// Verify the Relocation Directory is not empty.
|
|
//
|
|
if (RelocTableSize == 0) {
|
|
return RETURN_SUCCESS;
|
|
}
|
|
//
|
|
// Calculate the image displacement from its prefered load address.
|
|
//
|
|
Adjust = BaseAddress - OldBaseAddress;
|
|
//
|
|
// FIXME: RT driver check is removed in the hope we can force no relocs in
|
|
// writable segments.
|
|
//
|
|
// Skip explicit Relocation when the image is already loaded at its base
|
|
// address.
|
|
//
|
|
if (Adjust == 0 && !Chaining) {
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
RelocTarget = 0;
|
|
|
|
if (IsRuntime) {
|
|
FixupPointer = RuntimeContext->FixupData;
|
|
}
|
|
|
|
STATIC_ASSERT (
|
|
MIN_SIZE_OF_UE_FIXUP_ROOT <= UE_LOAD_TABLE_ALIGNMENT,
|
|
"The following arithmetic may overflow."
|
|
);
|
|
|
|
RootOffsetMax = RelocTableSize - MIN_SIZE_OF_UE_FIXUP_ROOT;
|
|
EntryOffsetMax = RelocTableSize - sizeof (*RelocRoot->Heads);
|
|
//
|
|
// Apply all relocation fixups of the Image.
|
|
//
|
|
for (TableOffset = 0; TableOffset <= RootOffsetMax;) {
|
|
RelocRoot = (CONST UE_FIXUP_ROOT *)(
|
|
(CONST UINT8 *)RelocTable + TableOffset
|
|
);
|
|
TableOffset += sizeof (*RelocRoot);
|
|
|
|
Overflow = BaseOverflowAddU32 (
|
|
RelocTarget,
|
|
RelocRoot->FirstOffset,
|
|
&RelocTarget
|
|
);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
Status = UnchainReloc (
|
|
RuntimeContext,
|
|
(CONST UINT8 *)RelocRoot,
|
|
sizeof (*RelocRoot),
|
|
IsRuntime,
|
|
0,
|
|
NULL
|
|
);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
//
|
|
// Process all relocation fixups of the current root.
|
|
//
|
|
while (TRUE) {
|
|
FixupInfo = *(CONST UINT16 *)((CONST UINT8 *)RelocTable + TableOffset);
|
|
//
|
|
// This cannot overflow due to the upper bound of TableOffset.
|
|
//
|
|
TableOffset += sizeof (*RelocRoot->Heads);
|
|
//
|
|
// Apply the image relocation fixup.
|
|
//
|
|
RelocType = UE_RELOC_FIXUP_TYPE (FixupInfo);
|
|
|
|
if (Chaining && !IsRuntime && (RelocType != UeReloc32NoMeta)) {
|
|
Status = InternalProcessRelocChain (
|
|
Image,
|
|
ImageSize,
|
|
Machine,
|
|
RelocType,
|
|
&RelocTarget,
|
|
Adjust,
|
|
RuntimeContext
|
|
);
|
|
} else {
|
|
Status = InternalApplyRelocation (
|
|
Image,
|
|
ImageSize,
|
|
Machine,
|
|
RelocType,
|
|
&RelocTarget,
|
|
Adjust,
|
|
IsRuntime ? FixupPointer : &FixupData,
|
|
IsRuntime
|
|
);
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
Status = UnchainReloc (
|
|
RuntimeContext,
|
|
(CONST UINT8 *)&FixupInfo,
|
|
sizeof (FixupInfo),
|
|
IsRuntime,
|
|
0,
|
|
&FixupData
|
|
);
|
|
|
|
if (IsRuntime) {
|
|
++FixupPointer;
|
|
}
|
|
}
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
RelocOffset = UE_RELOC_FIXUP_OFFSET (FixupInfo);
|
|
|
|
Status = UnchainReloc (
|
|
RuntimeContext,
|
|
NULL,
|
|
0,
|
|
IsRuntime,
|
|
RelocOffset,
|
|
NULL
|
|
);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
if (RelocOffset == UE_HEAD_FIXUP_OFFSET_END) {
|
|
break;
|
|
}
|
|
//
|
|
// It holds that ImageSize mod 4 KiB = 0, thus ImageSize <= 0xFFFFF000.
|
|
// Furthermore, it holds that RelocTarget <= ImageSize.
|
|
// Finally, it holds that RelocOffset <= 0xFFE.
|
|
// It follows that this cannot overflow.
|
|
//
|
|
RelocTarget += RelocOffset;
|
|
ASSERT (RelocOffset <= RelocTarget);
|
|
|
|
if (TableOffset > EntryOffsetMax) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
}
|
|
//
|
|
// This cannot overflow due to the TableOffset upper bounds and the
|
|
// alignment guarantee of RelocTableSize.
|
|
//
|
|
OldTableOffset = TableOffset;
|
|
TableOffset = ALIGN_VALUE (TableOffset, ALIGNOF (UE_FIXUP_ROOT));
|
|
ASSERT (OldTableOffset <= TableOffset);
|
|
}
|
|
|
|
STATIC_ASSERT (
|
|
sizeof (UE_FIXUP_ROOT) <= UE_LOAD_TABLE_ALIGNMENT,
|
|
"The following ASSERT may not hold."
|
|
);
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeRelocateImage (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
IN UINT64 BaseAddress,
|
|
OUT UE_LOADER_RUNTIME_CONTEXT *RuntimeContext OPTIONAL,
|
|
IN UINT32 RuntimeContextSize
|
|
)
|
|
{
|
|
CONST UE_HEADER *UeHdr;
|
|
BOOLEAN Chaining;
|
|
CONST VOID *RelocTable;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (IS_ALIGNED (Context->LoadTablesFileOffset, UE_LOAD_TABLE_ALIGNMENT));
|
|
ASSERT (IS_ALIGNED (Context->RelocTableSize, UE_LOAD_TABLE_ALIGNMENT));
|
|
ASSERT (IS_ALIGNED (BaseAddress, Context->SegmentAlignment));
|
|
ASSERT (RuntimeContext != NULL || RuntimeContextSize == 0);
|
|
ASSERT (RuntimeContextSize == 0 || sizeof (*RuntimeContext) <= RuntimeContextSize);
|
|
|
|
RelocTable = Context->FileBuffer + Context->LoadTablesFileOffset;
|
|
|
|
UeHdr = (CONST UE_HEADER *)Context->FileBuffer;
|
|
Chaining = (UeHdr->ImageInfo & UE_HEADER_IMAGE_INFO_CHAINED_FIXUPS) != 0;
|
|
|
|
if (RuntimeContext != NULL) {
|
|
RuntimeContext->Machine = Context->Machine;
|
|
}
|
|
|
|
if (Context->XIP) {
|
|
Context->EntryPointAddress -= Context->SegmentsFileOffset;
|
|
}
|
|
|
|
return InternaRelocateImage (
|
|
Context->ImageBuffer,
|
|
Context->ImageSize,
|
|
Context->Machine,
|
|
Context->XIP ? (Context->BaseAddress + Context->SegmentsFileOffset) : Context->BaseAddress,
|
|
RelocTable,
|
|
Context->RelocTableSize,
|
|
Chaining,
|
|
BaseAddress,
|
|
RuntimeContext,
|
|
FALSE
|
|
);
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeRelocateImageForRuntime (
|
|
IN OUT VOID *Image,
|
|
IN UINT32 ImageSize,
|
|
IN CONST UE_LOADER_RUNTIME_CONTEXT *RuntimeContext,
|
|
IN UINT64 BaseAddress
|
|
)
|
|
{
|
|
ASSERT (RuntimeContext != NULL);
|
|
|
|
return InternaRelocateImage (
|
|
Image,
|
|
ImageSize,
|
|
RuntimeContext->Machine,
|
|
(UINTN)Image,
|
|
RuntimeContext->UnchainedRelocs,
|
|
RuntimeContext->UnchainedRelocsSize,
|
|
FALSE,
|
|
BaseAddress,
|
|
(UE_LOADER_RUNTIME_CONTEXT *)RuntimeContext,
|
|
TRUE
|
|
);
|
|
}
|
|
|
|
STATIC
|
|
RETURN_STATUS
|
|
InternalGetDebugTable (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT CONST UE_DEBUG_TABLE **DebugTable
|
|
)
|
|
{
|
|
BOOLEAN Overflow;
|
|
CONST UE_LOAD_TABLE *LoadTable;
|
|
UINT8 LoadTableIndex;
|
|
UINT8 NumLoadTables;
|
|
UINT32 LoadTableFileOffset;
|
|
UINT32 LoadTableFileSize;
|
|
CONST UE_DEBUG_TABLE *DbgTable;
|
|
CONST UE_SEGMENT *Segments;
|
|
UINT16 NumSegments;
|
|
UINT32 LoadTableExtraSize;
|
|
UINT32 MinLoadTableExtraSize;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (DebugTable != NULL);
|
|
|
|
LoadTableFileOffset = Context->LoadTablesFileOffset;
|
|
NumLoadTables = Context->NumLoadTables;
|
|
|
|
for (LoadTableIndex = 0; LoadTableIndex < NumLoadTables; ++LoadTableIndex) {
|
|
LoadTable = Context->LoadTables + LoadTableIndex;
|
|
if (UE_LOAD_TABLE_ID (LoadTable->FileInfo) != UeLoadTableIdDebug) {
|
|
LoadTableFileOffset += UE_LOAD_TABLE_SIZE (LoadTable->FileInfo);
|
|
continue;
|
|
}
|
|
|
|
ASSERT (IS_ALIGNED (LoadTableFileOffset, ALIGNOF (UE_DEBUG_TABLE)));
|
|
|
|
LoadTableFileSize = UE_LOAD_TABLE_SIZE (LoadTable->FileInfo);
|
|
|
|
Overflow = BaseOverflowSubU32 (
|
|
LoadTableFileSize,
|
|
MIN_SIZE_OF_UE_DEBUG_TABLE,
|
|
&LoadTableExtraSize
|
|
);
|
|
if (Overflow) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
DbgTable = (CONST UE_DEBUG_TABLE *)(CONST VOID *)(
|
|
Context->FileBuffer + LoadTableFileOffset
|
|
);
|
|
|
|
NumSegments = UeGetSegments (Context, &Segments);
|
|
|
|
MinLoadTableExtraSize = DbgTable->SymbolsPathLength +
|
|
(UINT32)NumSegments * sizeof (UE_SEGMENT_NAME);
|
|
if (MinLoadTableExtraSize > LoadTableExtraSize) {
|
|
DEBUG_RAISE ();
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
*DebugTable = DbgTable;
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
return RETURN_NOT_FOUND;
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeGetSymbolsPath (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT CONST CHAR8 **SymbolsPath,
|
|
OUT UINT32 *SymbolsPathSize
|
|
)
|
|
{
|
|
RETURN_STATUS Status;
|
|
CONST UE_DEBUG_TABLE *DebugTable;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SymbolsPath != NULL);
|
|
ASSERT (SymbolsPathSize != NULL);
|
|
|
|
Status = InternalGetDebugTable (Context, &DebugTable);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
if (DebugTable->SymbolsPathLength == 0) {
|
|
return RETURN_NOT_FOUND;
|
|
}
|
|
|
|
if (DebugTable->SymbolsPath[DebugTable->SymbolsPathLength] != 0) {
|
|
return RETURN_VOLUME_CORRUPTED;
|
|
}
|
|
|
|
*SymbolsPath = (CONST CHAR8 *)DebugTable->SymbolsPath;
|
|
*SymbolsPathSize = (UINT32)DebugTable->SymbolsPathLength + 1;
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
UINTN
|
|
UeLoaderGetImageDebugAddress (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
RETURN_STATUS Status;
|
|
CONST UE_DEBUG_TABLE *DebugTable;
|
|
UINT8 SymOffsetFactor;
|
|
UINT32 SymOffsetSubtrahend;
|
|
|
|
ASSERT (Context != NULL);
|
|
|
|
SymOffsetSubtrahend = 0;
|
|
|
|
Status = InternalGetDebugTable (Context, &DebugTable);
|
|
if (!RETURN_ERROR (Status)) {
|
|
SymOffsetFactor = UE_DEBUG_TABLE_IMAGE_INFO_SYM_SUBTRAHEND_FACTOR (
|
|
DebugTable->ImageInfo
|
|
);
|
|
SymOffsetSubtrahend = (UINT32)SymOffsetFactor * Context->SegmentAlignment;
|
|
}
|
|
|
|
return UeLoaderGetImageAddress (Context) - SymOffsetSubtrahend;
|
|
}
|
|
|
|
RETURN_STATUS
|
|
UeGetSegmentNames (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT CONST UE_SEGMENT_NAME **SegmentNames
|
|
)
|
|
{
|
|
RETURN_STATUS Status;
|
|
CONST UE_DEBUG_TABLE *DebugTable;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SegmentNames != NULL);
|
|
|
|
Status = InternalGetDebugTable (Context, &DebugTable);
|
|
if (RETURN_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
*SegmentNames = UE_DEBUG_TABLE_SEGMENT_NAMES (DebugTable);
|
|
|
|
return RETURN_SUCCESS;
|
|
}
|
|
|
|
UINT32
|
|
UeGetEntryPointAddress (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->EntryPointAddress;
|
|
}
|
|
|
|
UINT16
|
|
UeGetMachine (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->Machine;
|
|
}
|
|
|
|
UINT16
|
|
UeGetSubsystem (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->Subsystem;
|
|
}
|
|
|
|
UINT32
|
|
UeGetSegmentAlignment (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->SegmentAlignment;
|
|
}
|
|
|
|
UINT32
|
|
UeGetImageSize (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->ImageSize;
|
|
}
|
|
|
|
UINT64
|
|
UeGetBaseAddress (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->BaseAddress;
|
|
}
|
|
|
|
BOOLEAN
|
|
UeGetRelocsStripped (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->RelocsStripped;
|
|
}
|
|
|
|
BOOLEAN
|
|
UeGetFixedAddress(
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return Context->FixedAddress;
|
|
}
|
|
|
|
UINTN
|
|
UeLoaderGetImageAddress (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
|
|
return (UINTN) Context->ImageBuffer;
|
|
}
|
|
|
|
UINT16
|
|
UeGetSegments (
|
|
IN CONST UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT CONST UE_SEGMENT **Segments
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
ASSERT (Segments != NULL);
|
|
|
|
*Segments = Context->Segments;
|
|
return (UINT16)Context->LastSegmentIndex + 1;
|
|
}
|
|
|
|
UINT16
|
|
UeGetSegmentImageInfos (
|
|
IN OUT UE_LOADER_IMAGE_CONTEXT *Context,
|
|
OUT CONST UINT32 **SegmentImageInfos,
|
|
OUT UINT8 *SegmentImageInfoIterSize
|
|
)
|
|
{
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SegmentImageInfos != NULL);
|
|
ASSERT (SegmentImageInfoIterSize != NULL);
|
|
|
|
ASSERT (IS_ALIGNED (Context->SegmentImageInfoIterSize, ALIGNOF (UINT32)));
|
|
|
|
*SegmentImageInfos = Context->Segments;
|
|
*SegmentImageInfoIterSize = Context->SegmentImageInfoIterSize;
|
|
return (UINT16)Context->LastSegmentIndex + 1;
|
|
}
|