Index: ELF/LinkerScript.h =================================================================== --- ELF/LinkerScript.h +++ ELF/LinkerScript.h @@ -10,6 +10,7 @@ #ifndef LLD_ELF_LINKER_SCRIPT_H #define LLD_ELF_LINKER_SCRIPT_H +#include "Symbols.h" #include "lld/Core/LLVM.h" #include "llvm/ADT/DenseMap.h" #include "llvm/ADT/MapVector.h" @@ -18,6 +19,7 @@ namespace lld { namespace elf { +template <class ELFT> class SymbolTable; // Parses a linker script. Calling this function updates // Config and ScriptConfig. @@ -40,7 +42,7 @@ // This enum represents what we can observe in SECTIONS tag of script: // ExprKind is a location counter change, like ". = . + 0x1000" // SectionKind is a description of output section, like ".data :..." -enum SectionsCommandKind { ExprKind, SectionKind }; +enum SectionsCommandKind { ExprKind, SectionKind, SymbolAssignmentKind }; struct SectionsCommand { SectionsCommandKind Kind; @@ -81,6 +83,7 @@ bool shouldKeep(InputSectionBase<ELFT> *S); void assignAddresses(ArrayRef<OutputSectionBase<ELFT> *> S); int compareSections(StringRef A, StringRef B); + void addScriptedSymbols(); private: // "ScriptConfig" is a bit too long, so define a short name for it. @@ -89,6 +92,8 @@ int getSectionIndex(StringRef Name); uintX_t Dot; + + std::vector<DefinedRegular<ELFT> *> Symbols; }; // Variable template is a C++14 feature, so we can't template Index: ELF/LinkerScript.cpp =================================================================== --- ELF/LinkerScript.cpp +++ ELF/LinkerScript.cpp @@ -223,12 +223,19 @@ // Assign addresses as instructed by linker script SECTIONS sub-commands. Dot = Out<ELFT>::ElfHeader->getSize() + Out<ELFT>::ProgramHeaders->getSize(); uintX_t MinVA = std::numeric_limits<uintX_t>::max(); + uintX_t SymIndex = 0; uintX_t ThreadBssOffset = 0; for (SectionsCommand &Cmd : Opt.Commands) { - if (Cmd.Kind == ExprKind) { + switch (Cmd.Kind) { + case ExprKind: Dot = evalExpr(Cmd.Expr, Dot); continue; + case SymbolAssignmentKind: + Symbols[SymIndex++]->Value = evalExpr(Cmd.Expr, Dot); + continue; + default: + break; } // Find all the sections with required name. There can be more than @@ -299,6 +306,17 @@ return I < J ? -1 : 1; } +template <class ELFT> +void LinkerScript<ELFT>::addScriptedSymbols() { + for (SectionsCommand &Cmd : Opt.Commands) { + if (Cmd.Kind == SymbolAssignmentKind) { + DefinedRegular<ELFT> *D = + Symtab<ELFT>::X->addAbsolute(Cmd.SectionName, STV_DEFAULT); + Symbols.push_back(D); + } + } +} + class elf::ScriptParser : public ScriptParserBase { typedef void (ScriptParser::*Handler)(); @@ -323,7 +341,9 @@ void readSections(); void readLocationCounterValue(); - void readOutputSectionDescription(); + void readOutputSectionDescription(StringRef OutSec); + void readSymbolAssignment(StringRef Name); + void readSectionsCommandExpr(); const static StringMap<Handler> Cmd; ScriptConfiguration &Opt = *ScriptConfig; @@ -487,10 +507,15 @@ expect("{"); while (!Error && !skip("}")) { StringRef Tok = peek(); - if (Tok == ".") + if (Tok == ".") { readLocationCounterValue(); + continue; + } + next(); + if (peek() == "=") + readSymbolAssignment(Tok); else - readOutputSectionDescription(); + readOutputSectionDescription(Tok); } } @@ -498,19 +523,10 @@ expect("."); expect("="); Opt.Commands.push_back({ExprKind, {}, ""}); - SectionsCommand &Cmd = Opt.Commands.back(); - while (!Error) { - StringRef Tok = next(); - if (Tok == ";") - break; - Cmd.Expr.push_back(Tok); - } - if (Cmd.Expr.empty()) - error("error in location counter expression"); + readSectionsCommandExpr(); } -void ScriptParser::readOutputSectionDescription() { - StringRef OutSec = next(); +void ScriptParser::readOutputSectionDescription(StringRef OutSec) { Opt.Commands.push_back({SectionKind, {}, OutSec}); expect(":"); expect("{"); @@ -548,6 +564,24 @@ } } +void ScriptParser::readSymbolAssignment(StringRef Name) { + expect("="); + Opt.Commands.push_back({SymbolAssignmentKind, {}, Name}); + readSectionsCommandExpr(); +} + +void ScriptParser::readSectionsCommandExpr() { + SectionsCommand &Cmd = Opt.Commands.back(); + while (!Error) { + StringRef Tok = next(); + if (Tok == ";") + break; + Cmd.Expr.push_back(Tok); + } + if (Cmd.Expr.empty()) + error("error in location counter expression"); +} + static bool isUnderSysroot(StringRef Path) { if (Config->Sysroot == "") return false; Index: ELF/Writer.cpp =================================================================== --- ELF/Writer.cpp +++ ELF/Writer.cpp @@ -785,6 +785,9 @@ // Define __rel[a]_iplt_{start,end} symbols if needed. addRelIpltSymbols(); + // Add scripted symbols with zero values now. + // Real values will be assigned later + Script<ELFT>::X->addScriptedSymbols(); if (!Out<ELFT>::EhFrame->empty()) { OutputSections.push_back(Out<ELFT>::EhFrame); Index: test/ELF/linkerscript-symbols.s =================================================================== --- test/ELF/linkerscript-symbols.s +++ test/ELF/linkerscript-symbols.s @@ -0,0 +1,12 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %s -o %t + +# RUN: echo "SECTIONS {.text : {*(.text.*)} text_end = .;}" > %t.script +# RUN: ld.lld -o %t1 --script %t.script %t +# RUN: llvm-objdump -t %t1 | FileCheck %s +# CHECK: 0000000000000121 *ABS* 00000000 text_end + +.global _start +_start: + nop +