2014-07-17 03:49:02 +08:00
|
|
|
//===- lib/FileFormat/MachO/ArchHandler_x86_64.cpp ------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "ArchHandler.h"
|
|
|
|
#include "Atoms.h"
|
|
|
|
#include "MachONormalizedFileBinaryUtils.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
|
|
#include "llvm/ADT/Triple.h"
|
2014-10-28 06:48:35 +08:00
|
|
|
#include "llvm/Support/Endian.h"
|
2014-07-17 03:49:02 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
|
|
|
|
using namespace llvm::MachO;
|
|
|
|
using namespace lld::mach_o::normalized;
|
|
|
|
|
|
|
|
namespace lld {
|
|
|
|
namespace mach_o {
|
|
|
|
|
2014-10-28 06:48:35 +08:00
|
|
|
using llvm::support::ulittle32_t;
|
|
|
|
using llvm::support::ulittle64_t;
|
|
|
|
|
|
|
|
using llvm::support::little32_t;
|
|
|
|
using llvm::support::little64_t;
|
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
class ArchHandler_x86_64 : public ArchHandler {
|
|
|
|
public:
|
|
|
|
ArchHandler_x86_64();
|
|
|
|
virtual ~ArchHandler_x86_64();
|
|
|
|
|
|
|
|
const Registry::KindStrings *kindStrings() override { return _sKindStrings; }
|
|
|
|
|
|
|
|
Reference::KindArch kindArch() override {
|
|
|
|
return Reference::KindArch::x86_64;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Used by GOTPass to locate GOT References
|
|
|
|
bool isGOTAccess(const Reference &ref, bool &canBypassGOT) override {
|
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
|
|
|
return false;
|
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
|
|
|
switch (ref.kindValue()) {
|
|
|
|
case ripRel32GotLoad:
|
|
|
|
canBypassGOT = true;
|
|
|
|
return true;
|
|
|
|
case ripRel32Got:
|
|
|
|
canBypassGOT = false;
|
|
|
|
return true;
|
2014-10-01 05:29:54 +08:00
|
|
|
case imageOffsetGot:
|
|
|
|
canBypassGOT = false;
|
|
|
|
return true;
|
2014-07-17 03:49:02 +08:00
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Used by GOTPass to update GOT References
|
|
|
|
void updateReferenceToGOT(const Reference *ref, bool targetNowGOT) override {
|
|
|
|
assert(ref->kindNamespace() == Reference::KindNamespace::mach_o);
|
|
|
|
assert(ref->kindArch() == Reference::KindArch::x86_64);
|
2014-10-01 05:29:54 +08:00
|
|
|
|
|
|
|
switch (ref->kindValue()) {
|
|
|
|
case ripRel32Got:
|
|
|
|
assert(targetNowGOT && "target must be GOT");
|
|
|
|
case ripRel32GotLoad:
|
|
|
|
const_cast<Reference *>(ref)
|
2014-07-17 03:49:02 +08:00
|
|
|
->setKindValue(targetNowGOT ? ripRel32 : ripRel32GotLoadNowLea);
|
2014-10-01 05:29:54 +08:00
|
|
|
break;
|
|
|
|
case imageOffsetGot:
|
|
|
|
const_cast<Reference *>(ref)->setKindValue(imageOffset);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("unknown GOT reference kind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool needsCompactUnwind() override {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
Reference::KindValue imageOffsetKind() override {
|
|
|
|
return imageOffset;
|
|
|
|
}
|
|
|
|
Reference::KindValue imageOffsetKindIndirect() override {
|
|
|
|
return imageOffsetGot;
|
2014-07-17 03:49:02 +08:00
|
|
|
}
|
|
|
|
|
2014-10-16 04:26:24 +08:00
|
|
|
Reference::KindValue unwindRefToCIEKind() override {
|
|
|
|
return negDelta32;
|
|
|
|
}
|
|
|
|
|
2014-10-16 02:19:31 +08:00
|
|
|
Reference::KindValue unwindRefToFunctionKind() override{
|
|
|
|
return unwindFDEToFunction;
|
|
|
|
}
|
|
|
|
|
2014-10-16 03:32:21 +08:00
|
|
|
Reference::KindValue unwindRefToEhFrameKind() override {
|
|
|
|
return unwindInfoToEhFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t dwarfCompactUnwindType() override {
|
|
|
|
return 0x04000000U;
|
|
|
|
}
|
2014-10-16 02:19:31 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
const StubInfo &stubInfo() override { return _sStubInfo; }
|
|
|
|
|
2014-10-14 09:51:42 +08:00
|
|
|
bool isNonCallBranch(const Reference &) override {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
bool isCallSite(const Reference &) override;
|
|
|
|
bool isPointer(const Reference &) override;
|
|
|
|
bool isPairedReloc(const normalized::Relocation &) override;
|
2014-10-01 05:29:54 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
std::error_code getReferenceInfo(const normalized::Relocation &reloc,
|
|
|
|
const DefinedAtom *inAtom,
|
|
|
|
uint32_t offsetInAtom,
|
|
|
|
uint64_t fixupAddress, bool swap,
|
|
|
|
FindAtomBySectionAndAddress atomFromAddress,
|
|
|
|
FindAtomBySymbolIndex atomFromSymbolIndex,
|
2015-02-23 07:54:38 +08:00
|
|
|
Reference::KindValue *kind,
|
|
|
|
const lld::Atom **target,
|
2014-07-17 03:49:02 +08:00
|
|
|
Reference::Addend *addend) override;
|
2015-02-23 07:54:38 +08:00
|
|
|
std::error_code
|
2014-07-17 03:49:02 +08:00
|
|
|
getPairReferenceInfo(const normalized::Relocation &reloc1,
|
|
|
|
const normalized::Relocation &reloc2,
|
|
|
|
const DefinedAtom *inAtom,
|
|
|
|
uint32_t offsetInAtom,
|
2014-10-22 07:45:37 +08:00
|
|
|
uint64_t fixupAddress, bool swap, bool scatterable,
|
2014-07-17 03:49:02 +08:00
|
|
|
FindAtomBySectionAndAddress atomFromAddress,
|
|
|
|
FindAtomBySymbolIndex atomFromSymbolIndex,
|
2015-02-23 07:54:38 +08:00
|
|
|
Reference::KindValue *kind,
|
|
|
|
const lld::Atom **target,
|
2014-07-17 03:49:02 +08:00
|
|
|
Reference::Addend *addend) override;
|
2014-07-18 07:16:21 +08:00
|
|
|
|
2014-11-04 01:14:46 +08:00
|
|
|
bool needsLocalSymbolInRelocatableFile(const DefinedAtom *atom) override {
|
2014-07-18 07:16:21 +08:00
|
|
|
return (atom->contentType() == DefinedAtom::typeCString);
|
|
|
|
}
|
|
|
|
|
|
|
|
void generateAtomContent(const DefinedAtom &atom, bool relocatable,
|
|
|
|
FindAddressForAtom findAddress,
|
2014-10-16 03:32:21 +08:00
|
|
|
FindAddressForAtom findSectionAddress,
|
2014-10-01 05:29:54 +08:00
|
|
|
uint64_t imageBase,
|
2014-07-18 07:16:21 +08:00
|
|
|
uint8_t *atomContentBuffer) override;
|
|
|
|
|
|
|
|
void appendSectionRelocations(const DefinedAtom &atom,
|
|
|
|
uint64_t atomSectionOffset,
|
|
|
|
const Reference &ref,
|
|
|
|
FindSymbolIndexForAtom symbolIndexForAtom,
|
|
|
|
FindSectionIndexForAtom sectionIndexForAtom,
|
|
|
|
FindAddressForAtom addressForAtom,
|
|
|
|
normalized::Relocations &relocs) override;
|
2014-07-17 03:49:02 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
static const Registry::KindStrings _sKindStrings[];
|
|
|
|
static const StubInfo _sStubInfo;
|
|
|
|
|
2015-02-19 20:38:54 +08:00
|
|
|
enum X86_64Kind: Reference::KindValue {
|
2014-07-17 03:49:02 +08:00
|
|
|
invalid, /// for error condition
|
2015-02-23 07:54:38 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// Kinds found in mach-o .o files:
|
|
|
|
branch32, /// ex: call _foo
|
|
|
|
ripRel32, /// ex: movq _foo(%rip), %rax
|
|
|
|
ripRel32Minus1, /// ex: movb $0x12, _foo(%rip)
|
|
|
|
ripRel32Minus2, /// ex: movw $0x1234, _foo(%rip)
|
|
|
|
ripRel32Minus4, /// ex: movl $0x12345678, _foo(%rip)
|
|
|
|
ripRel32Anon, /// ex: movq L1(%rip), %rax
|
|
|
|
ripRel32GotLoad, /// ex: movq _foo@GOTPCREL(%rip), %rax
|
|
|
|
ripRel32Got, /// ex: pushq _foo@GOTPCREL(%rip)
|
|
|
|
pointer64, /// ex: .quad _foo
|
|
|
|
pointer64Anon, /// ex: .quad L1
|
|
|
|
delta64, /// ex: .quad _foo - .
|
|
|
|
delta32, /// ex: .long _foo - .
|
|
|
|
delta64Anon, /// ex: .quad L1 - .
|
|
|
|
delta32Anon, /// ex: .long L1 - .
|
2014-10-16 04:26:24 +08:00
|
|
|
negDelta32, /// ex: .long . - _foo
|
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// Kinds introduced by Passes:
|
2015-02-23 07:54:38 +08:00
|
|
|
ripRel32GotLoadNowLea, /// Target of GOT load is in linkage unit so
|
2014-07-17 03:49:02 +08:00
|
|
|
/// "movq _foo@GOTPCREL(%rip), %rax" can be changed
|
|
|
|
/// to "leaq _foo(%rip), %rax
|
|
|
|
lazyPointer, /// Location contains a lazy pointer.
|
|
|
|
lazyImmediateLocation, /// Location contains immediate value used in stub.
|
2014-10-01 05:29:54 +08:00
|
|
|
|
|
|
|
imageOffset, /// Location contains offset of atom in final image
|
|
|
|
imageOffsetGot, /// Location contains offset of GOT entry for atom in
|
|
|
|
/// final image (typically personality function).
|
2014-10-16 02:19:31 +08:00
|
|
|
unwindFDEToFunction, /// Nearly delta64, but cannot be rematerialized in
|
|
|
|
/// relocatable object (yay for implicit contracts!).
|
2014-10-16 03:32:21 +08:00
|
|
|
unwindInfoToEhFrame, /// Fix low 24 bits of compact unwind encoding to
|
|
|
|
/// refer to __eh_frame entry.
|
2014-07-17 03:49:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
Reference::KindValue kindFromReloc(const normalized::Relocation &reloc);
|
|
|
|
Reference::KindValue kindFromRelocPair(const normalized::Relocation &reloc1,
|
|
|
|
const normalized::Relocation &reloc2);
|
|
|
|
|
2014-07-18 07:16:21 +08:00
|
|
|
void applyFixupFinal(const Reference &ref, uint8_t *location,
|
|
|
|
uint64_t fixupAddress, uint64_t targetAddress,
|
2014-10-16 03:32:21 +08:00
|
|
|
uint64_t inAtomAddress, uint64_t imageBaseAddress,
|
|
|
|
FindAddressForAtom findSectionAddress);
|
2014-07-18 07:16:21 +08:00
|
|
|
|
|
|
|
void applyFixupRelocatable(const Reference &ref, uint8_t *location,
|
|
|
|
uint64_t fixupAddress,
|
|
|
|
uint64_t targetAddress,
|
|
|
|
uint64_t inAtomAddress);
|
2014-07-17 03:49:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-10-28 06:48:35 +08:00
|
|
|
ArchHandler_x86_64::ArchHandler_x86_64() { }
|
2014-07-17 03:49:02 +08:00
|
|
|
|
|
|
|
ArchHandler_x86_64::~ArchHandler_x86_64() { }
|
|
|
|
|
|
|
|
const Registry::KindStrings ArchHandler_x86_64::_sKindStrings[] = {
|
|
|
|
LLD_KIND_STRING_ENTRY(invalid), LLD_KIND_STRING_ENTRY(branch32),
|
|
|
|
LLD_KIND_STRING_ENTRY(ripRel32), LLD_KIND_STRING_ENTRY(ripRel32Minus1),
|
|
|
|
LLD_KIND_STRING_ENTRY(ripRel32Minus2), LLD_KIND_STRING_ENTRY(ripRel32Minus4),
|
|
|
|
LLD_KIND_STRING_ENTRY(ripRel32Anon), LLD_KIND_STRING_ENTRY(ripRel32GotLoad),
|
|
|
|
LLD_KIND_STRING_ENTRY(ripRel32GotLoadNowLea),
|
|
|
|
LLD_KIND_STRING_ENTRY(ripRel32Got), LLD_KIND_STRING_ENTRY(lazyPointer),
|
|
|
|
LLD_KIND_STRING_ENTRY(lazyImmediateLocation),
|
|
|
|
LLD_KIND_STRING_ENTRY(pointer64), LLD_KIND_STRING_ENTRY(pointer64Anon),
|
|
|
|
LLD_KIND_STRING_ENTRY(delta32), LLD_KIND_STRING_ENTRY(delta64),
|
|
|
|
LLD_KIND_STRING_ENTRY(delta32Anon), LLD_KIND_STRING_ENTRY(delta64Anon),
|
2014-10-16 04:26:24 +08:00
|
|
|
LLD_KIND_STRING_ENTRY(negDelta32),
|
2014-10-01 05:29:54 +08:00
|
|
|
LLD_KIND_STRING_ENTRY(imageOffset), LLD_KIND_STRING_ENTRY(imageOffsetGot),
|
2014-10-16 02:19:31 +08:00
|
|
|
LLD_KIND_STRING_ENTRY(unwindFDEToFunction),
|
2014-10-16 03:32:21 +08:00
|
|
|
LLD_KIND_STRING_ENTRY(unwindInfoToEhFrame),
|
2014-07-17 03:49:02 +08:00
|
|
|
LLD_KIND_STRING_END
|
|
|
|
};
|
|
|
|
|
|
|
|
const ArchHandler::StubInfo ArchHandler_x86_64::_sStubInfo = {
|
|
|
|
"dyld_stub_binder",
|
|
|
|
|
2015-02-23 07:54:38 +08:00
|
|
|
// Lazy pointer references
|
2014-07-17 03:49:02 +08:00
|
|
|
{ Reference::KindArch::x86_64, pointer64, 0, 0 },
|
|
|
|
{ Reference::KindArch::x86_64, lazyPointer, 0, 0 },
|
2015-02-23 07:54:38 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// GOT pointer to dyld_stub_binder
|
|
|
|
{ Reference::KindArch::x86_64, pointer64, 0, 0 },
|
|
|
|
|
|
|
|
// x86_64 code alignment 2^1
|
2015-02-23 07:54:38 +08:00
|
|
|
1,
|
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// Stub size and code
|
2015-02-23 07:54:38 +08:00
|
|
|
6,
|
2014-07-17 03:49:02 +08:00
|
|
|
{ 0xff, 0x25, 0x00, 0x00, 0x00, 0x00 }, // jmp *lazyPointer
|
|
|
|
{ Reference::KindArch::x86_64, ripRel32, 2, 0 },
|
2014-09-10 07:52:59 +08:00
|
|
|
{ false, 0, 0, 0 },
|
2015-02-23 07:54:38 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// Stub Helper size and code
|
|
|
|
10,
|
|
|
|
{ 0x68, 0x00, 0x00, 0x00, 0x00, // pushq $lazy-info-offset
|
|
|
|
0xE9, 0x00, 0x00, 0x00, 0x00 }, // jmp helperhelper
|
|
|
|
{ Reference::KindArch::x86_64, lazyImmediateLocation, 1, 0 },
|
|
|
|
{ Reference::KindArch::x86_64, branch32, 6, 0 },
|
2015-02-23 07:54:38 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
// Stub Helper-Common size and code
|
|
|
|
16,
|
|
|
|
{ 0x4C, 0x8D, 0x1D, 0x00, 0x00, 0x00, 0x00, // leaq cache(%rip),%r11
|
|
|
|
0x41, 0x53, // push %r11
|
|
|
|
0xFF, 0x25, 0x00, 0x00, 0x00, 0x00, // jmp *binder(%rip)
|
|
|
|
0x90 }, // nop
|
|
|
|
{ Reference::KindArch::x86_64, ripRel32, 3, 0 },
|
2014-09-10 07:52:59 +08:00
|
|
|
{ false, 0, 0, 0 },
|
|
|
|
{ Reference::KindArch::x86_64, ripRel32, 11, 0 },
|
|
|
|
{ false, 0, 0, 0 }
|
2015-02-23 07:54:38 +08:00
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
bool ArchHandler_x86_64::isCallSite(const Reference &ref) {
|
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
|
|
|
return false;
|
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
|
|
|
return (ref.kindValue() == branch32);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ArchHandler_x86_64::isPointer(const Reference &ref) {
|
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
|
|
|
return false;
|
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
|
|
|
Reference::KindValue kind = ref.kindValue();
|
|
|
|
return (kind == pointer64 || kind == pointer64Anon);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ArchHandler_x86_64::isPairedReloc(const Relocation &reloc) {
|
|
|
|
return (reloc.type == X86_64_RELOC_SUBTRACTOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
Reference::KindValue
|
|
|
|
ArchHandler_x86_64::kindFromReloc(const Relocation &reloc) {
|
|
|
|
switch(relocPattern(reloc)) {
|
|
|
|
case X86_64_RELOC_BRANCH | rPcRel | rExtern | rLength4:
|
|
|
|
return branch32;
|
|
|
|
case X86_64_RELOC_SIGNED | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32;
|
|
|
|
case X86_64_RELOC_SIGNED | rPcRel | rLength4:
|
|
|
|
return ripRel32Anon;
|
|
|
|
case X86_64_RELOC_SIGNED_1 | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32Minus1;
|
|
|
|
case X86_64_RELOC_SIGNED_2 | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32Minus2;
|
|
|
|
case X86_64_RELOC_SIGNED_4 | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32Minus4;
|
|
|
|
case X86_64_RELOC_GOT_LOAD | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32GotLoad;
|
|
|
|
case X86_64_RELOC_GOT | rPcRel | rExtern | rLength4:
|
|
|
|
return ripRel32Got;
|
|
|
|
case X86_64_RELOC_UNSIGNED | rExtern | rLength8:
|
|
|
|
return pointer64;
|
|
|
|
case X86_64_RELOC_UNSIGNED | rLength8:
|
|
|
|
return pointer64Anon;
|
|
|
|
default:
|
|
|
|
return invalid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-23 07:54:38 +08:00
|
|
|
std::error_code
|
2014-07-17 03:49:02 +08:00
|
|
|
ArchHandler_x86_64::getReferenceInfo(const Relocation &reloc,
|
|
|
|
const DefinedAtom *inAtom,
|
|
|
|
uint32_t offsetInAtom,
|
|
|
|
uint64_t fixupAddress, bool swap,
|
|
|
|
FindAtomBySectionAndAddress atomFromAddress,
|
|
|
|
FindAtomBySymbolIndex atomFromSymbolIndex,
|
2015-02-23 07:54:38 +08:00
|
|
|
Reference::KindValue *kind,
|
|
|
|
const lld::Atom **target,
|
2014-07-17 03:49:02 +08:00
|
|
|
Reference::Addend *addend) {
|
|
|
|
typedef std::error_code E;
|
|
|
|
*kind = kindFromReloc(reloc);
|
|
|
|
if (*kind == invalid)
|
2015-04-14 10:34:09 +08:00
|
|
|
return make_dynamic_error_code("unknown type");
|
2014-07-17 03:49:02 +08:00
|
|
|
const uint8_t *fixupContent = &inAtom->rawContent()[offsetInAtom];
|
|
|
|
uint64_t targetAddress;
|
|
|
|
switch (*kind) {
|
|
|
|
case branch32:
|
|
|
|
case ripRel32:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = *(const little32_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case ripRel32Minus1:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = (int32_t)*(const little32_t *)fixupContent + 1;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case ripRel32Minus2:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = (int32_t)*(const little32_t *)fixupContent + 2;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case ripRel32Minus4:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = (int32_t)*(const little32_t *)fixupContent + 4;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case ripRel32Anon:
|
2014-11-14 15:15:43 +08:00
|
|
|
targetAddress = fixupAddress + 4 + *(const little32_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return atomFromAddress(reloc.symbol, targetAddress, target, addend);
|
|
|
|
case ripRel32GotLoad:
|
|
|
|
case ripRel32Got:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = *(const little32_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case pointer64:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = *(const little64_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case pointer64Anon:
|
2014-11-14 15:15:43 +08:00
|
|
|
targetAddress = *(const little64_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return atomFromAddress(reloc.symbol, targetAddress, target, addend);
|
|
|
|
default:
|
|
|
|
llvm_unreachable("bad reloc kind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Reference::KindValue
|
|
|
|
ArchHandler_x86_64::kindFromRelocPair(const normalized::Relocation &reloc1,
|
|
|
|
const normalized::Relocation &reloc2) {
|
|
|
|
switch(relocPattern(reloc1) << 16 | relocPattern(reloc2)) {
|
|
|
|
case ((X86_64_RELOC_SUBTRACTOR | rExtern | rLength8) << 16 |
|
|
|
|
X86_64_RELOC_UNSIGNED | rExtern | rLength8):
|
|
|
|
return delta64;
|
|
|
|
case ((X86_64_RELOC_SUBTRACTOR | rExtern | rLength4) << 16 |
|
|
|
|
X86_64_RELOC_UNSIGNED | rExtern | rLength4):
|
|
|
|
return delta32;
|
|
|
|
case ((X86_64_RELOC_SUBTRACTOR | rExtern | rLength8) << 16 |
|
|
|
|
X86_64_RELOC_UNSIGNED | rLength8):
|
|
|
|
return delta64Anon;
|
|
|
|
case ((X86_64_RELOC_SUBTRACTOR | rExtern | rLength4) << 16 |
|
|
|
|
X86_64_RELOC_UNSIGNED | rLength4):
|
|
|
|
return delta32Anon;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("bad reloc pairs");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::error_code
|
|
|
|
ArchHandler_x86_64::getPairReferenceInfo(const normalized::Relocation &reloc1,
|
|
|
|
const normalized::Relocation &reloc2,
|
|
|
|
const DefinedAtom *inAtom,
|
|
|
|
uint32_t offsetInAtom,
|
|
|
|
uint64_t fixupAddress, bool swap,
|
2014-10-22 07:45:37 +08:00
|
|
|
bool scatterable,
|
2014-07-17 03:49:02 +08:00
|
|
|
FindAtomBySectionAndAddress atomFromAddress,
|
|
|
|
FindAtomBySymbolIndex atomFromSymbolIndex,
|
2015-02-23 07:54:38 +08:00
|
|
|
Reference::KindValue *kind,
|
|
|
|
const lld::Atom **target,
|
2014-07-17 03:49:02 +08:00
|
|
|
Reference::Addend *addend) {
|
|
|
|
*kind = kindFromRelocPair(reloc1, reloc2);
|
|
|
|
if (*kind == invalid)
|
2015-04-14 10:34:09 +08:00
|
|
|
return make_dynamic_error_code("unknown pair");
|
2014-07-17 03:49:02 +08:00
|
|
|
const uint8_t *fixupContent = &inAtom->rawContent()[offsetInAtom];
|
|
|
|
typedef std::error_code E;
|
|
|
|
uint64_t targetAddress;
|
|
|
|
const lld::Atom *fromTarget;
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc1.symbol, &fromTarget))
|
|
|
|
return ec;
|
|
|
|
if (fromTarget != inAtom)
|
2015-04-14 10:34:09 +08:00
|
|
|
return make_dynamic_error_code("pointer diff not in base atom");
|
2014-07-17 03:49:02 +08:00
|
|
|
switch (*kind) {
|
|
|
|
case delta64:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc2.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = (int64_t)*(const little64_t *)fixupContent + offsetInAtom;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case delta32:
|
|
|
|
if (E ec = atomFromSymbolIndex(reloc2.symbol, target))
|
|
|
|
return ec;
|
2014-11-14 15:15:43 +08:00
|
|
|
*addend = (int32_t)*(const little32_t *)fixupContent + offsetInAtom;
|
2014-07-17 03:49:02 +08:00
|
|
|
return std::error_code();
|
|
|
|
case delta64Anon:
|
2014-11-14 15:15:43 +08:00
|
|
|
targetAddress = offsetInAtom + (int64_t)*(const little64_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return atomFromAddress(reloc2.symbol, targetAddress, target, addend);
|
|
|
|
case delta32Anon:
|
2014-11-14 15:15:43 +08:00
|
|
|
targetAddress = offsetInAtom + (int32_t)*(const little32_t *)fixupContent;
|
2014-07-17 03:49:02 +08:00
|
|
|
return atomFromAddress(reloc2.symbol, targetAddress, target, addend);
|
|
|
|
default:
|
|
|
|
llvm_unreachable("bad reloc pair kind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-16 03:32:21 +08:00
|
|
|
void ArchHandler_x86_64::generateAtomContent(
|
|
|
|
const DefinedAtom &atom, bool relocatable, FindAddressForAtom findAddress,
|
|
|
|
FindAddressForAtom findSectionAddress, uint64_t imageBaseAddress,
|
|
|
|
uint8_t *atomContentBuffer) {
|
2014-07-18 07:16:21 +08:00
|
|
|
// Copy raw bytes.
|
|
|
|
memcpy(atomContentBuffer, atom.rawContent().data(), atom.size());
|
|
|
|
// Apply fix-ups.
|
|
|
|
for (const Reference *ref : atom) {
|
|
|
|
uint32_t offset = ref->offsetInAtom();
|
|
|
|
const Atom *target = ref->target();
|
|
|
|
uint64_t targetAddress = 0;
|
|
|
|
if (isa<DefinedAtom>(target))
|
|
|
|
targetAddress = findAddress(*target);
|
|
|
|
uint64_t atomAddress = findAddress(atom);
|
|
|
|
uint64_t fixupAddress = atomAddress + offset;
|
|
|
|
if (relocatable) {
|
|
|
|
applyFixupRelocatable(*ref, &atomContentBuffer[offset],
|
|
|
|
fixupAddress, targetAddress,
|
|
|
|
atomAddress);
|
|
|
|
} else {
|
|
|
|
applyFixupFinal(*ref, &atomContentBuffer[offset],
|
2014-10-01 05:29:54 +08:00
|
|
|
fixupAddress, targetAddress,
|
2014-10-16 03:32:21 +08:00
|
|
|
atomAddress, imageBaseAddress, findSectionAddress);
|
2014-07-18 07:16:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-16 03:32:21 +08:00
|
|
|
void ArchHandler_x86_64::applyFixupFinal(
|
2014-10-28 06:48:35 +08:00
|
|
|
const Reference &ref, uint8_t *loc, uint64_t fixupAddress,
|
2014-10-16 03:32:21 +08:00
|
|
|
uint64_t targetAddress, uint64_t inAtomAddress, uint64_t imageBaseAddress,
|
|
|
|
FindAddressForAtom findSectionAddress) {
|
2014-07-18 07:16:21 +08:00
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
2014-07-17 03:49:02 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
2015-02-15 23:23:48 +08:00
|
|
|
ulittle32_t *loc32 = reinterpret_cast<ulittle32_t *>(loc);
|
|
|
|
ulittle64_t *loc64 = reinterpret_cast<ulittle64_t *>(loc);
|
2015-02-19 20:38:54 +08:00
|
|
|
switch (static_cast<X86_64Kind>(ref.kindValue())) {
|
2014-07-17 03:49:02 +08:00
|
|
|
case branch32:
|
|
|
|
case ripRel32:
|
2014-09-04 05:09:24 +08:00
|
|
|
case ripRel32Anon:
|
2014-07-17 03:49:02 +08:00
|
|
|
case ripRel32Got:
|
|
|
|
case ripRel32GotLoad:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = targetAddress - (fixupAddress + 4) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case pointer64:
|
|
|
|
case pointer64Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = targetAddress + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case ripRel32Minus1:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = targetAddress - (fixupAddress + 5) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case ripRel32Minus2:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = targetAddress - (fixupAddress + 6) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case ripRel32Minus4:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = targetAddress - (fixupAddress + 8) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case delta32:
|
|
|
|
case delta32Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = targetAddress - fixupAddress + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case delta64:
|
|
|
|
case delta64Anon:
|
2014-10-16 02:19:31 +08:00
|
|
|
case unwindFDEToFunction:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = targetAddress - fixupAddress + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case ripRel32GotLoadNowLea:
|
|
|
|
// Change MOVQ to LEA
|
2014-10-28 06:48:35 +08:00
|
|
|
assert(loc[-2] == 0x8B);
|
|
|
|
loc[-2] = 0x8D;
|
|
|
|
*loc32 = targetAddress - (fixupAddress + 4) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-16 04:26:24 +08:00
|
|
|
case negDelta32:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = fixupAddress - targetAddress + ref.addend();
|
2014-10-16 04:26:24 +08:00
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case lazyPointer:
|
2014-11-11 09:31:18 +08:00
|
|
|
// Do nothing
|
|
|
|
return;
|
2014-07-17 03:49:02 +08:00
|
|
|
case lazyImmediateLocation:
|
2014-11-11 09:31:18 +08:00
|
|
|
*loc32 = ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-01 05:29:54 +08:00
|
|
|
case imageOffset:
|
|
|
|
case imageOffsetGot:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = (targetAddress - imageBaseAddress) + ref.addend();
|
2014-10-01 05:29:54 +08:00
|
|
|
return;
|
2014-10-16 03:32:21 +08:00
|
|
|
case unwindInfoToEhFrame: {
|
|
|
|
uint64_t val = targetAddress - findSectionAddress(*ref.target()) + ref.addend();
|
|
|
|
assert(val < 0xffffffU && "offset in __eh_frame too large");
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = (*loc32 & 0xff000000U) | val;
|
2014-10-16 03:32:21 +08:00
|
|
|
return;
|
|
|
|
}
|
2014-09-04 05:09:24 +08:00
|
|
|
case invalid:
|
|
|
|
// Fall into llvm_unreachable().
|
2014-07-17 03:49:02 +08:00
|
|
|
break;
|
|
|
|
}
|
2014-09-04 05:09:24 +08:00
|
|
|
llvm_unreachable("invalid x86_64 Reference Kind");
|
2014-07-17 03:49:02 +08:00
|
|
|
}
|
|
|
|
|
2014-07-18 07:16:21 +08:00
|
|
|
|
|
|
|
void ArchHandler_x86_64::applyFixupRelocatable(const Reference &ref,
|
2014-10-28 06:48:35 +08:00
|
|
|
uint8_t *loc,
|
2014-07-18 07:16:21 +08:00
|
|
|
uint64_t fixupAddress,
|
|
|
|
uint64_t targetAddress,
|
|
|
|
uint64_t inAtomAddress) {
|
2015-02-15 23:23:48 +08:00
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
|
|
|
return;
|
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
2014-10-28 06:48:35 +08:00
|
|
|
ulittle32_t *loc32 = reinterpret_cast<ulittle32_t *>(loc);
|
|
|
|
ulittle64_t *loc64 = reinterpret_cast<ulittle64_t *>(loc);
|
2015-02-19 20:38:54 +08:00
|
|
|
switch (static_cast<X86_64Kind>(ref.kindValue())) {
|
2014-07-18 07:16:21 +08:00
|
|
|
case branch32:
|
|
|
|
case ripRel32:
|
|
|
|
case ripRel32Got:
|
|
|
|
case ripRel32GotLoad:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
|
|
|
case ripRel32Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = (targetAddress - (fixupAddress + 4)) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case pointer64:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case pointer64Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = targetAddress + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus1:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = ref.addend() - 1;
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus2:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = ref.addend() - 2;
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus4:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = ref.addend() - 4;
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta32:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = ref.addend() + inAtomAddress - fixupAddress;
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta32Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = (targetAddress - fixupAddress) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta64:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = ref.addend() + inAtomAddress - fixupAddress;
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta64Anon:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc64 = (targetAddress - fixupAddress) + ref.addend();
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-16 04:26:24 +08:00
|
|
|
case negDelta32:
|
2014-10-28 06:48:35 +08:00
|
|
|
*loc32 = fixupAddress - targetAddress + ref.addend();
|
2014-10-16 04:26:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32GotLoadNowLea:
|
|
|
|
llvm_unreachable("ripRel32GotLoadNowLea implies GOT pass was run");
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case lazyPointer:
|
|
|
|
case lazyImmediateLocation:
|
|
|
|
llvm_unreachable("lazy reference kind implies Stubs pass was run");
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-01 05:29:54 +08:00
|
|
|
case imageOffset:
|
|
|
|
case imageOffsetGot:
|
2014-10-16 03:32:21 +08:00
|
|
|
case unwindInfoToEhFrame:
|
|
|
|
llvm_unreachable("fixup implies __unwind_info");
|
2014-10-01 05:29:54 +08:00
|
|
|
return;
|
2014-10-16 02:19:31 +08:00
|
|
|
case unwindFDEToFunction:
|
|
|
|
// Do nothing for now
|
|
|
|
return;
|
2014-09-04 05:09:24 +08:00
|
|
|
case invalid:
|
|
|
|
// Fall into llvm_unreachable().
|
2014-07-18 07:16:21 +08:00
|
|
|
break;
|
|
|
|
}
|
2014-09-04 05:09:24 +08:00
|
|
|
llvm_unreachable("unknown x86_64 Reference Kind");
|
2014-07-18 07:16:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ArchHandler_x86_64::appendSectionRelocations(
|
|
|
|
const DefinedAtom &atom,
|
|
|
|
uint64_t atomSectionOffset,
|
|
|
|
const Reference &ref,
|
|
|
|
FindSymbolIndexForAtom symbolIndexForAtom,
|
|
|
|
FindSectionIndexForAtom sectionIndexForAtom,
|
|
|
|
FindAddressForAtom addressForAtom,
|
|
|
|
normalized::Relocations &relocs) {
|
|
|
|
if (ref.kindNamespace() != Reference::KindNamespace::mach_o)
|
|
|
|
return;
|
|
|
|
assert(ref.kindArch() == Reference::KindArch::x86_64);
|
|
|
|
uint32_t sectionOffset = atomSectionOffset + ref.offsetInAtom();
|
2015-02-19 20:38:54 +08:00
|
|
|
switch (static_cast<X86_64Kind>(ref.kindValue())) {
|
2014-07-18 07:16:21 +08:00
|
|
|
case branch32:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_BRANCH | rPcRel | rExtern | rLength4);
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_SIGNED | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
|
|
|
case ripRel32Anon:
|
|
|
|
appendReloc(relocs, sectionOffset, sectionIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_SIGNED | rPcRel | rLength4 );
|
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Got:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_GOT | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32GotLoad:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_GOT_LOAD | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case pointer64:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rExtern | rLength8);
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case pointer64Anon:
|
|
|
|
appendReloc(relocs, sectionOffset, sectionIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rLength8);
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus1:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_SIGNED_1 | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus2:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_SIGNED_2 | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32Minus4:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_SIGNED_4 | rPcRel | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta32:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(atom), 0,
|
|
|
|
X86_64_RELOC_SUBTRACTOR | rExtern | rLength4 );
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rExtern | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta32Anon:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(atom), 0,
|
|
|
|
X86_64_RELOC_SUBTRACTOR | rExtern | rLength4 );
|
|
|
|
appendReloc(relocs, sectionOffset, sectionIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rLength4 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta64:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(atom), 0,
|
|
|
|
X86_64_RELOC_SUBTRACTOR | rExtern | rLength8 );
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rExtern | rLength8 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case delta64Anon:
|
|
|
|
appendReloc(relocs, sectionOffset, symbolIndexForAtom(atom), 0,
|
|
|
|
X86_64_RELOC_SUBTRACTOR | rExtern | rLength8 );
|
|
|
|
appendReloc(relocs, sectionOffset, sectionIndexForAtom(*ref.target()), 0,
|
|
|
|
X86_64_RELOC_UNSIGNED | rLength8 );
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-16 02:19:31 +08:00
|
|
|
case unwindFDEToFunction:
|
2014-10-16 03:32:21 +08:00
|
|
|
case unwindInfoToEhFrame:
|
2014-10-16 04:26:24 +08:00
|
|
|
case negDelta32:
|
2014-10-16 02:19:31 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case ripRel32GotLoadNowLea:
|
|
|
|
llvm_unreachable("ripRel32GotLoadNowLea implies GOT pass was run");
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-07-18 07:16:21 +08:00
|
|
|
case lazyPointer:
|
|
|
|
case lazyImmediateLocation:
|
|
|
|
llvm_unreachable("lazy reference kind implies Stubs pass was run");
|
2014-09-04 05:09:24 +08:00
|
|
|
return;
|
2014-10-01 05:29:54 +08:00
|
|
|
case imageOffset:
|
|
|
|
case imageOffsetGot:
|
|
|
|
llvm_unreachable("__unwind_info references should have been resolved");
|
|
|
|
return;
|
2014-09-04 05:09:24 +08:00
|
|
|
case invalid:
|
|
|
|
// Fall into llvm_unreachable().
|
2014-07-18 07:16:21 +08:00
|
|
|
break;
|
|
|
|
}
|
2014-09-04 05:09:24 +08:00
|
|
|
llvm_unreachable("unknown x86_64 Reference Kind");
|
2014-07-18 07:16:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-17 03:49:02 +08:00
|
|
|
std::unique_ptr<mach_o::ArchHandler> ArchHandler::create_x86_64() {
|
|
|
|
return std::unique_ptr<mach_o::ArchHandler>(new ArchHandler_x86_64());
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace mach_o
|
|
|
|
} // namespace lld
|