Index: lib/ReaderWriter/ELF/ARM/ARMELFFile.h =================================================================== --- lib/ReaderWriter/ELF/ARM/ARMELFFile.h +++ lib/ReaderWriter/ELF/ARM/ARMELFFile.h @@ -51,6 +51,8 @@ }; template class ARMELFFile : public ELFFile { + typedef llvm::object::Elf_Rel_Impl Elf_Rel; + public: ARMELFFile(std::unique_ptr mb, ARMLinkingContext &ctx) : ELFFile(std::move(mb), ctx) {} @@ -61,6 +63,15 @@ new ARMELFFile(std::move(mb), ctx)); } +protected: + /// Returns initial addend; for ARM it is 0, because it is read + /// during the relocations applying + Reference::Addend getInitialAddend(ArrayRef, + uint64_t, + const Elf_Rel&) const override { + return 0; + } + private: typedef llvm::object::Elf_Sym_Impl Elf_Sym; typedef llvm::object::Elf_Shdr_Impl Elf_Shdr; Index: lib/ReaderWriter/ELF/ARM/ARMLinkingContext.h =================================================================== --- lib/ReaderWriter/ELF/ARM/ARMLinkingContext.h +++ lib/ReaderWriter/ELF/ARM/ARMLinkingContext.h @@ -22,6 +22,8 @@ static std::unique_ptr create(llvm::Triple); ARMLinkingContext(llvm::Triple); + bool isRelaOutputFormat() const override { return false; } + void addPasses(PassManager &) override; uint64_t getBaseAddress() const override { @@ -29,6 +31,19 @@ return 0x400000; return _baseAddress; } + + bool isPLTRelocation(const Reference &r) const override { + if (r.kindNamespace() != Reference::KindNamespace::ELF) + return false; + assert(r.kindArch() == Reference::KindArch::ARM); + switch (r.kindValue()) { + case llvm::ELF::R_ARM_JUMP_SLOT: + case llvm::ELF::R_ARM_IRELATIVE: + return true; + default: + return false; + } + } }; } // end namespace elf } // end namespace lld Index: lib/ReaderWriter/ELF/ARM/ARMRelocationHandler.cpp =================================================================== --- lib/ReaderWriter/ELF/ARM/ARMRelocationHandler.cpp +++ lib/ReaderWriter/ELF/ARM/ARMRelocationHandler.cpp @@ -379,6 +379,73 @@ return relocR_ARM_THM_MOV(location, arg); } +template +static void relocR_ARM_ALU_PC_GN_NC(uint8_t* location, uint32_t result) { + assert(lshift < 32 && lshift % 2 == 0); + + static const uint32_t rshift = 32 - lshift; + + result = ((result >> lshift) & 0xFF) | ((rshift / 2) << 8); + + applyArmReloc(location, result, 0xFFF); +} + +/// \brief R_ARM_ALU_PC_G0_NC - ((S + A) | T) - P => S + A - P +static void relocR_ARM_ALU_PC_G0_NC(uint8_t *location, uint64_t P, uint64_t S, + int64_t A) { + int32_t result = (uint32_t)((S + A) - P); + + if (result < 0) + llvm_unreachable("Negative offsets for group relocations has not been implemented"); + + DEBUG_WITH_TYPE( + "ARM", llvm::dbgs() << "\t\tHandle " << LLVM_FUNCTION_NAME << " -"; + llvm::dbgs() << " S: 0x" << Twine::utohexstr(S); + llvm::dbgs() << " A: 0x" << Twine::utohexstr(A); + llvm::dbgs() << " P: 0x" << Twine::utohexstr(P); + llvm::dbgs() << " result: 0x" << Twine::utohexstr((uint32_t)result) << "\n"); + + relocR_ARM_ALU_PC_GN_NC<20>(location, (uint32_t)result); +} + +/// \brief R_ARM_ALU_PC_G1_NC - ((S + A) | T) - P => S + A - P +static void relocR_ARM_ALU_PC_G1_NC(uint8_t *location, uint64_t P, uint64_t S, + int64_t A) { + int32_t result = (uint32_t)((S + A) - P); + + if (result < 0) + llvm_unreachable("Negative offsets for group relocations has not been implemented"); + + DEBUG_WITH_TYPE( + "ARM", llvm::dbgs() << "\t\tHandle " << LLVM_FUNCTION_NAME << " -"; + llvm::dbgs() << " S: 0x" << Twine::utohexstr(S); + llvm::dbgs() << " A: 0x" << Twine::utohexstr(A); + llvm::dbgs() << " P: 0x" << Twine::utohexstr(P); + llvm::dbgs() << " result: 0x" << Twine::utohexstr((uint32_t)result) << "\n"); + + relocR_ARM_ALU_PC_GN_NC<12>(location, (uint32_t)result); +} + +/// \brief R_ARM_LDR_PC_G2 - S + A - P +static void relocR_ARM_LDR_PC_G2(uint8_t *location, uint64_t P, uint64_t S, + int64_t A) { + int32_t result = (uint32_t)((S + A) - P); + + if (result < 0) + llvm_unreachable("Negative offsets for group relocations has not been implemented"); + + const uint32_t mask = 0xFFF; + result = result & mask; + + DEBUG_WITH_TYPE( + "ARM", llvm::dbgs() << "\t\tHandle " << LLVM_FUNCTION_NAME << " -"; + llvm::dbgs() << " S: 0x" << Twine::utohexstr(S); + llvm::dbgs() << " A: 0x" << Twine::utohexstr(A); + llvm::dbgs() << " P: 0x" << Twine::utohexstr(P); + llvm::dbgs() << " result: 0x" << Twine::utohexstr((uint32_t)result) << "\n"); + applyArmReloc(location, (uint32_t)result, mask); +} + std::error_code ARMTargetRelocationHandler::applyRelocation( ELFWriter &writer, llvm::FileOutputBuffer &buf, const lld::AtomLayout &atom, const Reference &ref) const { @@ -393,7 +460,7 @@ // Calculate proper initial addend for the relocation const Reference::Addend addend = - readAddend(location, ref.kindValue()); + readAddend(location, ref.kindValue()) + ref.addend(); // Flags that the relocation addresses Thumb instruction bool addressesThumb = false; @@ -451,6 +518,17 @@ case R_ARM_PREL31: relocR_ARM_PREL31(location, relocVAddress, targetVAddress, addend, addressesThumb); + case R_ARM_ALU_PC_G0_NC: + relocR_ARM_ALU_PC_G0_NC(location, relocVAddress, targetVAddress, addend); + break; + case R_ARM_ALU_PC_G1_NC: + relocR_ARM_ALU_PC_G1_NC(location, relocVAddress, targetVAddress, addend); + break; + case R_ARM_LDR_PC_G2: + relocR_ARM_LDR_PC_G2(location, relocVAddress, targetVAddress, addend); + break; + case R_ARM_IRELATIVE: + // Runtime only relocations. Ignore here. break; default: return make_unhandled_reloc_error(); Index: lib/ReaderWriter/ELF/ARM/ARMRelocationPass.cpp =================================================================== --- lib/ReaderWriter/ELF/ARM/ARMRelocationPass.cpp +++ lib/ReaderWriter/ELF/ARM/ARMRelocationPass.cpp @@ -32,7 +32,7 @@ // ARM B/BL instructions of static relocation veneer. // TODO: consider different instruction set for archs below ARMv5 // (one as for Thumb may be used though it's less optimal). -const uint8_t Veneer_ARM_B_BL_StaticAtomContent[8] = { +static const uint8_t Veneer_ARM_B_BL_StaticAtomContent[8] = { 0x04, 0xf0, 0x1f, 0xe5, // ldr pc, [pc, #-4] 0x00, 0x00, 0x00, 0x00 // }; @@ -40,12 +40,26 @@ // Thumb B/BL instructions of static relocation veneer. // TODO: consider different instruction set for archs above ARMv5 // (one as for ARM may be used since it's more optimal). -const uint8_t Veneer_THM_B_BL_StaticAtomContent[8] = { +static const uint8_t Veneer_THM_B_BL_StaticAtomContent[8] = { 0x78, 0x47, // bx pc 0x00, 0x00, // nop 0xfe, 0xff, 0xff, 0xea // b }; +// .got values +static const uint8_t ARMGotAtomContent[4] = {0, 0, 0, 0}; + +// .plt values (other entries) +static const uint8_t ARMPltAtomContent[12] = { + 0x00, 0xc0, 0x8f, + 0xe2, // add ip, pc, #offset[G0] + 0x00, 0xc0, 0x8c, + 0xe2, // add ip, ip, #offset[G1] + + 0x00, 0xf0, 0xbc, + 0xe5, // ldr pc, [ip, #offset[G2]]! +}; + /// \brief Atoms that hold veneer code. class VeneerAtom : public SimpleELFDefinedAtom { StringRef _section; @@ -104,6 +118,28 @@ } }; +/// \brief Atoms that are used by ARM dynamic linking +class ARMGOTAtom : public GOTAtom { +public: + ARMGOTAtom(const File &f, StringRef secName) : GOTAtom(f, secName) {} + + ArrayRef rawContent() const override { + return llvm::makeArrayRef(ARMGotAtomContent); + } + + Alignment alignment() const override { return Alignment(2); } +}; + +class ARMPLTAtom : public PLTAtom { +public: + ARMPLTAtom(const File &f, StringRef secName) : PLTAtom(f, secName) {} + + + ArrayRef rawContent() const override { + return llvm::makeArrayRef(ARMPltAtomContent); + } +}; + class ELFPassFile : public SimpleFile { public: ELFPassFile(const ELFLinkingContext &eti) : SimpleFile("ELFPassFile") { @@ -125,6 +161,15 @@ return; assert(ref.kindArch() == Reference::KindArch::ARM); switch (ref.kindValue()) { + case R_ARM_ABS32: + case R_ARM_REL32: + case R_ARM_TARGET1: + case R_ARM_THM_CALL: + case R_ARM_CALL: + case R_ARM_THM_MOVW_ABS_NC: + case R_ARM_THM_MOVT_ABS: + static_cast(this)->handleIFUNC(ref); + break; case R_ARM_JUMP24: case R_ARM_THM_JUMP24: static_cast(this)->handleVeneer(atom, ref); @@ -183,6 +228,64 @@ return std::error_code(); } + /// \brief Create a PLT entry referencing PLTGOT entry. + /// + /// The function creates the PLT entry object and passes ownership + /// over it to the caller. + PLTAtom *createPLTforGOT(const GOTAtom *ga) { + auto pa = new (_file._alloc) ARMPLTAtom(_file, ".plt"); + pa->addReferenceELF_ARM(R_ARM_ALU_PC_G0_NC, 0, ga, -8); + pa->addReferenceELF_ARM(R_ARM_ALU_PC_G1_NC, 4, ga, -4); + pa->addReferenceELF_ARM(R_ARM_LDR_PC_G2, 8, ga, 0); + return pa; + } + + /// \brief get the PLT entry for a given IFUNC Atom. + /// + /// If the entry does not exist. Both the GOT and PLT entry is created. + const PLTAtom *getIFUNCPLTEntry(const DefinedAtom *da) { + auto plt = _pltMap.find(da); + if (plt != _pltMap.end()) + return plt->second; + auto ga = new (_file._alloc) ARMGOTAtom(_file, ".got.plt"); + ga->addReferenceELF_ARM(R_ARM_ABS32, 0, da, 0); + ga->addReferenceELF_ARM(R_ARM_IRELATIVE, 0, da, 0); + auto pa = createPLTforGOT(ga); + +#ifndef NDEBUG + ga->_name = "__got_ifunc_"; + ga->_name += da->name(); + pa->_name = "__plt_ifunc_"; + pa->_name += da->name(); +#endif + + _gotMap[da] = ga; + _pltMap[da] = pa; + _gotVector.push_back(ga); + _pltVector.push_back(pa); + return pa; + } + + /// \brief Handle adding of PLT entry by marking the reference + /// as requiring veneer generation. + std::error_code handlePLTEntry(Reference &ref, const PLTAtom *pa) { + ref.setTarget(pa); + + return std::error_code(); + } + + /// \brief Redirect the call to the PLT stub for the target IFUNC. + /// + /// This create a PLT and GOT entry for the IFUNC if one does not exist. The + /// GOT entry and a IRELATIVE relocation to the original target resolver. + std::error_code handleIFUNC(const Reference &ref) { + auto target = dyn_cast_or_null(ref.target()); + if (target && target->contentType() == DefinedAtom::typeResolver) { + return handlePLTEntry(const_cast(ref), + getIFUNCPLTEntry(target)); + } + return std::error_code(); + } public: ARMRelocationPass(const ELFLinkingContext &ctx) : _file(ctx), _ctx(ctx) {} @@ -234,6 +337,14 @@ veneer->setOrdinal(ordinal++); mf->addAtom(*veneer); } + for (auto &plt : _pltVector) { + plt->setOrdinal(ordinal++); + mf->addAtom(*plt); + } + for (auto &got : _gotVector) { + got->setOrdinal(ordinal++); + mf->addAtom(*got); + } } protected: @@ -246,6 +357,15 @@ /// \brief the list of veneer atoms. std::vector _veneerVector; + + /// \brief Map Atoms to their GOT entries. + llvm::DenseMap _gotMap; + + /// \brief Map Atoms to their PLT entries. + llvm::DenseMap _pltMap; + /// \brief the list of GOT/PLT atoms + std::vector _gotVector; + std::vector _pltVector; }; /// This implements the static relocation model. Meaning GOT and PLT entries are Index: lib/ReaderWriter/ELF/ELFFile.h =================================================================== --- lib/ReaderWriter/ELF/ELFFile.h +++ lib/ReaderWriter/ELF/ELFFile.h @@ -315,6 +315,13 @@ return symbol->st_value; } + /// Returns initial addend + virtual Reference::Addend getInitialAddend(ArrayRef symContent, + uint64_t symbolValue, + const Elf_Rel& reference) const { + return *(symContent.data() + reference.r_offset - symbolValue); + } + /// Process the common symbol and create an atom for it. virtual ErrorOr *> handleCommonSymbol(StringRef symName, const Elf_Sym *sym) { @@ -1029,7 +1036,7 @@ auto elfRelocation = new (_readerStorage) ELFReference(rel.r_offset - symValue, kindArch(), rel.getType(isMips64EL), rel.getSymbol(isMips64EL)); - int32_t addend = *(symContent.data() + rel.r_offset - symValue); + Reference::Addend addend = getInitialAddend(symContent, symValue, rel); elfRelocation->setAddend(addend); addReferenceToSymbol(elfRelocation, symbol); _references.push_back(elfRelocation); Index: test/elf/ARM/rel-ifunc.test =================================================================== --- /dev/null +++ test/elf/ARM/rel-ifunc.test @@ -0,0 +1,139 @@ +# Check handling of IFUNC (gnu_indirect_function). +# RUN: yaml2obj -format=elf %s > %t-o.o +# RUN: lld -flavor gnu -target arm -m armelf_linux_eabi -Bstatic \ +# RUN: %t-o.o -o %t +# RUN: llvm-objdump -s -t %t | FileCheck %s + +# CHECK: Contents of section .rel.plt: +# CHECK: 400074 00104000 +# CHECK: Contents of section .plt: +# CHECK: 400080 00c68fe2 00ca8ce2 78ffbce5 +# CHECK-NEXT: Contents of section .text: +# CHECK: 40013c 80b500af fff7ccff fff79cef 00231846 +# CHECK: Contents of section .got.plt: +# CHECK: 401000 ad004000 +# CHECK: SYMBOL TABLE: +# CHECK: 004000ad g .text 00000017 myfunc +# CHECK: 00400074 g *ABS* 00000000 __rel_iplt_start +# CHECK: 0040007c g *ABS* 00000000 __rel_iplt_end + +--- +FileHeader: + Class: ELFCLASS32 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_ARM + Flags: [ EF_ARM_EABI_VER5 ] +Sections: + - Name: .text + Type: SHT_PROGBITS + Flags: [ SHF_ALLOC, SHF_EXECINSTR ] + AddressAlign: 0x0000000000000004 + Content: 80B400AF01231846BD465DF8047B704780B400AF02231846BD465DF8047B704780B400AF40F20003C0F200031846BD465DF8047B704700BF80B582B000AF78607B689847034618460837BD4680BD00BF80B584B000AF40F20003C0F20003FB601BE0FB68BB60BB681B687B60BB685B68DBB23B603B68A02B0CBF01230023DBB2002B07D07B681B681846FFF7D5FF02467B681A60FB680833FB60FA6840F20003C0F200039A42DCD31037BD4680BD00BF80B500AFFFF7CCFFFFF7FEFF0023184680BD00BF + - Name: .rel.text + Type: SHT_REL + Link: .symtab + AddressAlign: 0x0000000000000004 + Info: .text + Relocations: + - Offset: 0x0000000000000024 + Symbol: myfunc1 + Type: R_ARM_THM_MOVW_ABS_NC + - Offset: 0x0000000000000028 + Symbol: myfunc1 + Type: R_ARM_THM_MOVT_ABS + - Offset: 0x0000000000000056 + Symbol: __rel_iplt_start + Type: R_ARM_THM_MOVW_ABS_NC + - Offset: 0x000000000000005A + Symbol: __rel_iplt_start + Type: R_ARM_THM_MOVT_ABS + - Offset: 0x000000000000009C + Symbol: __rel_iplt_end + Type: R_ARM_THM_MOVW_ABS_NC + - Offset: 0x00000000000000A0 + Symbol: __rel_iplt_end + Type: R_ARM_THM_MOVT_ABS + - Offset: 0x00000000000000B8 + Symbol: myfunc + Type: R_ARM_THM_CALL + - Name: .data + Type: SHT_PROGBITS + Flags: [ SHF_WRITE, SHF_ALLOC ] + AddressAlign: 0x0000000000000001 + Content: '' + - Name: .bss + Type: SHT_NOBITS + Flags: [ SHF_WRITE, SHF_ALLOC ] + AddressAlign: 0x0000000000000001 + Content: '' + - Name: .comment + Type: SHT_PROGBITS + Flags: [ SHF_MERGE, SHF_STRINGS ] + AddressAlign: 0x0000000000000001 + Content: 004743433A202863726F7373746F6F6C2D4E47206C696E61726F2D312E31332E312D342E392D323031342E3039202D204C696E61726F2047434320342E392D323031342E30392920342E392E32203230313430393034202870726572656C656173652900 + - Name: .note.GNU-stack + Type: SHT_PROGBITS + AddressAlign: 0x0000000000000001 + Content: '' + - Name: .ARM.attributes + Type: SHT_ARM_ATTRIBUTES + AddressAlign: 0x0000000000000001 + Content: 4134000000616561626900012A00000005372D4100060A0741080109020A041204140115011703180119011A021B031C011E062201 +Symbols: + Local: + - Name: .text + Type: STT_SECTION + Section: .text + - Name: .data + Type: STT_SECTION + Section: .data + - Name: .bss + Type: STT_SECTION + Section: .bss + - Name: '$t' + Section: .text + - Name: elf_ifunc_invoke + Type: STT_FUNC + Section: .text + Value: 0x0000000000000039 + Size: 0x0000000000000016 + - Name: apply_irel + Type: STT_FUNC + Section: .text + Value: 0x0000000000000051 + Size: 0x000000000000005E + - Name: .note.GNU-stack + Type: STT_SECTION + Section: .note.GNU-stack + - Name: .comment + Type: STT_SECTION + Section: .comment + - Name: .ARM.attributes + Type: STT_SECTION + Section: .ARM.attributes + Global: + - Name: myfunc1 + Type: STT_FUNC + Section: .text + Value: 0x0000000000000001 + Size: 0x0000000000000010 + - Name: myfunc2 + Type: STT_FUNC + Section: .text + Value: 0x0000000000000011 + Size: 0x0000000000000010 + - Name: myfunc + Type: STT_GNU_IFUNC + Section: .text + Value: 0x0000000000000021 + Size: 0x0000000000000016 + - Name: _start + Type: STT_FUNC + Section: .text + Value: 0x00000000000000B1 + Size: 0x0000000000000012 + Weak: + - Name: __rel_iplt_start + - Name: __rel_iplt_end +...