1 //===- ARCRegisterInfo.cpp - ARC Register Information -----------*- C++ -*-===//
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 ARC implementation of the MRegisterInfo class.
11 //===----------------------------------------------------------------------===//
13 #include "ARCRegisterInfo.h"
15 #include "ARCInstrInfo.h"
16 #include "ARCMachineFunctionInfo.h"
17 #include "ARCSubtarget.h"
18 #include "llvm/ADT/BitVector.h"
19 #include "llvm/CodeGen/MachineFrameInfo.h"
20 #include "llvm/CodeGen/MachineFunction.h"
21 #include "llvm/CodeGen/MachineInstrBuilder.h"
22 #include "llvm/CodeGen/MachineModuleInfo.h"
23 #include "llvm/CodeGen/MachineRegisterInfo.h"
24 #include "llvm/CodeGen/RegisterScavenging.h"
25 #include "llvm/CodeGen/TargetFrameLowering.h"
26 #include "llvm/IR/Function.h"
27 #include "llvm/Support/Debug.h"
28 #include "llvm/Target/TargetMachine.h"
29 #include "llvm/Target/TargetOptions.h"
33 #define DEBUG_TYPE "arc-reg-info"
35 #define GET_REGINFO_TARGET_DESC
36 #include "ARCGenRegisterInfo.inc"
38 static void ReplaceFrameIndex(MachineBasicBlock::iterator II
,
39 const ARCInstrInfo
&TII
, unsigned Reg
,
40 unsigned FrameReg
, int Offset
, int StackSize
,
41 int ObjSize
, RegScavenger
*RS
, int SPAdj
) {
42 assert(RS
&& "Need register scavenger.");
43 MachineInstr
&MI
= *II
;
44 MachineBasicBlock
&MBB
= *MI
.getParent();
45 DebugLoc dl
= MI
.getDebugLoc();
46 unsigned BaseReg
= FrameReg
;
47 unsigned KillState
= 0;
48 if (MI
.getOpcode() == ARC::LD_rs9
&& (Offset
>= 256 || Offset
< -256)) {
49 // Loads can always be reached with LD_rlimm.
50 BuildMI(MBB
, II
, dl
, TII
.get(ARC::LD_rlimm
), Reg
)
53 .addMemOperand(*MI
.memoperands_begin());
58 if (MI
.getOpcode() != ARC::GETFI
&& (Offset
>= 256 || Offset
< -256)) {
59 // We need to use a scratch register to reach the far-away frame indexes.
60 BaseReg
= RS
->FindUnusedReg(&ARC::GPR32RegClass
);
62 // We can be sure that the scavenged-register slot is within the range
63 // of the load offset.
64 const TargetRegisterInfo
*TRI
=
65 MBB
.getParent()->getSubtarget().getRegisterInfo();
66 BaseReg
= RS
->scavengeRegister(&ARC::GPR32RegClass
, II
, SPAdj
);
67 assert(BaseReg
&& "Register scavenging failed.");
68 LLVM_DEBUG(dbgs() << "Scavenged register " << printReg(BaseReg
, TRI
)
69 << " for FrameReg=" << printReg(FrameReg
, TRI
)
70 << "+Offset=" << Offset
<< "\n");
72 RS
->setRegUsed(BaseReg
);
74 unsigned AddOpc
= isUInt
<6>(Offset
) ? ARC::ADD_rru6
: ARC::ADD_rrlimm
;
75 BuildMI(MBB
, II
, dl
, TII
.get(AddOpc
))
76 .addReg(BaseReg
, RegState::Define
)
80 KillState
= RegState::Kill
;
82 switch (MI
.getOpcode()) {
84 assert((Offset
% 4 == 0) && "LD needs 4 byte alignment.");
88 assert((Offset
% 2 == 0) && "LDH needs 2 byte alignment.");
92 LLVM_DEBUG(dbgs() << "Building LDFI\n");
93 BuildMI(MBB
, II
, dl
, TII
.get(MI
.getOpcode()), Reg
)
94 .addReg(BaseReg
, KillState
)
96 .addMemOperand(*MI
.memoperands_begin());
99 assert((Offset
% 4 == 0) && "ST needs 4 byte alignment.");
102 assert((Offset
% 2 == 0) && "STH needs 2 byte alignment.");
105 LLVM_DEBUG(dbgs() << "Building STFI\n");
106 BuildMI(MBB
, II
, dl
, TII
.get(MI
.getOpcode()))
107 .addReg(Reg
, getKillRegState(MI
.getOperand(0).isKill()))
108 .addReg(BaseReg
, KillState
)
110 .addMemOperand(*MI
.memoperands_begin());
113 LLVM_DEBUG(dbgs() << "Building GETFI\n");
115 TII
.get(isUInt
<6>(Offset
) ? ARC::ADD_rru6
: ARC::ADD_rrlimm
))
116 .addReg(Reg
, RegState::Define
)
121 llvm_unreachable("Unhandled opcode.");
124 // Erase old instruction.
128 ARCRegisterInfo::ARCRegisterInfo() : ARCGenRegisterInfo(ARC::BLINK
) {}
130 bool ARCRegisterInfo::needsFrameMoves(const MachineFunction
&MF
) {
131 return MF
.needsFrameMoves();
135 ARCRegisterInfo::getCalleeSavedRegs(const MachineFunction
*MF
) const {
136 return CSR_ARC_SaveList
;
139 BitVector
ARCRegisterInfo::getReservedRegs(const MachineFunction
&MF
) const {
140 BitVector
Reserved(getNumRegs());
142 Reserved
.set(ARC::ILINK
);
143 Reserved
.set(ARC::SP
);
144 Reserved
.set(ARC::GP
);
145 Reserved
.set(ARC::R25
);
146 Reserved
.set(ARC::BLINK
);
147 Reserved
.set(ARC::FP
);
151 bool ARCRegisterInfo::requiresRegisterScavenging(
152 const MachineFunction
&MF
) const {
156 bool ARCRegisterInfo::useFPForScavengingIndex(const MachineFunction
&MF
) const {
160 void ARCRegisterInfo::eliminateFrameIndex(MachineBasicBlock::iterator II
,
161 int SPAdj
, unsigned FIOperandNum
,
162 RegScavenger
*RS
) const {
163 assert(SPAdj
== 0 && "Unexpected");
164 MachineInstr
&MI
= *II
;
165 MachineOperand
&FrameOp
= MI
.getOperand(FIOperandNum
);
166 int FrameIndex
= FrameOp
.getIndex();
168 MachineFunction
&MF
= *MI
.getParent()->getParent();
169 const ARCInstrInfo
&TII
= *MF
.getSubtarget
<ARCSubtarget
>().getInstrInfo();
170 const ARCFrameLowering
*TFI
= getFrameLowering(MF
);
171 int Offset
= MF
.getFrameInfo().getObjectOffset(FrameIndex
);
172 int ObjSize
= MF
.getFrameInfo().getObjectSize(FrameIndex
);
173 int StackSize
= MF
.getFrameInfo().getStackSize();
174 int LocalFrameSize
= MF
.getFrameInfo().getLocalFrameSize();
176 LLVM_DEBUG(dbgs() << "\nFunction : " << MF
.getName() << "\n");
177 LLVM_DEBUG(dbgs() << "<--------->\n");
178 LLVM_DEBUG(dbgs() << MI
<< "\n");
179 LLVM_DEBUG(dbgs() << "FrameIndex : " << FrameIndex
<< "\n");
180 LLVM_DEBUG(dbgs() << "ObjSize : " << ObjSize
<< "\n");
181 LLVM_DEBUG(dbgs() << "FrameOffset : " << Offset
<< "\n");
182 LLVM_DEBUG(dbgs() << "StackSize : " << StackSize
<< "\n");
183 LLVM_DEBUG(dbgs() << "LocalFrameSize : " << LocalFrameSize
<< "\n");
184 (void)LocalFrameSize
;
186 // Special handling of DBG_VALUE instructions.
187 if (MI
.isDebugValue()) {
188 Register FrameReg
= getFrameRegister(MF
);
189 MI
.getOperand(FIOperandNum
).ChangeToRegister(FrameReg
, false /*isDef*/);
190 MI
.getOperand(FIOperandNum
+ 1).ChangeToImmediate(Offset
);
194 // fold constant into offset.
195 Offset
+= MI
.getOperand(FIOperandNum
+ 1).getImm();
197 // TODO: assert based on the load type:
198 // ldb needs no alignment,
199 // ldh needs 2 byte alignment
200 // ld needs 4 byte alignment
201 LLVM_DEBUG(dbgs() << "Offset : " << Offset
<< "\n"
204 Register Reg
= MI
.getOperand(0).getReg();
205 assert(ARC::GPR32RegClass
.contains(Reg
) && "Unexpected register operand");
207 if (!TFI
->hasFP(MF
)) {
208 Offset
= StackSize
+ Offset
;
210 assert((Offset
>= 0 && Offset
< StackSize
) && "SP Offset not in bounds.");
212 if (FrameIndex
>= 0) {
213 assert((Offset
< 0 && -Offset
<= StackSize
) &&
214 "FP Offset not in bounds.");
217 ReplaceFrameIndex(II
, TII
, Reg
, getFrameRegister(MF
), Offset
, StackSize
,
221 Register
ARCRegisterInfo::getFrameRegister(const MachineFunction
&MF
) const {
222 const ARCFrameLowering
*TFI
= getFrameLowering(MF
);
223 return TFI
->hasFP(MF
) ? ARC::FP
: ARC::SP
;
227 ARCRegisterInfo::getCallPreservedMask(const MachineFunction
&MF
,
228 CallingConv::ID CC
) const {
229 return CSR_ARC_RegMask
;