blob: 8ce8e322fe09bf1eb51ba65d3a918401b2116b9f [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
Diana Picus22274932016-11-11 08:27:37 +000014#include "ARMRegisterBankInfo.h"
15#include "ARMSubtarget.h"
16#include "ARMTargetMachine.h"
Diana Picus674888d2017-04-28 09:10:38 +000017#include "llvm/CodeGen/GlobalISel/InstructionSelector.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 Picus674888d2017-04-28 09:10:38 +000029namespace {
30class ARMInstructionSelector : public InstructionSelector {
31public:
32 ARMInstructionSelector(const ARMSubtarget &STI,
33 const ARMRegisterBankInfo &RBI);
34
35 bool select(MachineInstr &I) const override;
36
37private:
38 const ARMBaseInstrInfo &TII;
39 const ARMBaseRegisterInfo &TRI;
40 const ARMRegisterBankInfo &RBI;
41};
42} // end anonymous namespace
43
44namespace llvm {
45InstructionSelector *
46createARMInstructionSelector(const ARMSubtarget &STI,
47 const ARMRegisterBankInfo &RBI) {
48 return new ARMInstructionSelector(STI, RBI);
49}
50}
51
Diana Picus895c6aa2016-11-15 16:42:10 +000052ARMInstructionSelector::ARMInstructionSelector(const ARMSubtarget &STI,
Diana Picus22274932016-11-11 08:27:37 +000053 const ARMRegisterBankInfo &RBI)
Diana Picus895c6aa2016-11-15 16:42:10 +000054 : InstructionSelector(), TII(*STI.getInstrInfo()),
Diana Picus812caee2016-12-16 12:54:46 +000055 TRI(*STI.getRegisterInfo()), RBI(RBI) {}
Diana Picus22274932016-11-11 08:27:37 +000056
Diana Picus812caee2016-12-16 12:54:46 +000057static bool selectCopy(MachineInstr &I, const TargetInstrInfo &TII,
58 MachineRegisterInfo &MRI, const TargetRegisterInfo &TRI,
59 const RegisterBankInfo &RBI) {
60 unsigned DstReg = I.getOperand(0).getReg();
61 if (TargetRegisterInfo::isPhysicalRegister(DstReg))
62 return true;
63
64 const RegisterBank *RegBank = RBI.getRegBank(DstReg, MRI, TRI);
Benjamin Kramer24bf8682016-12-16 13:13:03 +000065 (void)RegBank;
Diana Picus812caee2016-12-16 12:54:46 +000066 assert(RegBank && "Can't get reg bank for virtual register");
67
Diana Picus36aa09f2016-12-19 14:07:50 +000068 const unsigned DstSize = MRI.getType(DstReg).getSizeInBits();
Daniel Jasper24218d52016-12-19 14:24:22 +000069 (void)DstSize;
Diana Picus36aa09f2016-12-19 14:07:50 +000070 unsigned SrcReg = I.getOperand(1).getReg();
71 const unsigned SrcSize = RBI.getSizeInBits(SrcReg, MRI, TRI);
72 (void)SrcSize;
Diana Picus64a33432017-04-21 13:16:50 +000073 // We use copies for trunc, so it's ok for the size of the destination to be
74 // smaller (the higher bits will just be undefined).
75 assert(DstSize <= SrcSize && "Copy with different width?!");
Diana Picus812caee2016-12-16 12:54:46 +000076
Diana Picus4fa83c02017-02-08 13:23:04 +000077 assert((RegBank->getID() == ARM::GPRRegBankID ||
78 RegBank->getID() == ARM::FPRRegBankID) &&
79 "Unsupported reg bank");
80
Diana Picus812caee2016-12-16 12:54:46 +000081 const TargetRegisterClass *RC = &ARM::GPRRegClass;
82
Diana Picus4fa83c02017-02-08 13:23:04 +000083 if (RegBank->getID() == ARM::FPRRegBankID) {
Diana Picus6beef3c2017-02-16 12:19:52 +000084 if (DstSize == 32)
85 RC = &ARM::SPRRegClass;
86 else if (DstSize == 64)
87 RC = &ARM::DPRRegClass;
88 else
89 llvm_unreachable("Unsupported destination size");
Diana Picus4fa83c02017-02-08 13:23:04 +000090 }
91
Diana Picus812caee2016-12-16 12:54:46 +000092 // No need to constrain SrcReg. It will get constrained when
93 // we hit another of its uses or its defs.
94 // Copies do not have constraints.
95 if (!RBI.constrainGenericRegister(DstReg, *RC, MRI)) {
96 DEBUG(dbgs() << "Failed to constrain " << TII.getName(I.getOpcode())
97 << " operand\n");
98 return false;
99 }
100 return true;
101}
102
Diana Picus6beef3c2017-02-16 12:19:52 +0000103static bool selectFAdd(MachineInstrBuilder &MIB, const ARMBaseInstrInfo &TII,
104 MachineRegisterInfo &MRI) {
105 assert(TII.getSubtarget().hasVFP2() && "Can't select fp add without vfp");
106
107 LLT Ty = MRI.getType(MIB->getOperand(0).getReg());
108 unsigned ValSize = Ty.getSizeInBits();
109
110 if (ValSize == 32) {
111 if (TII.getSubtarget().useNEONForSinglePrecisionFP())
112 return false;
113 MIB->setDesc(TII.get(ARM::VADDS));
114 } else {
115 assert(ValSize == 64 && "Unsupported size for floating point value");
116 if (TII.getSubtarget().isFPOnlySP())
117 return false;
118 MIB->setDesc(TII.get(ARM::VADDD));
119 }
120 MIB.add(predOps(ARMCC::AL));
121
122 return true;
123}
124
Diana Picusb1701e02017-02-16 12:19:57 +0000125static bool selectSequence(MachineInstrBuilder &MIB,
126 const ARMBaseInstrInfo &TII,
127 MachineRegisterInfo &MRI,
128 const TargetRegisterInfo &TRI,
129 const RegisterBankInfo &RBI) {
130 assert(TII.getSubtarget().hasVFP2() && "Can't select sequence without VFP");
131
132 // We only support G_SEQUENCE as a way to stick together two scalar GPRs
133 // into one DPR.
134 unsigned VReg0 = MIB->getOperand(0).getReg();
135 (void)VReg0;
136 assert(MRI.getType(VReg0).getSizeInBits() == 64 &&
137 RBI.getRegBank(VReg0, MRI, TRI)->getID() == ARM::FPRRegBankID &&
138 "Unsupported operand for G_SEQUENCE");
139 unsigned VReg1 = MIB->getOperand(1).getReg();
140 (void)VReg1;
141 assert(MRI.getType(VReg1).getSizeInBits() == 32 &&
142 RBI.getRegBank(VReg1, MRI, TRI)->getID() == ARM::GPRRegBankID &&
143 "Unsupported operand for G_SEQUENCE");
144 unsigned VReg2 = MIB->getOperand(3).getReg();
145 (void)VReg2;
146 assert(MRI.getType(VReg2).getSizeInBits() == 32 &&
147 RBI.getRegBank(VReg2, MRI, TRI)->getID() == ARM::GPRRegBankID &&
148 "Unsupported operand for G_SEQUENCE");
149
150 // Remove the operands corresponding to the offsets.
151 MIB->RemoveOperand(4);
152 MIB->RemoveOperand(2);
153
154 MIB->setDesc(TII.get(ARM::VMOVDRR));
155 MIB.add(predOps(ARMCC::AL));
156
157 return true;
158}
159
160static bool selectExtract(MachineInstrBuilder &MIB, const ARMBaseInstrInfo &TII,
161 MachineRegisterInfo &MRI,
162 const TargetRegisterInfo &TRI,
163 const RegisterBankInfo &RBI) {
164 assert(TII.getSubtarget().hasVFP2() && "Can't select extract without VFP");
165
166 // We only support G_EXTRACT as a way to break up one DPR into two GPRs.
167 unsigned VReg0 = MIB->getOperand(0).getReg();
168 (void)VReg0;
169 assert(MRI.getType(VReg0).getSizeInBits() == 32 &&
170 RBI.getRegBank(VReg0, MRI, TRI)->getID() == ARM::GPRRegBankID &&
Tim Northoverc2c545b2017-03-06 23:50:28 +0000171 "Unsupported operand for G_EXTRACT");
Diana Picusb1701e02017-02-16 12:19:57 +0000172 unsigned VReg1 = MIB->getOperand(1).getReg();
173 (void)VReg1;
Tim Northoverc2c545b2017-03-06 23:50:28 +0000174 assert(MRI.getType(VReg1).getSizeInBits() == 64 &&
175 RBI.getRegBank(VReg1, MRI, TRI)->getID() == ARM::FPRRegBankID &&
176 "Unsupported operand for G_EXTRACT");
177 assert(MIB->getOperand(2).getImm() % 32 == 0 &&
178 "Unsupported operand for G_EXTRACT");
Diana Picusb1701e02017-02-16 12:19:57 +0000179
180 // Remove the operands corresponding to the offsets.
Tim Northoverc2c545b2017-03-06 23:50:28 +0000181 MIB->getOperand(2).setImm(MIB->getOperand(2).getImm() / 32);
Diana Picusb1701e02017-02-16 12:19:57 +0000182
Tim Northoverc2c545b2017-03-06 23:50:28 +0000183 MIB->setDesc(TII.get(ARM::VGETLNi32));
Diana Picusb1701e02017-02-16 12:19:57 +0000184 MIB.add(predOps(ARMCC::AL));
185
186 return true;
187}
188
Diana Picus8b6c6be2017-01-25 08:10:40 +0000189/// Select the opcode for simple extensions (that translate to a single SXT/UXT
190/// instruction). Extension operations more complicated than that should not
Diana Picuse8368782017-02-17 13:44:19 +0000191/// invoke this. Returns the original opcode if it doesn't know how to select a
192/// better one.
Diana Picus8b6c6be2017-01-25 08:10:40 +0000193static unsigned selectSimpleExtOpc(unsigned Opc, unsigned Size) {
194 using namespace TargetOpcode;
195
Diana Picuse8368782017-02-17 13:44:19 +0000196 if (Size != 8 && Size != 16)
197 return Opc;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000198
199 if (Opc == G_SEXT)
200 return Size == 8 ? ARM::SXTB : ARM::SXTH;
201
202 if (Opc == G_ZEXT)
203 return Size == 8 ? ARM::UXTB : ARM::UXTH;
204
Diana Picuse8368782017-02-17 13:44:19 +0000205 return Opc;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000206}
207
Diana Picus3b99c642017-02-24 14:01:27 +0000208/// Select the opcode for simple loads and stores. For types smaller than 32
209/// bits, the value will be zero extended. Returns the original opcode if it
210/// doesn't know how to select a better one.
211static unsigned selectLoadStoreOpCode(unsigned Opc, unsigned RegBank,
212 unsigned Size) {
213 bool isStore = Opc == TargetOpcode::G_STORE;
214
Diana Picus1540b062017-02-16 14:10:50 +0000215 if (RegBank == ARM::GPRRegBankID) {
216 switch (Size) {
217 case 1:
218 case 8:
Diana Picus3b99c642017-02-24 14:01:27 +0000219 return isStore ? ARM::STRBi12 : ARM::LDRBi12;
Diana Picus1540b062017-02-16 14:10:50 +0000220 case 16:
Diana Picus3b99c642017-02-24 14:01:27 +0000221 return isStore ? ARM::STRH : ARM::LDRH;
Diana Picus1540b062017-02-16 14:10:50 +0000222 case 32:
Diana Picus3b99c642017-02-24 14:01:27 +0000223 return isStore ? ARM::STRi12 : ARM::LDRi12;
Diana Picuse8368782017-02-17 13:44:19 +0000224 default:
Diana Picus3b99c642017-02-24 14:01:27 +0000225 return Opc;
Diana Picus1540b062017-02-16 14:10:50 +0000226 }
Diana Picus1540b062017-02-16 14:10:50 +0000227 }
228
Diana Picuse8368782017-02-17 13:44:19 +0000229 if (RegBank == ARM::FPRRegBankID) {
230 switch (Size) {
231 case 32:
Diana Picus3b99c642017-02-24 14:01:27 +0000232 return isStore ? ARM::VSTRS : ARM::VLDRS;
Diana Picuse8368782017-02-17 13:44:19 +0000233 case 64:
Diana Picus3b99c642017-02-24 14:01:27 +0000234 return isStore ? ARM::VSTRD : ARM::VLDRD;
Diana Picuse8368782017-02-17 13:44:19 +0000235 default:
Diana Picus3b99c642017-02-24 14:01:27 +0000236 return Opc;
Diana Picuse8368782017-02-17 13:44:19 +0000237 }
Diana Picus278c7222017-01-26 09:20:47 +0000238 }
239
Diana Picus3b99c642017-02-24 14:01:27 +0000240 return Opc;
Diana Picus278c7222017-01-26 09:20:47 +0000241}
242
Diana Picus812caee2016-12-16 12:54:46 +0000243bool ARMInstructionSelector::select(MachineInstr &I) const {
244 assert(I.getParent() && "Instruction should be in a basic block!");
245 assert(I.getParent()->getParent() && "Instruction should be in a function!");
246
247 auto &MBB = *I.getParent();
248 auto &MF = *MBB.getParent();
249 auto &MRI = MF.getRegInfo();
250
251 if (!isPreISelGenericOpcode(I.getOpcode())) {
252 if (I.isCopy())
253 return selectCopy(I, TII, MRI, TRI, RBI);
254
255 return true;
256 }
257
Diana Picus519807f2016-12-19 11:26:31 +0000258 MachineInstrBuilder MIB{MF, I};
Diana Picusd83df5d2017-01-25 08:47:40 +0000259 bool isSExt = false;
Diana Picus519807f2016-12-19 11:26:31 +0000260
261 using namespace TargetOpcode;
262 switch (I.getOpcode()) {
Diana Picus8b6c6be2017-01-25 08:10:40 +0000263 case G_SEXT:
Diana Picusd83df5d2017-01-25 08:47:40 +0000264 isSExt = true;
265 LLVM_FALLTHROUGH;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000266 case G_ZEXT: {
267 LLT DstTy = MRI.getType(I.getOperand(0).getReg());
268 // FIXME: Smaller destination sizes coming soon!
269 if (DstTy.getSizeInBits() != 32) {
270 DEBUG(dbgs() << "Unsupported destination size for extension");
271 return false;
272 }
273
274 LLT SrcTy = MRI.getType(I.getOperand(1).getReg());
275 unsigned SrcSize = SrcTy.getSizeInBits();
276 switch (SrcSize) {
Diana Picusd83df5d2017-01-25 08:47:40 +0000277 case 1: {
278 // ZExt boils down to & 0x1; for SExt we also subtract that from 0
279 I.setDesc(TII.get(ARM::ANDri));
280 MIB.addImm(1).add(predOps(ARMCC::AL)).add(condCodeOp());
281
282 if (isSExt) {
283 unsigned SExtResult = I.getOperand(0).getReg();
284
285 // Use a new virtual register for the result of the AND
286 unsigned AndResult = MRI.createVirtualRegister(&ARM::GPRRegClass);
287 I.getOperand(0).setReg(AndResult);
288
289 auto InsertBefore = std::next(I.getIterator());
Martin Bohme8396e142017-01-25 14:28:19 +0000290 auto SubI =
Diana Picusd83df5d2017-01-25 08:47:40 +0000291 BuildMI(MBB, InsertBefore, I.getDebugLoc(), TII.get(ARM::RSBri))
292 .addDef(SExtResult)
293 .addUse(AndResult)
294 .addImm(0)
295 .add(predOps(ARMCC::AL))
296 .add(condCodeOp());
297 if (!constrainSelectedInstRegOperands(*SubI, TII, TRI, RBI))
298 return false;
299 }
300 break;
301 }
Diana Picus8b6c6be2017-01-25 08:10:40 +0000302 case 8:
303 case 16: {
304 unsigned NewOpc = selectSimpleExtOpc(I.getOpcode(), SrcSize);
Diana Picuse8368782017-02-17 13:44:19 +0000305 if (NewOpc == I.getOpcode())
306 return false;
Diana Picus8b6c6be2017-01-25 08:10:40 +0000307 I.setDesc(TII.get(NewOpc));
308 MIB.addImm(0).add(predOps(ARMCC::AL));
309 break;
310 }
311 default:
312 DEBUG(dbgs() << "Unsupported source size for extension");
313 return false;
314 }
315 break;
316 }
Diana Picus64a33432017-04-21 13:16:50 +0000317 case G_TRUNC: {
318 // The high bits are undefined, so there's nothing special to do, just
319 // treat it as a copy.
320 auto SrcReg = I.getOperand(1).getReg();
321 auto DstReg = I.getOperand(0).getReg();
322
323 const auto &SrcRegBank = *RBI.getRegBank(SrcReg, MRI, TRI);
324 const auto &DstRegBank = *RBI.getRegBank(DstReg, MRI, TRI);
325
326 if (SrcRegBank.getID() != DstRegBank.getID()) {
327 DEBUG(dbgs() << "G_TRUNC operands on different register banks\n");
328 return false;
329 }
330
331 if (SrcRegBank.getID() != ARM::GPRRegBankID) {
332 DEBUG(dbgs() << "G_TRUNC on non-GPR not supported yet\n");
333 return false;
334 }
335
336 I.setDesc(TII.get(COPY));
337 return selectCopy(I, TII, MRI, TRI, RBI);
338 }
Diana Picus519807f2016-12-19 11:26:31 +0000339 case G_ADD:
Diana Picus9d070942017-02-28 10:14:38 +0000340 case G_GEP:
Diana Picus812caee2016-12-16 12:54:46 +0000341 I.setDesc(TII.get(ARM::ADDrr));
Diana Picus8a73f552017-01-13 10:18:01 +0000342 MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
Diana Picus519807f2016-12-19 11:26:31 +0000343 break;
Diana Picusa3a0ccc2017-04-18 12:35:28 +0000344 case G_SUB:
345 I.setDesc(TII.get(ARM::SUBrr));
346 MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
347 break;
Diana Picus49472ff2017-04-19 07:29:46 +0000348 case G_MUL:
349 if (TII.getSubtarget().hasV6Ops()) {
350 I.setDesc(TII.get(ARM::MUL));
351 } else {
352 assert(TII.getSubtarget().useMulOps() && "Unsupported target");
353 I.setDesc(TII.get(ARM::MULv5));
354 MIB->getOperand(0).setIsEarlyClobber(true);
355 }
356 MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
357 break;
Diana Picusb70e88b2017-04-24 08:20:05 +0000358 case G_SDIV:
359 assert(TII.getSubtarget().hasDivideInARMMode() && "Unsupported operation");
360 I.setDesc(TII.get(ARM::SDIV));
361 MIB.add(predOps(ARMCC::AL));
362 break;
363 case G_UDIV:
364 assert(TII.getSubtarget().hasDivideInARMMode() && "Unsupported operation");
365 I.setDesc(TII.get(ARM::UDIV));
366 MIB.add(predOps(ARMCC::AL));
367 break;
Diana Picus4fa83c02017-02-08 13:23:04 +0000368 case G_FADD:
Diana Picus6beef3c2017-02-16 12:19:52 +0000369 if (!selectFAdd(MIB, TII, MRI))
Diana Picus4fa83c02017-02-08 13:23:04 +0000370 return false;
Diana Picus4fa83c02017-02-08 13:23:04 +0000371 break;
Diana Picus519807f2016-12-19 11:26:31 +0000372 case G_FRAME_INDEX:
373 // Add 0 to the given frame index and hope it will eventually be folded into
374 // the user(s).
375 I.setDesc(TII.get(ARM::ADDri));
Diana Picus8a73f552017-01-13 10:18:01 +0000376 MIB.addImm(0).add(predOps(ARMCC::AL)).add(condCodeOp());
Diana Picus519807f2016-12-19 11:26:31 +0000377 break;
Diana Picus5a7203a2017-02-28 13:05:42 +0000378 case G_CONSTANT: {
379 unsigned Reg = I.getOperand(0).getReg();
380 if (MRI.getType(Reg).getSizeInBits() != 32)
381 return false;
382
383 assert(RBI.getRegBank(Reg, MRI, TRI)->getID() == ARM::GPRRegBankID &&
384 "Expected constant to live in a GPR");
385 I.setDesc(TII.get(ARM::MOVi));
386 MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
Diana Picus95a8aa92017-04-24 06:30:56 +0000387
388 auto &Val = I.getOperand(1);
389 if (Val.isCImm()) {
390 if (Val.getCImm()->getBitWidth() > 32)
391 return false;
392 Val.ChangeToImmediate(Val.getCImm()->getZExtValue());
393 }
394
395 if (!Val.isImm()) {
396 return false;
397 }
398
Diana Picus5a7203a2017-02-28 13:05:42 +0000399 break;
400 }
Diana Picus3b99c642017-02-24 14:01:27 +0000401 case G_STORE:
Diana Picus278c7222017-01-26 09:20:47 +0000402 case G_LOAD: {
Diana Picus1c33c9f2017-02-20 14:45:58 +0000403 const auto &MemOp = **I.memoperands_begin();
404 if (MemOp.getOrdering() != AtomicOrdering::NotAtomic) {
405 DEBUG(dbgs() << "Atomic load/store not supported yet\n");
406 return false;
407 }
408
Diana Picus1540b062017-02-16 14:10:50 +0000409 unsigned Reg = I.getOperand(0).getReg();
410 unsigned RegBank = RBI.getRegBank(Reg, MRI, TRI)->getID();
411
412 LLT ValTy = MRI.getType(Reg);
Diana Picus278c7222017-01-26 09:20:47 +0000413 const auto ValSize = ValTy.getSizeInBits();
414
Diana Picus1540b062017-02-16 14:10:50 +0000415 assert((ValSize != 64 || TII.getSubtarget().hasVFP2()) &&
Diana Picus3b99c642017-02-24 14:01:27 +0000416 "Don't know how to load/store 64-bit value without VFP");
Diana Picus1540b062017-02-16 14:10:50 +0000417
Diana Picus3b99c642017-02-24 14:01:27 +0000418 const auto NewOpc = selectLoadStoreOpCode(I.getOpcode(), RegBank, ValSize);
419 if (NewOpc == G_LOAD || NewOpc == G_STORE)
Diana Picuse8368782017-02-17 13:44:19 +0000420 return false;
421
Diana Picus278c7222017-01-26 09:20:47 +0000422 I.setDesc(TII.get(NewOpc));
423
Diana Picus3b99c642017-02-24 14:01:27 +0000424 if (NewOpc == ARM::LDRH || NewOpc == ARM::STRH)
Diana Picus278c7222017-01-26 09:20:47 +0000425 // LDRH has a funny addressing mode (there's already a FIXME for it).
426 MIB.addReg(0);
Diana Picus4f8c3e12017-01-13 09:37:56 +0000427 MIB.addImm(0).add(predOps(ARMCC::AL));
Diana Picus519807f2016-12-19 11:26:31 +0000428 break;
Diana Picus278c7222017-01-26 09:20:47 +0000429 }
Diana Picusb1701e02017-02-16 12:19:57 +0000430 case G_SEQUENCE: {
431 if (!selectSequence(MIB, TII, MRI, TRI, RBI))
432 return false;
433 break;
434 }
435 case G_EXTRACT: {
436 if (!selectExtract(MIB, TII, MRI, TRI, RBI))
437 return false;
438 break;
439 }
Diana Picus519807f2016-12-19 11:26:31 +0000440 default:
441 return false;
Diana Picus812caee2016-12-16 12:54:46 +0000442 }
443
Diana Picus519807f2016-12-19 11:26:31 +0000444 return constrainSelectedInstRegOperands(I, TII, TRI, RBI);
Diana Picus22274932016-11-11 08:27:37 +0000445}