1 //===-- Mips16ISelDAGToDAG.cpp - A Dag to Dag Inst Selector for Mips16 ----===//
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 // Subclass of MipsDAGToDAGISel specialized for mips16.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "Mips16ISelDAGToDAG.h"
15 #include "MCTargetDesc/MipsBaseInfo.h"
16 #include "Mips.h"
17 #include "MipsAnalyzeImmediate.h"
18 #include "MipsMachineFunction.h"
19 #include "MipsRegisterInfo.h"
20 #include "llvm/CodeGen/MachineConstantPool.h"
21 #include "llvm/CodeGen/MachineFrameInfo.h"
22 #include "llvm/CodeGen/MachineFunction.h"
23 #include "llvm/CodeGen/MachineInstrBuilder.h"
24 #include "llvm/CodeGen/MachineRegisterInfo.h"
25 #include "llvm/CodeGen/SelectionDAGNodes.h"
26 #include "llvm/IR/CFG.h"
27 #include "llvm/IR/GlobalValue.h"
28 #include "llvm/IR/Instructions.h"
29 #include "llvm/IR/Intrinsics.h"
30 #include "llvm/IR/Type.h"
31 #include "llvm/Support/Debug.h"
32 #include "llvm/Support/ErrorHandling.h"
33 #include "llvm/Support/raw_ostream.h"
34 #include "llvm/Target/TargetMachine.h"
35 using namespace llvm;
36 
37 #define DEBUG_TYPE "mips-isel"
38 
runOnMachineFunction(MachineFunction & MF)39 bool Mips16DAGToDAGISel::runOnMachineFunction(MachineFunction &MF) {
40   Subtarget = &static_cast<const MipsSubtarget &>(MF.getSubtarget());
41   if (!Subtarget->inMips16Mode())
42     return false;
43   return MipsDAGToDAGISel::runOnMachineFunction(MF);
44 }
45 /// Select multiply instructions.
46 std::pair<SDNode*, SDNode*>
selectMULT(SDNode * N,unsigned Opc,SDLoc DL,EVT Ty,bool HasLo,bool HasHi)47 Mips16DAGToDAGISel::selectMULT(SDNode *N, unsigned Opc, SDLoc DL, EVT Ty,
48                                bool HasLo, bool HasHi) {
49   SDNode *Lo = nullptr, *Hi = nullptr;
50   SDNode *Mul = CurDAG->getMachineNode(Opc, DL, MVT::Glue, N->getOperand(0),
51                                        N->getOperand(1));
52   SDValue InFlag = SDValue(Mul, 0);
53 
54   if (HasLo) {
55     unsigned Opcode = Mips::Mflo16;
56     Lo = CurDAG->getMachineNode(Opcode, DL, Ty, MVT::Glue, InFlag);
57     InFlag = SDValue(Lo, 1);
58   }
59   if (HasHi) {
60     unsigned Opcode = Mips::Mfhi16;
61     Hi = CurDAG->getMachineNode(Opcode, DL, Ty, InFlag);
62   }
63   return std::make_pair(Lo, Hi);
64 }
65 
initGlobalBaseReg(MachineFunction & MF)66 void Mips16DAGToDAGISel::initGlobalBaseReg(MachineFunction &MF) {
67   MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
68 
69   if (!MipsFI->globalBaseRegSet())
70     return;
71 
72   MachineBasicBlock &MBB = MF.front();
73   MachineBasicBlock::iterator I = MBB.begin();
74   MachineRegisterInfo &RegInfo = MF.getRegInfo();
75   const TargetInstrInfo &TII = *Subtarget->getInstrInfo();
76   DebugLoc DL = I != MBB.end() ? I->getDebugLoc() : DebugLoc();
77   unsigned V0, V1, V2, GlobalBaseReg = MipsFI->getGlobalBaseReg();
78   const TargetRegisterClass *RC = &Mips::CPU16RegsRegClass;
79 
80   V0 = RegInfo.createVirtualRegister(RC);
81   V1 = RegInfo.createVirtualRegister(RC);
82   V2 = RegInfo.createVirtualRegister(RC);
83 
84   BuildMI(MBB, I, DL, TII.get(Mips::GotPrologue16), V0).
85     addReg(V1, RegState::Define).
86     addExternalSymbol("_gp_disp", MipsII::MO_ABS_HI).
87     addExternalSymbol("_gp_disp", MipsII::MO_ABS_LO);
88 
89   BuildMI(MBB, I, DL, TII.get(Mips::SllX16), V2).addReg(V0).addImm(16);
90   BuildMI(MBB, I, DL, TII.get(Mips::AdduRxRyRz16), GlobalBaseReg)
91     .addReg(V1).addReg(V2);
92 }
93 
94 // Insert instructions to initialize the Mips16 SP Alias register in the
95 // first MBB of the function.
96 //
initMips16SPAliasReg(MachineFunction & MF)97 void Mips16DAGToDAGISel::initMips16SPAliasReg(MachineFunction &MF) {
98   MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
99 
100   if (!MipsFI->mips16SPAliasRegSet())
101     return;
102 
103   MachineBasicBlock &MBB = MF.front();
104   MachineBasicBlock::iterator I = MBB.begin();
105   const TargetInstrInfo &TII = *Subtarget->getInstrInfo();
106   DebugLoc DL = I != MBB.end() ? I->getDebugLoc() : DebugLoc();
107   unsigned Mips16SPAliasReg = MipsFI->getMips16SPAliasReg();
108 
109   BuildMI(MBB, I, DL, TII.get(Mips::MoveR3216), Mips16SPAliasReg)
110     .addReg(Mips::SP);
111 }
112 
processFunctionAfterISel(MachineFunction & MF)113 void Mips16DAGToDAGISel::processFunctionAfterISel(MachineFunction &MF) {
114   initGlobalBaseReg(MF);
115   initMips16SPAliasReg(MF);
116 }
117 
118 /// getMips16SPAliasReg - Output the instructions required to put the
119 /// SP into a Mips16 accessible aliased register.
getMips16SPAliasReg()120 SDValue Mips16DAGToDAGISel::getMips16SPAliasReg() {
121   unsigned Mips16SPAliasReg =
122     MF->getInfo<MipsFunctionInfo>()->getMips16SPAliasReg();
123   return CurDAG->getRegister(Mips16SPAliasReg,
124                              getTargetLowering()->getPointerTy());
125 }
126 
getMips16SPRefReg(SDNode * Parent,SDValue & AliasReg)127 void Mips16DAGToDAGISel::getMips16SPRefReg(SDNode *Parent, SDValue &AliasReg) {
128   SDValue AliasFPReg = CurDAG->getRegister(Mips::S0,
129                                            getTargetLowering()->getPointerTy());
130   if (Parent) {
131     switch (Parent->getOpcode()) {
132       case ISD::LOAD: {
133         LoadSDNode *SD = dyn_cast<LoadSDNode>(Parent);
134         switch (SD->getMemoryVT().getSizeInBits()) {
135         case 8:
136         case 16:
137           AliasReg = Subtarget->getFrameLowering()->hasFP(*MF)
138                          ? AliasFPReg
139                          : getMips16SPAliasReg();
140           return;
141         }
142         break;
143       }
144       case ISD::STORE: {
145         StoreSDNode *SD = dyn_cast<StoreSDNode>(Parent);
146         switch (SD->getMemoryVT().getSizeInBits()) {
147         case 8:
148         case 16:
149           AliasReg = Subtarget->getFrameLowering()->hasFP(*MF)
150                          ? AliasFPReg
151                          : getMips16SPAliasReg();
152           return;
153         }
154         break;
155       }
156     }
157   }
158   AliasReg = CurDAG->getRegister(Mips::SP, getTargetLowering()->getPointerTy());
159   return;
160 
161 }
162 
selectAddr16(SDNode * Parent,SDValue Addr,SDValue & Base,SDValue & Offset,SDValue & Alias)163 bool Mips16DAGToDAGISel::selectAddr16(
164   SDNode *Parent, SDValue Addr, SDValue &Base, SDValue &Offset,
165   SDValue &Alias) {
166   EVT ValTy = Addr.getValueType();
167 
168   Alias = CurDAG->getTargetConstant(0, ValTy);
169 
170   // if Address is FI, get the TargetFrameIndex.
171   if (FrameIndexSDNode *FIN = dyn_cast<FrameIndexSDNode>(Addr)) {
172     Base   = CurDAG->getTargetFrameIndex(FIN->getIndex(), ValTy);
173     Offset = CurDAG->getTargetConstant(0, ValTy);
174     getMips16SPRefReg(Parent, Alias);
175     return true;
176   }
177   // on PIC code Load GA
178   if (Addr.getOpcode() == MipsISD::Wrapper) {
179     Base   = Addr.getOperand(0);
180     Offset = Addr.getOperand(1);
181     return true;
182   }
183   if (TM.getRelocationModel() != Reloc::PIC_) {
184     if ((Addr.getOpcode() == ISD::TargetExternalSymbol ||
185         Addr.getOpcode() == ISD::TargetGlobalAddress))
186       return false;
187   }
188   // Addresses of the form FI+const or FI|const
189   if (CurDAG->isBaseWithConstantOffset(Addr)) {
190     ConstantSDNode *CN = dyn_cast<ConstantSDNode>(Addr.getOperand(1));
191     if (isInt<16>(CN->getSExtValue())) {
192 
193       // If the first operand is a FI, get the TargetFI Node
194       if (FrameIndexSDNode *FIN = dyn_cast<FrameIndexSDNode>
195                                   (Addr.getOperand(0))) {
196         Base = CurDAG->getTargetFrameIndex(FIN->getIndex(), ValTy);
197         getMips16SPRefReg(Parent, Alias);
198       }
199       else
200         Base = Addr.getOperand(0);
201 
202       Offset = CurDAG->getTargetConstant(CN->getZExtValue(), ValTy);
203       return true;
204     }
205   }
206   // Operand is a result from an ADD.
207   if (Addr.getOpcode() == ISD::ADD) {
208     // When loading from constant pools, load the lower address part in
209     // the instruction itself. Example, instead of:
210     //  lui $2, %hi($CPI1_0)
211     //  addiu $2, $2, %lo($CPI1_0)
212     //  lwc1 $f0, 0($2)
213     // Generate:
214     //  lui $2, %hi($CPI1_0)
215     //  lwc1 $f0, %lo($CPI1_0)($2)
216     if (Addr.getOperand(1).getOpcode() == MipsISD::Lo ||
217         Addr.getOperand(1).getOpcode() == MipsISD::GPRel) {
218       SDValue Opnd0 = Addr.getOperand(1).getOperand(0);
219       if (isa<ConstantPoolSDNode>(Opnd0) || isa<GlobalAddressSDNode>(Opnd0) ||
220           isa<JumpTableSDNode>(Opnd0)) {
221         Base = Addr.getOperand(0);
222         Offset = Opnd0;
223         return true;
224       }
225     }
226 
227     // If an indexed floating point load/store can be emitted, return false.
228     const LSBaseSDNode *LS = dyn_cast<LSBaseSDNode>(Parent);
229 
230     if (LS) {
231       if (LS->getMemoryVT() == MVT::f32 && Subtarget->hasMips4_32r2())
232         return false;
233       if (LS->getMemoryVT() == MVT::f64 && Subtarget->hasMips4_32r2())
234         return false;
235     }
236   }
237   Base   = Addr;
238   Offset = CurDAG->getTargetConstant(0, ValTy);
239   return true;
240 }
241 
242 /// Select instructions not customized! Used for
243 /// expanded, promoted and normal instructions
selectNode(SDNode * Node)244 std::pair<bool, SDNode*> Mips16DAGToDAGISel::selectNode(SDNode *Node) {
245   unsigned Opcode = Node->getOpcode();
246   SDLoc DL(Node);
247 
248   ///
249   // Instruction Selection not handled by the auto-generated
250   // tablegen selection should be handled here.
251   ///
252   EVT NodeTy = Node->getValueType(0);
253   unsigned MultOpc;
254 
255   switch(Opcode) {
256   default: break;
257 
258   case ISD::SUBE:
259   case ISD::ADDE: {
260     SDValue InFlag = Node->getOperand(2), CmpLHS;
261     unsigned Opc = InFlag.getOpcode(); (void)Opc;
262     assert(((Opc == ISD::ADDC || Opc == ISD::ADDE) ||
263             (Opc == ISD::SUBC || Opc == ISD::SUBE)) &&
264            "(ADD|SUB)E flag operand must come from (ADD|SUB)C/E insn");
265 
266     unsigned MOp;
267     if (Opcode == ISD::ADDE) {
268       CmpLHS = InFlag.getValue(0);
269       MOp = Mips::AdduRxRyRz16;
270     } else {
271       CmpLHS = InFlag.getOperand(0);
272       MOp = Mips::SubuRxRyRz16;
273     }
274 
275     SDValue Ops[] = { CmpLHS, InFlag.getOperand(1) };
276 
277     SDValue LHS = Node->getOperand(0);
278     SDValue RHS = Node->getOperand(1);
279 
280     EVT VT = LHS.getValueType();
281 
282     unsigned Sltu_op = Mips::SltuRxRyRz16;
283     SDNode *Carry = CurDAG->getMachineNode(Sltu_op, DL, VT, Ops);
284     unsigned Addu_op = Mips::AdduRxRyRz16;
285     SDNode *AddCarry = CurDAG->getMachineNode(Addu_op, DL, VT,
286                                               SDValue(Carry,0), RHS);
287 
288     SDNode *Result = CurDAG->SelectNodeTo(Node, MOp, VT, MVT::Glue, LHS,
289                                           SDValue(AddCarry,0));
290     return std::make_pair(true, Result);
291   }
292 
293   /// Mul with two results
294   case ISD::SMUL_LOHI:
295   case ISD::UMUL_LOHI: {
296     MultOpc = (Opcode == ISD::UMUL_LOHI ? Mips::MultuRxRy16 : Mips::MultRxRy16);
297     std::pair<SDNode*, SDNode*> LoHi = selectMULT(Node, MultOpc, DL, NodeTy,
298                                                   true, true);
299     if (!SDValue(Node, 0).use_empty())
300       ReplaceUses(SDValue(Node, 0), SDValue(LoHi.first, 0));
301 
302     if (!SDValue(Node, 1).use_empty())
303       ReplaceUses(SDValue(Node, 1), SDValue(LoHi.second, 0));
304 
305     return std::make_pair(true, nullptr);
306   }
307 
308   case ISD::MULHS:
309   case ISD::MULHU: {
310     MultOpc = (Opcode == ISD::MULHU ? Mips::MultuRxRy16 : Mips::MultRxRy16);
311     SDNode *Result = selectMULT(Node, MultOpc, DL, NodeTy, false, true).second;
312     return std::make_pair(true, Result);
313   }
314   }
315 
316   return std::make_pair(false, nullptr);
317 }
318 
createMips16ISelDag(MipsTargetMachine & TM)319 FunctionPass *llvm::createMips16ISelDag(MipsTargetMachine &TM) {
320   return new Mips16DAGToDAGISel(TM);
321 }
322