Skip to content

Commit

Permalink
[LLD][ELF] - Do not remove empty sections referenced in LOADADDR/ADDR…
Browse files Browse the repository at this point in the history
… commands.

This is https://bugs.llvm.org//show_bug.cgi?id=38750.

If script references empty sections in LOADADDR/ADDR commands

.empty  : { *(.empty ) }
.text   : AT(LOADADDR (.empty) + SIZEOF (.empty)) { *(.text) }
then an empty section will be removed and LOADADDR/ADDR will evaluate to null.
It is not that user may expect from using of the generic script, what is a common case.

Differential revision: https://reviews.llvm.org/D54621

llvm-svn: 359279
  • Loading branch information
George Rimar committed Apr 26, 2019
1 parent 2aa0bde commit dee900a
Show file tree
Hide file tree
Showing 4 changed files with 36 additions and 3 deletions.
12 changes: 9 additions & 3 deletions lld/ELF/LinkerScript.cpp
Expand Up @@ -825,12 +825,18 @@ static bool isDiscardable(OutputSection &Sec) {
if (!Sec.Phdrs.empty())
return false;

// We do not want to remove sections that reference symbols in address and
// other expressions. We add script symbols as undefined, and want to ensure
// all of them are defined in the output, hence have to keep them.
// We do not want to remove OutputSections with expressions that reference
// symbols even if the OutputSection is empty. We want to ensure that the
// expressions can be evaluated and report an error if they cannot.
if (Sec.ExpressionsUseSymbols)
return false;

// OutputSections may be referenced by name in ADDR and LOADADDR expressions,
// as an empty Section can has a valid VMA and LMA we keep the OutputSection
// to maintain the integrity of the other Expression.
if (Sec.UsedInExpression)
return false;

for (BaseCommand *Base : Sec.SectionCommands) {
if (auto Cmd = dyn_cast<SymbolAssignment>(Base))
// Don't create empty output sections just for unreferenced PROVIDE
Expand Down
1 change: 1 addition & 0 deletions lld/ELF/OutputSections.h
Expand Up @@ -90,6 +90,7 @@ class OutputSection final : public BaseCommand, public SectionBase {
bool NonAlloc = false;
bool Noload = false;
bool ExpressionsUseSymbols = false;
bool UsedInExpression = false;
bool InOverlay = false;

void finalize();
Expand Down
2 changes: 2 additions & 0 deletions lld/ELF/ScriptParser.cpp
Expand Up @@ -1154,6 +1154,7 @@ Expr ScriptParser::readPrimary() {
if (Tok == "ADDR") {
StringRef Name = readParenLiteral();
OutputSection *Sec = Script->getOrCreateOutputSection(Name);
Sec->UsedInExpression = true;
return [=]() -> ExprValue {
checkIfExists(Sec, Location);
return {Sec, false, 0, Location};
Expand Down Expand Up @@ -1230,6 +1231,7 @@ Expr ScriptParser::readPrimary() {
if (Tok == "LOADADDR") {
StringRef Name = readParenLiteral();
OutputSection *Cmd = Script->getOrCreateOutputSection(Name);
Cmd->UsedInExpression = true;
return [=] {
checkIfExists(Cmd, Location);
return Cmd->getLMA();
Expand Down
24 changes: 24 additions & 0 deletions lld/test/ELF/linkerscript/empty-sections-expressions.test
@@ -0,0 +1,24 @@
# REQUIRES: x86
# RUN: echo ".text; nop; .data; .byte 0" \
# RUN: | llvm-mc -filetype=obj -triple=x86_64-pc-linux - -o %t.o
# RUN: ld.lld -o %t --script %s %t.o
# RUN: llvm-readelf -program-headers %t | FileCheck %s

## Check we do not remove the empty output sections used in LOADADDR/ADDR
## expressions and hence can evaluate the correct addresses.

# CHECK: Program Headers:
# CHECK-NEXT: Type Offset VirtAddr PhysAddr
# CHECK-NEXT: LOAD 0x001000 0x0000000000080000 0x0000000000080000
# CHECK-NEXT: LOAD 0x001001 0x0000000000080001 0x0000000000082000

# CHECK: Section to Segment mapping:
# CHECK: 00 .empty .text
# CHECK-NEXT: 01 .data

SECTIONS {
. = 0x00080000;
.empty : { *(.empty ) }
.text : AT(LOADADDR(.empty) + SIZEOF(.empty)) { *(.text) }
.data : AT(ADDR(.empty) + 0x2000) { *(.data) }
}

0 comments on commit dee900a

Please sign in to comment.