blob: a299e32c03a0818369f278dba5384f8cb3f8c501 [file] [log] [blame]
Pete Couperus2d1f6d62017-08-24 15:40:33 +00001//===- ARCInstrInfo.cpp - ARC Instruction Information -----------*- C++ -*-===//
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 the ARC implementation of the TargetInstrInfo class.
11//
12//===----------------------------------------------------------------------===//
13
14#include "ARCInstrInfo.h"
15#include "ARC.h"
16#include "ARCMachineFunctionInfo.h"
17#include "ARCSubtarget.h"
18#include "MCTargetDesc/ARCInfo.h"
19#include "llvm/CodeGen/MachineFrameInfo.h"
20#include "llvm/CodeGen/MachineInstrBuilder.h"
21#include "llvm/CodeGen/MachineMemOperand.h"
22#include "llvm/Support/Debug.h"
23#include "llvm/Support/TargetRegistry.h"
24
25using namespace llvm;
26
27#define GET_INSTRINFO_CTOR_DTOR
28#include "ARCGenInstrInfo.inc"
29
30#define DEBUG_TYPE "arc-inst-info"
31// Pin the vtable to this file.
32void ARCInstrInfo::anchor() {}
33
34ARCInstrInfo::ARCInstrInfo()
35 : ARCGenInstrInfo(ARC::ADJCALLSTACKDOWN, ARC::ADJCALLSTACKUP), RI() {}
36
37static bool isZeroImm(const MachineOperand &Op) {
38 return Op.isImm() && Op.getImm() == 0;
39}
40
41static bool isLoad(int Opcode) {
42 return Opcode == ARC::LD_rs9 || Opcode == ARC::LDH_rs9 ||
43 Opcode == ARC::LDB_rs9;
44}
45
46static bool isStore(int Opcode) {
47 return Opcode == ARC::ST_rs9 || Opcode == ARC::STH_rs9 ||
48 Opcode == ARC::STB_rs9;
49}
50
51/// If the specified machine instruction is a direct
52/// load from a stack slot, return the virtual or physical register number of
53/// the destination along with the FrameIndex of the loaded stack slot. If
54/// not, return 0. This predicate must return 0 if the instruction has
55/// any side effects other than loading from the stack slot.
56unsigned ARCInstrInfo::isLoadFromStackSlot(const MachineInstr &MI,
57 int &FrameIndex) const {
58 int Opcode = MI.getOpcode();
59 if (isLoad(Opcode)) {
60 if ((MI.getOperand(1).isFI()) && // is a stack slot
61 (MI.getOperand(2).isImm()) && // the imm is zero
62 (isZeroImm(MI.getOperand(2)))) {
63 FrameIndex = MI.getOperand(1).getIndex();
64 return MI.getOperand(0).getReg();
65 }
66 }
67 return 0;
68}
69
70/// If the specified machine instruction is a direct
71/// store to a stack slot, return the virtual or physical register number of
72/// the source reg along with the FrameIndex of the loaded stack slot. If
73/// not, return 0. This predicate must return 0 if the instruction has
74/// any side effects other than storing to the stack slot.
75unsigned ARCInstrInfo::isStoreToStackSlot(const MachineInstr &MI,
76 int &FrameIndex) const {
77 int Opcode = MI.getOpcode();
78 if (isStore(Opcode)) {
79 if ((MI.getOperand(1).isFI()) && // is a stack slot
80 (MI.getOperand(2).isImm()) && // the imm is zero
81 (isZeroImm(MI.getOperand(2)))) {
82 FrameIndex = MI.getOperand(1).getIndex();
83 return MI.getOperand(0).getReg();
84 }
85 }
86 return 0;
87}
88
89/// Return the inverse of passed condition, i.e. turning COND_E to COND_NE.
90static ARCCC::CondCode GetOppositeBranchCondition(ARCCC::CondCode CC) {
91 switch (CC) {
92 default:
93 llvm_unreachable("Illegal condition code!");
94 case ARCCC::EQ:
95 return ARCCC::NE;
96 case ARCCC::NE:
97 return ARCCC::EQ;
98 case ARCCC::LO:
99 return ARCCC::HS;
100 case ARCCC::HS:
101 return ARCCC::LO;
102 case ARCCC::GT:
103 return ARCCC::LE;
104 case ARCCC::GE:
105 return ARCCC::LT;
106 case ARCCC::LT:
107 return ARCCC::GE;
108 case ARCCC::LE:
109 return ARCCC::GT;
110 case ARCCC::HI:
111 return ARCCC::LS;
112 case ARCCC::LS:
113 return ARCCC::HI;
114 case ARCCC::NZ:
115 return ARCCC::Z;
116 case ARCCC::Z:
117 return ARCCC::NZ;
118 }
119}
120
121static bool isUncondBranchOpcode(int Opc) { return Opc == ARC::BR; }
122
123static bool isCondBranchOpcode(int Opc) {
124 return Opc == ARC::BRcc_rr_p || Opc == ARC::BRcc_ru6_p;
125}
126
127static bool isJumpOpcode(int Opc) { return Opc == ARC::J; }
128
129/// Analyze the branching code at the end of MBB, returning
130/// true if it cannot be understood (e.g. it's a switch dispatch or isn't
131/// implemented for a target). Upon success, this returns false and returns
132/// with the following information in various cases:
133///
134/// 1. If this block ends with no branches (it just falls through to its succ)
135/// just return false, leaving TBB/FBB null.
136/// 2. If this block ends with only an unconditional branch, it sets TBB to be
137/// the destination block.
138/// 3. If this block ends with a conditional branch and it falls through to a
139/// successor block, it sets TBB to be the branch destination block and a
140/// list of operands that evaluate the condition. These operands can be
141/// passed to other TargetInstrInfo methods to create new branches.
142/// 4. If this block ends with a conditional branch followed by an
143/// unconditional branch, it returns the 'true' destination in TBB, the
144/// 'false' destination in FBB, and a list of operands that evaluate the
145/// condition. These operands can be passed to other TargetInstrInfo
146/// methods to create new branches.
147///
148/// Note that RemoveBranch and InsertBranch must be implemented to support
149/// cases where this method returns success.
150///
151/// If AllowModify is true, then this routine is allowed to modify the basic
152/// block (e.g. delete instructions after the unconditional branch).
153
154bool ARCInstrInfo::analyzeBranch(MachineBasicBlock &MBB,
155 MachineBasicBlock *&TBB,
156 MachineBasicBlock *&FBB,
157 SmallVectorImpl<MachineOperand> &Cond,
158 bool AllowModify) const {
159 TBB = FBB = nullptr;
160 MachineBasicBlock::iterator I = MBB.end();
161 if (I == MBB.begin())
162 return false;
163 --I;
164
165 while (isPredicated(*I) || I->isTerminator() || I->isDebugValue()) {
166 // Flag to be raised on unanalyzeable instructions. This is useful in cases
167 // where we want to clean up on the end of the basic block before we bail
168 // out.
169 bool CantAnalyze = false;
170
171 // Skip over DEBUG values and predicated nonterminators.
172 while (I->isDebugValue() || !I->isTerminator()) {
173 if (I == MBB.begin())
174 return false;
175 --I;
176 }
177
178 if (isJumpOpcode(I->getOpcode())) {
179 // Indirect branches and jump tables can't be analyzed, but we still want
180 // to clean up any instructions at the tail of the basic block.
181 CantAnalyze = true;
182 } else if (isUncondBranchOpcode(I->getOpcode())) {
183 TBB = I->getOperand(0).getMBB();
184 } else if (isCondBranchOpcode(I->getOpcode())) {
185 // Bail out if we encounter multiple conditional branches.
186 if (!Cond.empty())
187 return true;
188
189 assert(!FBB && "FBB should have been null.");
190 FBB = TBB;
191 TBB = I->getOperand(0).getMBB();
192 Cond.push_back(I->getOperand(1));
193 Cond.push_back(I->getOperand(2));
194 Cond.push_back(I->getOperand(3));
195 } else if (I->isReturn()) {
196 // Returns can't be analyzed, but we should run cleanup.
197 CantAnalyze = !isPredicated(*I);
198 } else {
199 // We encountered other unrecognized terminator. Bail out immediately.
200 return true;
201 }
202
203 // Cleanup code - to be run for unpredicated unconditional branches and
204 // returns.
205 if (!isPredicated(*I) && (isUncondBranchOpcode(I->getOpcode()) ||
206 isJumpOpcode(I->getOpcode()) || I->isReturn())) {
207 // Forget any previous condition branch information - it no longer
208 // applies.
209 Cond.clear();
210 FBB = nullptr;
211
212 // If we can modify the function, delete everything below this
213 // unconditional branch.
214 if (AllowModify) {
215 MachineBasicBlock::iterator DI = std::next(I);
216 while (DI != MBB.end()) {
217 MachineInstr &InstToDelete = *DI;
218 ++DI;
219 InstToDelete.eraseFromParent();
220 }
221 }
222 }
223
224 if (CantAnalyze)
225 return true;
226
227 if (I == MBB.begin())
228 return false;
229
230 --I;
231 }
232
233 // We made it past the terminators without bailing out - we must have
234 // analyzed this branch successfully.
235 return false;
236}
237
238unsigned ARCInstrInfo::removeBranch(MachineBasicBlock &MBB,
239 int *BytesRemoved) const {
240 assert(!BytesRemoved && "Code size not handled");
241 MachineBasicBlock::iterator I = MBB.getLastNonDebugInstr();
242 if (I == MBB.end())
243 return 0;
244
245 if (!isUncondBranchOpcode(I->getOpcode()) &&
246 !isCondBranchOpcode(I->getOpcode()))
247 return 0;
248
249 // Remove the branch.
250 I->eraseFromParent();
251
252 I = MBB.end();
253
254 if (I == MBB.begin())
255 return 1;
256 --I;
257 if (!isCondBranchOpcode(I->getOpcode()))
258 return 1;
259
260 // Remove the branch.
261 I->eraseFromParent();
262 return 2;
263}
264
265void ARCInstrInfo::copyPhysReg(MachineBasicBlock &MBB,
266 MachineBasicBlock::iterator I,
267 const DebugLoc &dl, unsigned DestReg,
268 unsigned SrcReg, bool KillSrc) const {
269 assert(ARC::GPR32RegClass.contains(SrcReg) &&
270 "Only GPR32 src copy supported.");
271 assert(ARC::GPR32RegClass.contains(DestReg) &&
272 "Only GPR32 dest copy supported.");
273 BuildMI(MBB, I, dl, get(ARC::MOV_rr), DestReg)
274 .addReg(SrcReg, getKillRegState(KillSrc));
275}
276
277void ARCInstrInfo::storeRegToStackSlot(MachineBasicBlock &MBB,
278 MachineBasicBlock::iterator I,
279 unsigned SrcReg, bool isKill,
280 int FrameIndex,
281 const TargetRegisterClass *RC,
282 const TargetRegisterInfo *TRI) const {
283 DebugLoc dl = MBB.findDebugLoc(I);
284 MachineFunction &MF = *MBB.getParent();
285 MachineFrameInfo &MFI = MF.getFrameInfo();
286 unsigned Align = MFI.getObjectAlignment(FrameIndex);
287
288 MachineMemOperand *MMO = MF.getMachineMemOperand(
289 MachinePointerInfo::getFixedStack(MF, FrameIndex),
290 MachineMemOperand::MOStore, MFI.getObjectSize(FrameIndex), Align);
291
292 assert(MMO && "Couldn't get MachineMemOperand for store to stack.");
293 assert(TRI->getSpillSize(*RC) == 4 &&
294 "Only support 4-byte stores to stack now.");
295 assert(ARC::GPR32RegClass.hasSubClassEq(RC) &&
296 "Only support GPR32 stores to stack now.");
Francis Visoiu Mistrih9d419d32017-11-28 12:42:37 +0000297 DEBUG(dbgs() << "Created store reg=" << printReg(SrcReg, TRI)
Pete Couperus2d1f6d62017-08-24 15:40:33 +0000298 << " to FrameIndex=" << FrameIndex << "\n");
299 BuildMI(MBB, I, dl, get(ARC::ST_rs9))
300 .addReg(SrcReg, getKillRegState(isKill))
301 .addFrameIndex(FrameIndex)
302 .addImm(0)
303 .addMemOperand(MMO);
304}
305
306void ARCInstrInfo::loadRegFromStackSlot(MachineBasicBlock &MBB,
307 MachineBasicBlock::iterator I,
308 unsigned DestReg, int FrameIndex,
309 const TargetRegisterClass *RC,
310 const TargetRegisterInfo *TRI) const {
311 DebugLoc dl = MBB.findDebugLoc(I);
312 MachineFunction &MF = *MBB.getParent();
313 MachineFrameInfo &MFI = MF.getFrameInfo();
314 unsigned Align = MFI.getObjectAlignment(FrameIndex);
315 MachineMemOperand *MMO = MF.getMachineMemOperand(
316 MachinePointerInfo::getFixedStack(MF, FrameIndex),
317 MachineMemOperand::MOLoad, MFI.getObjectSize(FrameIndex), Align);
318
319 assert(MMO && "Couldn't get MachineMemOperand for store to stack.");
320 assert(TRI->getSpillSize(*RC) == 4 &&
321 "Only support 4-byte loads from stack now.");
322 assert(ARC::GPR32RegClass.hasSubClassEq(RC) &&
323 "Only support GPR32 stores to stack now.");
Francis Visoiu Mistrih9d419d32017-11-28 12:42:37 +0000324 DEBUG(dbgs() << "Created load reg=" << printReg(DestReg, TRI)
Pete Couperus2d1f6d62017-08-24 15:40:33 +0000325 << " from FrameIndex=" << FrameIndex << "\n");
326 BuildMI(MBB, I, dl, get(ARC::LD_rs9))
327 .addReg(DestReg, RegState::Define)
328 .addFrameIndex(FrameIndex)
329 .addImm(0)
330 .addMemOperand(MMO);
331}
332
333/// Return the inverse opcode of the specified Branch instruction.
334bool ARCInstrInfo::reverseBranchCondition(
335 SmallVectorImpl<MachineOperand> &Cond) const {
336 assert((Cond.size() == 3) && "Invalid ARC branch condition!");
337 Cond[2].setImm(GetOppositeBranchCondition((ARCCC::CondCode)Cond[2].getImm()));
338 return false;
339}
340
341MachineBasicBlock::iterator
342ARCInstrInfo::loadImmediate(MachineBasicBlock &MBB,
343 MachineBasicBlock::iterator MI, unsigned Reg,
344 uint64_t Value) const {
345 DebugLoc dl = MBB.findDebugLoc(MI);
346 if (isInt<12>(Value)) {
347 return BuildMI(MBB, MI, dl, get(ARC::MOV_rs12), Reg)
348 .addImm(Value)
349 .getInstr();
350 }
351 llvm_unreachable("Need Arc long immediate instructions.");
352}
353
354unsigned ARCInstrInfo::insertBranch(MachineBasicBlock &MBB,
355 MachineBasicBlock *TBB,
356 MachineBasicBlock *FBB,
357 ArrayRef<MachineOperand> Cond,
358 const DebugLoc &dl, int *BytesAdded) const {
359 assert(!BytesAdded && "Code size not handled.");
360
361 // Shouldn't be a fall through.
362 assert(TBB && "InsertBranch must not be told to insert a fallthrough");
363 assert((Cond.size() == 3 || Cond.size() == 0) &&
364 "ARC branch conditions have two components!");
365
366 if (Cond.empty()) {
367 BuildMI(&MBB, dl, get(ARC::BR)).addMBB(TBB);
368 return 1;
369 }
370 int BccOpc = Cond[1].isImm() ? ARC::BRcc_ru6_p : ARC::BRcc_rr_p;
371 MachineInstrBuilder MIB = BuildMI(&MBB, dl, get(BccOpc));
372 MIB.addMBB(TBB);
373 for (unsigned i = 0; i < 3; i++) {
374 MIB.add(Cond[i]);
375 }
376
377 // One-way conditional branch.
378 if (!FBB) {
379 return 1;
380 }
381
382 // Two-way conditional branch.
383 BuildMI(&MBB, dl, get(ARC::BR)).addMBB(FBB);
384 return 2;
385}
386
387unsigned ARCInstrInfo::getInstSizeInBytes(const MachineInstr &MI) const {
388 if (MI.getOpcode() == TargetOpcode::INLINEASM) {
389 const MachineFunction *MF = MI.getParent()->getParent();
390 const char *AsmStr = MI.getOperand(0).getSymbolName();
391 return getInlineAsmLength(AsmStr, *MF->getTarget().getMCAsmInfo());
392 }
393 return MI.getDesc().getSize();
394}