• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- HexagonAsmPrinter.h - Print machine code to an Hexagon .s file ----===//
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 // Hexagon Assembly printer class.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #ifndef HEXAGONASMPRINTER_H
15 #define HEXAGONASMPRINTER_H
16 
17 #include "Hexagon.h"
18 #include "HexagonTargetMachine.h"
19 #include "llvm/CodeGen/AsmPrinter.h"
20 #include "llvm/Support/Compiler.h"
21 #include "llvm/Support/raw_ostream.h"
22 
23 namespace llvm {
24   class HexagonAsmPrinter : public AsmPrinter {
25     const HexagonSubtarget *Subtarget;
26 
27   public:
HexagonAsmPrinter(TargetMachine & TM,MCStreamer & Streamer)28     explicit HexagonAsmPrinter(TargetMachine &TM, MCStreamer &Streamer)
29       : AsmPrinter(TM, Streamer) {
30       Subtarget = &TM.getSubtarget<HexagonSubtarget>();
31     }
32 
getPassName()33     virtual const char *getPassName() const {
34       return "Hexagon Assembly Printer";
35     }
36 
37     bool isBlockOnlyReachableByFallthrough(const MachineBasicBlock *MBB) const;
38 
39     virtual void EmitInstruction(const MachineInstr *MI);
40     virtual void EmitAlignment(unsigned NumBits,
41                                const GlobalValue *GV = 0) const;
42 
43     void printOperand(const MachineInstr *MI, unsigned OpNo, raw_ostream &O);
44     bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,
45                          unsigned AsmVariant, const char *ExtraCode,
46                          raw_ostream &OS);
47     bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo,
48                                unsigned AsmVariant, const char *ExtraCode,
49                                raw_ostream &OS);
50 
51     /// printInstruction - This method is automatically generated by tablegen
52     /// from the instruction set description.  This method returns true if the
53     /// machine instruction was sufficiently described to print it, otherwise it
54     /// returns false.
55     void printInstruction(const MachineInstr *MI, raw_ostream &O);
56 
57     //    void printMachineInstruction(const MachineInstr *MI);
58     void printOp(const MachineOperand &MO, raw_ostream &O);
59 
60     /// printRegister - Print register according to target requirements.
61     ///
printRegister(const MachineOperand & MO,bool R0AsZero,raw_ostream & O)62     void printRegister(const MachineOperand &MO, bool R0AsZero,
63                        raw_ostream &O) {
64       unsigned RegNo = MO.getReg();
65       assert(TargetRegisterInfo::isPhysicalRegister(RegNo) && "Not physreg??");
66       O << getRegisterName(RegNo);
67     }
68 
printImmOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)69     void printImmOperand(const MachineInstr *MI, unsigned OpNo,
70                                 raw_ostream &O) {
71       int value = MI->getOperand(OpNo).getImm();
72       O << value;
73     }
74 
printNegImmOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)75     void printNegImmOperand(const MachineInstr *MI, unsigned OpNo,
76                                    raw_ostream &O) {
77       int value = MI->getOperand(OpNo).getImm();
78       O << -value;
79     }
80 
printMEMriOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)81     void printMEMriOperand(const MachineInstr *MI, unsigned OpNo,
82                                   raw_ostream &O) {
83       const MachineOperand &MO1 = MI->getOperand(OpNo);
84       const MachineOperand &MO2 = MI->getOperand(OpNo+1);
85 
86       O << getRegisterName(MO1.getReg())
87         << " + #"
88         << (int) MO2.getImm();
89     }
90 
printFrameIndexOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)91     void printFrameIndexOperand(const MachineInstr *MI, unsigned OpNo,
92                                        raw_ostream &O) {
93       const MachineOperand &MO1 = MI->getOperand(OpNo);
94       const MachineOperand &MO2 = MI->getOperand(OpNo+1);
95 
96       O << getRegisterName(MO1.getReg())
97         << ", #"
98         << MO2.getImm();
99     }
100 
printBranchOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)101     void printBranchOperand(const MachineInstr *MI, unsigned OpNo,
102                             raw_ostream &O) {
103       // Branches can take an immediate operand.  This is used by the branch
104       // selection pass to print $+8, an eight byte displacement from the PC.
105       if (MI->getOperand(OpNo).isImm()) {
106         O << "$+" << MI->getOperand(OpNo).getImm()*4;
107       } else {
108         printOp(MI->getOperand(OpNo), O);
109       }
110     }
111 
printCallOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)112     void printCallOperand(const MachineInstr *MI, unsigned OpNo,
113                           raw_ostream &O) {
114     }
115 
printAbsAddrOperand(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)116     void printAbsAddrOperand(const MachineInstr *MI, unsigned OpNo,
117                              raw_ostream &O) {
118     }
119 
printSymbolHi(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)120     void printSymbolHi(const MachineInstr *MI, unsigned OpNo, raw_ostream &O) {
121       O << "#HI(";
122       if (MI->getOperand(OpNo).isImm()) {
123         printImmOperand(MI, OpNo, O);
124       }
125       else {
126         printOp(MI->getOperand(OpNo), O);
127       }
128       O << ")";
129     }
130 
printSymbolLo(const MachineInstr * MI,unsigned OpNo,raw_ostream & O)131     void printSymbolLo(const MachineInstr *MI, unsigned OpNo, raw_ostream &O) {
132       O << "#HI(";
133       if (MI->getOperand(OpNo).isImm()) {
134         printImmOperand(MI, OpNo, O);
135       }
136       else {
137         printOp(MI->getOperand(OpNo), O);
138       }
139       O << ")";
140     }
141 
142     void printPredicateOperand(const MachineInstr *MI, unsigned OpNo,
143                                raw_ostream &O);
144 
145 #if 0
146     void printModuleLevelGV(const GlobalVariable* GVar, raw_ostream &O);
147 #endif
148 
149     void printAddrModeBasePlusOffset(const MachineInstr *MI, int OpNo,
150                                      raw_ostream &O);
151 
152     void printGlobalOperand(const MachineInstr *MI, int OpNo, raw_ostream &O);
153     void printJumpTable(const MachineInstr *MI, int OpNo, raw_ostream &O);
154     void printConstantPool(const MachineInstr *MI, int OpNo, raw_ostream &O);
155 
156     static const char *getRegisterName(unsigned RegNo);
157 
158 #if 0
159     void EmitStartOfAsmFile(Module &M);
160 #endif
161   };
162 
163 } // end of llvm namespace
164 
165 #endif
166