blob: bdf8e5d840b308d26c5b95aaa93233a0fb50f433 [file] [log] [blame]
Alex Bradbury89718422017-10-19 21:37:38 +00001//===-- RISCVAsmPrinter.cpp - RISCV LLVM assembly writer ------------------===//
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// This file contains a printer that converts from our internal representation
11// of machine-dependent LLVM code to the RISCV assembly language.
12//
13//===----------------------------------------------------------------------===//
14
15#include "RISCV.h"
16#include "InstPrinter/RISCVInstPrinter.h"
Sameer AbuAsalc1b0e662018-04-06 21:07:05 +000017#include "MCTargetDesc/RISCVMCExpr.h"
Alex Bradbury89718422017-10-19 21:37:38 +000018#include "RISCVTargetMachine.h"
19#include "llvm/CodeGen/AsmPrinter.h"
20#include "llvm/CodeGen/MachineConstantPool.h"
21#include "llvm/CodeGen/MachineFunctionPass.h"
22#include "llvm/CodeGen/MachineInstr.h"
23#include "llvm/CodeGen/MachineModuleInfo.h"
24#include "llvm/MC/MCAsmInfo.h"
25#include "llvm/MC/MCInst.h"
26#include "llvm/MC/MCStreamer.h"
27#include "llvm/MC/MCSymbol.h"
28#include "llvm/Support/TargetRegistry.h"
29#include "llvm/Support/raw_ostream.h"
30using namespace llvm;
31
32#define DEBUG_TYPE "asm-printer"
33
34namespace {
35class RISCVAsmPrinter : public AsmPrinter {
36public:
37 explicit RISCVAsmPrinter(TargetMachine &TM,
38 std::unique_ptr<MCStreamer> Streamer)
39 : AsmPrinter(TM, std::move(Streamer)) {}
40
41 StringRef getPassName() const override { return "RISCV Assembly Printer"; }
42
43 void EmitInstruction(const MachineInstr *MI) override;
44
Alex Bradbury9330e642018-01-10 20:05:09 +000045 bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,
46 unsigned AsmVariant, const char *ExtraCode,
47 raw_ostream &OS) override;
48 bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo,
49 unsigned AsmVariant, const char *ExtraCode,
50 raw_ostream &OS) override;
51
Sameer AbuAsalc1b0e662018-04-06 21:07:05 +000052 void EmitToStreamer(MCStreamer &S, const MCInst &Inst);
Alex Bradbury89718422017-10-19 21:37:38 +000053 bool emitPseudoExpansionLowering(MCStreamer &OutStreamer,
54 const MachineInstr *MI);
Alex Bradburyec8aa912017-11-08 13:24:21 +000055
56 // Wrapper needed for tblgenned pseudo lowering.
57 bool lowerOperand(const MachineOperand &MO, MCOperand &MCOp) const {
58 return LowerRISCVMachineOperandToMCOperand(MO, MCOp, *this);
59 }
Alex Bradbury89718422017-10-19 21:37:38 +000060};
61}
62
Sameer AbuAsalc1b0e662018-04-06 21:07:05 +000063#define GEN_COMPRESS_INSTR
64#include "RISCVGenCompressInstEmitter.inc"
65void RISCVAsmPrinter::EmitToStreamer(MCStreamer &S, const MCInst &Inst) {
66 MCInst CInst;
67 bool Res = compressInst(CInst, Inst, *TM.getMCSubtargetInfo(),
68 OutStreamer->getContext());
69 AsmPrinter::EmitToStreamer(*OutStreamer, Res ? CInst : Inst);
70}
71
Alex Bradbury89718422017-10-19 21:37:38 +000072// Simple pseudo-instructions have their lowering (with expansion to real
73// instructions) auto-generated.
74#include "RISCVGenMCPseudoLowering.inc"
75
76void RISCVAsmPrinter::EmitInstruction(const MachineInstr *MI) {
77 // Do any auto-generated pseudo lowerings.
78 if (emitPseudoExpansionLowering(*OutStreamer, MI))
79 return;
80
81 MCInst TmpInst;
Alex Bradburyec8aa912017-11-08 13:24:21 +000082 LowerRISCVMachineInstrToMCInst(MI, TmpInst, *this);
Alex Bradbury89718422017-10-19 21:37:38 +000083 EmitToStreamer(*OutStreamer, TmpInst);
84}
85
Alex Bradbury9330e642018-01-10 20:05:09 +000086bool RISCVAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,
87 unsigned AsmVariant,
88 const char *ExtraCode, raw_ostream &OS) {
89 if (AsmVariant != 0)
90 report_fatal_error("There are no defined alternate asm variants");
91
92 // First try the generic code, which knows about modifiers like 'c' and 'n'.
93 if (!AsmPrinter::PrintAsmOperand(MI, OpNo, AsmVariant, ExtraCode, OS))
94 return false;
95
96 if (!ExtraCode) {
97 const MachineOperand &MO = MI->getOperand(OpNo);
98 switch (MO.getType()) {
99 case MachineOperand::MO_Immediate:
100 OS << MO.getImm();
101 return false;
102 case MachineOperand::MO_Register:
103 OS << RISCVInstPrinter::getRegisterName(MO.getReg());
104 return false;
105 default:
106 break;
107 }
108 }
109
110 return true;
111}
112
113bool RISCVAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI,
114 unsigned OpNo, unsigned AsmVariant,
115 const char *ExtraCode,
116 raw_ostream &OS) {
117 if (AsmVariant != 0)
118 report_fatal_error("There are no defined alternate asm variants");
119
120 if (!ExtraCode) {
121 const MachineOperand &MO = MI->getOperand(OpNo);
122 // For now, we only support register memory operands in registers and
123 // assume there is no addend
124 if (!MO.isReg())
125 return true;
126
127 OS << "0(" << RISCVInstPrinter::getRegisterName(MO.getReg()) << ")";
128 return false;
129 }
130
131 return AsmPrinter::PrintAsmMemoryOperand(MI, OpNo, AsmVariant, ExtraCode, OS);
132}
133
Alex Bradbury89718422017-10-19 21:37:38 +0000134// Force static initialization.
135extern "C" void LLVMInitializeRISCVAsmPrinter() {
136 RegisterAsmPrinter<RISCVAsmPrinter> X(getTheRISCV32Target());
137 RegisterAsmPrinter<RISCVAsmPrinter> Y(getTheRISCV64Target());
138}