blob: dccd717eb995ef1d0a1017237b5c45dfc7b33693 [file] [log] [blame]
Diana Picus22274932016-11-11 08:27:37 +00001//===- ARMInstructionSelector.cpp ----------------------------*- 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/// \file
10/// This file implements the targeting of the InstructionSelector class for ARM.
11/// \todo This should be generated by TableGen.
12//===----------------------------------------------------------------------===//
13
14#include "ARMInstructionSelector.h"
15#include "ARMRegisterBankInfo.h"
16#include "ARMSubtarget.h"
17#include "ARMTargetMachine.h"
Diana Picus812caee2016-12-16 12:54:46 +000018#include "llvm/CodeGen/MachineRegisterInfo.h"
Diana Picus22274932016-11-11 08:27:37 +000019#include "llvm/Support/Debug.h"
20
21#define DEBUG_TYPE "arm-isel"
22
23using namespace llvm;
24
25#ifndef LLVM_BUILD_GLOBAL_ISEL
26#error "You shouldn't build this"
27#endif
28
Diana Picus895c6aa2016-11-15 16:42:10 +000029ARMInstructionSelector::ARMInstructionSelector(const ARMSubtarget &STI,
Diana Picus22274932016-11-11 08:27:37 +000030 const ARMRegisterBankInfo &RBI)
Diana Picus895c6aa2016-11-15 16:42:10 +000031 : InstructionSelector(), TII(*STI.getInstrInfo()),
Diana Picus812caee2016-12-16 12:54:46 +000032 TRI(*STI.getRegisterInfo()), RBI(RBI) {}
Diana Picus22274932016-11-11 08:27:37 +000033
Diana Picus812caee2016-12-16 12:54:46 +000034static bool selectCopy(MachineInstr &I, const TargetInstrInfo &TII,
35 MachineRegisterInfo &MRI, const TargetRegisterInfo &TRI,
36 const RegisterBankInfo &RBI) {
37 unsigned DstReg = I.getOperand(0).getReg();
38 if (TargetRegisterInfo::isPhysicalRegister(DstReg))
39 return true;
40
41 const RegisterBank *RegBank = RBI.getRegBank(DstReg, MRI, TRI);
Benjamin Kramer24bf8682016-12-16 13:13:03 +000042 (void)RegBank;
Diana Picus812caee2016-12-16 12:54:46 +000043 assert(RegBank && "Can't get reg bank for virtual register");
44
Diana Picus36aa09f2016-12-19 14:07:50 +000045 const unsigned DstSize = MRI.getType(DstReg).getSizeInBits();
Daniel Jasper24218d52016-12-19 14:24:22 +000046 (void)DstSize;
Diana Picus36aa09f2016-12-19 14:07:50 +000047 unsigned SrcReg = I.getOperand(1).getReg();
48 const unsigned SrcSize = RBI.getSizeInBits(SrcReg, MRI, TRI);
49 (void)SrcSize;
50 assert((DstSize == SrcSize ||
51 // Copies are a means to setup initial types, the number of
52 // bits may not exactly match.
53 (TargetRegisterInfo::isPhysicalRegister(SrcReg) &&
54 DstSize <= SrcSize)) &&
Benjamin Kramer24bf8682016-12-16 13:13:03 +000055 "Copy with different width?!");
Diana Picus812caee2016-12-16 12:54:46 +000056
Diana Picus4fa83c02017-02-08 13:23:04 +000057 assert((RegBank->getID() == ARM::GPRRegBankID ||
58 RegBank->getID() == ARM::FPRRegBankID) &&
59 "Unsupported reg bank");
60
Diana Picus812caee2016-12-16 12:54:46 +000061 const TargetRegisterClass *RC = &ARM::GPRRegClass;
62
Diana Picus4fa83c02017-02-08 13:23:04 +000063 if (RegBank->getID() == ARM::FPRRegBankID) {
Diana Picus6beef3c2017-02-16 12:19:52 +000064 if (DstSize == 32)
65 RC = &ARM::SPRRegClass;
66 else if (DstSize == 64)
67 RC = &ARM::DPRRegClass;
68 else
69 llvm_unreachable("Unsupported destination size");
Diana Picus4fa83c02017-02-08 13:23:04 +000070 }
71
Diana Picus812caee2016-12-16 12:54:46 +000072 // No need to constrain SrcReg. It will get constrained when
73 // we hit another of its uses or its defs.
74 // Copies do not have constraints.
75 if (!RBI.constrainGenericRegister(DstReg, *RC, MRI)) {
76 DEBUG(dbgs() << "Failed to constrain " << TII.getName(I.getOpcode())
77 << " operand\n");
78 return false;
79 }
80 return true;
81}
82
Diana Picus6beef3c2017-02-16 12:19:52 +000083static bool selectFAdd(MachineInstrBuilder &MIB, const ARMBaseInstrInfo &TII,
84 MachineRegisterInfo &MRI) {
85 assert(TII.getSubtarget().hasVFP2() && "Can't select fp add without vfp");
86
87 LLT Ty = MRI.getType(MIB->getOperand(0).getReg());
88 unsigned ValSize = Ty.getSizeInBits();
89
90 if (ValSize == 32) {
91 if (TII.getSubtarget().useNEONForSinglePrecisionFP())
92 return false;
93 MIB->setDesc(TII.get(ARM::VADDS));
94 } else {
95 assert(ValSize == 64 && "Unsupported size for floating point value");
96 if (TII.getSubtarget().isFPOnlySP())
97 return false;
98 MIB->setDesc(TII.get(ARM::VADDD));
99 }
100 MIB.add(predOps(ARMCC::AL));
101
102 return true;
103}
104
Diana Picusb1701e02017-02-16 12:19:57 +0000105static bool selectSequence(MachineInstrBuilder &MIB,
106 const ARMBaseInstrInfo &TII,
107 MachineRegisterInfo &MRI,
108 const TargetRegisterInfo &TRI,
109 const RegisterBankInfo &RBI) {
110 assert(TII.getSubtarget().hasVFP2() && "Can't select sequence without VFP");
111
112 // We only support G_SEQUENCE as a way to stick together two scalar GPRs
113 // into one DPR.
114 unsigned VReg0 = MIB->getOperand(0).getReg();
115 (void)VReg0;
116 assert(MRI.getType(VReg0).getSizeInBits() == 64 &&
117 RBI.getRegBank(VReg0, MRI, TRI)->getID() == ARM::FPRRegBankID &&
118 "Unsupported operand for G_SEQUENCE");
119 unsigned VReg1 = MIB->getOperand(1).getReg();
120 (void)VReg1;
121 assert(MRI.getType(VReg1).getSizeInBits() == 32 &&
122 RBI.getRegBank(VReg1, MRI, TRI)->getID() == ARM::GPRRegBankID &&
123 "Unsupported operand for G_SEQUENCE");
124 unsigned VReg2 = MIB->getOperand(3).getReg();
125 (void)VReg2;
126 assert(MRI.getType(VReg2).getSizeInBits() == 32 &&
127 RBI.getRegBank(VReg2, MRI, TRI)->getID() == ARM::GPRRegBankID &&
128 "Unsupported operand for G_SEQUENCE");
129
130 // Remove the operands corresponding to the offsets.
131 MIB->RemoveOperand(4);
132 MIB->RemoveOperand(2);
133
134 MIB->setDesc(TII.get(ARM::VMOVDRR));
135 MIB.add(predOps(ARMCC::AL));
136
137 return true;
138}
139
140static bool selectExtract(MachineInstrBuilder &MIB, const ARMBaseInstrInfo &TII,
141 MachineRegisterInfo &MRI,
142 const TargetRegisterInfo &TRI,
143 const RegisterBankInfo &RBI) {
144 assert(TII.getSubtarget().hasVFP2() && "Can't select extract without VFP");
145
146 // We only support G_EXTRACT as a way to break up one DPR into two GPRs.
147 unsigned VReg0 = MIB->getOperand(0).getReg();
148 (void)VReg0;
149 assert(MRI.getType(VReg0).getSizeInBits() == 32 &&
150 RBI.getRegBank(VReg0, MRI, TRI)->getID() == ARM::GPRRegBankID &&
151 "Unsupported operand for G_SEQUENCE");
152 unsigned VReg1 = MIB->getOperand(1).getReg();
153 (void)VReg1;
154 assert(MRI.getType(VReg1).getSizeInBits() == 32 &&
155 RBI.getRegBank(VReg1, MRI, TRI)->getID() == ARM::GPRRegBankID &&
156 "Unsupported operand for G_SEQUENCE");
157 unsigned VReg2 = MIB->getOperand(2).getReg();
158 (void)VReg2;
159 assert(MRI.getType(VReg2).getSizeInBits() == 64 &&
160 RBI.getRegBank(VReg2, MRI, TRI)->getID() == ARM::FPRRegBankID &&
161 "Unsupported operand for G_SEQUENCE");
162
163 // Remove the operands corresponding to the offsets.
164 MIB->RemoveOperand(4);
165 MIB->RemoveOperand(3);
166
167 MIB->setDesc(TII.get(ARM::VMOVRRD));
168 MIB.add(predOps(ARMCC::AL));
169
170 return true;
171}
172
Diana Picus8b6c6be2017-01-25 08:10:40 +0000173/// Select the opcode for simple extensions (that translate to a single SXT/UXT
174/// instruction). Extension operations more complicated than that should not
Diana Picuse8368782017-02-17 13:44:19 +0000175/// invoke this. Returns the original opcode if it doesn't know how to select a
176/// better one.
Diana Picus8b6c6be2017-01-25 08:10:40 +0000177static unsigned selectSimpleExtOpc(unsigned Opc, unsigned Size) {
178 using namespace TargetOpcode;
179
Diana Picuse8368782017-02-17 13:44:19 +0000180 if (Size != 8 && Size != 16)
181 return Opc;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000182
183 if (Opc == G_SEXT)
184 return Size == 8 ? ARM::SXTB : ARM::SXTH;
185
186 if (Opc == G_ZEXT)
187 return Size == 8 ? ARM::UXTB : ARM::UXTH;
188
Diana Picuse8368782017-02-17 13:44:19 +0000189 return Opc;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000190}
191
Diana Picus3b99c642017-02-24 14:01:27 +0000192/// Select the opcode for simple loads and stores. For types smaller than 32
193/// bits, the value will be zero extended. Returns the original opcode if it
194/// doesn't know how to select a better one.
195static unsigned selectLoadStoreOpCode(unsigned Opc, unsigned RegBank,
196 unsigned Size) {
197 bool isStore = Opc == TargetOpcode::G_STORE;
198
Diana Picus1540b062017-02-16 14:10:50 +0000199 if (RegBank == ARM::GPRRegBankID) {
200 switch (Size) {
201 case 1:
202 case 8:
Diana Picus3b99c642017-02-24 14:01:27 +0000203 return isStore ? ARM::STRBi12 : ARM::LDRBi12;
Diana Picus1540b062017-02-16 14:10:50 +0000204 case 16:
Diana Picus3b99c642017-02-24 14:01:27 +0000205 return isStore ? ARM::STRH : ARM::LDRH;
Diana Picus1540b062017-02-16 14:10:50 +0000206 case 32:
Diana Picus3b99c642017-02-24 14:01:27 +0000207 return isStore ? ARM::STRi12 : ARM::LDRi12;
Diana Picuse8368782017-02-17 13:44:19 +0000208 default:
Diana Picus3b99c642017-02-24 14:01:27 +0000209 return Opc;
Diana Picus1540b062017-02-16 14:10:50 +0000210 }
Diana Picus1540b062017-02-16 14:10:50 +0000211 }
212
Diana Picuse8368782017-02-17 13:44:19 +0000213 if (RegBank == ARM::FPRRegBankID) {
214 switch (Size) {
215 case 32:
Diana Picus3b99c642017-02-24 14:01:27 +0000216 return isStore ? ARM::VSTRS : ARM::VLDRS;
Diana Picuse8368782017-02-17 13:44:19 +0000217 case 64:
Diana Picus3b99c642017-02-24 14:01:27 +0000218 return isStore ? ARM::VSTRD : ARM::VLDRD;
Diana Picuse8368782017-02-17 13:44:19 +0000219 default:
Diana Picus3b99c642017-02-24 14:01:27 +0000220 return Opc;
Diana Picuse8368782017-02-17 13:44:19 +0000221 }
Diana Picus278c7222017-01-26 09:20:47 +0000222 }
223
Diana Picus3b99c642017-02-24 14:01:27 +0000224 return Opc;
Diana Picus278c7222017-01-26 09:20:47 +0000225}
226
Diana Picus812caee2016-12-16 12:54:46 +0000227bool ARMInstructionSelector::select(MachineInstr &I) const {
228 assert(I.getParent() && "Instruction should be in a basic block!");
229 assert(I.getParent()->getParent() && "Instruction should be in a function!");
230
231 auto &MBB = *I.getParent();
232 auto &MF = *MBB.getParent();
233 auto &MRI = MF.getRegInfo();
234
235 if (!isPreISelGenericOpcode(I.getOpcode())) {
236 if (I.isCopy())
237 return selectCopy(I, TII, MRI, TRI, RBI);
238
239 return true;
240 }
241
Diana Picus519807f2016-12-19 11:26:31 +0000242 MachineInstrBuilder MIB{MF, I};
Diana Picusd83df5d2017-01-25 08:47:40 +0000243 bool isSExt = false;
Diana Picus519807f2016-12-19 11:26:31 +0000244
245 using namespace TargetOpcode;
246 switch (I.getOpcode()) {
Diana Picus8b6c6be2017-01-25 08:10:40 +0000247 case G_SEXT:
Diana Picusd83df5d2017-01-25 08:47:40 +0000248 isSExt = true;
249 LLVM_FALLTHROUGH;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000250 case G_ZEXT: {
251 LLT DstTy = MRI.getType(I.getOperand(0).getReg());
252 // FIXME: Smaller destination sizes coming soon!
253 if (DstTy.getSizeInBits() != 32) {
254 DEBUG(dbgs() << "Unsupported destination size for extension");
255 return false;
256 }
257
258 LLT SrcTy = MRI.getType(I.getOperand(1).getReg());
259 unsigned SrcSize = SrcTy.getSizeInBits();
260 switch (SrcSize) {
Diana Picusd83df5d2017-01-25 08:47:40 +0000261 case 1: {
262 // ZExt boils down to & 0x1; for SExt we also subtract that from 0
263 I.setDesc(TII.get(ARM::ANDri));
264 MIB.addImm(1).add(predOps(ARMCC::AL)).add(condCodeOp());
265
266 if (isSExt) {
267 unsigned SExtResult = I.getOperand(0).getReg();
268
269 // Use a new virtual register for the result of the AND
270 unsigned AndResult = MRI.createVirtualRegister(&ARM::GPRRegClass);
271 I.getOperand(0).setReg(AndResult);
272
273 auto InsertBefore = std::next(I.getIterator());
Martin Bohme8396e142017-01-25 14:28:19 +0000274 auto SubI =
Diana Picusd83df5d2017-01-25 08:47:40 +0000275 BuildMI(MBB, InsertBefore, I.getDebugLoc(), TII.get(ARM::RSBri))
276 .addDef(SExtResult)
277 .addUse(AndResult)
278 .addImm(0)
279 .add(predOps(ARMCC::AL))
280 .add(condCodeOp());
281 if (!constrainSelectedInstRegOperands(*SubI, TII, TRI, RBI))
282 return false;
283 }
284 break;
285 }
Diana Picus8b6c6be2017-01-25 08:10:40 +0000286 case 8:
287 case 16: {
288 unsigned NewOpc = selectSimpleExtOpc(I.getOpcode(), SrcSize);
Diana Picuse8368782017-02-17 13:44:19 +0000289 if (NewOpc == I.getOpcode())
290 return false;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000291 I.setDesc(TII.get(NewOpc));
292 MIB.addImm(0).add(predOps(ARMCC::AL));
293 break;
294 }
295 default:
296 DEBUG(dbgs() << "Unsupported source size for extension");
297 return false;
298 }
299 break;
300 }
Diana Picus519807f2016-12-19 11:26:31 +0000301 case G_ADD:
Diana Picus9d070942017-02-28 10:14:38 +0000302 case G_GEP:
Diana Picus812caee2016-12-16 12:54:46 +0000303 I.setDesc(TII.get(ARM::ADDrr));
Diana Picus8a73f552017-01-13 10:18:01 +0000304 MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
Diana Picus519807f2016-12-19 11:26:31 +0000305 break;
Diana Picus4fa83c02017-02-08 13:23:04 +0000306 case G_FADD:
Diana Picus6beef3c2017-02-16 12:19:52 +0000307 if (!selectFAdd(MIB, TII, MRI))
Diana Picus4fa83c02017-02-08 13:23:04 +0000308 return false;
Diana Picus4fa83c02017-02-08 13:23:04 +0000309 break;
Diana Picus519807f2016-12-19 11:26:31 +0000310 case G_FRAME_INDEX:
311 // Add 0 to the given frame index and hope it will eventually be folded into
312 // the user(s).
313 I.setDesc(TII.get(ARM::ADDri));
Diana Picus8a73f552017-01-13 10:18:01 +0000314 MIB.addImm(0).add(predOps(ARMCC::AL)).add(condCodeOp());
Diana Picus519807f2016-12-19 11:26:31 +0000315 break;
Diana Picus3b99c642017-02-24 14:01:27 +0000316 case G_STORE:
Diana Picus278c7222017-01-26 09:20:47 +0000317 case G_LOAD: {
Diana Picus1c33c9f2017-02-20 14:45:58 +0000318 const auto &MemOp = **I.memoperands_begin();
319 if (MemOp.getOrdering() != AtomicOrdering::NotAtomic) {
320 DEBUG(dbgs() << "Atomic load/store not supported yet\n");
321 return false;
322 }
323
Diana Picus1540b062017-02-16 14:10:50 +0000324 unsigned Reg = I.getOperand(0).getReg();
325 unsigned RegBank = RBI.getRegBank(Reg, MRI, TRI)->getID();
326
327 LLT ValTy = MRI.getType(Reg);
Diana Picus278c7222017-01-26 09:20:47 +0000328 const auto ValSize = ValTy.getSizeInBits();
329
Diana Picus1540b062017-02-16 14:10:50 +0000330 assert((ValSize != 64 || TII.getSubtarget().hasVFP2()) &&
Diana Picus3b99c642017-02-24 14:01:27 +0000331 "Don't know how to load/store 64-bit value without VFP");
Diana Picus1540b062017-02-16 14:10:50 +0000332
Diana Picus3b99c642017-02-24 14:01:27 +0000333 const auto NewOpc = selectLoadStoreOpCode(I.getOpcode(), RegBank, ValSize);
334 if (NewOpc == G_LOAD || NewOpc == G_STORE)
Diana Picuse8368782017-02-17 13:44:19 +0000335 return false;
336
Diana Picus278c7222017-01-26 09:20:47 +0000337 I.setDesc(TII.get(NewOpc));
338
Diana Picus3b99c642017-02-24 14:01:27 +0000339 if (NewOpc == ARM::LDRH || NewOpc == ARM::STRH)
Diana Picus278c7222017-01-26 09:20:47 +0000340 // LDRH has a funny addressing mode (there's already a FIXME for it).
341 MIB.addReg(0);
Diana Picus4f8c3e12017-01-13 09:37:56 +0000342 MIB.addImm(0).add(predOps(ARMCC::AL));
Diana Picus519807f2016-12-19 11:26:31 +0000343 break;
Diana Picus278c7222017-01-26 09:20:47 +0000344 }
Diana Picusb1701e02017-02-16 12:19:57 +0000345 case G_SEQUENCE: {
346 if (!selectSequence(MIB, TII, MRI, TRI, RBI))
347 return false;
348 break;
349 }
350 case G_EXTRACT: {
351 if (!selectExtract(MIB, TII, MRI, TRI, RBI))
352 return false;
353 break;
354 }
Diana Picus519807f2016-12-19 11:26:31 +0000355 default:
356 return false;
Diana Picus812caee2016-12-16 12:54:46 +0000357 }
358
Diana Picus519807f2016-12-19 11:26:31 +0000359 return constrainSelectedInstRegOperands(I, TII, TRI, RBI);
Diana Picus22274932016-11-11 08:27:37 +0000360}