mirror of https://github.com/acidanthera/audk.git
UefiPayloadPkg/PayloadLoader: Fix bug in locating relocation section
Per ELF spec, the DT_REL/DT_RELA tag in dynamic section stores the virtual address of the relocation section. But today's code logic treats it as the section offset and finds the relocation section whose offset equals to DT_REL/DT_RELA. The logic can work when the section offset equals to the section virtual address. But when the ELF is generated from the link script that reserves a sizeof(pe_header) in the file beginning, the section offset doesn't equal to section virtual address. Such logic can not find the relocation section. The patch fixes this bug. Signed-off-by: Ray Ni <ray.ni@intel.com> Cc: Maurice Ma <maurice.ma@intel.com> Reviewed-by: Guo Dong <guo.dong@intel.com> Cc: Benjamin You <benjamin.you@intel.com>
This commit is contained in:
parent
580b11201e
commit
939ed3a592
|
@ -206,7 +206,7 @@ RelocateElf32Dynamic (
|
||||||
Elf32_Shdr *DynShdr;
|
Elf32_Shdr *DynShdr;
|
||||||
Elf32_Shdr *RelShdr;
|
Elf32_Shdr *RelShdr;
|
||||||
Elf32_Dyn *Dyn;
|
Elf32_Dyn *Dyn;
|
||||||
UINT32 RelaOffset;
|
UINT32 RelaAddress;
|
||||||
UINT32 RelaCount;
|
UINT32 RelaCount;
|
||||||
UINT32 RelaSize;
|
UINT32 RelaSize;
|
||||||
UINT32 RelaEntrySize;
|
UINT32 RelaEntrySize;
|
||||||
|
@ -246,7 +246,7 @@ RelocateElf32Dynamic (
|
||||||
//
|
//
|
||||||
// 2. Locate the relocation section from the dynamic section.
|
// 2. Locate the relocation section from the dynamic section.
|
||||||
//
|
//
|
||||||
RelaOffset = MAX_UINT32;
|
RelaAddress = MAX_UINT32;
|
||||||
RelaSize = 0;
|
RelaSize = 0;
|
||||||
RelaCount = 0;
|
RelaCount = 0;
|
||||||
RelaEntrySize = 0;
|
RelaEntrySize = 0;
|
||||||
|
@ -265,7 +265,7 @@ RelocateElf32Dynamic (
|
||||||
// based on the original file value and the memory base address.
|
// based on the original file value and the memory base address.
|
||||||
// For consistency, files do not contain relocation entries to ``correct'' addresses in the dynamic structure.
|
// For consistency, files do not contain relocation entries to ``correct'' addresses in the dynamic structure.
|
||||||
//
|
//
|
||||||
RelaOffset = Dyn->d_un.d_ptr - (UINT32) (UINTN) ElfCt->PreferredImageAddress;
|
RelaAddress = Dyn->d_un.d_ptr;
|
||||||
RelaType = (Dyn->d_tag == DT_RELA) ? SHT_RELA: SHT_REL;
|
RelaType = (Dyn->d_tag == DT_RELA) ? SHT_RELA: SHT_REL;
|
||||||
break;
|
break;
|
||||||
case DT_RELACOUNT:
|
case DT_RELACOUNT:
|
||||||
|
@ -285,7 +285,7 @@ RelocateElf32Dynamic (
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (RelaOffset == MAX_UINT64) {
|
if (RelaAddress == MAX_UINT64) {
|
||||||
ASSERT (RelaCount == 0);
|
ASSERT (RelaCount == 0);
|
||||||
ASSERT (RelaEntrySize == 0);
|
ASSERT (RelaEntrySize == 0);
|
||||||
ASSERT (RelaSize == 0);
|
ASSERT (RelaSize == 0);
|
||||||
|
@ -298,8 +298,16 @@ RelocateElf32Dynamic (
|
||||||
//
|
//
|
||||||
// Verify the existence of the relocation section.
|
// Verify the existence of the relocation section.
|
||||||
//
|
//
|
||||||
RelShdr = GetElf32SectionByRange (ElfCt->FileBase, RelaOffset, RelaSize);
|
RelShdr = NULL;
|
||||||
|
for (Index = 0; Index < ElfCt->ShNum; Index++) {
|
||||||
|
RelShdr = GetElf32SectionByIndex (ElfCt->FileBase, Index);
|
||||||
ASSERT (RelShdr != NULL);
|
ASSERT (RelShdr != NULL);
|
||||||
|
if ((RelShdr->sh_addr == RelaAddress) && (RelShdr->sh_size == RelaSize)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
RelShdr = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (RelShdr == NULL) {
|
if (RelShdr == NULL) {
|
||||||
return EFI_UNSUPPORTED;
|
return EFI_UNSUPPORTED;
|
||||||
}
|
}
|
||||||
|
|
|
@ -215,7 +215,7 @@ RelocateElf64Dynamic (
|
||||||
Elf64_Shdr *DynShdr;
|
Elf64_Shdr *DynShdr;
|
||||||
Elf64_Shdr *RelShdr;
|
Elf64_Shdr *RelShdr;
|
||||||
Elf64_Dyn *Dyn;
|
Elf64_Dyn *Dyn;
|
||||||
UINT64 RelaOffset;
|
UINT64 RelaAddress;
|
||||||
UINT64 RelaCount;
|
UINT64 RelaCount;
|
||||||
UINT64 RelaSize;
|
UINT64 RelaSize;
|
||||||
UINT64 RelaEntrySize;
|
UINT64 RelaEntrySize;
|
||||||
|
@ -255,7 +255,7 @@ RelocateElf64Dynamic (
|
||||||
//
|
//
|
||||||
// 2. Locate the relocation section from the dynamic section.
|
// 2. Locate the relocation section from the dynamic section.
|
||||||
//
|
//
|
||||||
RelaOffset = MAX_UINT64;
|
RelaAddress = MAX_UINT64;
|
||||||
RelaSize = 0;
|
RelaSize = 0;
|
||||||
RelaCount = 0;
|
RelaCount = 0;
|
||||||
RelaEntrySize = 0;
|
RelaEntrySize = 0;
|
||||||
|
@ -274,7 +274,7 @@ RelocateElf64Dynamic (
|
||||||
// based on the original file value and the memory base address.
|
// based on the original file value and the memory base address.
|
||||||
// For consistency, files do not contain relocation entries to ``correct'' addresses in the dynamic structure.
|
// For consistency, files do not contain relocation entries to ``correct'' addresses in the dynamic structure.
|
||||||
//
|
//
|
||||||
RelaOffset = Dyn->d_un.d_ptr - (UINTN) ElfCt->PreferredImageAddress;
|
RelaAddress = Dyn->d_un.d_ptr;
|
||||||
RelaType = (Dyn->d_tag == DT_RELA) ? SHT_RELA: SHT_REL;
|
RelaType = (Dyn->d_tag == DT_RELA) ? SHT_RELA: SHT_REL;
|
||||||
break;
|
break;
|
||||||
case DT_RELACOUNT:
|
case DT_RELACOUNT:
|
||||||
|
@ -294,7 +294,7 @@ RelocateElf64Dynamic (
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (RelaOffset == MAX_UINT64) {
|
if (RelaAddress == MAX_UINT64) {
|
||||||
ASSERT (RelaCount == 0);
|
ASSERT (RelaCount == 0);
|
||||||
ASSERT (RelaEntrySize == 0);
|
ASSERT (RelaEntrySize == 0);
|
||||||
ASSERT (RelaSize == 0);
|
ASSERT (RelaSize == 0);
|
||||||
|
@ -307,8 +307,16 @@ RelocateElf64Dynamic (
|
||||||
//
|
//
|
||||||
// Verify the existence of the relocation section.
|
// Verify the existence of the relocation section.
|
||||||
//
|
//
|
||||||
RelShdr = GetElf64SectionByRange (ElfCt->FileBase, RelaOffset, RelaSize);
|
RelShdr = NULL;
|
||||||
|
for (Index = 0; Index < ElfCt->ShNum; Index++) {
|
||||||
|
RelShdr = GetElf64SectionByIndex (ElfCt->FileBase, Index);
|
||||||
ASSERT (RelShdr != NULL);
|
ASSERT (RelShdr != NULL);
|
||||||
|
if ((RelShdr->sh_addr == RelaAddress) && (RelShdr->sh_size == RelaSize)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
RelShdr = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (RelShdr == NULL) {
|
if (RelShdr == NULL) {
|
||||||
return EFI_UNSUPPORTED;
|
return EFI_UNSUPPORTED;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue