diff --git a/lld/MachO/Arch/X86_64.cpp b/lld/MachO/Arch/X86_64.cpp --- a/lld/MachO/Arch/X86_64.cpp +++ b/lld/MachO/Arch/X86_64.cpp @@ -25,8 +25,9 @@ struct X86_64 : TargetInfo { X86_64(); - uint64_t getImplicitAddend(MemoryBufferRef, const section_64 &, - const relocation_info &) const override; + bool isPairedReloc(const relocation_info) const override; + uint64_t getAddend(MemoryBufferRef, const section_64 &, const relocation_info, + const relocation_info) const override; void relocateOne(uint8_t *loc, const Reloc &, uint64_t val) const override; void writeStub(uint8_t *buf, const macho::Symbol &) const override; @@ -69,8 +70,13 @@ fatal(msg); } -uint64_t X86_64::getImplicitAddend(MemoryBufferRef mb, const section_64 &sec, - const relocation_info &rel) const { +bool X86_64::isPairedReloc(const relocation_info rel) const { + return (rel.r_type == X86_64_RELOC_SUBTRACTOR); +} + +uint64_t X86_64::getAddend(MemoryBufferRef mb, const section_64 &sec, + const relocation_info rel, + const relocation_info pairedRel) const { auto *buf = reinterpret_cast(mb.getBufferStart()); const uint8_t *loc = buf + sec.offset + rel.r_address; @@ -140,7 +146,7 @@ break; default: llvm_unreachable( - "getImplicitAddend should have flagged all unhandled relocation types"); + "getAddend should have flagged all unhandled relocation types"); } switch (r.length) { diff --git a/lld/MachO/InputFiles.cpp b/lld/MachO/InputFiles.cpp --- a/lld/MachO/InputFiles.cpp +++ b/lld/MachO/InputFiles.cpp @@ -172,31 +172,28 @@ void InputFile::parseRelocations(const section_64 &sec, SubsectionMap &subsecMap) { auto *buf = reinterpret_cast(mb.getBufferStart()); - ArrayRef anyRelInfos( - reinterpret_cast(buf + sec.reloff), - sec.nreloc); - - for (const any_relocation_info &anyRelInfo : anyRelInfos) { - if (anyRelInfo.r_word0 & R_SCATTERED) + ArrayRef relInfos( + reinterpret_cast(buf + sec.reloff), sec.nreloc); + + for (size_t i = 0; i < relInfos.size(); i++) { + const relocation_info pairedInfo = relInfos[i]; + const relocation_info relInfo = + (target->isPairedReloc(pairedInfo) ? relInfos[++i] : pairedInfo); + assert(i < relInfos.size()); + if (relInfo.r_address & R_SCATTERED) fatal("TODO: Scattered relocations not supported"); - auto relInfo = reinterpret_cast(anyRelInfo); - Reloc r; r.type = relInfo.r_type; r.pcrel = relInfo.r_pcrel; r.length = relInfo.r_length; - uint64_t rawAddend = target->getImplicitAddend(mb, sec, relInfo); - + r.offset = relInfo.r_address; + uint64_t rawAddend = target->getAddend(mb, sec, relInfo, pairedInfo); if (relInfo.r_extern) { r.referent = symbols[relInfo.r_symbolnum]; r.addend = rawAddend; } else { - if (relInfo.r_symbolnum == 0 || relInfo.r_symbolnum > subsections.size()) - fatal("invalid section index in relocation for offset " + - std::to_string(r.offset) + " in section " + sec.sectname + - " of " + getName()); - + assert(relInfo.r_symbolnum && relInfo.r_symbolnum <= subsections.size()); SubsectionMap &referentSubsecMap = subsections[relInfo.r_symbolnum - 1]; const section_64 &referentSec = sectionHeaders[relInfo.r_symbolnum - 1]; uint32_t referentOffset; @@ -216,7 +213,6 @@ r.addend = referentOffset; } - r.offset = relInfo.r_address; InputSection *subsec = findContainingSubsection(subsecMap, &r.offset); subsec->relocs.push_back(r); } diff --git a/lld/MachO/Target.h b/lld/MachO/Target.h --- a/lld/MachO/Target.h +++ b/lld/MachO/Target.h @@ -37,9 +37,11 @@ virtual ~TargetInfo() = default; // Validate the relocation structure and get its addend. - virtual uint64_t - getImplicitAddend(llvm::MemoryBufferRef, const llvm::MachO::section_64 &, - const llvm::MachO::relocation_info &) const = 0; + virtual uint64_t getAddend(llvm::MemoryBufferRef, + const llvm::MachO::section_64 &, + const llvm::MachO::relocation_info, + const llvm::MachO::relocation_info) const = 0; + virtual bool isPairedReloc(const llvm::MachO::relocation_info) const = 0; virtual void relocateOne(uint8_t *loc, const Reloc &, uint64_t val) const = 0; // Write code for lazy binding. See the comments on StubsSection for more