LLVM 23.0.0git
RISCVELFStreamer.cpp
Go to the documentation of this file.
1//===-- RISCVELFStreamer.cpp - RISC-V ELF Target Streamer Methods ---------===//
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// This file provides RISC-V specific target streamer methods.
10//
11//===----------------------------------------------------------------------===//
12
13#include "RISCVELFStreamer.h"
14#include "RISCVAsmBackend.h"
15#include "RISCVBaseInfo.h"
16#include "RISCVMCTargetDesc.h"
19#include "llvm/MC/MCAssembler.h"
21#include "llvm/MC/MCContext.h"
24
25using namespace llvm;
26
27// This part is for ELF object output.
29 const MCSubtargetInfo &STI)
30 : RISCVTargetStreamer(S), CurrentVendor("riscv") {
32 const FeatureBitset &Features = STI.getFeatureBits();
33 auto &MAB = static_cast<RISCVAsmBackend &>(MCA.getBackend());
35 MAB.getTargetOptions().getABIName()));
37
38 // Compute the initial ISA string. This serves two purposes:
39 // 1. Deduplication: subsequent .option arch/rvc/norvc directives compare
40 // against ArchString to avoid propagating redundant ISA updates.
41 // 2. Initial symbol: seed the streamer's active ISA so a "$x<ArchString>"
42 // mapping symbol is emitted before the first instruction, recording
43 // the full ISA in the object even when no .option directive is present.
45 STI.hasFeature(RISCV::Feature64Bit), Features)) {
46 InitialArchString = (*ParseResult)->toString();
47 ArchString = InitialArchString;
49 }
50}
51
53 std::unique_ptr<MCAsmBackend> MAB,
54 std::unique_ptr<MCObjectWriter> MOW,
55 std::unique_ptr<MCCodeEmitter> MCE)
56 : MCELFStreamer(C, std::move(MAB), std::move(MOW), std::move(MCE)) {}
57
61
63 if (Arch == ArchString)
64 return;
65 ArchString = std::string(Arch);
67}
68
70 ArchStringStack.push_back(ArchString);
71}
72
74 if (!ArchStringStack.empty())
75 setArchString(ArchStringStack.pop_back_val());
76}
77
86
87void RISCVTargetELFStreamer::emitAttribute(unsigned Attribute, unsigned Value) {
88 getStreamer().setAttributeItem(Attribute, Value, /*OverwriteExisting=*/true);
89}
90
91void RISCVTargetELFStreamer::emitTextAttribute(unsigned Attribute,
93 getStreamer().setAttributeItem(Attribute, String, /*OverwriteExisting=*/true);
94}
95
96void RISCVTargetELFStreamer::emitIntTextAttribute(unsigned Attribute,
97 unsigned IntValue,
98 StringRef StringValue) {
99 getStreamer().setAttributeItems(Attribute, IntValue, StringValue,
100 /*OverwriteExisting=*/true);
101}
102
104 RISCVELFStreamer &S = getStreamer();
105 if (S.Contents.empty())
106 return;
107
108 S.emitAttributesSection(CurrentVendor, ".riscv.attributes",
109 ELF::SHT_RISCV_ATTRIBUTES, AttributeSection);
110}
111
116
117 unsigned EFlags = W.getELFHeaderEFlags();
118
119 if (hasRVC())
120 EFlags |= ELF::EF_RISCV_RVC;
121 if (hasTSO())
122 EFlags |= ELF::EF_RISCV_TSO;
123
124 switch (ABI) {
129 break;
135 break;
141 break;
145 EFlags |= ELF::EF_RISCV_RVE;
146 break;
148 llvm_unreachable("Improperly initialised target ABI");
149 }
150
151 W.setELFHeaderEFlags(EFlags);
152}
153
155 AttributeSection = nullptr;
156 ArchString = InitialArchString;
157 ArchStringStack.clear();
158 // Re-seed the streamer's active ISA so the first instruction after reset
159 // still records the full ISA via "$x<ISA>", matching the behaviour set up
160 // in the constructor.
161 if (!InitialArchString.empty())
162 getStreamer().setMappingSymbolArch(InitialArchString);
163}
164
169
172 LastMappingSymbols.clear();
173 LastEMS = EMS_None;
174 MappingSymbolArch.clear();
175 LastEmittedArch.clear();
176 LastEmittedArchInSection.clear();
177 // Call target streamer reset last: it may call setMappingSymbolArch to
178 // re-seed the initial ISA after our state has been cleared.
179 static_cast<RISCVTargetStreamer *>(getTargetStreamer())->reset();
180}
181
182void RISCVELFStreamer::emitDataMappingSymbol() {
183 if (LastEMS == EMS_Data)
184 return;
185 emitMappingSymbol("$d");
186 LastEMS = EMS_Data;
187}
188
189void RISCVELFStreamer::emitInstructionsMappingSymbol() {
190 // Emit a mapping symbol at the start of each instruction run, and whenever
191 // the active ISA has changed since the last one emitted in this section.
192 // The symbol takes the form "$x<ISA>" when MappingSymbolArch is known, or
193 // plain "$x" as a fallback. The comparison with LastEmittedArch provides
194 // deduplication: repeating .option arch with the same ISA, or re-entering a
195 // section whose last mapping symbol already matches the active ISA, emits
196 // no redundant symbol.
197 bool NeedSymbol =
198 LastEMS != EMS_Instructions || LastEmittedArch != MappingSymbolArch;
199 if (NeedSymbol) {
200 if (MappingSymbolArch.empty())
201 emitMappingSymbol("$x");
202 else
203 emitMappingSymbol("$x" + MappingSymbolArch);
204 LastEmittedArch = MappingSymbolArch;
205 }
206 LastEMS = EMS_Instructions;
207}
208
209void RISCVELFStreamer::emitMappingSymbol(StringRef Name) {
210 auto *Symbol =
211 static_cast<MCSymbolELF *>(getContext().createLocalSymbol(Name));
212 emitLabel(Symbol);
213 Symbol->setType(ELF::STT_NOTYPE);
214 Symbol->setBinding(ELF::STB_LOCAL);
215}
216
218 MappingSymbolArch = std::string(Arch);
219}
220
222 // We have to keep track of the mapping symbol state of any sections we
223 // use. Each one should start off as EMS_None, which is provided as the
224 // default constructor by DenseMap::lookup. The last ISA suffix emitted in
225 // each section is also preserved so that re-entering a section only emits a
226 // new "$x<ISA>" symbol when the active ISA has actually changed.
227 const MCSection *Prev = getPreviousSection().first;
228 LastMappingSymbols[Prev] = LastEMS;
229 LastEmittedArchInSection[Prev] = LastEmittedArch;
230 LastEMS = LastMappingSymbols.lookup(Section);
231 auto It = LastEmittedArchInSection.find(Section);
232 LastEmittedArch = It != LastEmittedArchInSection.end() ? It->second : "";
233
234 MCELFStreamer::changeSection(Section, Subsection);
235}
236
238 const MCSubtargetInfo &STI) {
239 emitInstructionsMappingSymbol();
241}
242
244 emitDataMappingSymbol();
246}
247
248void RISCVELFStreamer::emitFill(const MCExpr &NumBytes, uint64_t FillValue,
249 SMLoc Loc) {
250 emitDataMappingSymbol();
251 MCELFStreamer::emitFill(NumBytes, FillValue, Loc);
252}
253
255 SMLoc Loc) {
256 emitDataMappingSymbol();
258}
259
261 std::unique_ptr<MCAsmBackend> &&MAB,
262 std::unique_ptr<MCObjectWriter> &&MOW,
263 std::unique_ptr<MCCodeEmitter> &&MCE) {
264 return new RISCVELFStreamer(C, std::move(MAB), std::move(MOW),
265 std::move(MCE));
266}
267
269 const uint32_t Feature1And) {
270 MCStreamer &OutStreamer = getStreamer();
271 MCContext &Ctx = OutStreamer.getContext();
272
273 const Triple &Triple = Ctx.getTargetTriple();
274 Align NoteAlign;
275 uint64_t DescSize;
276 if (Triple.isArch64Bit()) {
277 NoteAlign = Align(8);
278 DescSize = 16;
279 } else {
281 NoteAlign = Align(4);
282 DescSize = 12;
283 }
284
285 assert(Ctx.getObjectFileType() == MCContext::Environment::IsELF);
286 MCSection *const NoteSection =
287 Ctx.getELFSection(".note.gnu.property", ELF::SHT_NOTE, ELF::SHF_ALLOC);
288 OutStreamer.pushSection();
289 OutStreamer.switchSection(NoteSection);
290
291 // Emit the note header
292 OutStreamer.emitValueToAlignment(NoteAlign);
293 OutStreamer.emitIntValue(4, 4); // n_namsz
294 OutStreamer.emitIntValue(DescSize, 4); // n_descsz
295 OutStreamer.emitIntValue(ELF::NT_GNU_PROPERTY_TYPE_0, 4); // n_type
296 OutStreamer.emitBytes(StringRef("GNU", 4)); // n_name
297
298 // Emit n_desc field
299
300 // Emit the feature_1_and property
301 OutStreamer.emitIntValue(ELF::GNU_PROPERTY_RISCV_FEATURE_1_AND, 4); // pr_type
302 OutStreamer.emitIntValue(4, 4); // pr_datasz
303 OutStreamer.emitIntValue(Feature1And, 4); // pr_data
304 OutStreamer.emitValueToAlignment(NoteAlign); // pr_padding
305
306 OutStreamer.popSection();
307}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Functions, function parameters, and return types can have attributes to indicate how they should be t...
Definition Attributes.h:105
Container class for subtarget features.
MCAsmBackend & getBackend() const
LLVM_ABI bool registerSymbol(const MCSymbol &Symbol)
Context object for machine code objects.
Definition MCContext.h:83
LLVM_ABI MCSymbol * createLocalSymbol(StringRef Name)
Create a local, non-temporary symbol like an ELF mapping symbol.
SmallVector< AttributeItem, 64 > Contents
void emitAttributesSection(StringRef Vendor, const Twine &Section, unsigned Type, MCSection *&AttributeSection)
void changeSection(MCSection *Section, uint32_t Subsection=0) override
This is called by popSection and switchSection, if the current section changes.
void setAttributeItems(unsigned Attribute, unsigned IntValue, StringRef StringValue, bool OverwriteExisting)
ELFObjectWriter & getWriter()
void setAttributeItem(unsigned Attribute, unsigned Value, bool OverwriteExisting)
void reset() override
state management
void emitLabel(MCSymbol *Symbol, SMLoc Loc=SMLoc()) override
Emit a label for Symbol into the current section.
MCELFStreamer(MCContext &Context, std::unique_ptr< MCAsmBackend > TAB, std::unique_ptr< MCObjectWriter > OW, std::unique_ptr< MCCodeEmitter > Emitter)
Base class for the full range of assembler expressions which are needed for parsing.
Definition MCExpr.h:34
Instances of this class represent a single low-level machine instruction.
Definition MCInst.h:188
void emitFill(const MCExpr &NumBytes, uint64_t FillValue, SMLoc Loc=SMLoc()) override
Emit Size bytes worth of the value specified by FillValue.
MCAssembler & getAssembler()
void emitBytes(StringRef Data) override
Emit the bytes in Data into the output.
void emitInstruction(const MCInst &Inst, const MCSubtargetInfo &STI) override
Emit the given Instruction into the current section.
void emitValueImpl(const MCExpr *Value, unsigned Size, SMLoc Loc=SMLoc()) override
Emit the expression Value into the output as a native integer of the given Size bytes.
Instances of this class represent a uniqued identifier for a section in the current translation unit.
Definition MCSection.h:573
Streaming machine code generation interface.
Definition MCStreamer.h:222
MCSectionSubPair getPreviousSection() const
Return the previous section that the streamer is emitting code to.
Definition MCStreamer.h:433
virtual bool popSection()
Restore the current and previous section from the section stack.
MCContext & getContext() const
Definition MCStreamer.h:323
virtual void emitValueToAlignment(Align Alignment, int64_t Fill=0, uint8_t FillLen=1, unsigned MaxBytesToEmit=0)
Emit some number of copies of Value until the byte alignment ByteAlignment is reached.
MCTargetStreamer * getTargetStreamer()
Definition MCStreamer.h:333
virtual void emitIntValue(uint64_t Value, unsigned Size)
Special case of EmitValue that avoids the client having to pass in a MCExpr for constant integers.
void pushSection()
Save the current and previous section on the section stack.
Definition MCStreamer.h:450
virtual void switchSection(MCSection *Section, uint32_t Subsec=0)
Set the current section where code is being emitted to Section.
virtual void emitBytes(StringRef Data)
Emit the bytes in Data into the output.
Generic base class for all target subtargets.
bool hasFeature(unsigned Feature) const
const Triple & getTargetTriple() const
const FeatureBitset & getFeatureBits() const
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
Definition MCSymbol.h:42
MCStreamer & Streamer
Definition MCStreamer.h:97
This class represents success/failure for parsing-like operations that find it important to chain tog...
void setMappingSymbolArch(StringRef Arch)
void changeSection(MCSection *Section, uint32_t Subsection) override
This is called by popSection and switchSection, if the current section changes.
void emitBytes(StringRef Data) override
Emit the bytes in Data into the output.
RISCVELFStreamer(MCContext &C, std::unique_ptr< MCAsmBackend > MAB, std::unique_ptr< MCObjectWriter > MOW, std::unique_ptr< MCCodeEmitter > MCE)
void emitInstruction(const MCInst &Inst, const MCSubtargetInfo &STI) override
Emit the given Instruction into the current section.
void emitFill(const MCExpr &NumBytes, uint64_t FillValue, SMLoc Loc) override
Emit Size bytes worth of the value specified by FillValue.
void emitValueImpl(const MCExpr *Value, unsigned Size, SMLoc Loc) override
Emit the expression Value into the output as a native integer of the given Size bytes.
void setArchString(StringRef Arch) override
void emitNoteGnuPropertySection(const uint32_t Feature1And)
void emitDirectiveOptionNoExact() override
RISCVTargetELFStreamer(MCStreamer &S, const MCSubtargetInfo &STI)
void emitDirectiveVariantCC(MCSymbol &Symbol) override
RISCVELFStreamer & getStreamer()
void emitDirectiveOptionNoRelax() override
RISCVABI::ABI getTargetABI() const
void setFlagsFromFeatures(const MCSubtargetInfo &STI)
void setTargetABI(RISCVABI::ABI ABI)
Represents a location in source code.
Definition SMLoc.h:22
Represent a constant reference to a string, i.e.
Definition StringRef.h:56
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
LLVM_ABI bool isArch64Bit() const
Test whether the architecture is 64-bit.
Definition Triple.cpp:2058
LLVM_ABI bool isArch32Bit() const
Test whether the architecture is 32-bit.
Definition Triple.cpp:2062
LLVM Value Representation.
Definition Value.h:75
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ C
The default llvm calling convention, compatible with C.
Definition CallingConv.h:34
@ NT_GNU_PROPERTY_TYPE_0
Definition ELF.h:1813
@ SHF_ALLOC
Definition ELF.h:1251
@ SHT_RISCV_ATTRIBUTES
Definition ELF.h:1234
@ SHT_NOTE
Definition ELF.h:1156
@ STB_LOCAL
Definition ELF.h:1407
@ STT_NOTYPE
Definition ELF.h:1419
@ STO_RISCV_VARIANT_CC
Definition ELF.h:733
@ GNU_PROPERTY_RISCV_FEATURE_1_AND
Definition ELF.h:1849
@ EF_RISCV_RVE
Definition ELF.h:718
@ EF_RISCV_RVC
Definition ELF.h:712
@ EF_RISCV_TSO
Definition ELF.h:719
@ EF_RISCV_FLOAT_ABI_SINGLE
Definition ELF.h:715
@ EF_RISCV_FLOAT_ABI_DOUBLE
Definition ELF.h:716
ABI computeTargetABI(const Triple &TT, const FeatureBitset &FeatureBits, StringRef ABIName)
llvm::Expected< std::unique_ptr< RISCVISAInfo > > parseFeatureBits(bool IsRV64, const FeatureBitset &FeatureBits)
This is an optimization pass for GlobalISel generic memory operations.
MCStreamer * createRISCVELFStreamer(const Triple &, MCContext &C, std::unique_ptr< MCAsmBackend > &&MAB, std::unique_ptr< MCObjectWriter > &&MOW, std::unique_ptr< MCCodeEmitter > &&MCE)
FunctionAddr VTableAddr uintptr_t uintptr_t Data
Definition InstrProf.h:221
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1916
Implement std::hash so that hash_code can be used in STL containers.
Definition BitVector.h:874
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition Alignment.h:39