LLVM 20.0.0git
MCELFObjectWriter.h
Go to the documentation of this file.
1//===- llvm/MC/MCELFObjectWriter.h - ELF Object Writer ----------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#ifndef LLVM_MC_MCELFOBJECTWRITER_H
10#define LLVM_MC_MCELFOBJECTWRITER_H
11
12#include "llvm/ADT/DenseMap.h"
20#include <cstdint>
21#include <memory>
22#include <optional>
23#include <vector>
24
25namespace llvm {
26
27class MCAssembler;
28class MCContext;
29class MCFixup;
30class MCSymbol;
31class MCSymbolELF;
32class MCTargetOptions;
33class MCValue;
34
36 uint64_t Offset; // Where is the relocation.
37 const MCSymbolELF *Symbol; // The symbol to relocate with.
38 unsigned Type; // The type of the relocation.
39 uint64_t Addend; // The addend to use.
40
44
45 void print(raw_ostream &Out) const {
46 Out << "Off=" << Offset << ", Sym=" << Symbol << ", Type=" << Type
47 << ", Addend=" << Addend;
48 }
49
50 LLVM_DUMP_METHOD void dump() const { print(errs()); }
51};
52
54 const uint8_t OSABI;
55 const uint8_t ABIVersion;
56 const uint16_t EMachine;
57 const unsigned HasRelocationAddend : 1;
58 const unsigned Is64Bit : 1;
59
60protected:
61 MCELFObjectTargetWriter(bool Is64Bit_, uint8_t OSABI_, uint16_t EMachine_,
62 bool HasRelocationAddend_, uint8_t ABIVersion_ = 0);
63
64public:
65 virtual ~MCELFObjectTargetWriter() = default;
66
67 Triple::ObjectFormatType getFormat() const override { return Triple::ELF; }
68 static bool classof(const MCObjectTargetWriter *W) {
69 return W->getFormat() == Triple::ELF;
70 }
71
73 switch (OSType) {
76 case Triple::PS4:
77 case Triple::FreeBSD:
79 case Triple::Solaris:
81 case Triple::OpenBSD:
83 default:
84 return ELF::ELFOSABI_NONE;
85 }
86 }
87
88 virtual unsigned getRelocType(MCContext &Ctx, const MCValue &Target,
89 const MCFixup &Fixup, bool IsPCRel) const = 0;
90
91 virtual bool needsRelocateWithSymbol(const MCValue &Val, const MCSymbol &Sym,
92 unsigned Type) const;
93
94 virtual void sortRelocs(const MCAssembler &Asm,
95 std::vector<ELFRelocationEntry> &Relocs);
96
97 /// \name Accessors
98 /// @{
99 uint8_t getOSABI() const { return OSABI; }
100 uint8_t getABIVersion() const { return ABIVersion; }
101 uint16_t getEMachine() const { return EMachine; }
102 bool hasRelocationAddend() const { return HasRelocationAddend; }
103 bool is64Bit() const { return Is64Bit; }
104 /// @}
105
106 // Instead of changing everyone's API we pack the N64 Type fields
107 // into the existing 32 bit data unsigned.
108#define R_TYPE_SHIFT 0
109#define R_TYPE_MASK 0xffffff00
110#define R_TYPE2_SHIFT 8
111#define R_TYPE2_MASK 0xffff00ff
112#define R_TYPE3_SHIFT 16
113#define R_TYPE3_MASK 0xff00ffff
114#define R_SSYM_SHIFT 24
115#define R_SSYM_MASK 0x00ffffff
116
117 // N64 relocation type accessors
119 return (unsigned)((Type >> R_TYPE_SHIFT) & 0xff);
120 }
122 return (unsigned)((Type >> R_TYPE2_SHIFT) & 0xff);
123 }
125 return (unsigned)((Type >> R_TYPE3_SHIFT) & 0xff);
126 }
128 return (unsigned)((Type >> R_SSYM_SHIFT) & 0xff);
129 }
130
131 // N64 relocation type setting
132 static unsigned setRTypes(unsigned Value1, unsigned Value2, unsigned Value3) {
133 return ((Value1 & 0xff) << R_TYPE_SHIFT) |
134 ((Value2 & 0xff) << R_TYPE2_SHIFT) |
135 ((Value3 & 0xff) << R_TYPE3_SHIFT);
136 }
137 unsigned setRSsym(unsigned Value, unsigned Type) const {
138 return (Type & R_SSYM_MASK) | ((Value & 0xff) << R_SSYM_SHIFT);
139 }
140};
141
142class ELFObjectWriter final : public MCObjectWriter {
143 unsigned ELFHeaderEFlags = 0;
144
145public:
146 std::unique_ptr<MCELFObjectTargetWriter> TargetObjectWriter;
149
152 bool IsLittleEndian = false;
153 bool SeenGnuAbi = false;
154 std::optional<uint8_t> OverrideABIVersion;
155
156 struct Symver {
158 const MCSymbol *Sym;
160 // True if .symver *, *@@@* or .symver *, *, remove.
162 };
164
165 ELFObjectWriter(std::unique_ptr<MCELFObjectTargetWriter> MOTW,
167 ELFObjectWriter(std::unique_ptr<MCELFObjectTargetWriter> MOTW,
169 bool IsLittleEndian);
170
171 void reset() override;
172 void executePostLayoutBinding(MCAssembler &Asm) override;
173 void recordRelocation(MCAssembler &Asm, const MCFragment *Fragment,
174 const MCFixup &Fixup, MCValue Target,
175 uint64_t &FixedValue) override;
177 const MCSymbol &SymA,
178 const MCFragment &FB, bool InSet,
179 bool IsPCRel) const override;
180 uint64_t writeObject(MCAssembler &Asm) override;
181
182 bool hasRelocationAddend() const;
183 bool usesRela(const MCTargetOptions *TO, const MCSectionELF &Sec) const;
184
185 bool shouldRelocateWithSymbol(const MCAssembler &Asm, const MCValue &Val,
186 const MCSymbolELF *Sym, uint64_t C,
187 unsigned Type) const;
188
189 bool checkRelocation(MCContext &Ctx, SMLoc Loc, const MCSectionELF *From,
190 const MCSectionELF *To);
191
192 unsigned getELFHeaderEFlags() const { return ELFHeaderEFlags; }
193 void setELFHeaderEFlags(unsigned Flags) { ELFHeaderEFlags = Flags; }
194
195 // Mark that we have seen GNU ABI usage (e.g. SHF_GNU_RETAIN, STB_GNU_UNIQUE).
196 void markGnuAbi() { SeenGnuAbi = true; }
197 bool seenGnuAbi() const { return SeenGnuAbi; }
198
199 // Override the default e_ident[EI_ABIVERSION] in the ELF header.
201};
202} // end namespace llvm
203
204#endif // LLVM_MC_MCELFOBJECTWRITER_H
BlockVerifier::State From
#define LLVM_DUMP_METHOD
Mark debug helper function definitions like dump() that should not be stripped from debug builds.
Definition: Compiler.h:622
This file defines the DenseMap class.
Symbol * Sym
Definition: ELF_riscv.cpp:479
#define R_TYPE2_SHIFT
#define R_SSYM_MASK
#define R_TYPE_SHIFT
#define R_TYPE3_SHIFT
#define R_SSYM_SHIFT
PowerPC TLS Dynamic Call Fixup
This file defines the SmallVector class.
std::unique_ptr< MCELFObjectTargetWriter > TargetObjectWriter
unsigned getELFHeaderEFlags() const
bool shouldRelocateWithSymbol(const MCAssembler &Asm, const MCValue &Val, const MCSymbolELF *Sym, uint64_t C, unsigned Type) const
uint64_t writeObject(MCAssembler &Asm) override
Write the object file and returns the number of bytes written.
void executePostLayoutBinding(MCAssembler &Asm) override
Perform any late binding of symbols (for example, to assign symbol indices for use when generating re...
void reset() override
lifetime management
void setELFHeaderEFlags(unsigned Flags)
void setOverrideABIVersion(uint8_t V)
std::optional< uint8_t > OverrideABIVersion
void recordRelocation(MCAssembler &Asm, const MCFragment *Fragment, const MCFixup &Fixup, MCValue Target, uint64_t &FixedValue) override
Record a relocation entry.
DenseMap< const MCSectionELF *, std::vector< ELFRelocationEntry > > Relocations
SmallVector< Symver, 0 > Symvers
raw_pwrite_stream & OS
bool isSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, const MCSymbol &SymA, const MCFragment &FB, bool InSet, bool IsPCRel) const override
bool hasRelocationAddend() const
bool checkRelocation(MCContext &Ctx, SMLoc Loc, const MCSectionELF *From, const MCSectionELF *To)
raw_pwrite_stream * DwoOS
bool usesRela(const MCTargetOptions *TO, const MCSectionELF &Sec) const
DenseMap< const MCSymbolELF *, const MCSymbolELF * > Renames
Context object for machine code objects.
Definition: MCContext.h:83
static bool classof(const MCObjectTargetWriter *W)
virtual void sortRelocs(const MCAssembler &Asm, std::vector< ELFRelocationEntry > &Relocs)
virtual bool needsRelocateWithSymbol(const MCValue &Val, const MCSymbol &Sym, unsigned Type) const
Triple::ObjectFormatType getFormat() const override
static unsigned setRTypes(unsigned Value1, unsigned Value2, unsigned Value3)
uint8_t getRType(uint32_t Type) const
virtual unsigned getRelocType(MCContext &Ctx, const MCValue &Target, const MCFixup &Fixup, bool IsPCRel) const =0
uint8_t getRSsym(uint32_t Type) const
virtual ~MCELFObjectTargetWriter()=default
static uint8_t getOSABI(Triple::OSType OSType)
unsigned setRSsym(unsigned Value, unsigned Type) const
uint8_t getRType2(uint32_t Type) const
uint8_t getRType3(uint32_t Type) const
Encode information on a single operation to perform on a byte sequence (e.g., an encoded instruction)...
Definition: MCFixup.h:71
Base class for classes that define behaviour that is specific to both the target and the object forma...
Defines the object file and target independent interfaces used by the assembler backend to write nati...
This represents a section on linux, lots of unix variants and some bare metal systems.
Definition: MCSectionELF.h:27
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
Definition: MCSymbol.h:41
This represents an "assembler immediate".
Definition: MCValue.h:36
Represents a location in source code.
Definition: SMLoc.h:23
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Definition: SmallVector.h:1196
StringRef - Represent a constant reference to a string, i.e.
Definition: StringRef.h:51
Target - Wrapper for Target specific information.
@ HermitCore
Definition: Triple.h:234
ObjectFormatType
Definition: Triple.h:307
The instances of the Type class are immutable: once they are created, they are never changed.
Definition: Type.h:45
LLVM Value Representation.
Definition: Value.h:74
This class implements an extremely fast bulk output stream that can only output to a stream.
Definition: raw_ostream.h:52
An abstract base class for streams implementations that also support a pwrite operation.
Definition: raw_ostream.h:434
@ C
The default llvm calling convention, compatible with C.
Definition: CallingConv.h:34
@ ELFOSABI_OPENBSD
Definition: ELF.h:356
@ ELFOSABI_SOLARIS
Definition: ELF.h:350
@ ELFOSABI_FREEBSD
Definition: ELF.h:353
@ ELFOSABI_STANDALONE
Definition: ELF.h:371
@ ELFOSABI_NONE
Definition: ELF.h:344
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
ELFRelocationEntry(uint64_t Offset, const MCSymbolELF *Symbol, unsigned Type, uint64_t Addend)
const MCSymbolELF * Symbol
void print(raw_ostream &Out) const
LLVM_DUMP_METHOD void dump() const