[ELF][ARM] Fix crash when discarding InputSections that have .ARM.exidx

When /DISCARD/ is used on an input section, that input section may have
a .ARM.exidx metadata section that depends on it. As the discard handling
comes after the .ARM.exidx synthetic section is created we need to make
sure that we account for the case where the .ARM.exidx output section
should be removed because there are no more live input sections.

Differential Revision: https://reviews.llvm.org/D67848

llvm-svn: 372781
This commit is contained in:
Peter Smith 2019-09-24 21:44:14 +00:00
parent 4278a9e6b5
commit 06b3e3421a
4 changed files with 38 additions and 10 deletions

View File

@ -3160,17 +3160,23 @@ static InputSection *findExidxSection(InputSection *isec) {
return nullptr;
}
static bool isValidExidxSectionDep(InputSection *isec) {
return (isec->flags & SHF_ALLOC) && (isec->flags & SHF_EXECINSTR) &&
isec->getSize() > 0;
}
bool ARMExidxSyntheticSection::addSection(InputSection *isec) {
if (isec->type == SHT_ARM_EXIDX) {
if (InputSection* dep = isec->getLinkOrderDep())
if (isValidExidxSectionDep(dep)) {
exidxSections.push_back(isec);
return true;
}
return false;
}
if ((isec->flags & SHF_ALLOC) && (isec->flags & SHF_EXECINSTR) &&
isec->getSize() > 0) {
if (isValidExidxSectionDep(isec)) {
executableSections.push_back(isec);
if (empty && findExidxSection(isec))
empty = false;
return false;
}
@ -3337,6 +3343,12 @@ void ARMExidxSyntheticSection::writeTo(uint8_t *buf) {
assert(size == offset + 8);
}
bool ARMExidxSyntheticSection::isNeeded() const {
return llvm::find_if(exidxSections, [](InputSection *isec) {
return isec->isLive();
}) != exidxSections.end();
}
bool ARMExidxSyntheticSection::classof(const SectionBase *d) {
return d->kind() == InputSectionBase::Synthetic && d->type == SHT_ARM_EXIDX;
}

View File

@ -994,7 +994,7 @@ public:
size_t getSize() const override { return size; }
void writeTo(uint8_t *buf) override;
bool isNeeded() const override { return !empty; }
bool isNeeded() const override;
// Sort and remove duplicate entries.
void finalizeContents() override;
InputSection *getLinkOrderDep() const;
@ -1008,9 +1008,6 @@ public:
private:
size_t size;
// Empty if ExecutableSections contains no dependent .ARM.exidx sections.
bool empty = true;
// Instead of storing pointers to the .ARM.exidx InputSections from
// InputObjects, we store pointers to the executable sections that need
// .ARM.exidx sections. We can then use the dependentSections of these to

View File

@ -0,0 +1,19 @@
// REQUIRES: arm
// RUN: llvm-mc -filetype=obj -triple arm-gnu-linux-eabi -mcpu cortex-a7 -arm-add-build-attributes %s -o %t.o
// RUN: echo "ENTRY(__entrypoint) SECTIONS { /DISCARD/ : { *(.text.1) } }" > %t.script
// RUN: ld.lld -T %t.script %t.o -o %t.elf
// RUN: llvm-readobj --sections %t.elf | FileCheck %s
/// Test that when we /DISCARD/ all the input sections with associated
/// .ARM.exidx sections then we also discard all the .ARM.exidx sections.
.section .text.1, "ax", %progbits
.global foo
.type foo, %function
.fnstart
foo:
bx lr
.cantunwind
.fnend
// CHECK-NOT: .ARM.exidx