audk/MdePkg/Library/BasePeCoffLib2/PeCoffDebug.c

265 lines
8.1 KiB
C

/** @file
Implements APIs to load PE/COFF debug information.
Copyright (c) 2020 - 2021, Marvin Häuser. All rights reserved.<BR>
Copyright (c) 2020, Vitaly Cheptsov. All rights reserved.<BR>
Copyright (c) 2020, ISP RAS. All rights reserved.<BR>
Portions copyright (c) 2006 - 2019, Intel Corporation. All rights reserved.<BR>
Portions copyright (c) 2008 - 2010, Apple Inc. All rights reserved.<BR>
Portions copyright (c) 2020, Hewlett Packard Enterprise Development LP. All rights reserved.<BR>
SPDX-License-Identifier: BSD-3-Clause
**/
#include <Base.h>
#include <Uefi/UefiBaseType.h>
#include <IndustryStandard/PeImage2.h>
#include <Library/BaseMemoryLib.h>
#include <Library/BaseOverflowLib.h>
#include <Library/DebugLib.h>
#include <Library/PcdLib.h>
#include <Library/PeCoffLib2.h>
#include "BasePeCoffLib2Internals.h"
RETURN_STATUS
PeCoffGetPdbPath (
IN CONST PE_COFF_LOADER_IMAGE_CONTEXT *Context,
OUT CONST CHAR8 **PdbPath,
OUT UINT32 *PdbPathSize
)
{
BOOLEAN Overflow;
CONST EFI_IMAGE_DATA_DIRECTORY *DebugDir;
CONST EFI_TE_IMAGE_HEADER *TeHdr;
CONST EFI_IMAGE_NT_HEADERS32 *Pe32Hdr;
CONST EFI_IMAGE_NT_HEADERS64 *Pe32PlusHdr;
CONST EFI_IMAGE_DEBUG_DIRECTORY_ENTRY *DebugEntries;
UINT32 NumDebugEntries;
UINT32 DebugIndex;
CONST EFI_IMAGE_DEBUG_DIRECTORY_ENTRY *CodeViewEntry;
UINT32 DebugDirTop;
UINT32 DebugEntryFileOffset;
UINT32 DebugEntryFileOffsetTop;
CONST CHAR8 *CodeView;
UINT32 PdbOffset;
CONST CHAR8 *PdbName;
UINT32 PdbNameSize;
ASSERT (Context != NULL);
ASSERT (PdbPath != NULL);
ASSERT (PdbPathSize != NULL);
if (!PcdGetBool (PcdImageLoaderDebugSupport)) {
return RETURN_NOT_FOUND;
}
//
// Retrieve the Debug Directory of the Image.
//
switch (Context->ImageType) {
case PeCoffLoaderTypeTe:
if (PcdGetBool (PcdImageLoaderProhibitTe)) {
ASSERT (FALSE);
return RETURN_UNSUPPORTED;
}
TeHdr = (CONST EFI_TE_IMAGE_HEADER *) (CONST VOID *) (
(CONST CHAR8 *) Context->FileBuffer
);
DebugDir = &TeHdr->DataDirectory[1];
break;
case PeCoffLoaderTypePe32:
Pe32Hdr = (CONST EFI_IMAGE_NT_HEADERS32 *) (CONST VOID *) (
(CONST CHAR8 *) Context->FileBuffer + Context->ExeHdrOffset
);
if (Pe32Hdr->NumberOfRvaAndSizes <= EFI_IMAGE_DIRECTORY_ENTRY_DEBUG) {
return RETURN_NOT_FOUND;
}
DebugDir = Pe32Hdr->DataDirectory + EFI_IMAGE_DIRECTORY_ENTRY_DEBUG;
break;
case PeCoffLoaderTypePe32Plus:
Pe32PlusHdr = (CONST EFI_IMAGE_NT_HEADERS64 *) (CONST VOID *) (
(CONST CHAR8 *) Context->FileBuffer + Context->ExeHdrOffset
);
if (Pe32PlusHdr->NumberOfRvaAndSizes <= EFI_IMAGE_DIRECTORY_ENTRY_DEBUG) {
return RETURN_NOT_FOUND;
}
DebugDir = Pe32PlusHdr->DataDirectory + EFI_IMAGE_DIRECTORY_ENTRY_DEBUG;
break;
default:
ASSERT (FALSE);
return RETURN_VOLUME_CORRUPTED;
}
//
// Verify the Debug Directory is not empty.
//
if (DebugDir->Size == 0) {
return RETURN_NOT_FOUND;
}
//
// Verify the Debug Directory has a well-formed size.
//
if (DebugDir->Size % sizeof (*DebugEntries) != 0) {
//
// Some Apple-made images contain a sum of EFI_IMAGE_DEBUG_DIRECTORY_ENTRY
// and EFI_IMAGE_DEBUG_CODEVIEW_MTOC_ENTRY sizes in DebugDir size.
// Since this violates the spec and nobody but Apple has access
// to the DEBUG symbols, just ignore this debug information.
//
return RETURN_VOLUME_CORRUPTED;
}
//
// Verify the Debug Directory is in bounds of the Image buffer.
//
Overflow = BaseOverflowAddU32 (
DebugDir->VirtualAddress,
DebugDir->Size,
&DebugDirTop
);
if (Overflow || DebugDirTop > Context->SizeOfImage) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
//
// Verify the Debug Directory Image address is sufficiently aligned.
//
if (!IS_ALIGNED (DebugDir->VirtualAddress, ALIGNOF (EFI_IMAGE_DEBUG_DIRECTORY_ENTRY))) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
DebugEntries = (CONST EFI_IMAGE_DEBUG_DIRECTORY_ENTRY *) (CONST VOID *) (
(CONST CHAR8 *) Context->ImageBuffer + DebugDir->VirtualAddress
);
NumDebugEntries = DebugDir->Size / sizeof (*DebugEntries);
for (DebugIndex = 0; DebugIndex < NumDebugEntries; ++DebugIndex) {
if (DebugEntries[DebugIndex].Type == EFI_IMAGE_DEBUG_TYPE_CODEVIEW) {
break;
}
}
//
// Verify the CodeView entry has been found in the Debug Directory.
//
if (DebugIndex == NumDebugEntries) {
return RETURN_NOT_FOUND;
}
//
// Verify the CodeView entry has sufficient space for the signature.
//
CodeViewEntry = &DebugEntries[DebugIndex];
if (CodeViewEntry->SizeOfData < sizeof (UINT32)) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
DebugEntryFileOffset = CodeViewEntry->FileOffset;
if (!PcdGetBool (PcdImageLoaderProhibitTe)) {
Overflow = BaseOverflowSubU32 (
DebugEntryFileOffset,
Context->TeStrippedOffset,
&DebugEntryFileOffset
);
if (Overflow) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
} else {
ASSERT (Context->TeStrippedOffset == 0);
}
//
// Verify the CodeView entry is in bounds of the raw file, and the
// CodeView entry raw file offset is sufficiently aligned.
//
Overflow = BaseOverflowAddU32 (
DebugEntryFileOffset,
CodeViewEntry->SizeOfData,
&DebugEntryFileOffsetTop
);
if (Overflow || DebugEntryFileOffsetTop > Context->FileSize
|| !IS_ALIGNED (DebugEntryFileOffset, ALIGNOF (UINT32))) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
CodeView = (CONST CHAR8 *) Context->FileBuffer + DebugEntryFileOffset;
//
// This memory access is safe because we know that
// 1) IS_ALIGNED (DebugEntryFileOffset, ALIGNOF (UINT32))
// 2) sizeof (UINT32) <= CodeViewEntry->SizeOfData.
//
switch (*(CONST UINT32 *) (CONST VOID *) CodeView) {
case CODEVIEW_SIGNATURE_NB10:
PdbOffset = sizeof (EFI_IMAGE_DEBUG_CODEVIEW_NB10_ENTRY);
STATIC_ASSERT (
ALIGNOF (EFI_IMAGE_DEBUG_CODEVIEW_NB10_ENTRY) <= ALIGNOF (UINT32),
"The structure may be misalignedd."
);
break;
case CODEVIEW_SIGNATURE_RSDS:
PdbOffset = sizeof (EFI_IMAGE_DEBUG_CODEVIEW_RSDS_ENTRY);
STATIC_ASSERT (
ALIGNOF (EFI_IMAGE_DEBUG_CODEVIEW_RSDS_ENTRY) <= ALIGNOF (UINT32),
"The structure may be misalignedd."
);
break;
case CODEVIEW_SIGNATURE_MTOC:
PdbOffset = sizeof (EFI_IMAGE_DEBUG_CODEVIEW_MTOC_ENTRY);
STATIC_ASSERT (
ALIGNOF (EFI_IMAGE_DEBUG_CODEVIEW_MTOC_ENTRY) <= ALIGNOF (UINT32),
"The structure may be misalignedd."
);
break;
default:
return RETURN_UNSUPPORTED;
}
//
// Verify the PDB path exists and is in bounds of the Image buffer.
//
Overflow = BaseOverflowSubU32 (
CodeViewEntry->SizeOfData,
PdbOffset,
&PdbNameSize
);
if (Overflow || PdbNameSize == 0) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
//
// Verify the PDB path is correctly terminated.
//
PdbName = CodeView + PdbOffset;
if (PdbName[PdbNameSize - 1] != 0) {
DEBUG_RAISE ();
return RETURN_VOLUME_CORRUPTED;
}
*PdbPath = PdbName;
*PdbPathSize = PdbNameSize;
return RETURN_SUCCESS;
}