• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- SystemZInstPrinter.cpp - Convert SystemZ MCInst to assembly syntax ===//
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 #define DEBUG_TYPE "asm-printer"
11 
12 #include "SystemZInstPrinter.h"
13 #include "llvm/MC/MCExpr.h"
14 #include "llvm/MC/MCInstrInfo.h"
15 #include "llvm/Support/raw_ostream.h"
16 
17 using namespace llvm;
18 
19 #include "SystemZGenAsmWriter.inc"
20 
printAddress(unsigned Base,int64_t Disp,unsigned Index,raw_ostream & O)21 void SystemZInstPrinter::printAddress(unsigned Base, int64_t Disp,
22                                       unsigned Index, raw_ostream &O) {
23   O << Disp;
24   if (Base) {
25     O << '(';
26     if (Index)
27       O << '%' << getRegisterName(Index) << ',';
28     O << '%' << getRegisterName(Base) << ')';
29   } else
30     assert(!Index && "Shouldn't have an index without a base");
31 }
32 
printOperand(const MCOperand & MO,raw_ostream & O)33 void SystemZInstPrinter::printOperand(const MCOperand &MO, raw_ostream &O) {
34   if (MO.isReg())
35     O << '%' << getRegisterName(MO.getReg());
36   else if (MO.isImm())
37     O << MO.getImm();
38   else if (MO.isExpr())
39     O << *MO.getExpr();
40   else
41     llvm_unreachable("Invalid operand");
42 }
43 
printInst(const MCInst * MI,raw_ostream & O,StringRef Annot)44 void SystemZInstPrinter::printInst(const MCInst *MI, raw_ostream &O,
45                                    StringRef Annot) {
46   printInstruction(MI, O);
47   printAnnotation(O, Annot);
48 }
49 
printRegName(raw_ostream & O,unsigned RegNo) const50 void SystemZInstPrinter::printRegName(raw_ostream &O, unsigned RegNo) const {
51   O << '%' << getRegisterName(RegNo);
52 }
53 
printU4ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)54 void SystemZInstPrinter::printU4ImmOperand(const MCInst *MI, int OpNum,
55                                            raw_ostream &O) {
56   int64_t Value = MI->getOperand(OpNum).getImm();
57   assert(isUInt<4>(Value) && "Invalid u4imm argument");
58   O << Value;
59 }
60 
printU6ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)61 void SystemZInstPrinter::printU6ImmOperand(const MCInst *MI, int OpNum,
62                                            raw_ostream &O) {
63   int64_t Value = MI->getOperand(OpNum).getImm();
64   assert(isUInt<6>(Value) && "Invalid u6imm argument");
65   O << Value;
66 }
67 
printS8ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)68 void SystemZInstPrinter::printS8ImmOperand(const MCInst *MI, int OpNum,
69                                            raw_ostream &O) {
70   int64_t Value = MI->getOperand(OpNum).getImm();
71   assert(isInt<8>(Value) && "Invalid s8imm argument");
72   O << Value;
73 }
74 
printU8ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)75 void SystemZInstPrinter::printU8ImmOperand(const MCInst *MI, int OpNum,
76                                            raw_ostream &O) {
77   int64_t Value = MI->getOperand(OpNum).getImm();
78   assert(isUInt<8>(Value) && "Invalid u8imm argument");
79   O << Value;
80 }
81 
printS16ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)82 void SystemZInstPrinter::printS16ImmOperand(const MCInst *MI, int OpNum,
83                                             raw_ostream &O) {
84   int64_t Value = MI->getOperand(OpNum).getImm();
85   assert(isInt<16>(Value) && "Invalid s16imm argument");
86   O << Value;
87 }
88 
printU16ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)89 void SystemZInstPrinter::printU16ImmOperand(const MCInst *MI, int OpNum,
90                                             raw_ostream &O) {
91   int64_t Value = MI->getOperand(OpNum).getImm();
92   assert(isUInt<16>(Value) && "Invalid u16imm argument");
93   O << Value;
94 }
95 
printS32ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)96 void SystemZInstPrinter::printS32ImmOperand(const MCInst *MI, int OpNum,
97                                             raw_ostream &O) {
98   int64_t Value = MI->getOperand(OpNum).getImm();
99   assert(isInt<32>(Value) && "Invalid s32imm argument");
100   O << Value;
101 }
102 
printU32ImmOperand(const MCInst * MI,int OpNum,raw_ostream & O)103 void SystemZInstPrinter::printU32ImmOperand(const MCInst *MI, int OpNum,
104                                             raw_ostream &O) {
105   int64_t Value = MI->getOperand(OpNum).getImm();
106   assert(isUInt<32>(Value) && "Invalid u32imm argument");
107   O << Value;
108 }
109 
printAccessRegOperand(const MCInst * MI,int OpNum,raw_ostream & O)110 void SystemZInstPrinter::printAccessRegOperand(const MCInst *MI, int OpNum,
111                                                raw_ostream &O) {
112   uint64_t Value = MI->getOperand(OpNum).getImm();
113   assert(Value < 16 && "Invalid access register number");
114   O << "%a" << (unsigned int)Value;
115 }
116 
printPCRelOperand(const MCInst * MI,int OpNum,raw_ostream & O)117 void SystemZInstPrinter::printPCRelOperand(const MCInst *MI, int OpNum,
118                                            raw_ostream &O) {
119   const MCOperand &MO = MI->getOperand(OpNum);
120   if (MO.isImm()) {
121     O << "0x";
122     O.write_hex(MO.getImm());
123   } else
124     O << *MO.getExpr();
125 }
126 
printCallOperand(const MCInst * MI,int OpNum,raw_ostream & O)127 void SystemZInstPrinter::printCallOperand(const MCInst *MI, int OpNum,
128                                           raw_ostream &O) {
129   const MCOperand &MO = MI->getOperand(OpNum);
130   if (MO.isImm()) {
131     O << "0x";
132     O.write_hex(MO.getImm());
133   } else {
134     O << *MO.getExpr();
135     O << "@PLT";
136   }
137 }
138 
printOperand(const MCInst * MI,int OpNum,raw_ostream & O)139 void SystemZInstPrinter::printOperand(const MCInst *MI, int OpNum,
140                                       raw_ostream &O) {
141   printOperand(MI->getOperand(OpNum), O);
142 }
143 
printBDAddrOperand(const MCInst * MI,int OpNum,raw_ostream & O)144 void SystemZInstPrinter::printBDAddrOperand(const MCInst *MI, int OpNum,
145                                             raw_ostream &O) {
146   printAddress(MI->getOperand(OpNum).getReg(),
147                MI->getOperand(OpNum + 1).getImm(), 0, O);
148 }
149 
printBDXAddrOperand(const MCInst * MI,int OpNum,raw_ostream & O)150 void SystemZInstPrinter::printBDXAddrOperand(const MCInst *MI, int OpNum,
151                                              raw_ostream &O) {
152   printAddress(MI->getOperand(OpNum).getReg(),
153                MI->getOperand(OpNum + 1).getImm(),
154                MI->getOperand(OpNum + 2).getReg(), O);
155 }
156 
printBDLAddrOperand(const MCInst * MI,int OpNum,raw_ostream & O)157 void SystemZInstPrinter::printBDLAddrOperand(const MCInst *MI, int OpNum,
158                                              raw_ostream &O) {
159   unsigned Base = MI->getOperand(OpNum).getReg();
160   uint64_t Disp = MI->getOperand(OpNum + 1).getImm();
161   uint64_t Length = MI->getOperand(OpNum + 2).getImm();
162   O << Disp << '(' << Length;
163   if (Base)
164     O << ",%" << getRegisterName(Base);
165   O << ')';
166 }
167 
printCond4Operand(const MCInst * MI,int OpNum,raw_ostream & O)168 void SystemZInstPrinter::printCond4Operand(const MCInst *MI, int OpNum,
169                                            raw_ostream &O) {
170   static const char *const CondNames[] = {
171     "o", "h", "nle", "l", "nhe", "lh", "ne",
172     "e", "nlh", "he", "nl", "le", "nh", "no"
173   };
174   uint64_t Imm = MI->getOperand(OpNum).getImm();
175   assert(Imm > 0 && Imm < 15 && "Invalid condition");
176   O << CondNames[Imm - 1];
177 }
178