• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-------- MipsELFStreamer.cpp - ELF Object Output ---------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "MipsELFStreamer.h"
11 #include "MipsOptionRecord.h"
12 #include "MipsTargetStreamer.h"
13 #include "llvm/BinaryFormat/ELF.h"
14 #include "llvm/MC/MCAsmBackend.h"
15 #include "llvm/MC/MCAssembler.h"
16 #include "llvm/MC/MCCodeEmitter.h"
17 #include "llvm/MC/MCContext.h"
18 #include "llvm/MC/MCDwarf.h"
19 #include "llvm/MC/MCInst.h"
20 #include "llvm/MC/MCObjectWriter.h"
21 #include "llvm/MC/MCSymbolELF.h"
22 #include "llvm/Support/Casting.h"
23 
24 using namespace llvm;
25 
MipsELFStreamer(MCContext & Context,std::unique_ptr<MCAsmBackend> MAB,std::unique_ptr<MCObjectWriter> OW,std::unique_ptr<MCCodeEmitter> Emitter)26 MipsELFStreamer::MipsELFStreamer(MCContext &Context,
27                                  std::unique_ptr<MCAsmBackend> MAB,
28                                  std::unique_ptr<MCObjectWriter> OW,
29                                  std::unique_ptr<MCCodeEmitter> Emitter)
30     : MCELFStreamer(Context, std::move(MAB), std::move(OW),
31                     std::move(Emitter)) {
32   RegInfoRecord = new MipsRegInfoRecord(this, Context);
33   MipsOptionRecords.push_back(
34       std::unique_ptr<MipsRegInfoRecord>(RegInfoRecord));
35 }
36 
EmitInstruction(const MCInst & Inst,const MCSubtargetInfo & STI,bool)37 void MipsELFStreamer::EmitInstruction(const MCInst &Inst,
38                                       const MCSubtargetInfo &STI, bool) {
39   MCELFStreamer::EmitInstruction(Inst, STI);
40 
41   MCContext &Context = getContext();
42   const MCRegisterInfo *MCRegInfo = Context.getRegisterInfo();
43 
44   for (unsigned OpIndex = 0; OpIndex < Inst.getNumOperands(); ++OpIndex) {
45     const MCOperand &Op = Inst.getOperand(OpIndex);
46 
47     if (!Op.isReg())
48       continue;
49 
50     unsigned Reg = Op.getReg();
51     RegInfoRecord->SetPhysRegUsed(Reg, MCRegInfo);
52   }
53 
54   createPendingLabelRelocs();
55 }
56 
EmitCFIStartProcImpl(MCDwarfFrameInfo & Frame)57 void MipsELFStreamer::EmitCFIStartProcImpl(MCDwarfFrameInfo &Frame) {
58   Frame.Begin = getContext().createTempSymbol();
59   MCELFStreamer::EmitLabel(Frame.Begin);
60 }
61 
EmitCFILabel()62 MCSymbol *MipsELFStreamer::EmitCFILabel() {
63   MCSymbol *Label = getContext().createTempSymbol("cfi", true);
64   MCELFStreamer::EmitLabel(Label);
65   return Label;
66 }
67 
EmitCFIEndProcImpl(MCDwarfFrameInfo & Frame)68 void MipsELFStreamer::EmitCFIEndProcImpl(MCDwarfFrameInfo &Frame) {
69   Frame.End = getContext().createTempSymbol();
70   MCELFStreamer::EmitLabel(Frame.End);
71 }
72 
createPendingLabelRelocs()73 void MipsELFStreamer::createPendingLabelRelocs() {
74   MipsTargetELFStreamer *ELFTargetStreamer =
75       static_cast<MipsTargetELFStreamer *>(getTargetStreamer());
76 
77   // FIXME: Also mark labels when in MIPS16 mode.
78   if (ELFTargetStreamer->isMicroMipsEnabled()) {
79     for (auto *L : Labels) {
80       auto *Label = cast<MCSymbolELF>(L);
81       getAssembler().registerSymbol(*Label);
82       Label->setOther(ELF::STO_MIPS_MICROMIPS);
83     }
84   }
85 
86   Labels.clear();
87 }
88 
EmitLabel(MCSymbol * Symbol,SMLoc Loc)89 void MipsELFStreamer::EmitLabel(MCSymbol *Symbol, SMLoc Loc) {
90   MCELFStreamer::EmitLabel(Symbol);
91   Labels.push_back(Symbol);
92 }
93 
SwitchSection(MCSection * Section,const MCExpr * Subsection)94 void MipsELFStreamer::SwitchSection(MCSection *Section,
95                                     const MCExpr *Subsection) {
96   MCELFStreamer::SwitchSection(Section, Subsection);
97   Labels.clear();
98 }
99 
EmitValueImpl(const MCExpr * Value,unsigned Size,SMLoc Loc)100 void MipsELFStreamer::EmitValueImpl(const MCExpr *Value, unsigned Size,
101                                     SMLoc Loc) {
102   MCELFStreamer::EmitValueImpl(Value, Size, Loc);
103   Labels.clear();
104 }
105 
EmitIntValue(uint64_t Value,unsigned Size)106 void MipsELFStreamer::EmitIntValue(uint64_t Value, unsigned Size) {
107   MCELFStreamer::EmitIntValue(Value, Size);
108   Labels.clear();
109 }
110 
EmitMipsOptionRecords()111 void MipsELFStreamer::EmitMipsOptionRecords() {
112   for (const auto &I : MipsOptionRecords)
113     I->EmitMipsOptionRecord();
114 }
115 
createMipsELFStreamer(MCContext & Context,std::unique_ptr<MCAsmBackend> MAB,std::unique_ptr<MCObjectWriter> OW,std::unique_ptr<MCCodeEmitter> Emitter,bool RelaxAll)116 MCELFStreamer *llvm::createMipsELFStreamer(
117     MCContext &Context, std::unique_ptr<MCAsmBackend> MAB,
118     std::unique_ptr<MCObjectWriter> OW, std::unique_ptr<MCCodeEmitter> Emitter,
119     bool RelaxAll) {
120   return new MipsELFStreamer(Context, std::move(MAB), std::move(OW),
121                              std::move(Emitter));
122 }
123