mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-26 12:36:07 +00:00

otherwise operator<<(const ELFSyncStream &s, RelType type) applies to non-reloc-type uint32_t, which can be confusing.
603 lines
19 KiB
C++
603 lines
19 KiB
C++
//===- SystemZ.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 "OutputSections.h"
|
|
#include "Symbols.h"
|
|
#include "SyntheticSections.h"
|
|
#include "Target.h"
|
|
#include "lld/Common/ErrorHandler.h"
|
|
#include "llvm/BinaryFormat/ELF.h"
|
|
#include "llvm/Support/Endian.h"
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::support::endian;
|
|
using namespace llvm::ELF;
|
|
using namespace lld;
|
|
using namespace lld::elf;
|
|
|
|
namespace {
|
|
class SystemZ : public TargetInfo {
|
|
public:
|
|
SystemZ(Ctx &);
|
|
int getTlsGdRelaxSkip(RelType type) const override;
|
|
RelExpr getRelExpr(RelType type, const Symbol &s,
|
|
const uint8_t *loc) const override;
|
|
RelType getDynRel(RelType type) const override;
|
|
void writeGotHeader(uint8_t *buf) const override;
|
|
void writeGotPlt(uint8_t *buf, const Symbol &s) const override;
|
|
void writeIgotPlt(uint8_t *buf, const Symbol &s) const override;
|
|
void writePltHeader(uint8_t *buf) const override;
|
|
void addPltHeaderSymbols(InputSection &isd) const override;
|
|
void writePlt(uint8_t *buf, const Symbol &sym,
|
|
uint64_t pltEntryAddr) const override;
|
|
RelExpr adjustTlsExpr(RelType type, RelExpr expr) const override;
|
|
RelExpr adjustGotPcExpr(RelType type, int64_t addend,
|
|
const uint8_t *loc) const override;
|
|
bool relaxOnce(int pass) const override;
|
|
void relocate(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const override;
|
|
int64_t getImplicitAddend(const uint8_t *buf, RelType type) const override;
|
|
|
|
private:
|
|
void relaxGot(uint8_t *loc, const Relocation &rel, uint64_t val) const;
|
|
void relaxTlsGdToIe(uint8_t *loc, const Relocation &rel, uint64_t val) const;
|
|
void relaxTlsGdToLe(uint8_t *loc, const Relocation &rel, uint64_t val) const;
|
|
void relaxTlsLdToLe(uint8_t *loc, const Relocation &rel, uint64_t val) const;
|
|
};
|
|
} // namespace
|
|
|
|
SystemZ::SystemZ(Ctx &ctx) : TargetInfo(ctx) {
|
|
copyRel = R_390_COPY;
|
|
gotRel = R_390_GLOB_DAT;
|
|
pltRel = R_390_JMP_SLOT;
|
|
relativeRel = R_390_RELATIVE;
|
|
iRelativeRel = R_390_IRELATIVE;
|
|
symbolicRel = R_390_64;
|
|
tlsGotRel = R_390_TLS_TPOFF;
|
|
tlsModuleIndexRel = R_390_TLS_DTPMOD;
|
|
tlsOffsetRel = R_390_TLS_DTPOFF;
|
|
gotHeaderEntriesNum = 3;
|
|
gotPltHeaderEntriesNum = 0;
|
|
gotEntrySize = 8;
|
|
pltHeaderSize = 32;
|
|
pltEntrySize = 32;
|
|
ipltEntrySize = 32;
|
|
|
|
// This "trap instruction" is used to fill gaps between sections.
|
|
// On SystemZ, the behavior of the GNU ld is to fill those gaps
|
|
// with nop instructions instead - and unfortunately the default
|
|
// glibc crt object files (used to) rely on that behavior since
|
|
// they use an alignment on the .init section fragments that causes
|
|
// gaps which must be filled with nops as they are being executed.
|
|
// Therefore, we provide a nop instruction as "trapInstr" here.
|
|
trapInstr = {0x07, 0x07, 0x07, 0x07};
|
|
|
|
defaultImageBase = 0x1000000;
|
|
}
|
|
|
|
RelExpr SystemZ::getRelExpr(RelType type, const Symbol &s,
|
|
const uint8_t *loc) const {
|
|
switch (type) {
|
|
case R_390_NONE:
|
|
return R_NONE;
|
|
// Relocations targeting the symbol value.
|
|
case R_390_8:
|
|
case R_390_12:
|
|
case R_390_16:
|
|
case R_390_20:
|
|
case R_390_32:
|
|
case R_390_64:
|
|
return R_ABS;
|
|
case R_390_PC16:
|
|
case R_390_PC32:
|
|
case R_390_PC64:
|
|
case R_390_PC12DBL:
|
|
case R_390_PC16DBL:
|
|
case R_390_PC24DBL:
|
|
case R_390_PC32DBL:
|
|
return R_PC;
|
|
case R_390_GOTOFF16:
|
|
case R_390_GOTOFF: // a.k.a. R_390_GOTOFF32
|
|
case R_390_GOTOFF64:
|
|
return R_GOTREL;
|
|
// Relocations targeting the PLT associated with the symbol.
|
|
case R_390_PLT32:
|
|
case R_390_PLT64:
|
|
case R_390_PLT12DBL:
|
|
case R_390_PLT16DBL:
|
|
case R_390_PLT24DBL:
|
|
case R_390_PLT32DBL:
|
|
return R_PLT_PC;
|
|
case R_390_PLTOFF16:
|
|
case R_390_PLTOFF32:
|
|
case R_390_PLTOFF64:
|
|
return R_PLT_GOTREL;
|
|
// Relocations targeting the GOT entry associated with the symbol.
|
|
case R_390_GOTENT:
|
|
return R_GOT_PC;
|
|
case R_390_GOT12:
|
|
case R_390_GOT16:
|
|
case R_390_GOT20:
|
|
case R_390_GOT32:
|
|
case R_390_GOT64:
|
|
return R_GOT_OFF;
|
|
// Relocations targeting the GOTPLT entry associated with the symbol.
|
|
case R_390_GOTPLTENT:
|
|
return R_GOTPLT_PC;
|
|
case R_390_GOTPLT12:
|
|
case R_390_GOTPLT16:
|
|
case R_390_GOTPLT20:
|
|
case R_390_GOTPLT32:
|
|
case R_390_GOTPLT64:
|
|
return R_GOTPLT_GOTREL;
|
|
// Relocations targeting _GLOBAL_OFFSET_TABLE_.
|
|
case R_390_GOTPC:
|
|
case R_390_GOTPCDBL:
|
|
return R_GOTONLY_PC;
|
|
// TLS-related relocations.
|
|
case R_390_TLS_LOAD:
|
|
return R_NONE;
|
|
case R_390_TLS_GDCALL:
|
|
return R_TLSGD_PC;
|
|
case R_390_TLS_LDCALL:
|
|
return R_TLSLD_PC;
|
|
case R_390_TLS_GD32:
|
|
case R_390_TLS_GD64:
|
|
return R_TLSGD_GOT;
|
|
case R_390_TLS_LDM32:
|
|
case R_390_TLS_LDM64:
|
|
return R_TLSLD_GOT;
|
|
case R_390_TLS_LDO32:
|
|
case R_390_TLS_LDO64:
|
|
return R_DTPREL;
|
|
case R_390_TLS_LE32:
|
|
case R_390_TLS_LE64:
|
|
return R_TPREL;
|
|
case R_390_TLS_IE32:
|
|
case R_390_TLS_IE64:
|
|
return R_GOT;
|
|
case R_390_TLS_GOTIE12:
|
|
case R_390_TLS_GOTIE20:
|
|
case R_390_TLS_GOTIE32:
|
|
case R_390_TLS_GOTIE64:
|
|
return R_GOT_OFF;
|
|
case R_390_TLS_IEENT:
|
|
return R_GOT_PC;
|
|
|
|
default:
|
|
Err(ctx) << getErrorLoc(ctx, loc) << "unknown relocation (" << type.v
|
|
<< ") against symbol " << &s;
|
|
return R_NONE;
|
|
}
|
|
}
|
|
|
|
void SystemZ::writeGotHeader(uint8_t *buf) const {
|
|
// _GLOBAL_OFFSET_TABLE_[0] holds the value of _DYNAMIC.
|
|
// _GLOBAL_OFFSET_TABLE_[1] and [2] are reserved.
|
|
write64be(buf, ctx.mainPart->dynamic->getVA());
|
|
}
|
|
|
|
void SystemZ::writeGotPlt(uint8_t *buf, const Symbol &s) const {
|
|
write64be(buf, s.getPltVA(ctx) + 14);
|
|
}
|
|
|
|
void SystemZ::writeIgotPlt(uint8_t *buf, const Symbol &s) const {
|
|
if (ctx.arg.writeAddends)
|
|
write64be(buf, s.getVA(ctx));
|
|
}
|
|
|
|
void SystemZ::writePltHeader(uint8_t *buf) const {
|
|
const uint8_t pltData[] = {
|
|
0xe3, 0x10, 0xf0, 0x38, 0x00, 0x24, // stg %r1,56(%r15)
|
|
0xc0, 0x10, 0x00, 0x00, 0x00, 0x00, // larl %r1,_GLOBAL_OFFSET_TABLE_
|
|
0xd2, 0x07, 0xf0, 0x30, 0x10, 0x08, // mvc 48(8,%r15),8(%r1)
|
|
0xe3, 0x10, 0x10, 0x10, 0x00, 0x04, // lg %r1,16(%r1)
|
|
0x07, 0xf1, // br %r1
|
|
0x07, 0x00, // nopr
|
|
0x07, 0x00, // nopr
|
|
0x07, 0x00, // nopr
|
|
};
|
|
memcpy(buf, pltData, sizeof(pltData));
|
|
uint64_t got = ctx.in.got->getVA();
|
|
uint64_t plt = ctx.in.plt->getVA();
|
|
write32be(buf + 8, (got - plt - 6) >> 1);
|
|
}
|
|
|
|
void SystemZ::addPltHeaderSymbols(InputSection &isec) const {
|
|
// The PLT header needs a reference to _GLOBAL_OFFSET_TABLE_, so we
|
|
// must ensure the .got section is created even if otherwise unused.
|
|
ctx.in.got->hasGotOffRel.store(true, std::memory_order_relaxed);
|
|
}
|
|
|
|
void SystemZ::writePlt(uint8_t *buf, const Symbol &sym,
|
|
uint64_t pltEntryAddr) const {
|
|
const uint8_t inst[] = {
|
|
0xc0, 0x10, 0x00, 0x00, 0x00, 0x00, // larl %r1,<.got.plt slot>
|
|
0xe3, 0x10, 0x10, 0x00, 0x00, 0x04, // lg %r1,0(%r1)
|
|
0x07, 0xf1, // br %r1
|
|
0x0d, 0x10, // basr %r1,%r0
|
|
0xe3, 0x10, 0x10, 0x0c, 0x00, 0x14, // lgf %r1,12(%r1)
|
|
0xc0, 0xf4, 0x00, 0x00, 0x00, 0x00, // jg <plt header>
|
|
0x00, 0x00, 0x00, 0x00, // <relocation offset>
|
|
};
|
|
memcpy(buf, inst, sizeof(inst));
|
|
|
|
write32be(buf + 2, (sym.getGotPltVA(ctx) - pltEntryAddr) >> 1);
|
|
write32be(buf + 24, (ctx.in.plt->getVA() - pltEntryAddr - 22) >> 1);
|
|
write32be(buf + 28, ctx.in.relaPlt->entsize * sym.getPltIdx(ctx));
|
|
}
|
|
|
|
int64_t SystemZ::getImplicitAddend(const uint8_t *buf, RelType type) const {
|
|
switch (type) {
|
|
case R_390_8:
|
|
return SignExtend64<8>(*buf);
|
|
case R_390_16:
|
|
case R_390_PC16:
|
|
return SignExtend64<16>(read16be(buf));
|
|
case R_390_PC16DBL:
|
|
return SignExtend64<16>(read16be(buf)) << 1;
|
|
case R_390_32:
|
|
case R_390_PC32:
|
|
return SignExtend64<32>(read32be(buf));
|
|
case R_390_PC32DBL:
|
|
return SignExtend64<32>(read32be(buf)) << 1;
|
|
case R_390_64:
|
|
case R_390_PC64:
|
|
case R_390_TLS_DTPMOD:
|
|
case R_390_TLS_DTPOFF:
|
|
case R_390_TLS_TPOFF:
|
|
case R_390_GLOB_DAT:
|
|
case R_390_RELATIVE:
|
|
case R_390_IRELATIVE:
|
|
return read64be(buf);
|
|
case R_390_COPY:
|
|
case R_390_JMP_SLOT:
|
|
case R_390_NONE:
|
|
// These relocations are defined as not having an implicit addend.
|
|
return 0;
|
|
default:
|
|
InternalErr(ctx, buf) << "cannot read addend for relocation " << type;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
RelType SystemZ::getDynRel(RelType type) const {
|
|
if (type == R_390_64 || type == R_390_PC64)
|
|
return type;
|
|
return R_390_NONE;
|
|
}
|
|
|
|
RelExpr SystemZ::adjustTlsExpr(RelType type, RelExpr expr) const {
|
|
if (expr == R_RELAX_TLS_GD_TO_IE)
|
|
return R_RELAX_TLS_GD_TO_IE_GOT_OFF;
|
|
return expr;
|
|
}
|
|
|
|
int SystemZ::getTlsGdRelaxSkip(RelType type) const {
|
|
// A __tls_get_offset call instruction is marked with 2 relocations:
|
|
//
|
|
// R_390_TLS_GDCALL / R_390_TLS_LDCALL: marker relocation
|
|
// R_390_PLT32DBL: __tls_get_offset
|
|
//
|
|
// After the relaxation we no longer call __tls_get_offset and should skip
|
|
// both relocations to not create a false dependence on __tls_get_offset
|
|
// being defined.
|
|
//
|
|
// Note that this mechanism only works correctly if the R_390_TLS_[GL]DCALL
|
|
// is seen immediately *before* the R_390_PLT32DBL. Unfortunately, current
|
|
// compilers on the platform will typically generate the inverse sequence.
|
|
// To fix this, we sort relocations by offset in RelocationScanner::scan;
|
|
// this ensures the correct sequence as the R_390_TLS_[GL]DCALL applies to
|
|
// the first byte of the brasl instruction, while the R_390_PLT32DBL applies
|
|
// to its third byte (the relative displacement).
|
|
|
|
if (type == R_390_TLS_GDCALL || type == R_390_TLS_LDCALL)
|
|
return 2;
|
|
return 1;
|
|
}
|
|
|
|
void SystemZ::relaxTlsGdToIe(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const {
|
|
// The general-dynamic code sequence for a global `x`:
|
|
//
|
|
// Instruction Relocation Symbol
|
|
// ear %rX,%a0
|
|
// sllg %rX,%rX,32
|
|
// ear %rX,%a1
|
|
// larl %r12,_GLOBAL_OFFSET_TABLE_ R_390_GOTPCDBL _GLOBAL_OFFSET_TABLE_
|
|
// lgrl %r2,.LC0 R_390_PC32DBL .LC0
|
|
// brasl %r14,__tls_get_offset@plt R_390_TLS_GDCALL x
|
|
// :tls_gdcall:x R_390_PLT32DBL __tls_get_offset
|
|
// la %r2,0(%r2,%rX)
|
|
//
|
|
// .LC0:
|
|
// .quad x@TLSGD R_390_TLS_GD64 x
|
|
//
|
|
// Relaxing to initial-exec entails:
|
|
// 1) Replacing the call by a load from the GOT.
|
|
// 2) Replacing the relocation on the constant LC0 by R_390_TLS_GOTIE64.
|
|
|
|
switch (rel.type) {
|
|
case R_390_TLS_GDCALL:
|
|
// brasl %r14,__tls_get_offset@plt -> lg %r2,0(%r2,%r12)
|
|
write16be(loc, 0xe322);
|
|
write32be(loc + 2, 0xc0000004);
|
|
break;
|
|
case R_390_TLS_GD64:
|
|
relocateNoSym(loc, R_390_TLS_GOTIE64, val);
|
|
break;
|
|
default:
|
|
llvm_unreachable("unsupported relocation for TLS GD to IE relaxation");
|
|
}
|
|
}
|
|
|
|
void SystemZ::relaxTlsGdToLe(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const {
|
|
// The general-dynamic code sequence for a global `x`:
|
|
//
|
|
// Instruction Relocation Symbol
|
|
// ear %rX,%a0
|
|
// sllg %rX,%rX,32
|
|
// ear %rX,%a1
|
|
// larl %r12,_GLOBAL_OFFSET_TABLE_ R_390_GOTPCDBL _GLOBAL_OFFSET_TABLE_
|
|
// lgrl %r2,.LC0 R_390_PC32DBL .LC0
|
|
// brasl %r14,__tls_get_offset@plt R_390_TLS_GDCALL x
|
|
// :tls_gdcall:x R_390_PLT32DBL __tls_get_offset
|
|
// la %r2,0(%r2,%rX)
|
|
//
|
|
// .LC0:
|
|
// .quad x@tlsgd R_390_TLS_GD64 x
|
|
//
|
|
// Relaxing to local-exec entails:
|
|
// 1) Replacing the call by a nop.
|
|
// 2) Replacing the relocation on the constant LC0 by R_390_TLS_LE64.
|
|
|
|
switch (rel.type) {
|
|
case R_390_TLS_GDCALL:
|
|
// brasl %r14,__tls_get_offset@plt -> brcl 0,.
|
|
write16be(loc, 0xc004);
|
|
write32be(loc + 2, 0x00000000);
|
|
break;
|
|
case R_390_TLS_GD64:
|
|
relocateNoSym(loc, R_390_TLS_LE64, val);
|
|
break;
|
|
default:
|
|
llvm_unreachable("unsupported relocation for TLS GD to LE relaxation");
|
|
}
|
|
}
|
|
|
|
void SystemZ::relaxTlsLdToLe(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const {
|
|
// The local-dynamic code sequence for a global `x`:
|
|
//
|
|
// Instruction Relocation Symbol
|
|
// ear %rX,%a0
|
|
// sllg %rX,%rX,32
|
|
// ear %rX,%a1
|
|
// larl %r12,_GLOBAL_OFFSET_TABLE_ R_390_GOTPCDBL _GLOBAL_OFFSET_TABLE_
|
|
// lgrl %r2,.LC0 R_390_PC32DBL .LC0
|
|
// brasl %r14,__tls_get_offset@plt R_390_TLS_LDCALL <sym>
|
|
// :tls_ldcall:<sym> R_390_PLT32DBL __tls_get_offset
|
|
// la %r2,0(%r2,%rX)
|
|
// lgrl %rY,.LC1 R_390_PC32DBL .LC1
|
|
// la %r2,0(%r2,%rY)
|
|
//
|
|
// .LC0:
|
|
// .quad <sym>@tlsldm R_390_TLS_LDM64 <sym>
|
|
// .LC1:
|
|
// .quad x@dtpoff R_390_TLS_LDO64 x
|
|
//
|
|
// Relaxing to local-exec entails:
|
|
// 1) Replacing the call by a nop.
|
|
// 2) Replacing the constant LC0 by 0 (i.e. ignoring the relocation).
|
|
// 3) Replacing the relocation on the constant LC1 by R_390_TLS_LE64.
|
|
|
|
switch (rel.type) {
|
|
case R_390_TLS_LDCALL:
|
|
// brasl %r14,__tls_get_offset@plt -> brcl 0,.
|
|
write16be(loc, 0xc004);
|
|
write32be(loc + 2, 0x00000000);
|
|
break;
|
|
case R_390_TLS_LDM64:
|
|
break;
|
|
case R_390_TLS_LDO64:
|
|
relocateNoSym(loc, R_390_TLS_LE64, val);
|
|
break;
|
|
default:
|
|
llvm_unreachable("unsupported relocation for TLS LD to LE relaxation");
|
|
}
|
|
}
|
|
|
|
RelExpr SystemZ::adjustGotPcExpr(RelType type, int64_t addend,
|
|
const uint8_t *loc) const {
|
|
// Only R_390_GOTENT with addend 2 can be relaxed.
|
|
if (!ctx.arg.relax || addend != 2 || type != R_390_GOTENT)
|
|
return R_GOT_PC;
|
|
const uint16_t op = read16be(loc - 2);
|
|
|
|
// lgrl rx,sym@GOTENT -> larl rx, sym
|
|
// This relaxation is legal if "sym" binds locally (which was already
|
|
// verified by our caller) and is in-range and properly aligned for a
|
|
// LARL instruction. We cannot verify the latter constraint here, so
|
|
// we assume it is true and revert the decision later on in relaxOnce
|
|
// if necessary.
|
|
if ((op & 0xff0f) == 0xc408)
|
|
return R_RELAX_GOT_PC;
|
|
|
|
return R_GOT_PC;
|
|
}
|
|
|
|
bool SystemZ::relaxOnce(int pass) const {
|
|
// If we decided in adjustGotPcExpr to relax a R_390_GOTENT,
|
|
// we need to validate the target symbol is in-range and aligned.
|
|
SmallVector<InputSection *, 0> storage;
|
|
bool changed = false;
|
|
for (OutputSection *osec : ctx.outputSections) {
|
|
if (!(osec->flags & SHF_EXECINSTR))
|
|
continue;
|
|
for (InputSection *sec : getInputSections(*osec, storage)) {
|
|
for (Relocation &rel : sec->relocs()) {
|
|
if (rel.expr != R_RELAX_GOT_PC)
|
|
continue;
|
|
|
|
uint64_t v = sec->getRelocTargetVA(
|
|
ctx, rel, sec->getOutputSection()->addr + rel.offset);
|
|
if (isInt<33>(v) && !(v & 1))
|
|
continue;
|
|
if (rel.sym->auxIdx == 0) {
|
|
rel.sym->allocateAux(ctx);
|
|
addGotEntry(ctx, *rel.sym);
|
|
changed = true;
|
|
}
|
|
rel.expr = R_GOT_PC;
|
|
}
|
|
}
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
void SystemZ::relaxGot(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const {
|
|
assert(isInt<33>(val) &&
|
|
"R_390_GOTENT should not have been relaxed if it overflows");
|
|
assert(!(val & 1) &&
|
|
"R_390_GOTENT should not have been relaxed if it is misaligned");
|
|
const uint16_t op = read16be(loc - 2);
|
|
|
|
// lgrl rx,sym@GOTENT -> larl rx, sym
|
|
if ((op & 0xff0f) == 0xc408) {
|
|
write16be(loc - 2, 0xc000 | (op & 0x00f0));
|
|
write32be(loc, val >> 1);
|
|
}
|
|
}
|
|
|
|
void SystemZ::relocate(uint8_t *loc, const Relocation &rel,
|
|
uint64_t val) const {
|
|
switch (rel.expr) {
|
|
case R_RELAX_GOT_PC:
|
|
return relaxGot(loc, rel, val);
|
|
case R_RELAX_TLS_GD_TO_IE_GOT_OFF:
|
|
return relaxTlsGdToIe(loc, rel, val);
|
|
case R_RELAX_TLS_GD_TO_LE:
|
|
return relaxTlsGdToLe(loc, rel, val);
|
|
case R_RELAX_TLS_LD_TO_LE:
|
|
return relaxTlsLdToLe(loc, rel, val);
|
|
default:
|
|
break;
|
|
}
|
|
switch (rel.type) {
|
|
case R_390_8:
|
|
checkIntUInt(ctx, loc, val, 8, rel);
|
|
*loc = val;
|
|
break;
|
|
case R_390_12:
|
|
case R_390_GOT12:
|
|
case R_390_GOTPLT12:
|
|
case R_390_TLS_GOTIE12:
|
|
checkUInt(ctx, loc, val, 12, rel);
|
|
write16be(loc, (read16be(loc) & 0xF000) | val);
|
|
break;
|
|
case R_390_PC12DBL:
|
|
case R_390_PLT12DBL:
|
|
checkInt(ctx, loc, val, 13, rel);
|
|
checkAlignment(ctx, loc, val, 2, rel);
|
|
write16be(loc, (read16be(loc) & 0xF000) | ((val >> 1) & 0x0FFF));
|
|
break;
|
|
case R_390_16:
|
|
case R_390_GOT16:
|
|
case R_390_GOTPLT16:
|
|
case R_390_GOTOFF16:
|
|
case R_390_PLTOFF16:
|
|
checkIntUInt(ctx, loc, val, 16, rel);
|
|
write16be(loc, val);
|
|
break;
|
|
case R_390_PC16:
|
|
checkInt(ctx, loc, val, 16, rel);
|
|
write16be(loc, val);
|
|
break;
|
|
case R_390_PC16DBL:
|
|
case R_390_PLT16DBL:
|
|
checkInt(ctx, loc, val, 17, rel);
|
|
checkAlignment(ctx, loc, val, 2, rel);
|
|
write16be(loc, val >> 1);
|
|
break;
|
|
case R_390_20:
|
|
case R_390_GOT20:
|
|
case R_390_GOTPLT20:
|
|
case R_390_TLS_GOTIE20:
|
|
checkInt(ctx, loc, val, 20, rel);
|
|
write32be(loc, (read32be(loc) & 0xF00000FF) | ((val & 0xFFF) << 16) |
|
|
((val & 0xFF000) >> 4));
|
|
break;
|
|
case R_390_PC24DBL:
|
|
case R_390_PLT24DBL:
|
|
checkInt(ctx, loc, val, 25, rel);
|
|
checkAlignment(ctx, loc, val, 2, rel);
|
|
loc[0] = val >> 17;
|
|
loc[1] = val >> 9;
|
|
loc[2] = val >> 1;
|
|
break;
|
|
case R_390_32:
|
|
case R_390_GOT32:
|
|
case R_390_GOTPLT32:
|
|
case R_390_GOTOFF:
|
|
case R_390_PLTOFF32:
|
|
case R_390_TLS_IE32:
|
|
case R_390_TLS_GOTIE32:
|
|
case R_390_TLS_GD32:
|
|
case R_390_TLS_LDM32:
|
|
case R_390_TLS_LDO32:
|
|
case R_390_TLS_LE32:
|
|
checkIntUInt(ctx, loc, val, 32, rel);
|
|
write32be(loc, val);
|
|
break;
|
|
case R_390_PC32:
|
|
case R_390_PLT32:
|
|
checkInt(ctx, loc, val, 32, rel);
|
|
write32be(loc, val);
|
|
break;
|
|
case R_390_PC32DBL:
|
|
case R_390_PLT32DBL:
|
|
case R_390_GOTPCDBL:
|
|
case R_390_GOTENT:
|
|
case R_390_GOTPLTENT:
|
|
case R_390_TLS_IEENT:
|
|
checkInt(ctx, loc, val, 33, rel);
|
|
checkAlignment(ctx, loc, val, 2, rel);
|
|
write32be(loc, val >> 1);
|
|
break;
|
|
case R_390_64:
|
|
case R_390_PC64:
|
|
case R_390_PLT64:
|
|
case R_390_GOT64:
|
|
case R_390_GOTPLT64:
|
|
case R_390_GOTOFF64:
|
|
case R_390_PLTOFF64:
|
|
case R_390_GOTPC:
|
|
case R_390_TLS_IE64:
|
|
case R_390_TLS_GOTIE64:
|
|
case R_390_TLS_GD64:
|
|
case R_390_TLS_LDM64:
|
|
case R_390_TLS_LDO64:
|
|
case R_390_TLS_LE64:
|
|
case R_390_TLS_DTPMOD:
|
|
case R_390_TLS_DTPOFF:
|
|
case R_390_TLS_TPOFF:
|
|
write64be(loc, val);
|
|
break;
|
|
case R_390_TLS_LOAD:
|
|
case R_390_TLS_GDCALL:
|
|
case R_390_TLS_LDCALL:
|
|
break;
|
|
default:
|
|
llvm_unreachable("unknown relocation");
|
|
}
|
|
}
|
|
|
|
void elf::setSystemZTargetInfo(Ctx &ctx) { ctx.target.reset(new SystemZ(ctx)); }
|