Index: ELF/LinkerScript.h =================================================================== --- ELF/LinkerScript.h +++ ELF/LinkerScript.h @@ -114,6 +114,13 @@ unsigned Flags; }; +class ScriptMethods { +public: + 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. @@ -129,12 +136,14 @@ // List of section patterns specified with KEEP commands. They will // be kept even if they are unused and --gc-sections is specified. std::vector KeptSections; + + ScriptMethods* Methods; }; extern ScriptConfiguration *ScriptConfig; // This is a runner of the linker script. -template class LinkerScript { +template class LinkerScript final : public ScriptMethods { typedef typename ELFT::uint uintX_t; public: @@ -151,8 +160,10 @@ void assignAddresses(); int compareSections(StringRef A, StringRef B); bool hasPhdrsCommands(); - uintX_t getOutputSectionSize(StringRef Name); - uintX_t getHeaderSize(); + + uint64_t getOutputSectionSize(StringRef Name) override; + uint64_t getHeaderSize() override; + uint64_t getSymbolValue(StringRef S) override; std::vector *> *OutputSections; Index: ELF/LinkerScript.cpp =================================================================== --- ELF/LinkerScript.cpp +++ ELF/LinkerScript.cpp @@ -87,8 +87,10 @@ return !S || !S->Live; } -template LinkerScript::LinkerScript() {} -template LinkerScript::~LinkerScript() {} +template LinkerScript::LinkerScript() { Opt.Methods = this; } +template LinkerScript::~LinkerScript() { + Opt.Methods = nullptr; +} template bool LinkerScript::shouldKeep(InputSectionBase *S) { @@ -533,7 +535,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(); @@ -541,11 +543,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. @@ -1004,59 +1013,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 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 ScriptConfig->Methods->getSymbolValue(S); } SymbolAssignment *ScriptParser::readAssignment(StringRef Name) { @@ -1204,10 +1161,12 @@ expect("("); StringRef Name = next(); expect(")"); - return [=](uint64_t Dot) { return getSectionSize(Name); }; + return [=](uint64_t Dot) { + return ScriptConfig->Methods->getOutputSectionSize(Name); + }; } if (Tok == "SIZEOF_HEADERS") - return [=](uint64_t Dot) { return getHeaderSize(); }; + return [=](uint64_t Dot) { return ScriptConfig->Methods->getHeaderSize(); }; // Parse a symbol name or a number literal. uint64_t V = 0;