1 //===-- ThumbRegisterInfo.cpp - Thumb-1 Register Information -------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file contains the Thumb-1 implementation of the TargetRegisterInfo
12 //===----------------------------------------------------------------------===//
14 #include "ThumbRegisterInfo.h"
15 #include "ARMBaseInstrInfo.h"
16 #include "ARMMachineFunctionInfo.h"
17 #include "ARMSubtarget.h"
18 #include "MCTargetDesc/ARMAddressingModes.h"
19 #include "llvm/CodeGen/MachineConstantPool.h"
20 #include "llvm/CodeGen/MachineFrameInfo.h"
21 #include "llvm/CodeGen/MachineFunction.h"
22 #include "llvm/CodeGen/MachineInstrBuilder.h"
23 #include "llvm/CodeGen/MachineRegisterInfo.h"
24 #include "llvm/CodeGen/RegisterScavenging.h"
25 #include "llvm/IR/Constants.h"
26 #include "llvm/IR/DerivedTypes.h"
27 #include "llvm/IR/Function.h"
28 #include "llvm/IR/LLVMContext.h"
29 #include "llvm/Support/CommandLine.h"
30 #include "llvm/Support/ErrorHandling.h"
31 #include "llvm/CodeGen/TargetFrameLowering.h"
32 #include "llvm/Target/TargetMachine.h"
35 extern cl::opt
<bool> ReuseFrameIndexVals
;
40 ThumbRegisterInfo::ThumbRegisterInfo() : ARMBaseRegisterInfo() {}
42 const TargetRegisterClass
*
43 ThumbRegisterInfo::getLargestLegalSuperClass(const TargetRegisterClass
*RC
,
44 const MachineFunction
&MF
) const {
45 if (!MF
.getSubtarget
<ARMSubtarget
>().isThumb1Only())
46 return ARMBaseRegisterInfo::getLargestLegalSuperClass(RC
, MF
);
48 if (ARM::tGPRRegClass
.hasSubClassEq(RC
))
49 return &ARM::tGPRRegClass
;
50 return ARMBaseRegisterInfo::getLargestLegalSuperClass(RC
, MF
);
53 const TargetRegisterClass
*
54 ThumbRegisterInfo::getPointerRegClass(const MachineFunction
&MF
,
55 unsigned Kind
) const {
56 if (!MF
.getSubtarget
<ARMSubtarget
>().isThumb1Only())
57 return ARMBaseRegisterInfo::getPointerRegClass(MF
, Kind
);
58 return &ARM::tGPRRegClass
;
61 static void emitThumb1LoadConstPool(MachineBasicBlock
&MBB
,
62 MachineBasicBlock::iterator
&MBBI
,
63 const DebugLoc
&dl
, unsigned DestReg
,
64 unsigned SubIdx
, int Val
,
65 ARMCC::CondCodes Pred
, unsigned PredReg
,
67 MachineFunction
&MF
= *MBB
.getParent();
68 const ARMSubtarget
&STI
= MF
.getSubtarget
<ARMSubtarget
>();
69 const TargetInstrInfo
&TII
= *STI
.getInstrInfo();
70 MachineConstantPool
*ConstantPool
= MF
.getConstantPool();
71 const Constant
*C
= ConstantInt::get(
72 Type::getInt32Ty(MBB
.getParent()->getFunction().getContext()), Val
);
73 unsigned Idx
= ConstantPool
->getConstantPoolIndex(C
, 4);
75 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::tLDRpci
))
76 .addReg(DestReg
, getDefRegState(true), SubIdx
)
77 .addConstantPoolIndex(Idx
).addImm(Pred
).addReg(PredReg
)
81 static void emitThumb2LoadConstPool(MachineBasicBlock
&MBB
,
82 MachineBasicBlock::iterator
&MBBI
,
83 const DebugLoc
&dl
, unsigned DestReg
,
84 unsigned SubIdx
, int Val
,
85 ARMCC::CondCodes Pred
, unsigned PredReg
,
87 MachineFunction
&MF
= *MBB
.getParent();
88 const TargetInstrInfo
&TII
= *MF
.getSubtarget().getInstrInfo();
89 MachineConstantPool
*ConstantPool
= MF
.getConstantPool();
90 const Constant
*C
= ConstantInt::get(
91 Type::getInt32Ty(MBB
.getParent()->getFunction().getContext()), Val
);
92 unsigned Idx
= ConstantPool
->getConstantPoolIndex(C
, 4);
94 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::t2LDRpci
))
95 .addReg(DestReg
, getDefRegState(true), SubIdx
)
96 .addConstantPoolIndex(Idx
)
97 .add(predOps(ARMCC::AL
))
101 /// emitLoadConstPool - Emits a load from constpool to materialize the
102 /// specified immediate.
103 void ThumbRegisterInfo::emitLoadConstPool(
104 MachineBasicBlock
&MBB
, MachineBasicBlock::iterator
&MBBI
,
105 const DebugLoc
&dl
, unsigned DestReg
, unsigned SubIdx
, int Val
,
106 ARMCC::CondCodes Pred
, unsigned PredReg
, unsigned MIFlags
) const {
107 MachineFunction
&MF
= *MBB
.getParent();
108 const ARMSubtarget
&STI
= MF
.getSubtarget
<ARMSubtarget
>();
109 if (STI
.isThumb1Only()) {
110 assert((isARMLowRegister(DestReg
) || isVirtualRegister(DestReg
)) &&
111 "Thumb1 does not have ldr to high register");
112 return emitThumb1LoadConstPool(MBB
, MBBI
, dl
, DestReg
, SubIdx
, Val
, Pred
,
115 return emitThumb2LoadConstPool(MBB
, MBBI
, dl
, DestReg
, SubIdx
, Val
, Pred
,
119 /// emitThumbRegPlusImmInReg - Emits a series of instructions to materialize
120 /// a destreg = basereg + immediate in Thumb code. Materialize the immediate
121 /// in a register using mov / mvn sequences or load the immediate from a
123 static void emitThumbRegPlusImmInReg(
124 MachineBasicBlock
&MBB
, MachineBasicBlock::iterator
&MBBI
,
125 const DebugLoc
&dl
, unsigned DestReg
, unsigned BaseReg
, int NumBytes
,
126 bool CanChangeCC
, const TargetInstrInfo
&TII
,
127 const ARMBaseRegisterInfo
&MRI
, unsigned MIFlags
= MachineInstr::NoFlags
) {
128 MachineFunction
&MF
= *MBB
.getParent();
129 const ARMSubtarget
&ST
= MF
.getSubtarget
<ARMSubtarget
>();
130 bool isHigh
= !isARMLowRegister(DestReg
) ||
131 (BaseReg
!= 0 && !isARMLowRegister(BaseReg
));
133 // Subtract doesn't have high register version. Load the negative value
134 // if either base or dest register is a high register. Also, if do not
135 // issue sub as part of the sequence if condition register is to be
137 if (NumBytes
< 0 && !isHigh
&& CanChangeCC
) {
139 NumBytes
= -NumBytes
;
141 unsigned LdReg
= DestReg
;
142 if (DestReg
== ARM::SP
)
143 assert(BaseReg
== ARM::SP
&& "Unexpected!");
144 if (!isARMLowRegister(DestReg
) && !MRI
.isVirtualRegister(DestReg
))
145 LdReg
= MF
.getRegInfo().createVirtualRegister(&ARM::tGPRRegClass
);
147 if (NumBytes
<= 255 && NumBytes
>= 0 && CanChangeCC
) {
148 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::tMOVi8
), LdReg
)
151 .setMIFlags(MIFlags
);
152 } else if (NumBytes
< 0 && NumBytes
>= -255 && CanChangeCC
) {
153 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::tMOVi8
), LdReg
)
156 .setMIFlags(MIFlags
);
157 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::tRSB
), LdReg
)
159 .addReg(LdReg
, RegState::Kill
)
160 .setMIFlags(MIFlags
);
161 } else if (ST
.genExecuteOnly()) {
162 BuildMI(MBB
, MBBI
, dl
, TII
.get(ARM::t2MOVi32imm
), LdReg
)
163 .addImm(NumBytes
).setMIFlags(MIFlags
);
165 MRI
.emitLoadConstPool(MBB
, MBBI
, dl
, LdReg
, 0, NumBytes
, ARMCC::AL
, 0,
169 int Opc
= (isSub
) ? ARM::tSUBrr
170 : ((isHigh
|| !CanChangeCC
) ? ARM::tADDhirr
: ARM::tADDrr
);
171 MachineInstrBuilder MIB
= BuildMI(MBB
, MBBI
, dl
, TII
.get(Opc
), DestReg
);
172 if (Opc
!= ARM::tADDhirr
)
173 MIB
= MIB
.add(t1CondCodeOp());
174 if (DestReg
== ARM::SP
|| isSub
)
175 MIB
.addReg(BaseReg
).addReg(LdReg
, RegState::Kill
);
177 MIB
.addReg(LdReg
).addReg(BaseReg
, RegState::Kill
);
178 MIB
.add(predOps(ARMCC::AL
));
181 /// emitThumbRegPlusImmediate - Emits a series of instructions to materialize
182 /// a destreg = basereg + immediate in Thumb code. Tries a series of ADDs or
183 /// SUBs first, and uses a constant pool value if the instruction sequence would
184 /// be too long. This is allowed to modify the condition flags.
185 void llvm::emitThumbRegPlusImmediate(MachineBasicBlock
&MBB
,
186 MachineBasicBlock::iterator
&MBBI
,
187 const DebugLoc
&dl
, unsigned DestReg
,
188 unsigned BaseReg
, int NumBytes
,
189 const TargetInstrInfo
&TII
,
190 const ARMBaseRegisterInfo
&MRI
,
192 bool isSub
= NumBytes
< 0;
193 unsigned Bytes
= (unsigned)NumBytes
;
194 if (isSub
) Bytes
= -NumBytes
;
197 unsigned CopyBits
= 0;
198 unsigned CopyScale
= 1;
199 bool CopyNeedsCC
= false;
201 unsigned ExtraBits
= 0;
202 unsigned ExtraScale
= 1;
203 bool ExtraNeedsCC
= false;
206 // We need to select two types of instruction, maximizing the available
207 // immediate range of each. The instructions we use will depend on whether
208 // DestReg and BaseReg are low, high or the stack pointer.
209 // * CopyOpc - DestReg = BaseReg + imm
210 // This will be emitted once if DestReg != BaseReg, and never if
211 // DestReg == BaseReg.
212 // * ExtraOpc - DestReg = DestReg + imm
213 // This will be emitted as many times as necessary to add the
215 // If the immediate ranges of these instructions are not large enough to cover
216 // NumBytes with a reasonable number of instructions, we fall back to using a
217 // value loaded from a constant pool.
218 if (DestReg
== ARM::SP
) {
219 if (BaseReg
== ARM::SP
) {
221 // Already in right reg, no copy needed
223 // low -> sp or high -> sp
224 CopyOpc
= ARM::tMOVr
;
227 ExtraOpc
= isSub
? ARM::tSUBspi
: ARM::tADDspi
;
230 } else if (isARMLowRegister(DestReg
)) {
231 if (BaseReg
== ARM::SP
) {
233 assert(!isSub
&& "Thumb1 does not have tSUBrSPi");
234 CopyOpc
= ARM::tADDrSPi
;
237 } else if (DestReg
== BaseReg
) {
239 // Already in right reg, no copy needed
240 } else if (isARMLowRegister(BaseReg
)) {
241 // low -> different low
242 CopyOpc
= isSub
? ARM::tSUBi3
: ARM::tADDi3
;
247 CopyOpc
= ARM::tMOVr
;
250 ExtraOpc
= isSub
? ARM::tSUBi8
: ARM::tADDi8
;
253 } else /* DestReg is high */ {
254 if (DestReg
== BaseReg
) {
256 // Already in right reg, no copy needed
258 // {low,high,sp} -> high
259 CopyOpc
= ARM::tMOVr
;
265 // We could handle an unaligned immediate with an unaligned copy instruction
266 // and an aligned extra instruction, but this case is not currently needed.
267 assert(((Bytes
& 3) == 0 || ExtraScale
== 1) &&
268 "Unaligned offset, but all instructions require alignment");
270 unsigned CopyRange
= ((1 << CopyBits
) - 1) * CopyScale
;
271 // If we would emit the copy with an immediate of 0, just use tMOVr.
272 if (CopyOpc
&& Bytes
< CopyScale
) {
273 CopyOpc
= ARM::tMOVr
;
278 unsigned ExtraRange
= ((1 << ExtraBits
) - 1) * ExtraScale
; // per instruction
279 unsigned RequiredCopyInstrs
= CopyOpc
? 1 : 0;
280 unsigned RangeAfterCopy
= (CopyRange
> Bytes
) ? 0 : (Bytes
- CopyRange
);
282 // We could handle this case when the copy instruction does not require an
283 // aligned immediate, but we do not currently do this.
284 assert(RangeAfterCopy
% ExtraScale
== 0 &&
285 "Extra instruction requires immediate to be aligned");
287 unsigned RequiredExtraInstrs
;
289 RequiredExtraInstrs
= alignTo(RangeAfterCopy
, ExtraRange
) / ExtraRange
;
290 else if (RangeAfterCopy
> 0)
291 // We need an extra instruction but none is available
292 RequiredExtraInstrs
= 1000000;
294 RequiredExtraInstrs
= 0;
295 unsigned RequiredInstrs
= RequiredCopyInstrs
+ RequiredExtraInstrs
;
296 unsigned Threshold
= (DestReg
== ARM::SP
) ? 3 : 2;
298 // Use a constant pool, if the sequence of ADDs/SUBs is too expensive.
299 if (RequiredInstrs
> Threshold
) {
300 emitThumbRegPlusImmInReg(MBB
, MBBI
, dl
,
301 DestReg
, BaseReg
, NumBytes
, true,
306 // Emit zero or one copy instructions
308 unsigned CopyImm
= std::min(Bytes
, CopyRange
) / CopyScale
;
309 Bytes
-= CopyImm
* CopyScale
;
311 MachineInstrBuilder MIB
= BuildMI(MBB
, MBBI
, dl
, TII
.get(CopyOpc
), DestReg
);
313 MIB
= MIB
.add(t1CondCodeOp());
314 MIB
.addReg(BaseReg
, RegState::Kill
);
315 if (CopyOpc
!= ARM::tMOVr
) {
318 MIB
.setMIFlags(MIFlags
).add(predOps(ARMCC::AL
));
323 // Emit zero or more in-place add/sub instructions
325 unsigned ExtraImm
= std::min(Bytes
, ExtraRange
) / ExtraScale
;
326 Bytes
-= ExtraImm
* ExtraScale
;
328 MachineInstrBuilder MIB
= BuildMI(MBB
, MBBI
, dl
, TII
.get(ExtraOpc
), DestReg
);
330 MIB
= MIB
.add(t1CondCodeOp());
333 .add(predOps(ARMCC::AL
))
334 .setMIFlags(MIFlags
);
338 static void removeOperands(MachineInstr
&MI
, unsigned i
) {
340 for (unsigned e
= MI
.getNumOperands(); i
!= e
; ++i
)
341 MI
.RemoveOperand(Op
);
344 /// convertToNonSPOpcode - Change the opcode to the non-SP version, because
345 /// we're replacing the frame index with a non-SP register.
346 static unsigned convertToNonSPOpcode(unsigned Opcode
) {
358 bool ThumbRegisterInfo::rewriteFrameIndex(MachineBasicBlock::iterator II
,
359 unsigned FrameRegIdx
,
360 unsigned FrameReg
, int &Offset
,
361 const ARMBaseInstrInfo
&TII
) const {
362 MachineInstr
&MI
= *II
;
363 MachineBasicBlock
&MBB
= *MI
.getParent();
364 assert(MBB
.getParent()->getSubtarget
<ARMSubtarget
>().isThumb1Only() &&
365 "This isn't needed for thumb2!");
366 DebugLoc dl
= MI
.getDebugLoc();
367 MachineInstrBuilder
MIB(*MBB
.getParent(), &MI
);
368 unsigned Opcode
= MI
.getOpcode();
369 const MCInstrDesc
&Desc
= MI
.getDesc();
370 unsigned AddrMode
= (Desc
.TSFlags
& ARMII::AddrModeMask
);
372 if (Opcode
== ARM::tADDframe
) {
373 Offset
+= MI
.getOperand(FrameRegIdx
+1).getImm();
374 unsigned DestReg
= MI
.getOperand(0).getReg();
376 emitThumbRegPlusImmediate(MBB
, II
, dl
, DestReg
, FrameReg
, Offset
, TII
,
381 if (AddrMode
!= ARMII::AddrModeT1_s
)
382 llvm_unreachable("Unsupported addressing mode!");
384 unsigned ImmIdx
= FrameRegIdx
+ 1;
385 int InstrOffs
= MI
.getOperand(ImmIdx
).getImm();
386 unsigned NumBits
= (FrameReg
== ARM::SP
) ? 8 : 5;
389 Offset
+= InstrOffs
* Scale
;
390 assert((Offset
& (Scale
- 1)) == 0 && "Can't encode this offset!");
392 // Common case: small offset, fits into instruction.
393 MachineOperand
&ImmOp
= MI
.getOperand(ImmIdx
);
394 int ImmedOffset
= Offset
/ Scale
;
395 unsigned Mask
= (1 << NumBits
) - 1;
397 if ((unsigned)Offset
<= Mask
* Scale
) {
398 // Replace the FrameIndex with the frame register (e.g., sp).
399 MI
.getOperand(FrameRegIdx
).ChangeToRegister(FrameReg
, false);
400 ImmOp
.ChangeToImmediate(ImmedOffset
);
402 // If we're using a register where sp was stored, convert the instruction
403 // to the non-SP version.
404 unsigned NewOpc
= convertToNonSPOpcode(Opcode
);
405 if (NewOpc
!= Opcode
&& FrameReg
!= ARM::SP
)
406 MI
.setDesc(TII
.get(NewOpc
));
412 Mask
= (1 << NumBits
) - 1;
414 // If this is a thumb spill / restore, we will be using a constpool load to
415 // materialize the offset.
416 if (Opcode
== ARM::tLDRspi
|| Opcode
== ARM::tSTRspi
) {
417 ImmOp
.ChangeToImmediate(0);
419 // Otherwise, it didn't fit. Pull in what we can to simplify the immed.
420 ImmedOffset
= ImmedOffset
& Mask
;
421 ImmOp
.ChangeToImmediate(ImmedOffset
);
422 Offset
&= ~(Mask
* Scale
);
429 void ThumbRegisterInfo::resolveFrameIndex(MachineInstr
&MI
, unsigned BaseReg
,
430 int64_t Offset
) const {
431 const MachineFunction
&MF
= *MI
.getParent()->getParent();
432 const ARMSubtarget
&STI
= MF
.getSubtarget
<ARMSubtarget
>();
433 if (!STI
.isThumb1Only())
434 return ARMBaseRegisterInfo::resolveFrameIndex(MI
, BaseReg
, Offset
);
436 const ARMBaseInstrInfo
&TII
= *STI
.getInstrInfo();
437 int Off
= Offset
; // ARM doesn't need the general 64-bit offsets
440 while (!MI
.getOperand(i
).isFI()) {
442 assert(i
< MI
.getNumOperands() && "Instr doesn't have FrameIndex operand!");
444 bool Done
= rewriteFrameIndex(MI
, i
, BaseReg
, Off
, TII
);
445 assert (Done
&& "Unable to resolve frame index!");
449 void ThumbRegisterInfo::eliminateFrameIndex(MachineBasicBlock::iterator II
,
450 int SPAdj
, unsigned FIOperandNum
,
451 RegScavenger
*RS
) const {
452 MachineInstr
&MI
= *II
;
453 MachineBasicBlock
&MBB
= *MI
.getParent();
454 MachineFunction
&MF
= *MBB
.getParent();
455 const ARMSubtarget
&STI
= MF
.getSubtarget
<ARMSubtarget
>();
456 if (!STI
.isThumb1Only())
457 return ARMBaseRegisterInfo::eliminateFrameIndex(II
, SPAdj
, FIOperandNum
,
461 const ARMBaseInstrInfo
&TII
= *STI
.getInstrInfo();
462 DebugLoc dl
= MI
.getDebugLoc();
463 MachineInstrBuilder
MIB(*MBB
.getParent(), &MI
);
466 int FrameIndex
= MI
.getOperand(FIOperandNum
).getIndex();
467 const ARMFrameLowering
*TFI
= getFrameLowering(MF
);
468 int Offset
= TFI
->ResolveFrameIndexReference(MF
, FrameIndex
, FrameReg
, SPAdj
);
470 // PEI::scavengeFrameVirtualRegs() cannot accurately track SPAdj because the
471 // call frame setup/destroy instructions have already been eliminated. That
472 // means the stack pointer cannot be used to access the emergency spill slot
473 // when !hasReservedCallFrame().
475 if (RS
&& FrameReg
== ARM::SP
&& RS
->isScavengingFrameIndex(FrameIndex
)){
476 assert(STI
.getFrameLowering()->hasReservedCallFrame(MF
) &&
477 "Cannot use SP to access the emergency spill slot in "
478 "functions without a reserved call frame");
479 assert(!MF
.getFrameInfo().hasVarSizedObjects() &&
480 "Cannot use SP to access the emergency spill slot in "
481 "functions with variable sized frame objects");
485 // Special handling of dbg_value instructions.
486 if (MI
.isDebugValue()) {
487 MI
.getOperand(FIOperandNum
). ChangeToRegister(FrameReg
, false /*isDef*/);
488 MI
.getOperand(FIOperandNum
+1).ChangeToImmediate(Offset
);
492 // Modify MI as necessary to handle as much of 'Offset' as possible
493 assert(MF
.getInfo
<ARMFunctionInfo
>()->isThumbFunction() &&
494 "This eliminateFrameIndex only supports Thumb1!");
495 if (rewriteFrameIndex(MI
, FIOperandNum
, FrameReg
, Offset
, TII
))
498 // If we get here, the immediate doesn't fit into the instruction. We folded
499 // as much as possible above, handle the rest, providing a register that is
501 assert(Offset
&& "This code isn't needed if offset already handled!");
503 unsigned Opcode
= MI
.getOpcode();
505 // Remove predicate first.
506 int PIdx
= MI
.findFirstPredOperandIdx();
508 removeOperands(MI
, PIdx
);
511 // Use the destination register to materialize sp + offset.
512 unsigned TmpReg
= MI
.getOperand(0).getReg();
514 if (Opcode
== ARM::tLDRspi
) {
515 if (FrameReg
== ARM::SP
|| STI
.genExecuteOnly())
516 emitThumbRegPlusImmInReg(MBB
, II
, dl
, TmpReg
, FrameReg
,
517 Offset
, false, TII
, *this);
519 emitLoadConstPool(MBB
, II
, dl
, TmpReg
, 0, Offset
);
523 emitThumbRegPlusImmediate(MBB
, II
, dl
, TmpReg
, FrameReg
, Offset
, TII
,
527 MI
.setDesc(TII
.get(UseRR
? ARM::tLDRr
: ARM::tLDRi
));
528 MI
.getOperand(FIOperandNum
).ChangeToRegister(TmpReg
, false, false, true);
530 // Use [reg, reg] addrmode. Replace the immediate operand w/ the frame
531 // register. The offset is already handled in the vreg value.
532 MI
.getOperand(FIOperandNum
+1).ChangeToRegister(FrameReg
, false, false,
534 } else if (MI
.mayStore()) {
535 VReg
= MF
.getRegInfo().createVirtualRegister(&ARM::tGPRRegClass
);
538 if (Opcode
== ARM::tSTRspi
) {
539 if (FrameReg
== ARM::SP
|| STI
.genExecuteOnly())
540 emitThumbRegPlusImmInReg(MBB
, II
, dl
, VReg
, FrameReg
,
541 Offset
, false, TII
, *this);
543 emitLoadConstPool(MBB
, II
, dl
, VReg
, 0, Offset
);
547 emitThumbRegPlusImmediate(MBB
, II
, dl
, VReg
, FrameReg
, Offset
, TII
,
549 MI
.setDesc(TII
.get(UseRR
? ARM::tSTRr
: ARM::tSTRi
));
550 MI
.getOperand(FIOperandNum
).ChangeToRegister(VReg
, false, false, true);
552 // Use [reg, reg] addrmode. Replace the immediate operand w/ the frame
553 // register. The offset is already handled in the vreg value.
554 MI
.getOperand(FIOperandNum
+1).ChangeToRegister(FrameReg
, false, false,
557 llvm_unreachable("Unexpected opcode!");
560 // Add predicate back if it's needed.
561 if (MI
.isPredicable())
562 MIB
.add(predOps(ARMCC::AL
));
566 ThumbRegisterInfo::useFPForScavengingIndex(const MachineFunction
&MF
) const {
567 if (MF
.getSubtarget
<ARMSubtarget
>().isThumb1Only()) {
568 // For Thumb1, the emergency spill slot must be some small positive
569 // offset from the base/stack pointer.
572 // For Thumb2, put the emergency spill slot next to FP.