2021-03-12 02:28:09 +08:00
|
|
|
//===- Relocations.h --------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLD_MACHO_RELOCATIONS_H
|
|
|
|
#define LLD_MACHO_RELOCATIONS_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/BitmaskEnum.h"
|
|
|
|
#include "llvm/ADT/PointerUnion.h"
|
|
|
|
#include "llvm/BinaryFormat/MachO.h"
|
2021-04-16 09:14:33 +08:00
|
|
|
#include "llvm/Support/Endian.h"
|
2021-03-12 02:28:09 +08:00
|
|
|
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
|
|
|
|
namespace lld {
|
|
|
|
namespace macho {
|
|
|
|
LLVM_ENABLE_BITMASK_ENUMS_IN_NAMESPACE();
|
|
|
|
|
|
|
|
class Symbol;
|
|
|
|
class InputSection;
|
|
|
|
|
|
|
|
enum class RelocAttrBits {
|
|
|
|
_0 = 0, // invalid
|
|
|
|
PCREL = 1 << 0, // Value is PC-relative offset
|
|
|
|
ABSOLUTE = 1 << 1, // Value is an absolute address or fixed offset
|
|
|
|
BYTE4 = 1 << 2, // 4 byte datum
|
|
|
|
BYTE8 = 1 << 3, // 8 byte datum
|
|
|
|
EXTERN = 1 << 4, // Can have an external symbol
|
|
|
|
LOCAL = 1 << 5, // Can have a local symbol
|
|
|
|
ADDEND = 1 << 6, // *_ADDEND paired prefix reloc
|
|
|
|
SUBTRAHEND = 1 << 7, // *_SUBTRACTOR paired prefix reloc
|
|
|
|
BRANCH = 1 << 8, // Value is branch target
|
|
|
|
GOT = 1 << 9, // References a symbol in the Global Offset Table
|
|
|
|
TLV = 1 << 10, // References a thread-local symbol
|
2021-04-10 07:47:10 +08:00
|
|
|
LOAD = 1 << 11, // Relaxable indirect load
|
|
|
|
POINTER = 1 << 12, // Non-relaxable indirect load (pointer is taken)
|
|
|
|
UNSIGNED = 1 << 13, // *_UNSIGNED relocs
|
|
|
|
LLVM_MARK_AS_BITMASK_ENUM(/*LargestValue*/ (1 << 14) - 1),
|
2021-03-12 02:28:09 +08:00
|
|
|
};
|
|
|
|
// Note: SUBTRACTOR always pairs with UNSIGNED (a delta between two symbols).
|
|
|
|
|
|
|
|
struct RelocAttrs {
|
|
|
|
llvm::StringRef name;
|
|
|
|
RelocAttrBits bits;
|
|
|
|
bool hasAttr(RelocAttrBits b) const { return (bits & b) == b; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Reloc {
|
|
|
|
uint8_t type = llvm::MachO::GENERIC_RELOC_INVALID;
|
|
|
|
bool pcrel = false;
|
|
|
|
uint8_t length = 0;
|
|
|
|
// The offset from the start of the subsection that this relocation belongs
|
|
|
|
// to.
|
2021-11-13 08:07:07 +08:00
|
|
|
uint32_t offset = 0;
|
2021-03-12 02:28:09 +08:00
|
|
|
// Adding this offset to the address of the referent symbol or subsection
|
|
|
|
// gives the destination that this relocation refers to.
|
2021-03-13 06:26:11 +08:00
|
|
|
int64_t addend = 0;
|
2021-03-12 02:28:09 +08:00
|
|
|
llvm::PointerUnion<Symbol *, InputSection *> referent = nullptr;
|
2022-06-13 09:56:45 +08:00
|
|
|
|
|
|
|
Reloc() = default;
|
|
|
|
|
|
|
|
Reloc(uint8_t type, bool pcrel, uint8_t length, uint32_t offset,
|
|
|
|
int64_t addend, llvm::PointerUnion<Symbol *, InputSection *> referent)
|
|
|
|
: type(type), pcrel(pcrel), length(length), offset(offset),
|
|
|
|
addend(addend), referent(referent) {}
|
2021-03-12 02:28:09 +08:00
|
|
|
};
|
|
|
|
|
2022-06-17 23:21:59 +08:00
|
|
|
struct OptimizationHint {
|
|
|
|
// Offset of the first address within the containing InputSection.
|
|
|
|
uint64_t offset0;
|
|
|
|
// Offset of the other addresses relative to the first one.
|
|
|
|
int16_t delta[2];
|
|
|
|
uint8_t type;
|
|
|
|
};
|
|
|
|
|
2021-03-12 02:28:09 +08:00
|
|
|
bool validateSymbolRelocation(const Symbol *, const InputSection *,
|
|
|
|
const Reloc &);
|
|
|
|
|
2021-03-13 06:26:11 +08:00
|
|
|
/*
|
|
|
|
* v: The value the relocation is attempting to encode
|
|
|
|
* bits: The number of bits actually available to encode this relocation
|
|
|
|
*/
|
2022-03-01 10:56:38 +08:00
|
|
|
void reportRangeError(void *loc, const Reloc &, const llvm::Twine &v,
|
|
|
|
uint8_t bits, int64_t min, uint64_t max);
|
2021-03-13 06:26:11 +08:00
|
|
|
|
|
|
|
struct SymbolDiagnostic {
|
|
|
|
const Symbol *symbol;
|
|
|
|
llvm::StringRef reason;
|
|
|
|
};
|
|
|
|
|
2022-03-01 10:56:38 +08:00
|
|
|
void reportRangeError(void *loc, SymbolDiagnostic, const llvm::Twine &v,
|
|
|
|
uint8_t bits, int64_t min, uint64_t max);
|
2021-03-13 06:26:11 +08:00
|
|
|
|
|
|
|
template <typename Diagnostic>
|
2022-03-01 10:56:38 +08:00
|
|
|
inline void checkInt(void *loc, Diagnostic d, int64_t v, int bits) {
|
2021-03-13 06:26:11 +08:00
|
|
|
if (v != llvm::SignExtend64(v, bits))
|
2022-03-01 10:56:38 +08:00
|
|
|
reportRangeError(loc, d, llvm::Twine(v), bits, llvm::minIntN(bits),
|
2021-03-13 06:26:11 +08:00
|
|
|
llvm::maxIntN(bits));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Diagnostic>
|
2022-03-01 10:56:38 +08:00
|
|
|
inline void checkUInt(void *loc, Diagnostic d, uint64_t v, int bits) {
|
2021-03-13 06:26:11 +08:00
|
|
|
if ((v >> bits) != 0)
|
2022-03-01 10:56:38 +08:00
|
|
|
reportRangeError(loc, d, llvm::Twine(v), bits, 0, llvm::maxUIntN(bits));
|
2021-03-13 06:26:11 +08:00
|
|
|
}
|
|
|
|
|
2021-04-16 09:14:33 +08:00
|
|
|
inline void writeAddress(uint8_t *loc, uint64_t addr, uint8_t length) {
|
|
|
|
switch (length) {
|
|
|
|
case 2:
|
|
|
|
llvm::support::endian::write32le(loc, addr);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
llvm::support::endian::write64le(loc, addr);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("invalid r_length");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-12 02:28:09 +08:00
|
|
|
extern const RelocAttrs invalidRelocAttrs;
|
|
|
|
|
|
|
|
} // namespace macho
|
|
|
|
} // namespace lld
|
|
|
|
|
|
|
|
#endif
|