1 //===-- LanaiRegisterInfo.cpp - Lanai Register Information ------*- C++ -*-===//
2 //
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
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file contains the Lanai implementation of the TargetRegisterInfo class.
10 //
11 //===----------------------------------------------------------------------===//
12
13 #include "LanaiRegisterInfo.h"
14 #include "LanaiAluCode.h"
15 #include "LanaiCondCode.h"
16 #include "LanaiFrameLowering.h"
17 #include "LanaiInstrInfo.h"
18 #include "llvm/ADT/BitVector.h"
19 #include "llvm/ADT/STLExtras.h"
20 #include "llvm/CodeGen/MachineFrameInfo.h"
21 #include "llvm/CodeGen/MachineFunction.h"
22 #include "llvm/CodeGen/MachineInstrBuilder.h"
23 #include "llvm/CodeGen/RegisterScavenging.h"
24 #include "llvm/CodeGen/TargetFrameLowering.h"
25 #include "llvm/CodeGen/TargetInstrInfo.h"
26 #include "llvm/IR/Function.h"
27 #include "llvm/IR/Type.h"
28 #include "llvm/Support/ErrorHandling.h"
29
30 #define GET_REGINFO_TARGET_DESC
31 #include "LanaiGenRegisterInfo.inc"
32
33 using namespace llvm;
34
LanaiRegisterInfo()35 LanaiRegisterInfo::LanaiRegisterInfo() : LanaiGenRegisterInfo(Lanai::RCA) {}
36
37 const uint16_t *
getCalleeSavedRegs(const MachineFunction *) const38 LanaiRegisterInfo::getCalleeSavedRegs(const MachineFunction * /*MF*/) const {
39 return CSR_SaveList;
40 }
41
getReservedRegs(const MachineFunction & MF) const42 BitVector LanaiRegisterInfo::getReservedRegs(const MachineFunction &MF) const {
43 BitVector Reserved(getNumRegs());
44
45 Reserved.set(Lanai::R0);
46 Reserved.set(Lanai::R1);
47 Reserved.set(Lanai::PC);
48 Reserved.set(Lanai::R2);
49 Reserved.set(Lanai::SP);
50 Reserved.set(Lanai::R4);
51 Reserved.set(Lanai::FP);
52 Reserved.set(Lanai::R5);
53 Reserved.set(Lanai::RR1);
54 Reserved.set(Lanai::R10);
55 Reserved.set(Lanai::RR2);
56 Reserved.set(Lanai::R11);
57 Reserved.set(Lanai::RCA);
58 Reserved.set(Lanai::R15);
59 if (hasBasePointer(MF))
60 Reserved.set(getBaseRegister());
61 return Reserved;
62 }
63
requiresRegisterScavenging(const MachineFunction &) const64 bool LanaiRegisterInfo::requiresRegisterScavenging(
65 const MachineFunction & /*MF*/) const {
66 return true;
67 }
68
isALUArithLoOpcode(unsigned Opcode)69 static bool isALUArithLoOpcode(unsigned Opcode) {
70 switch (Opcode) {
71 case Lanai::ADD_I_LO:
72 case Lanai::SUB_I_LO:
73 case Lanai::ADD_F_I_LO:
74 case Lanai::SUB_F_I_LO:
75 case Lanai::ADDC_I_LO:
76 case Lanai::SUBB_I_LO:
77 case Lanai::ADDC_F_I_LO:
78 case Lanai::SUBB_F_I_LO:
79 return true;
80 default:
81 return false;
82 }
83 }
84
getOppositeALULoOpcode(unsigned Opcode)85 static unsigned getOppositeALULoOpcode(unsigned Opcode) {
86 switch (Opcode) {
87 case Lanai::ADD_I_LO:
88 return Lanai::SUB_I_LO;
89 case Lanai::SUB_I_LO:
90 return Lanai::ADD_I_LO;
91 case Lanai::ADD_F_I_LO:
92 return Lanai::SUB_F_I_LO;
93 case Lanai::SUB_F_I_LO:
94 return Lanai::ADD_F_I_LO;
95 case Lanai::ADDC_I_LO:
96 return Lanai::SUBB_I_LO;
97 case Lanai::SUBB_I_LO:
98 return Lanai::ADDC_I_LO;
99 case Lanai::ADDC_F_I_LO:
100 return Lanai::SUBB_F_I_LO;
101 case Lanai::SUBB_F_I_LO:
102 return Lanai::ADDC_F_I_LO;
103 default:
104 llvm_unreachable("Invalid ALU lo opcode");
105 }
106 }
107
getRRMOpcodeVariant(unsigned Opcode)108 static unsigned getRRMOpcodeVariant(unsigned Opcode) {
109 switch (Opcode) {
110 case Lanai::LDBs_RI:
111 return Lanai::LDBs_RR;
112 case Lanai::LDBz_RI:
113 return Lanai::LDBz_RR;
114 case Lanai::LDHs_RI:
115 return Lanai::LDHs_RR;
116 case Lanai::LDHz_RI:
117 return Lanai::LDHz_RR;
118 case Lanai::LDW_RI:
119 return Lanai::LDW_RR;
120 case Lanai::STB_RI:
121 return Lanai::STB_RR;
122 case Lanai::STH_RI:
123 return Lanai::STH_RR;
124 case Lanai::SW_RI:
125 return Lanai::SW_RR;
126 default:
127 llvm_unreachable("Opcode has no RRM variant");
128 }
129 }
130
eliminateFrameIndex(MachineBasicBlock::iterator II,int SPAdj,unsigned FIOperandNum,RegScavenger * RS) const131 void LanaiRegisterInfo::eliminateFrameIndex(MachineBasicBlock::iterator II,
132 int SPAdj, unsigned FIOperandNum,
133 RegScavenger *RS) const {
134 assert(SPAdj == 0 && "Unexpected");
135
136 MachineInstr &MI = *II;
137 MachineFunction &MF = *MI.getParent()->getParent();
138 const TargetInstrInfo *TII = MF.getSubtarget().getInstrInfo();
139 const TargetFrameLowering *TFI = MF.getSubtarget().getFrameLowering();
140 bool HasFP = TFI->hasFP(MF);
141 DebugLoc DL = MI.getDebugLoc();
142
143 int FrameIndex = MI.getOperand(FIOperandNum).getIndex();
144
145 int Offset = MF.getFrameInfo().getObjectOffset(FrameIndex) +
146 MI.getOperand(FIOperandNum + 1).getImm();
147
148 // Addressable stack objects are addressed using neg. offsets from fp
149 // or pos. offsets from sp/basepointer
150 if (!HasFP || (needsStackRealignment(MF) && FrameIndex >= 0))
151 Offset += MF.getFrameInfo().getStackSize();
152
153 Register FrameReg = getFrameRegister(MF);
154 if (FrameIndex >= 0) {
155 if (hasBasePointer(MF))
156 FrameReg = getBaseRegister();
157 else if (needsStackRealignment(MF))
158 FrameReg = Lanai::SP;
159 }
160
161 // Replace frame index with a frame pointer reference.
162 // If the offset is small enough to fit in the immediate field, directly
163 // encode it.
164 // Otherwise scavenge a register and encode it into a MOVHI, OR_I_LO sequence.
165 if ((isSPLSOpcode(MI.getOpcode()) && !isInt<10>(Offset)) ||
166 !isInt<16>(Offset)) {
167 assert(RS && "Register scavenging must be on");
168 unsigned Reg = RS->FindUnusedReg(&Lanai::GPRRegClass);
169 if (!Reg)
170 Reg = RS->scavengeRegister(&Lanai::GPRRegClass, II, SPAdj);
171 assert(Reg && "Register scavenger failed");
172
173 bool HasNegOffset = false;
174 // ALU ops have unsigned immediate values. If the Offset is negative, we
175 // negate it here and reverse the opcode later.
176 if (Offset < 0) {
177 HasNegOffset = true;
178 Offset = -Offset;
179 }
180
181 if (!isInt<16>(Offset)) {
182 // Reg = hi(offset) | lo(offset)
183 BuildMI(*MI.getParent(), II, DL, TII->get(Lanai::MOVHI), Reg)
184 .addImm(static_cast<uint32_t>(Offset) >> 16);
185 BuildMI(*MI.getParent(), II, DL, TII->get(Lanai::OR_I_LO), Reg)
186 .addReg(Reg)
187 .addImm(Offset & 0xffffU);
188 } else {
189 // Reg = mov(offset)
190 BuildMI(*MI.getParent(), II, DL, TII->get(Lanai::ADD_I_LO), Reg)
191 .addImm(0)
192 .addImm(Offset);
193 }
194 // Reg = FrameReg OP Reg
195 if (MI.getOpcode() == Lanai::ADD_I_LO) {
196 BuildMI(*MI.getParent(), II, DL,
197 HasNegOffset ? TII->get(Lanai::SUB_R) : TII->get(Lanai::ADD_R),
198 MI.getOperand(0).getReg())
199 .addReg(FrameReg)
200 .addReg(Reg)
201 .addImm(LPCC::ICC_T);
202 MI.eraseFromParent();
203 return;
204 }
205 if (isSPLSOpcode(MI.getOpcode()) || isRMOpcode(MI.getOpcode())) {
206 MI.setDesc(TII->get(getRRMOpcodeVariant(MI.getOpcode())));
207 if (HasNegOffset) {
208 // Change the ALU op (operand 3) from LPAC::ADD (the default) to
209 // LPAC::SUB with the already negated offset.
210 assert((MI.getOperand(3).getImm() == LPAC::ADD) &&
211 "Unexpected ALU op in RRM instruction");
212 MI.getOperand(3).setImm(LPAC::SUB);
213 }
214 } else
215 llvm_unreachable("Unexpected opcode in frame index operation");
216
217 MI.getOperand(FIOperandNum).ChangeToRegister(FrameReg, /*isDef=*/false);
218 MI.getOperand(FIOperandNum + 1)
219 .ChangeToRegister(Reg, /*isDef=*/false, /*isImp=*/false,
220 /*isKill=*/true);
221 return;
222 }
223
224 // ALU arithmetic ops take unsigned immediates. If the offset is negative,
225 // we replace the instruction with one that inverts the opcode and negates
226 // the immediate.
227 if ((Offset < 0) && isALUArithLoOpcode(MI.getOpcode())) {
228 unsigned NewOpcode = getOppositeALULoOpcode(MI.getOpcode());
229 // We know this is an ALU op, so we know the operands are as follows:
230 // 0: destination register
231 // 1: source register (frame register)
232 // 2: immediate
233 BuildMI(*MI.getParent(), II, DL, TII->get(NewOpcode),
234 MI.getOperand(0).getReg())
235 .addReg(FrameReg)
236 .addImm(-Offset);
237 MI.eraseFromParent();
238 } else {
239 MI.getOperand(FIOperandNum).ChangeToRegister(FrameReg, /*isDef=*/false);
240 MI.getOperand(FIOperandNum + 1).ChangeToImmediate(Offset);
241 }
242 }
243
hasBasePointer(const MachineFunction & MF) const244 bool LanaiRegisterInfo::hasBasePointer(const MachineFunction &MF) const {
245 const MachineFrameInfo &MFI = MF.getFrameInfo();
246 // When we need stack realignment and there are dynamic allocas, we can't
247 // reference off of the stack pointer, so we reserve a base pointer.
248 if (needsStackRealignment(MF) && MFI.hasVarSizedObjects())
249 return true;
250
251 return false;
252 }
253
getRARegister() const254 unsigned LanaiRegisterInfo::getRARegister() const { return Lanai::RCA; }
255
256 Register
getFrameRegister(const MachineFunction &) const257 LanaiRegisterInfo::getFrameRegister(const MachineFunction & /*MF*/) const {
258 return Lanai::FP;
259 }
260
getBaseRegister() const261 Register LanaiRegisterInfo::getBaseRegister() const { return Lanai::R14; }
262
263 const uint32_t *
getCallPreservedMask(const MachineFunction &,CallingConv::ID) const264 LanaiRegisterInfo::getCallPreservedMask(const MachineFunction & /*MF*/,
265 CallingConv::ID /*CC*/) const {
266 return CSR_RegMask;
267 }
268