Files
clang-p2996/lld/MachO/Arch/ARM.cpp
Daniel Bertalan 0d30e92f59 [lld-macho] Add support for emitting chained fixups
This commit adds support for chained fixups, which were introduced in
Apple's late 2020 OS releases. This format replaces the dyld opcodes
used for supplying rebase and binding information, and encodes most of
that data directly in the memory location that will have the fixup
applied.

This reduces binary size and is a requirement for page-in linking, which
will be available starting with macOS 13.

A high-level overview of the format and my implementation can be found
in SyntheticSections.h.

This feature is currently gated behind the `-fixup_chains` flag, and
will be enabled by default for supported targets in a later commit.

Like in ld64, lazy binding is disabled when chained fixups are in use,
and the `-init_offsets` transformation is performed by default.

Differential Revision: https://reviews.llvm.org/D132560
2022-10-04 11:48:45 +02:00

215 lines
7.2 KiB
C++

//===- ARM.cpp ------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "InputFiles.h"
#include "Symbols.h"
#include "SyntheticSections.h"
#include "Target.h"
#include "lld/Common/ErrorHandler.h"
#include "llvm/ADT/Bitfields.h"
#include "llvm/BinaryFormat/MachO.h"
#include "llvm/Support/Endian.h"
using namespace llvm;
using namespace llvm::MachO;
using namespace llvm::support::endian;
using namespace lld;
using namespace lld::macho;
namespace {
struct ARM : TargetInfo {
ARM(uint32_t cpuSubtype);
int64_t getEmbeddedAddend(MemoryBufferRef, uint64_t offset,
const relocation_info) const override;
void relocateOne(uint8_t *loc, const Reloc &, uint64_t va,
uint64_t pc) const override;
void writeStub(uint8_t *buf, const Symbol &, uint64_t) const override;
void writeStubHelperHeader(uint8_t *buf) const override;
void writeStubHelperEntry(uint8_t *buf, const Symbol &,
uint64_t entryAddr) const override;
void writeObjCMsgSendStub(uint8_t *buf, Symbol *sym, uint64_t stubsAddr,
uint64_t stubOffset, uint64_t selrefsVA,
uint64_t selectorIndex, uint64_t gotAddr,
uint64_t msgSendIndex) const override;
void relaxGotLoad(uint8_t *loc, uint8_t type) const override;
uint64_t getPageSize() const override { return 4 * 1024; }
void handleDtraceReloc(const Symbol *sym, const Reloc &r,
uint8_t *loc) const override;
};
} // namespace
static constexpr std::array<RelocAttrs, 10> relocAttrsArray{{
#define B(x) RelocAttrBits::x
{"VANILLA", /* FIXME populate this */ B(_0)},
{"PAIR", /* FIXME populate this */ B(_0)},
{"SECTDIFF", /* FIXME populate this */ B(_0)},
{"LOCAL_SECTDIFF", /* FIXME populate this */ B(_0)},
{"PB_LA_PTR", /* FIXME populate this */ B(_0)},
{"BR24", B(PCREL) | B(LOCAL) | B(EXTERN) | B(BRANCH) | B(BYTE4)},
{"BR22", B(PCREL) | B(LOCAL) | B(EXTERN) | B(BRANCH) | B(BYTE4)},
{"32BIT_BRANCH", /* FIXME populate this */ B(_0)},
{"HALF", /* FIXME populate this */ B(_0)},
{"HALF_SECTDIFF", /* FIXME populate this */ B(_0)},
#undef B
}};
int64_t ARM::getEmbeddedAddend(MemoryBufferRef mb, uint64_t offset,
relocation_info rel) const {
// FIXME: implement this
return 0;
}
template <int N> using BitfieldFlag = Bitfield::Element<bool, N, 1>;
// ARM BL encoding:
//
// 30 28 24 0
// +---------+---------+----------------------------------------------+
// | cond | 1 0 1 1 | imm24 |
// +---------+---------+----------------------------------------------+
//
// `cond` here varies depending on whether we have bleq, blne, etc.
// `imm24` encodes a 26-bit pcrel offset -- last 2 bits are zero as ARM
// functions are 4-byte-aligned.
//
// ARM BLX encoding:
//
// 30 28 24 0
// +---------+---------+----------------------------------------------+
// | 1 1 1 1 | 1 0 1 H | imm24 |
// +---------+---------+----------------------------------------------+
//
// Since Thumb functions are 2-byte-aligned, we need one extra bit to encode
// the offset -- that is the H bit.
//
// BLX is always unconditional, so while we can convert directly from BLX to BL,
// we need to insert a shim if a BL's target is a Thumb function.
//
// Helper aliases for decoding BL / BLX:
using Cond = Bitfield::Element<uint32_t, 28, 4>;
using Imm24 = Bitfield::Element<int32_t, 0, 24>;
void ARM::relocateOne(uint8_t *loc, const Reloc &r, uint64_t value,
uint64_t pc) const {
switch (r.type) {
case ARM_RELOC_BR24: {
uint32_t base = read32le(loc);
bool isBlx = Bitfield::get<Cond>(base) == 0xf;
const Symbol *sym = r.referent.get<Symbol *>();
int32_t offset = value - (pc + 8);
if (auto *defined = dyn_cast<Defined>(sym)) {
if (!isBlx && defined->thumb) {
error("TODO: implement interworking shim");
return;
} else if (isBlx && !defined->thumb) {
Bitfield::set<Cond>(base, 0xe); // unconditional BL
Bitfield::set<BitfieldFlag<24>>(base, true);
isBlx = false;
}
} else {
error("TODO: Implement ARM_RELOC_BR24 for dylib symbols");
return;
}
if (isBlx) {
assert((0x1 & value) == 0);
Bitfield::set<Imm24>(base, offset >> 2);
Bitfield::set<BitfieldFlag<24>>(base, (offset >> 1) & 1); // H bit
} else {
assert((0x3 & value) == 0);
Bitfield::set<Imm24>(base, offset >> 2);
}
write32le(loc, base);
break;
}
default:
fatal("unhandled relocation type");
}
}
void ARM::writeStub(uint8_t *buf, const Symbol &sym, uint64_t) const {
fatal("TODO: implement this");
}
void ARM::writeStubHelperHeader(uint8_t *buf) const {
fatal("TODO: implement this");
}
void ARM::writeStubHelperEntry(uint8_t *buf, const Symbol &sym,
uint64_t entryAddr) const {
fatal("TODO: implement this");
}
void ARM::writeObjCMsgSendStub(uint8_t *buf, Symbol *sym, uint64_t stubsAddr,
uint64_t stubOffset, uint64_t selrefsVA,
uint64_t selectorIndex, uint64_t gotAddr,
uint64_t msgSendIndex) const {
fatal("TODO: implement this");
}
void ARM::relaxGotLoad(uint8_t *loc, uint8_t type) const {
fatal("TODO: implement this");
}
ARM::ARM(uint32_t cpuSubtype) : TargetInfo(ILP32()) {
cpuType = CPU_TYPE_ARM;
this->cpuSubtype = cpuSubtype;
stubSize = 0 /* FIXME */;
stubHelperHeaderSize = 0 /* FIXME */;
stubHelperEntrySize = 0 /* FIXME */;
relocAttrs = {relocAttrsArray.data(), relocAttrsArray.size()};
}
TargetInfo *macho::createARMTargetInfo(uint32_t cpuSubtype) {
static ARM t(cpuSubtype);
return &t;
}
void ARM::handleDtraceReloc(const Symbol *sym, const Reloc &r,
uint8_t *loc) const {
if (config->outputType == MH_OBJECT)
return;
switch (r.type) {
case ARM_RELOC_BR24:
if (sym->getName().startswith("___dtrace_probe")) {
// change call site to a NOP
write32le(loc, 0xE1A00000);
} else if (sym->getName().startswith("___dtrace_isenabled")) {
// change call site to 'eor r0, r0, r0'
write32le(loc, 0xE0200000);
} else {
error("Unrecognized dtrace symbol prefix: " + toString(*sym));
}
break;
case ARM_THUMB_RELOC_BR22:
if (sym->getName().startswith("___dtrace_probe")) {
// change 32-bit blx call site to two thumb NOPs
write32le(loc, 0x46C046C0);
} else if (sym->getName().startswith("___dtrace_isenabled")) {
// change 32-bit blx call site to 'nop', 'eor r0, r0'
write32le(loc, 0x46C04040);
} else {
error("Unrecognized dtrace symbol prefix: " + toString(*sym));
}
break;
default:
llvm_unreachable("Unsupported dtrace relocation type for ARM");
}
}