Index: ELF/Driver.cpp =================================================================== --- ELF/Driver.cpp +++ ELF/Driver.cpp @@ -569,7 +569,7 @@ std::unique_ptr TI(createTarget()); Target = TI.get(); LinkerScript LS; - Script::X = &LS; + ScriptBase = Script::X = &LS; Config->Rela = ELFT::Is64Bits || Config->EMachine == EM_X86_64; Config->Mips64EL = Index: ELF/LinkerScript.h =================================================================== --- ELF/LinkerScript.h +++ ELF/LinkerScript.h @@ -116,6 +116,14 @@ unsigned Flags; }; +class LinkerScriptBase { +public: + virtual uint64_t getOutputSectionAddress(StringRef Name) = 0; + virtual uint64_t getOutputSectionSize(StringRef Name) = 0; + virtual uint64_t getHeaderSize() = 0; + virtual uint64_t getSymbolValue(StringRef S) = 0; +}; + // ScriptConfiguration holds linker script parse results. struct ScriptConfiguration { // Used to assign addresses to sections. @@ -136,7 +144,7 @@ extern ScriptConfiguration *ScriptConfig; // This is a runner of the linker script. -template class LinkerScript { +template class LinkerScript final : public LinkerScriptBase { typedef typename ELFT::uint uintX_t; public: @@ -153,9 +161,10 @@ void assignAddresses(); int compareSections(StringRef A, StringRef B); bool hasPhdrsCommands(); - uintX_t getOutputSectionAddress(StringRef Name); - uintX_t getOutputSectionSize(StringRef Name); - uintX_t getHeaderSize(); + uint64_t getOutputSectionAddress(StringRef Name) override; + uint64_t getOutputSectionSize(StringRef Name) override; + uint64_t getHeaderSize() override; + uint64_t getSymbolValue(StringRef S) override; std::vector *> *OutputSections; @@ -182,6 +191,7 @@ // Variable template is a C++14 feature, so we can't template // a global variable. Use a struct to workaround. +extern LinkerScriptBase* ScriptBase; template struct Script { static LinkerScript *X; }; template LinkerScript *Script::X; Index: ELF/LinkerScript.cpp =================================================================== --- ELF/LinkerScript.cpp +++ ELF/LinkerScript.cpp @@ -41,6 +41,7 @@ using namespace lld; using namespace lld::elf; +LinkerScriptBase* elf::ScriptBase; ScriptConfiguration *elf::ScriptConfig; template @@ -566,8 +567,7 @@ } template -typename ELFT::uint -LinkerScript::getOutputSectionAddress(StringRef Name) { +uint64_t LinkerScript::getOutputSectionAddress(StringRef Name) { for (OutputSectionBase *Sec : *OutputSections) if (Sec->getName() == Name) return Sec->getVA(); @@ -576,7 +576,7 @@ } template -typename ELFT::uint LinkerScript::getOutputSectionSize(StringRef Name) { +uint64_t LinkerScript::getOutputSectionSize(StringRef Name) { for (OutputSectionBase *Sec : *OutputSections) if (Sec->getName() == Name) return Sec->getSize(); @@ -584,11 +584,18 @@ return 0; } -template -typename ELFT::uint LinkerScript::getHeaderSize() { +template uint64_t LinkerScript::getHeaderSize() { return Out::ElfHeader->getSize() + Out::ProgramHeaders->getSize(); } +template +uint64_t LinkerScript::getSymbolValue(StringRef S) { + if (SymbolBody *B = Symtab::X->find(S)) + return B->getVA(); + error("symbol not found: " + S); + return 0; +} + // Returns indices of ELF headers containing specific section, identified // by Name. Each index is a zero based number of ELF header listed within // PHDRS {} script block. @@ -619,6 +626,10 @@ return 0; } +template LinkerScript *getScript() { + return static_cast *>(ScriptBase); +} + class elf::ScriptParser : public ScriptParserBase { typedef void (ScriptParser::*Handler)(); @@ -1075,74 +1086,7 @@ static uint64_t getSymbolValue(StringRef S, uint64_t Dot) { if (S == ".") return Dot; - - switch (Config->EKind) { - case ELF32LEKind: - if (SymbolBody *B = Symtab::X->find(S)) - return B->getVA(); - break; - case ELF32BEKind: - if (SymbolBody *B = Symtab::X->find(S)) - return B->getVA(); - break; - case ELF64LEKind: - if (SymbolBody *B = Symtab::X->find(S)) - return B->getVA(); - break; - case ELF64BEKind: - if (SymbolBody *B = Symtab::X->find(S)) - return B->getVA(); - break; - default: - llvm_unreachable("unsupported target"); - } - error("symbol not found: " + S); - return 0; -} - -static uint64_t getSectionSize(StringRef Name) { - switch (Config->EKind) { - case ELF32LEKind: - return Script::X->getOutputSectionSize(Name); - case ELF32BEKind: - return Script::X->getOutputSectionSize(Name); - case ELF64LEKind: - return Script::X->getOutputSectionSize(Name); - case ELF64BEKind: - return Script::X->getOutputSectionSize(Name); - default: - llvm_unreachable("unsupported target"); - } -} - -static uint64_t getSectionAddress(StringRef Name) { - switch (Config->EKind) { - case ELF32LEKind: - return Script::X->getOutputSectionAddress(Name); - case ELF32BEKind: - return Script::X->getOutputSectionAddress(Name); - case ELF64LEKind: - return Script::X->getOutputSectionAddress(Name); - case ELF64BEKind: - return Script::X->getOutputSectionAddress(Name); - default: - llvm_unreachable("unsupported target"); - } -} - -static uint64_t getHeaderSize() { - switch (Config->EKind) { - case ELF32LEKind: - return Script::X->getHeaderSize(); - case ELF32BEKind: - return Script::X->getHeaderSize(); - case ELF64LEKind: - return Script::X->getHeaderSize(); - case ELF64BEKind: - return Script::X->getHeaderSize(); - default: - llvm_unreachable("unsupported target"); - } + return ScriptBase->getSymbolValue(S); } SymbolAssignment *ScriptParser::readAssignment(StringRef Name) { @@ -1244,7 +1188,8 @@ expect("("); StringRef Name = next(); expect(")"); - return [=](uint64_t Dot) { return getSectionAddress(Name); }; + return + [=](uint64_t Dot) { return ScriptBase->getOutputSectionAddress(Name); }; } if (Tok == "ASSERT") return readAssert(); @@ -1296,10 +1241,10 @@ expect("("); StringRef Name = next(); expect(")"); - return [=](uint64_t Dot) { return getSectionSize(Name); }; + return [=](uint64_t Dot) { return ScriptBase->getOutputSectionSize(Name); }; } if (Tok == "SIZEOF_HEADERS") - return [=](uint64_t Dot) { return getHeaderSize(); }; + return [=](uint64_t Dot) { return ScriptBase->getHeaderSize(); }; // Parse a symbol name or a number literal. uint64_t V = 0;