xref: /freebsd/contrib/llvm-project/llvm/lib/MC/MCObjectStreamer.cpp (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
1 //===- lib/MC/MCObjectStreamer.cpp - Object File MCStreamer Interface -----===//
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 #include "llvm/MC/MCObjectStreamer.h"
10 #include "llvm/MC/MCAsmBackend.h"
11 #include "llvm/MC/MCAsmInfo.h"
12 #include "llvm/MC/MCAssembler.h"
13 #include "llvm/MC/MCCodeEmitter.h"
14 #include "llvm/MC/MCCodeView.h"
15 #include "llvm/MC/MCContext.h"
16 #include "llvm/MC/MCDwarf.h"
17 #include "llvm/MC/MCExpr.h"
18 #include "llvm/MC/MCObjectFileInfo.h"
19 #include "llvm/MC/MCObjectWriter.h"
20 #include "llvm/MC/MCSection.h"
21 #include "llvm/MC/MCSymbol.h"
22 #include "llvm/MC/MCValue.h"
23 #include "llvm/Support/ErrorHandling.h"
24 #include "llvm/Support/SourceMgr.h"
25 using namespace llvm;
26 
MCObjectStreamer(MCContext & Context,std::unique_ptr<MCAsmBackend> TAB,std::unique_ptr<MCObjectWriter> OW,std::unique_ptr<MCCodeEmitter> Emitter)27 MCObjectStreamer::MCObjectStreamer(MCContext &Context,
28                                    std::unique_ptr<MCAsmBackend> TAB,
29                                    std::unique_ptr<MCObjectWriter> OW,
30                                    std::unique_ptr<MCCodeEmitter> Emitter)
31     : MCStreamer(Context),
32       Assembler(std::make_unique<MCAssembler>(
33           Context, std::move(TAB), std::move(Emitter), std::move(OW))),
34       EmitEHFrame(true), EmitDebugFrame(false) {
35   assert(Assembler->getBackendPtr() && Assembler->getEmitterPtr());
36   setAllowAutoPadding(Assembler->getBackend().allowAutoPadding());
37   if (Context.getTargetOptions() && Context.getTargetOptions()->MCRelaxAll)
38     Assembler->setRelaxAll(true);
39 }
40 
41 MCObjectStreamer::~MCObjectStreamer() = default;
42 
getAssemblerPtr()43 MCAssembler *MCObjectStreamer::getAssemblerPtr() {
44   if (getUseAssemblerInfoForParsing())
45     return Assembler.get();
46   return nullptr;
47 }
48 
49 // When fixup's offset is a forward declared label, e.g.:
50 //
51 //   .reloc 1f, R_MIPS_JALR, foo
52 // 1: nop
53 //
54 // postpone adding it to Fixups vector until the label is defined and its offset
55 // is known.
resolvePendingFixups()56 void MCObjectStreamer::resolvePendingFixups() {
57   for (PendingMCFixup &PendingFixup : PendingFixups) {
58     if (!PendingFixup.Sym || PendingFixup.Sym->isUndefined ()) {
59       getContext().reportError(PendingFixup.Fixup.getLoc(),
60                                "unresolved relocation offset");
61       continue;
62     }
63     PendingFixup.Fixup.setOffset(PendingFixup.Sym->getOffset() +
64                                  PendingFixup.Fixup.getOffset());
65 
66     // If the location symbol to relocate is in MCEncodedFragmentWithFixups,
67     // put the Fixup into location symbol's fragment. Otherwise
68     // put into PendingFixup.DF
69     MCFragment *SymFragment = PendingFixup.Sym->getFragment();
70     switch (SymFragment->getKind()) {
71     case MCFragment::FT_Relaxable:
72     case MCFragment::FT_Dwarf:
73     case MCFragment::FT_PseudoProbe:
74       cast<MCEncodedFragmentWithFixups<8, 1>>(SymFragment)
75           ->getFixups()
76           .push_back(PendingFixup.Fixup);
77       break;
78     case MCFragment::FT_Data:
79     case MCFragment::FT_CVDefRange:
80       cast<MCEncodedFragmentWithFixups<32, 4>>(SymFragment)
81           ->getFixups()
82           .push_back(PendingFixup.Fixup);
83       break;
84     default:
85       PendingFixup.DF->getFixups().push_back(PendingFixup.Fixup);
86       break;
87     }
88   }
89   PendingFixups.clear();
90 }
91 
92 // As a compile-time optimization, avoid allocating and evaluating an MCExpr
93 // tree for (Hi - Lo) when Hi and Lo are offsets into the same fragment.
absoluteSymbolDiff(const MCSymbol * Hi,const MCSymbol * Lo)94 static std::optional<uint64_t> absoluteSymbolDiff(const MCSymbol *Hi,
95                                                   const MCSymbol *Lo) {
96   assert(Hi && Lo);
97   if (!Hi->getFragment() || Hi->getFragment() != Lo->getFragment() ||
98       Hi->isVariable() || Lo->isVariable())
99     return std::nullopt;
100 
101   return Hi->getOffset() - Lo->getOffset();
102 }
103 
emitAbsoluteSymbolDiff(const MCSymbol * Hi,const MCSymbol * Lo,unsigned Size)104 void MCObjectStreamer::emitAbsoluteSymbolDiff(const MCSymbol *Hi,
105                                               const MCSymbol *Lo,
106                                               unsigned Size) {
107   if (!getAssembler().getContext().getTargetTriple().isRISCV())
108     if (std::optional<uint64_t> Diff = absoluteSymbolDiff(Hi, Lo))
109       return emitIntValue(*Diff, Size);
110   MCStreamer::emitAbsoluteSymbolDiff(Hi, Lo, Size);
111 }
112 
emitAbsoluteSymbolDiffAsULEB128(const MCSymbol * Hi,const MCSymbol * Lo)113 void MCObjectStreamer::emitAbsoluteSymbolDiffAsULEB128(const MCSymbol *Hi,
114                                                        const MCSymbol *Lo) {
115   if (!getAssembler().getContext().getTargetTriple().isRISCV())
116     if (std::optional<uint64_t> Diff = absoluteSymbolDiff(Hi, Lo)) {
117       emitULEB128IntValue(*Diff);
118       return;
119     }
120   MCStreamer::emitAbsoluteSymbolDiffAsULEB128(Hi, Lo);
121 }
122 
reset()123 void MCObjectStreamer::reset() {
124   if (Assembler) {
125     Assembler->reset();
126     if (getContext().getTargetOptions())
127       Assembler->setRelaxAll(getContext().getTargetOptions()->MCRelaxAll);
128   }
129   EmitEHFrame = true;
130   EmitDebugFrame = false;
131   MCStreamer::reset();
132 }
133 
emitFrames(MCAsmBackend * MAB)134 void MCObjectStreamer::emitFrames(MCAsmBackend *MAB) {
135   if (!getNumFrameInfos())
136     return;
137 
138   if (EmitEHFrame)
139     MCDwarfFrameEmitter::Emit(*this, MAB, true);
140 
141   if (EmitDebugFrame)
142     MCDwarfFrameEmitter::Emit(*this, MAB, false);
143 }
144 
canReuseDataFragment(const MCDataFragment & F,const MCAssembler & Assembler,const MCSubtargetInfo * STI)145 static bool canReuseDataFragment(const MCDataFragment &F,
146                                  const MCAssembler &Assembler,
147                                  const MCSubtargetInfo *STI) {
148   if (!F.hasInstructions())
149     return true;
150   // Do not add data after a linker-relaxable instruction. The difference
151   // between a new label and a label at or before the linker-relaxable
152   // instruction cannot be resolved at assemble-time.
153   if (F.isLinkerRelaxable())
154     return false;
155   // When bundling is enabled, we don't want to add data to a fragment that
156   // already has instructions (see MCELFStreamer::emitInstToData for details)
157   if (Assembler.isBundlingEnabled())
158     return false;
159   // If the subtarget is changed mid fragment we start a new fragment to record
160   // the new STI.
161   return !STI || F.getSubtargetInfo() == STI;
162 }
163 
164 MCDataFragment *
getOrCreateDataFragment(const MCSubtargetInfo * STI)165 MCObjectStreamer::getOrCreateDataFragment(const MCSubtargetInfo *STI) {
166   auto *F = dyn_cast<MCDataFragment>(getCurrentFragment());
167   if (!F || !canReuseDataFragment(*F, *Assembler, STI)) {
168     F = getContext().allocFragment<MCDataFragment>();
169     insert(F);
170   }
171   return F;
172 }
173 
visitUsedSymbol(const MCSymbol & Sym)174 void MCObjectStreamer::visitUsedSymbol(const MCSymbol &Sym) {
175   Assembler->registerSymbol(Sym);
176 }
177 
emitCFISections(bool EH,bool Debug)178 void MCObjectStreamer::emitCFISections(bool EH, bool Debug) {
179   MCStreamer::emitCFISections(EH, Debug);
180   EmitEHFrame = EH;
181   EmitDebugFrame = Debug;
182 }
183 
emitValueImpl(const MCExpr * Value,unsigned Size,SMLoc Loc)184 void MCObjectStreamer::emitValueImpl(const MCExpr *Value, unsigned Size,
185                                      SMLoc Loc) {
186   MCStreamer::emitValueImpl(Value, Size, Loc);
187   MCDataFragment *DF = getOrCreateDataFragment();
188 
189   MCDwarfLineEntry::make(this, getCurrentSectionOnly());
190 
191   // Avoid fixups when possible.
192   int64_t AbsValue;
193   if (Value->evaluateAsAbsolute(AbsValue, getAssemblerPtr())) {
194     if (!isUIntN(8 * Size, AbsValue) && !isIntN(8 * Size, AbsValue)) {
195       getContext().reportError(
196           Loc, "value evaluated as " + Twine(AbsValue) + " is out of range.");
197       return;
198     }
199     emitIntValue(AbsValue, Size);
200     return;
201   }
202   DF->getFixups().push_back(
203       MCFixup::create(DF->getContents().size(), Value,
204                       MCFixup::getKindForSize(Size, false), Loc));
205   DF->getContents().resize(DF->getContents().size() + Size, 0);
206 }
207 
emitCFILabel()208 MCSymbol *MCObjectStreamer::emitCFILabel() {
209   MCSymbol *Label = getContext().createTempSymbol("cfi");
210   emitLabel(Label);
211   return Label;
212 }
213 
emitCFIStartProcImpl(MCDwarfFrameInfo & Frame)214 void MCObjectStreamer::emitCFIStartProcImpl(MCDwarfFrameInfo &Frame) {
215   // We need to create a local symbol to avoid relocations.
216   Frame.Begin = getContext().createTempSymbol();
217   emitLabel(Frame.Begin);
218 }
219 
emitCFIEndProcImpl(MCDwarfFrameInfo & Frame)220 void MCObjectStreamer::emitCFIEndProcImpl(MCDwarfFrameInfo &Frame) {
221   Frame.End = getContext().createTempSymbol();
222   emitLabel(Frame.End);
223 }
224 
emitLabel(MCSymbol * Symbol,SMLoc Loc)225 void MCObjectStreamer::emitLabel(MCSymbol *Symbol, SMLoc Loc) {
226   MCStreamer::emitLabel(Symbol, Loc);
227 
228   getAssembler().registerSymbol(*Symbol);
229 
230   // If there is a current fragment, mark the symbol as pointing into it.
231   // Otherwise queue the label and set its fragment pointer when we emit the
232   // next fragment.
233   MCDataFragment *F = getOrCreateDataFragment();
234   Symbol->setFragment(F);
235   Symbol->setOffset(F->getContents().size());
236 
237   emitPendingAssignments(Symbol);
238 }
239 
emitPendingAssignments(MCSymbol * Symbol)240 void MCObjectStreamer::emitPendingAssignments(MCSymbol *Symbol) {
241   auto Assignments = pendingAssignments.find(Symbol);
242   if (Assignments != pendingAssignments.end()) {
243     for (const PendingAssignment &A : Assignments->second)
244       emitAssignment(A.Symbol, A.Value);
245 
246     pendingAssignments.erase(Assignments);
247   }
248 }
249 
250 // Emit a label at a previously emitted fragment/offset position. This must be
251 // within the currently-active section.
emitLabelAtPos(MCSymbol * Symbol,SMLoc Loc,MCDataFragment & F,uint64_t Offset)252 void MCObjectStreamer::emitLabelAtPos(MCSymbol *Symbol, SMLoc Loc,
253                                       MCDataFragment &F, uint64_t Offset) {
254   assert(F.getParent() == getCurrentSectionOnly());
255   MCStreamer::emitLabel(Symbol, Loc);
256   getAssembler().registerSymbol(*Symbol);
257   Symbol->setFragment(&F);
258   Symbol->setOffset(Offset);
259 }
260 
emitULEB128Value(const MCExpr * Value)261 void MCObjectStreamer::emitULEB128Value(const MCExpr *Value) {
262   int64_t IntValue;
263   if (Value->evaluateAsAbsolute(IntValue, getAssemblerPtr())) {
264     emitULEB128IntValue(IntValue);
265     return;
266   }
267   insert(getContext().allocFragment<MCLEBFragment>(*Value, false));
268 }
269 
emitSLEB128Value(const MCExpr * Value)270 void MCObjectStreamer::emitSLEB128Value(const MCExpr *Value) {
271   int64_t IntValue;
272   if (Value->evaluateAsAbsolute(IntValue, getAssemblerPtr())) {
273     emitSLEB128IntValue(IntValue);
274     return;
275   }
276   insert(getContext().allocFragment<MCLEBFragment>(*Value, true));
277 }
278 
emitWeakReference(MCSymbol * Alias,const MCSymbol * Symbol)279 void MCObjectStreamer::emitWeakReference(MCSymbol *Alias,
280                                          const MCSymbol *Symbol) {
281   report_fatal_error("This file format doesn't support weak aliases.");
282 }
283 
changeSection(MCSection * Section,uint32_t Subsection)284 void MCObjectStreamer::changeSection(MCSection *Section, uint32_t Subsection) {
285   changeSectionImpl(Section, Subsection);
286 }
287 
changeSectionImpl(MCSection * Section,uint32_t Subsection)288 bool MCObjectStreamer::changeSectionImpl(MCSection *Section,
289                                          uint32_t Subsection) {
290   assert(Section && "Cannot switch to a null section!");
291   getContext().clearDwarfLocSeen();
292 
293   auto &Subsections = Section->Subsections;
294   size_t I = 0, E = Subsections.size();
295   while (I != E && Subsections[I].first < Subsection)
296     ++I;
297   // If the subsection number is not in the sorted Subsections list, create a
298   // new fragment list.
299   if (I == E || Subsections[I].first != Subsection) {
300     auto *F = getContext().allocFragment<MCDataFragment>();
301     F->setParent(Section);
302     Subsections.insert(Subsections.begin() + I,
303                        {Subsection, MCSection::FragList{F, F}});
304   }
305   Section->CurFragList = &Subsections[I].second;
306   CurFrag = Section->CurFragList->Tail;
307 
308   return getAssembler().registerSection(*Section);
309 }
310 
switchSectionNoPrint(MCSection * Section)311 void MCObjectStreamer::switchSectionNoPrint(MCSection *Section) {
312   MCStreamer::switchSectionNoPrint(Section);
313   changeSection(Section, 0);
314 }
315 
emitAssignment(MCSymbol * Symbol,const MCExpr * Value)316 void MCObjectStreamer::emitAssignment(MCSymbol *Symbol, const MCExpr *Value) {
317   getAssembler().registerSymbol(*Symbol);
318   MCStreamer::emitAssignment(Symbol, Value);
319   emitPendingAssignments(Symbol);
320 }
321 
emitConditionalAssignment(MCSymbol * Symbol,const MCExpr * Value)322 void MCObjectStreamer::emitConditionalAssignment(MCSymbol *Symbol,
323                                                  const MCExpr *Value) {
324   const MCSymbol *Target = &cast<MCSymbolRefExpr>(*Value).getSymbol();
325 
326   // If the symbol already exists, emit the assignment. Otherwise, emit it
327   // later only if the symbol is also emitted.
328   if (Target->isRegistered())
329     emitAssignment(Symbol, Value);
330   else
331     pendingAssignments[Target].push_back({Symbol, Value});
332 }
333 
mayHaveInstructions(MCSection & Sec) const334 bool MCObjectStreamer::mayHaveInstructions(MCSection &Sec) const {
335   return Sec.hasInstructions();
336 }
337 
emitInstruction(const MCInst & Inst,const MCSubtargetInfo & STI)338 void MCObjectStreamer::emitInstruction(const MCInst &Inst,
339                                        const MCSubtargetInfo &STI) {
340   const MCSection &Sec = *getCurrentSectionOnly();
341   if (Sec.isVirtualSection()) {
342     getContext().reportError(Inst.getLoc(), Twine(Sec.getVirtualSectionKind()) +
343                                                 " section '" + Sec.getName() +
344                                                 "' cannot have instructions");
345     return;
346   }
347   emitInstructionImpl(Inst, STI);
348 }
349 
emitInstructionImpl(const MCInst & Inst,const MCSubtargetInfo & STI)350 void MCObjectStreamer::emitInstructionImpl(const MCInst &Inst,
351                                            const MCSubtargetInfo &STI) {
352   MCStreamer::emitInstruction(Inst, STI);
353 
354   MCSection *Sec = getCurrentSectionOnly();
355   Sec->setHasInstructions(true);
356 
357   // Now that a machine instruction has been assembled into this section, make
358   // a line entry for any .loc directive that has been seen.
359   MCDwarfLineEntry::make(this, getCurrentSectionOnly());
360 
361   // If this instruction doesn't need relaxation, just emit it as data.
362   MCAssembler &Assembler = getAssembler();
363   MCAsmBackend &Backend = Assembler.getBackend();
364   if (!(Backend.mayNeedRelaxation(Inst, STI) ||
365         Backend.allowEnhancedRelaxation())) {
366     emitInstToData(Inst, STI);
367     return;
368   }
369 
370   // Otherwise, relax and emit it as data if either:
371   // - The RelaxAll flag was passed
372   // - Bundling is enabled and this instruction is inside a bundle-locked
373   //   group. We want to emit all such instructions into the same data
374   //   fragment.
375   if (Assembler.getRelaxAll() ||
376       (Assembler.isBundlingEnabled() && Sec->isBundleLocked())) {
377     MCInst Relaxed = Inst;
378     while (Backend.mayNeedRelaxation(Relaxed, STI))
379       Backend.relaxInstruction(Relaxed, STI);
380     emitInstToData(Relaxed, STI);
381     return;
382   }
383 
384   // Otherwise emit to a separate fragment.
385   emitInstToFragment(Inst, STI);
386 }
387 
emitInstToFragment(const MCInst & Inst,const MCSubtargetInfo & STI)388 void MCObjectStreamer::emitInstToFragment(const MCInst &Inst,
389                                           const MCSubtargetInfo &STI) {
390   // Always create a new, separate fragment here, because its size can change
391   // during relaxation.
392   MCRelaxableFragment *IF =
393       getContext().allocFragment<MCRelaxableFragment>(Inst, STI);
394   insert(IF);
395 
396   getAssembler().getEmitter().encodeInstruction(Inst, IF->getContents(),
397                                                 IF->getFixups(), STI);
398 }
399 
400 #ifndef NDEBUG
401 static const char *const BundlingNotImplementedMsg =
402   "Aligned bundling is not implemented for this object format";
403 #endif
404 
emitBundleAlignMode(Align Alignment)405 void MCObjectStreamer::emitBundleAlignMode(Align Alignment) {
406   llvm_unreachable(BundlingNotImplementedMsg);
407 }
408 
emitBundleLock(bool AlignToEnd)409 void MCObjectStreamer::emitBundleLock(bool AlignToEnd) {
410   llvm_unreachable(BundlingNotImplementedMsg);
411 }
412 
emitBundleUnlock()413 void MCObjectStreamer::emitBundleUnlock() {
414   llvm_unreachable(BundlingNotImplementedMsg);
415 }
416 
emitDwarfLocDirective(unsigned FileNo,unsigned Line,unsigned Column,unsigned Flags,unsigned Isa,unsigned Discriminator,StringRef FileName)417 void MCObjectStreamer::emitDwarfLocDirective(unsigned FileNo, unsigned Line,
418                                              unsigned Column, unsigned Flags,
419                                              unsigned Isa,
420                                              unsigned Discriminator,
421                                              StringRef FileName) {
422   // In case we see two .loc directives in a row, make sure the
423   // first one gets a line entry.
424   MCDwarfLineEntry::make(this, getCurrentSectionOnly());
425 
426   this->MCStreamer::emitDwarfLocDirective(FileNo, Line, Column, Flags, Isa,
427                                           Discriminator, FileName);
428 }
429 
buildSymbolDiff(MCObjectStreamer & OS,const MCSymbol * A,const MCSymbol * B,SMLoc Loc)430 static const MCExpr *buildSymbolDiff(MCObjectStreamer &OS, const MCSymbol *A,
431                                      const MCSymbol *B, SMLoc Loc) {
432   MCContext &Context = OS.getContext();
433   MCSymbolRefExpr::VariantKind Variant = MCSymbolRefExpr::VK_None;
434   const MCExpr *ARef = MCSymbolRefExpr::create(A, Variant, Context);
435   const MCExpr *BRef = MCSymbolRefExpr::create(B, Variant, Context);
436   const MCExpr *AddrDelta =
437       MCBinaryExpr::create(MCBinaryExpr::Sub, ARef, BRef, Context, Loc);
438   return AddrDelta;
439 }
440 
emitDwarfSetLineAddr(MCObjectStreamer & OS,MCDwarfLineTableParams Params,int64_t LineDelta,const MCSymbol * Label,int PointerSize)441 static void emitDwarfSetLineAddr(MCObjectStreamer &OS,
442                                  MCDwarfLineTableParams Params,
443                                  int64_t LineDelta, const MCSymbol *Label,
444                                  int PointerSize) {
445   // emit the sequence to set the address
446   OS.emitIntValue(dwarf::DW_LNS_extended_op, 1);
447   OS.emitULEB128IntValue(PointerSize + 1);
448   OS.emitIntValue(dwarf::DW_LNE_set_address, 1);
449   OS.emitSymbolValue(Label, PointerSize);
450 
451   // emit the sequence for the LineDelta (from 1) and a zero address delta.
452   MCDwarfLineAddr::Emit(&OS, Params, LineDelta, 0);
453 }
454 
emitDwarfAdvanceLineAddr(int64_t LineDelta,const MCSymbol * LastLabel,const MCSymbol * Label,unsigned PointerSize)455 void MCObjectStreamer::emitDwarfAdvanceLineAddr(int64_t LineDelta,
456                                                 const MCSymbol *LastLabel,
457                                                 const MCSymbol *Label,
458                                                 unsigned PointerSize) {
459   if (!LastLabel) {
460     emitDwarfSetLineAddr(*this, Assembler->getDWARFLinetableParams(), LineDelta,
461                          Label, PointerSize);
462     return;
463   }
464   const MCExpr *AddrDelta = buildSymbolDiff(*this, Label, LastLabel, SMLoc());
465   insert(getContext().allocFragment<MCDwarfLineAddrFragment>(LineDelta,
466                                                              *AddrDelta));
467 }
468 
emitDwarfLineEndEntry(MCSection * Section,MCSymbol * LastLabel)469 void MCObjectStreamer::emitDwarfLineEndEntry(MCSection *Section,
470                                              MCSymbol *LastLabel) {
471   // Emit a DW_LNE_end_sequence for the end of the section.
472   // Use the section end label to compute the address delta and use INT64_MAX
473   // as the line delta which is the signal that this is actually a
474   // DW_LNE_end_sequence.
475   MCSymbol *SectionEnd = endSection(Section);
476 
477   // Switch back the dwarf line section, in case endSection had to switch the
478   // section.
479   MCContext &Ctx = getContext();
480   switchSection(Ctx.getObjectFileInfo()->getDwarfLineSection());
481 
482   const MCAsmInfo *AsmInfo = Ctx.getAsmInfo();
483   emitDwarfAdvanceLineAddr(INT64_MAX, LastLabel, SectionEnd,
484                            AsmInfo->getCodePointerSize());
485 }
486 
emitDwarfAdvanceFrameAddr(const MCSymbol * LastLabel,const MCSymbol * Label,SMLoc Loc)487 void MCObjectStreamer::emitDwarfAdvanceFrameAddr(const MCSymbol *LastLabel,
488                                                  const MCSymbol *Label,
489                                                  SMLoc Loc) {
490   const MCExpr *AddrDelta = buildSymbolDiff(*this, Label, LastLabel, Loc);
491   insert(getContext().allocFragment<MCDwarfCallFrameFragment>(*AddrDelta));
492 }
493 
emitCVLocDirective(unsigned FunctionId,unsigned FileNo,unsigned Line,unsigned Column,bool PrologueEnd,bool IsStmt,StringRef FileName,SMLoc Loc)494 void MCObjectStreamer::emitCVLocDirective(unsigned FunctionId, unsigned FileNo,
495                                           unsigned Line, unsigned Column,
496                                           bool PrologueEnd, bool IsStmt,
497                                           StringRef FileName, SMLoc Loc) {
498   // Validate the directive.
499   if (!checkCVLocSection(FunctionId, FileNo, Loc))
500     return;
501 
502   // Emit a label at the current position and record it in the CodeViewContext.
503   MCSymbol *LineSym = getContext().createTempSymbol();
504   emitLabel(LineSym);
505   getContext().getCVContext().recordCVLoc(getContext(), LineSym, FunctionId,
506                                           FileNo, Line, Column, PrologueEnd,
507                                           IsStmt);
508 }
509 
emitCVLinetableDirective(unsigned FunctionId,const MCSymbol * Begin,const MCSymbol * End)510 void MCObjectStreamer::emitCVLinetableDirective(unsigned FunctionId,
511                                                 const MCSymbol *Begin,
512                                                 const MCSymbol *End) {
513   getContext().getCVContext().emitLineTableForFunction(*this, FunctionId, Begin,
514                                                        End);
515   this->MCStreamer::emitCVLinetableDirective(FunctionId, Begin, End);
516 }
517 
emitCVInlineLinetableDirective(unsigned PrimaryFunctionId,unsigned SourceFileId,unsigned SourceLineNum,const MCSymbol * FnStartSym,const MCSymbol * FnEndSym)518 void MCObjectStreamer::emitCVInlineLinetableDirective(
519     unsigned PrimaryFunctionId, unsigned SourceFileId, unsigned SourceLineNum,
520     const MCSymbol *FnStartSym, const MCSymbol *FnEndSym) {
521   getContext().getCVContext().emitInlineLineTableForFunction(
522       *this, PrimaryFunctionId, SourceFileId, SourceLineNum, FnStartSym,
523       FnEndSym);
524   this->MCStreamer::emitCVInlineLinetableDirective(
525       PrimaryFunctionId, SourceFileId, SourceLineNum, FnStartSym, FnEndSym);
526 }
527 
emitCVDefRangeDirective(ArrayRef<std::pair<const MCSymbol *,const MCSymbol * >> Ranges,StringRef FixedSizePortion)528 void MCObjectStreamer::emitCVDefRangeDirective(
529     ArrayRef<std::pair<const MCSymbol *, const MCSymbol *>> Ranges,
530     StringRef FixedSizePortion) {
531   getContext().getCVContext().emitDefRange(*this, Ranges, FixedSizePortion);
532   // Attach labels that were pending before we created the defrange fragment to
533   // the beginning of the new fragment.
534   this->MCStreamer::emitCVDefRangeDirective(Ranges, FixedSizePortion);
535 }
536 
emitCVStringTableDirective()537 void MCObjectStreamer::emitCVStringTableDirective() {
538   getContext().getCVContext().emitStringTable(*this);
539 }
emitCVFileChecksumsDirective()540 void MCObjectStreamer::emitCVFileChecksumsDirective() {
541   getContext().getCVContext().emitFileChecksums(*this);
542 }
543 
emitCVFileChecksumOffsetDirective(unsigned FileNo)544 void MCObjectStreamer::emitCVFileChecksumOffsetDirective(unsigned FileNo) {
545   getContext().getCVContext().emitFileChecksumOffset(*this, FileNo);
546 }
547 
emitBytes(StringRef Data)548 void MCObjectStreamer::emitBytes(StringRef Data) {
549   MCDwarfLineEntry::make(this, getCurrentSectionOnly());
550   MCDataFragment *DF = getOrCreateDataFragment();
551   DF->getContents().append(Data.begin(), Data.end());
552 }
553 
emitValueToAlignment(Align Alignment,int64_t Value,unsigned ValueSize,unsigned MaxBytesToEmit)554 void MCObjectStreamer::emitValueToAlignment(Align Alignment, int64_t Value,
555                                             unsigned ValueSize,
556                                             unsigned MaxBytesToEmit) {
557   if (MaxBytesToEmit == 0)
558     MaxBytesToEmit = Alignment.value();
559   insert(getContext().allocFragment<MCAlignFragment>(
560       Alignment, Value, ValueSize, MaxBytesToEmit));
561 
562   // Update the maximum alignment on the current section if necessary.
563   MCSection *CurSec = getCurrentSectionOnly();
564   CurSec->ensureMinAlignment(Alignment);
565 }
566 
emitCodeAlignment(Align Alignment,const MCSubtargetInfo * STI,unsigned MaxBytesToEmit)567 void MCObjectStreamer::emitCodeAlignment(Align Alignment,
568                                          const MCSubtargetInfo *STI,
569                                          unsigned MaxBytesToEmit) {
570   emitValueToAlignment(Alignment, 0, 1, MaxBytesToEmit);
571   cast<MCAlignFragment>(getCurrentFragment())->setEmitNops(true, STI);
572 }
573 
emitValueToOffset(const MCExpr * Offset,unsigned char Value,SMLoc Loc)574 void MCObjectStreamer::emitValueToOffset(const MCExpr *Offset,
575                                          unsigned char Value,
576                                          SMLoc Loc) {
577   insert(getContext().allocFragment<MCOrgFragment>(*Offset, Value, Loc));
578 }
579 
580 // Associate DTPRel32 fixup with data and resize data area
emitDTPRel32Value(const MCExpr * Value)581 void MCObjectStreamer::emitDTPRel32Value(const MCExpr *Value) {
582   MCDataFragment *DF = getOrCreateDataFragment();
583   DF->getFixups().push_back(MCFixup::create(DF->getContents().size(),
584                                             Value, FK_DTPRel_4));
585   DF->getContents().resize(DF->getContents().size() + 4, 0);
586 }
587 
588 // Associate DTPRel64 fixup with data and resize data area
emitDTPRel64Value(const MCExpr * Value)589 void MCObjectStreamer::emitDTPRel64Value(const MCExpr *Value) {
590   MCDataFragment *DF = getOrCreateDataFragment();
591   DF->getFixups().push_back(MCFixup::create(DF->getContents().size(),
592                                             Value, FK_DTPRel_8));
593   DF->getContents().resize(DF->getContents().size() + 8, 0);
594 }
595 
596 // Associate TPRel32 fixup with data and resize data area
emitTPRel32Value(const MCExpr * Value)597 void MCObjectStreamer::emitTPRel32Value(const MCExpr *Value) {
598   MCDataFragment *DF = getOrCreateDataFragment();
599   DF->getFixups().push_back(MCFixup::create(DF->getContents().size(),
600                                             Value, FK_TPRel_4));
601   DF->getContents().resize(DF->getContents().size() + 4, 0);
602 }
603 
604 // Associate TPRel64 fixup with data and resize data area
emitTPRel64Value(const MCExpr * Value)605 void MCObjectStreamer::emitTPRel64Value(const MCExpr *Value) {
606   MCDataFragment *DF = getOrCreateDataFragment();
607   DF->getFixups().push_back(MCFixup::create(DF->getContents().size(),
608                                             Value, FK_TPRel_8));
609   DF->getContents().resize(DF->getContents().size() + 8, 0);
610 }
611 
612 // Associate GPRel32 fixup with data and resize data area
emitGPRel32Value(const MCExpr * Value)613 void MCObjectStreamer::emitGPRel32Value(const MCExpr *Value) {
614   MCDataFragment *DF = getOrCreateDataFragment();
615   DF->getFixups().push_back(
616       MCFixup::create(DF->getContents().size(), Value, FK_GPRel_4));
617   DF->getContents().resize(DF->getContents().size() + 4, 0);
618 }
619 
620 // Associate GPRel64 fixup with data and resize data area
emitGPRel64Value(const MCExpr * Value)621 void MCObjectStreamer::emitGPRel64Value(const MCExpr *Value) {
622   MCDataFragment *DF = getOrCreateDataFragment();
623   DF->getFixups().push_back(
624       MCFixup::create(DF->getContents().size(), Value, FK_GPRel_4));
625   DF->getContents().resize(DF->getContents().size() + 8, 0);
626 }
627 
628 static std::optional<std::pair<bool, std::string>>
getOffsetAndDataFragment(const MCSymbol & Symbol,uint32_t & RelocOffset,MCDataFragment * & DF)629 getOffsetAndDataFragment(const MCSymbol &Symbol, uint32_t &RelocOffset,
630                          MCDataFragment *&DF) {
631   if (Symbol.isVariable()) {
632     const MCExpr *SymbolExpr = Symbol.getVariableValue();
633     MCValue OffsetVal;
634     if(!SymbolExpr->evaluateAsRelocatable(OffsetVal, nullptr, nullptr))
635       return std::make_pair(false,
636                             std::string("symbol in .reloc offset is not "
637                                         "relocatable"));
638     if (OffsetVal.isAbsolute()) {
639       RelocOffset = OffsetVal.getConstant();
640       MCFragment *Fragment = Symbol.getFragment();
641       // FIXME Support symbols with no DF. For example:
642       // .reloc .data, ENUM_VALUE, <some expr>
643       if (!Fragment || Fragment->getKind() != MCFragment::FT_Data)
644         return std::make_pair(false,
645                               std::string("symbol in offset has no data "
646                                           "fragment"));
647       DF = cast<MCDataFragment>(Fragment);
648       return std::nullopt;
649     }
650 
651     if (OffsetVal.getSymB())
652       return std::make_pair(false,
653                             std::string(".reloc symbol offset is not "
654                                         "representable"));
655 
656     const MCSymbolRefExpr &SRE = cast<MCSymbolRefExpr>(*OffsetVal.getSymA());
657     if (!SRE.getSymbol().isDefined())
658       return std::make_pair(false,
659                             std::string("symbol used in the .reloc offset is "
660                                         "not defined"));
661 
662     if (SRE.getSymbol().isVariable())
663       return std::make_pair(false,
664                             std::string("symbol used in the .reloc offset is "
665                                         "variable"));
666 
667     MCFragment *Fragment = SRE.getSymbol().getFragment();
668     // FIXME Support symbols with no DF. For example:
669     // .reloc .data, ENUM_VALUE, <some expr>
670     if (!Fragment || Fragment->getKind() != MCFragment::FT_Data)
671       return std::make_pair(false,
672                             std::string("symbol in offset has no data "
673                                         "fragment"));
674     RelocOffset = SRE.getSymbol().getOffset() + OffsetVal.getConstant();
675     DF = cast<MCDataFragment>(Fragment);
676   } else {
677     RelocOffset = Symbol.getOffset();
678     MCFragment *Fragment = Symbol.getFragment();
679     // FIXME Support symbols with no DF. For example:
680     // .reloc .data, ENUM_VALUE, <some expr>
681     if (!Fragment || Fragment->getKind() != MCFragment::FT_Data)
682       return std::make_pair(false,
683                             std::string("symbol in offset has no data "
684                                         "fragment"));
685     DF = cast<MCDataFragment>(Fragment);
686   }
687   return std::nullopt;
688 }
689 
690 std::optional<std::pair<bool, std::string>>
emitRelocDirective(const MCExpr & Offset,StringRef Name,const MCExpr * Expr,SMLoc Loc,const MCSubtargetInfo & STI)691 MCObjectStreamer::emitRelocDirective(const MCExpr &Offset, StringRef Name,
692                                      const MCExpr *Expr, SMLoc Loc,
693                                      const MCSubtargetInfo &STI) {
694   std::optional<MCFixupKind> MaybeKind =
695       Assembler->getBackend().getFixupKind(Name);
696   if (!MaybeKind)
697     return std::make_pair(true, std::string("unknown relocation name"));
698 
699   MCFixupKind Kind = *MaybeKind;
700   if (Expr)
701     visitUsedExpr(*Expr);
702   else
703     Expr =
704         MCSymbolRefExpr::create(getContext().createTempSymbol(), getContext());
705 
706   MCDataFragment *DF = getOrCreateDataFragment(&STI);
707   MCValue OffsetVal;
708   if (!Offset.evaluateAsRelocatable(OffsetVal, nullptr, nullptr))
709     return std::make_pair(false,
710                           std::string(".reloc offset is not relocatable"));
711   if (OffsetVal.isAbsolute()) {
712     if (OffsetVal.getConstant() < 0)
713       return std::make_pair(false, std::string(".reloc offset is negative"));
714     DF->getFixups().push_back(
715         MCFixup::create(OffsetVal.getConstant(), Expr, Kind, Loc));
716     return std::nullopt;
717   }
718   if (OffsetVal.getSymB())
719     return std::make_pair(false,
720                           std::string(".reloc offset is not representable"));
721 
722   const MCSymbolRefExpr &SRE = cast<MCSymbolRefExpr>(*OffsetVal.getSymA());
723   const MCSymbol &Symbol = SRE.getSymbol();
724   if (Symbol.isDefined()) {
725     uint32_t SymbolOffset = 0;
726     std::optional<std::pair<bool, std::string>> Error =
727         getOffsetAndDataFragment(Symbol, SymbolOffset, DF);
728 
729     if (Error != std::nullopt)
730       return Error;
731 
732     DF->getFixups().push_back(
733         MCFixup::create(SymbolOffset + OffsetVal.getConstant(),
734                         Expr, Kind, Loc));
735     return std::nullopt;
736   }
737 
738   PendingFixups.emplace_back(
739       &SRE.getSymbol(), DF,
740       MCFixup::create(OffsetVal.getConstant(), Expr, Kind, Loc));
741   return std::nullopt;
742 }
743 
emitFill(const MCExpr & NumBytes,uint64_t FillValue,SMLoc Loc)744 void MCObjectStreamer::emitFill(const MCExpr &NumBytes, uint64_t FillValue,
745                                 SMLoc Loc) {
746   assert(getCurrentSectionOnly() && "need a section");
747   insert(
748       getContext().allocFragment<MCFillFragment>(FillValue, 1, NumBytes, Loc));
749 }
750 
emitFill(const MCExpr & NumValues,int64_t Size,int64_t Expr,SMLoc Loc)751 void MCObjectStreamer::emitFill(const MCExpr &NumValues, int64_t Size,
752                                 int64_t Expr, SMLoc Loc) {
753   int64_t IntNumValues;
754   // Do additional checking now if we can resolve the value.
755   if (NumValues.evaluateAsAbsolute(IntNumValues, getAssemblerPtr())) {
756     if (IntNumValues < 0) {
757       getContext().getSourceManager()->PrintMessage(
758           Loc, SourceMgr::DK_Warning,
759           "'.fill' directive with negative repeat count has no effect");
760       return;
761     }
762     // Emit now if we can for better errors.
763     int64_t NonZeroSize = Size > 4 ? 4 : Size;
764     Expr &= ~0ULL >> (64 - NonZeroSize * 8);
765     for (uint64_t i = 0, e = IntNumValues; i != e; ++i) {
766       emitIntValue(Expr, NonZeroSize);
767       if (NonZeroSize < Size)
768         emitIntValue(0, Size - NonZeroSize);
769     }
770     return;
771   }
772 
773   // Otherwise emit as fragment.
774   assert(getCurrentSectionOnly() && "need a section");
775   insert(
776       getContext().allocFragment<MCFillFragment>(Expr, Size, NumValues, Loc));
777 }
778 
emitNops(int64_t NumBytes,int64_t ControlledNopLength,SMLoc Loc,const MCSubtargetInfo & STI)779 void MCObjectStreamer::emitNops(int64_t NumBytes, int64_t ControlledNopLength,
780                                 SMLoc Loc, const MCSubtargetInfo &STI) {
781   assert(getCurrentSectionOnly() && "need a section");
782   insert(getContext().allocFragment<MCNopsFragment>(
783       NumBytes, ControlledNopLength, Loc, STI));
784 }
785 
emitFileDirective(StringRef Filename)786 void MCObjectStreamer::emitFileDirective(StringRef Filename) {
787   MCAssembler &Asm = getAssembler();
788   Asm.getWriter().addFileName(Asm, Filename);
789 }
790 
emitFileDirective(StringRef Filename,StringRef CompilerVersion,StringRef TimeStamp,StringRef Description)791 void MCObjectStreamer::emitFileDirective(StringRef Filename,
792                                          StringRef CompilerVersion,
793                                          StringRef TimeStamp,
794                                          StringRef Description) {
795   MCObjectWriter &W = getAssembler().getWriter();
796   W.addFileName(getAssembler(), Filename);
797   if (CompilerVersion.size())
798     W.setCompilerVersion(CompilerVersion);
799   // TODO: add TimeStamp and Description to .file symbol table entry
800   // with the integrated assembler.
801 }
802 
emitAddrsig()803 void MCObjectStreamer::emitAddrsig() {
804   getAssembler().getWriter().emitAddrsigSection();
805 }
806 
emitAddrsigSym(const MCSymbol * Sym)807 void MCObjectStreamer::emitAddrsigSym(const MCSymbol *Sym) {
808   getAssembler().getWriter().addAddrsigSymbol(Sym);
809 }
810 
finishImpl()811 void MCObjectStreamer::finishImpl() {
812   getContext().RemapDebugPaths();
813 
814   // If we are generating dwarf for assembly source files dump out the sections.
815   if (getContext().getGenDwarfForAssembly())
816     MCGenDwarfInfo::Emit(this);
817 
818   // Dump out the dwarf file & directory tables and line tables.
819   MCDwarfLineTable::emit(this, getAssembler().getDWARFLinetableParams());
820 
821   // Emit pseudo probes for the current module.
822   MCPseudoProbeTable::emit(this);
823 
824   resolvePendingFixups();
825   getAssembler().Finish();
826 }
827