[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / lib / Target / XCore / XCoreRegisterInfo.cpp
blob6799823f6fcb75882cb178df7cf590c7158a7a53
1 //===-- XCoreRegisterInfo.cpp - XCore Register Information ----------------===//
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 XCore implementation of the MRegisterInfo class.
11 //===----------------------------------------------------------------------===//
13 #include "XCoreRegisterInfo.h"
14 #include "XCore.h"
15 #include "XCoreInstrInfo.h"
16 #include "XCoreMachineFunctionInfo.h"
17 #include "XCoreSubtarget.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/MachineModuleInfo.h"
24 #include "llvm/CodeGen/MachineRegisterInfo.h"
25 #include "llvm/CodeGen/RegisterScavenging.h"
26 #include "llvm/IR/Function.h"
27 #include "llvm/IR/Type.h"
28 #include "llvm/Support/Debug.h"
29 #include "llvm/Support/ErrorHandling.h"
30 #include "llvm/Support/MathExtras.h"
31 #include "llvm/Support/raw_ostream.h"
32 #include "llvm/CodeGen/TargetFrameLowering.h"
33 #include "llvm/Target/TargetMachine.h"
34 #include "llvm/Target/TargetOptions.h"
36 using namespace llvm;
38 #define DEBUG_TYPE "xcore-reg-info"
40 #define GET_REGINFO_TARGET_DESC
41 #include "XCoreGenRegisterInfo.inc"
43 XCoreRegisterInfo::XCoreRegisterInfo()
44 : XCoreGenRegisterInfo(XCore::LR) {
47 // helper functions
48 static inline bool isImmUs(unsigned val) {
49 return val <= 11;
52 static inline bool isImmU6(unsigned val) {
53 return val < (1 << 6);
56 static inline bool isImmU16(unsigned val) {
57 return val < (1 << 16);
61 static void InsertFPImmInst(MachineBasicBlock::iterator II,
62 const XCoreInstrInfo &TII,
63 unsigned Reg, unsigned FrameReg, int Offset ) {
64 MachineInstr &MI = *II;
65 MachineBasicBlock &MBB = *MI.getParent();
66 DebugLoc dl = MI.getDebugLoc();
68 switch (MI.getOpcode()) {
69 case XCore::LDWFI:
70 BuildMI(MBB, II, dl, TII.get(XCore::LDW_2rus), Reg)
71 .addReg(FrameReg)
72 .addImm(Offset)
73 .addMemOperand(*MI.memoperands_begin());
74 break;
75 case XCore::STWFI:
76 BuildMI(MBB, II, dl, TII.get(XCore::STW_2rus))
77 .addReg(Reg, getKillRegState(MI.getOperand(0).isKill()))
78 .addReg(FrameReg)
79 .addImm(Offset)
80 .addMemOperand(*MI.memoperands_begin());
81 break;
82 case XCore::LDAWFI:
83 BuildMI(MBB, II, dl, TII.get(XCore::LDAWF_l2rus), Reg)
84 .addReg(FrameReg)
85 .addImm(Offset);
86 break;
87 default:
88 llvm_unreachable("Unexpected Opcode");
92 static void InsertFPConstInst(MachineBasicBlock::iterator II,
93 const XCoreInstrInfo &TII,
94 unsigned Reg, unsigned FrameReg,
95 int Offset, RegScavenger *RS ) {
96 assert(RS && "requiresRegisterScavenging failed");
97 MachineInstr &MI = *II;
98 MachineBasicBlock &MBB = *MI.getParent();
99 DebugLoc dl = MI.getDebugLoc();
100 unsigned ScratchOffset = RS->scavengeRegister(&XCore::GRRegsRegClass, II, 0);
101 RS->setRegUsed(ScratchOffset);
102 TII.loadImmediate(MBB, II, ScratchOffset, Offset);
104 switch (MI.getOpcode()) {
105 case XCore::LDWFI:
106 BuildMI(MBB, II, dl, TII.get(XCore::LDW_3r), Reg)
107 .addReg(FrameReg)
108 .addReg(ScratchOffset, RegState::Kill)
109 .addMemOperand(*MI.memoperands_begin());
110 break;
111 case XCore::STWFI:
112 BuildMI(MBB, II, dl, TII.get(XCore::STW_l3r))
113 .addReg(Reg, getKillRegState(MI.getOperand(0).isKill()))
114 .addReg(FrameReg)
115 .addReg(ScratchOffset, RegState::Kill)
116 .addMemOperand(*MI.memoperands_begin());
117 break;
118 case XCore::LDAWFI:
119 BuildMI(MBB, II, dl, TII.get(XCore::LDAWF_l3r), Reg)
120 .addReg(FrameReg)
121 .addReg(ScratchOffset, RegState::Kill);
122 break;
123 default:
124 llvm_unreachable("Unexpected Opcode");
128 static void InsertSPImmInst(MachineBasicBlock::iterator II,
129 const XCoreInstrInfo &TII,
130 unsigned Reg, int Offset) {
131 MachineInstr &MI = *II;
132 MachineBasicBlock &MBB = *MI.getParent();
133 DebugLoc dl = MI.getDebugLoc();
134 bool isU6 = isImmU6(Offset);
136 switch (MI.getOpcode()) {
137 int NewOpcode;
138 case XCore::LDWFI:
139 NewOpcode = (isU6) ? XCore::LDWSP_ru6 : XCore::LDWSP_lru6;
140 BuildMI(MBB, II, dl, TII.get(NewOpcode), Reg)
141 .addImm(Offset)
142 .addMemOperand(*MI.memoperands_begin());
143 break;
144 case XCore::STWFI:
145 NewOpcode = (isU6) ? XCore::STWSP_ru6 : XCore::STWSP_lru6;
146 BuildMI(MBB, II, dl, TII.get(NewOpcode))
147 .addReg(Reg, getKillRegState(MI.getOperand(0).isKill()))
148 .addImm(Offset)
149 .addMemOperand(*MI.memoperands_begin());
150 break;
151 case XCore::LDAWFI:
152 NewOpcode = (isU6) ? XCore::LDAWSP_ru6 : XCore::LDAWSP_lru6;
153 BuildMI(MBB, II, dl, TII.get(NewOpcode), Reg)
154 .addImm(Offset);
155 break;
156 default:
157 llvm_unreachable("Unexpected Opcode");
161 static void InsertSPConstInst(MachineBasicBlock::iterator II,
162 const XCoreInstrInfo &TII,
163 unsigned Reg, int Offset, RegScavenger *RS ) {
164 assert(RS && "requiresRegisterScavenging failed");
165 MachineInstr &MI = *II;
166 MachineBasicBlock &MBB = *MI.getParent();
167 DebugLoc dl = MI.getDebugLoc();
168 unsigned OpCode = MI.getOpcode();
170 unsigned ScratchBase;
171 if (OpCode==XCore::STWFI) {
172 ScratchBase = RS->scavengeRegister(&XCore::GRRegsRegClass, II, 0);
173 RS->setRegUsed(ScratchBase);
174 } else
175 ScratchBase = Reg;
176 BuildMI(MBB, II, dl, TII.get(XCore::LDAWSP_ru6), ScratchBase).addImm(0);
177 unsigned ScratchOffset = RS->scavengeRegister(&XCore::GRRegsRegClass, II, 0);
178 RS->setRegUsed(ScratchOffset);
179 TII.loadImmediate(MBB, II, ScratchOffset, Offset);
181 switch (OpCode) {
182 case XCore::LDWFI:
183 BuildMI(MBB, II, dl, TII.get(XCore::LDW_3r), Reg)
184 .addReg(ScratchBase, RegState::Kill)
185 .addReg(ScratchOffset, RegState::Kill)
186 .addMemOperand(*MI.memoperands_begin());
187 break;
188 case XCore::STWFI:
189 BuildMI(MBB, II, dl, TII.get(XCore::STW_l3r))
190 .addReg(Reg, getKillRegState(MI.getOperand(0).isKill()))
191 .addReg(ScratchBase, RegState::Kill)
192 .addReg(ScratchOffset, RegState::Kill)
193 .addMemOperand(*MI.memoperands_begin());
194 break;
195 case XCore::LDAWFI:
196 BuildMI(MBB, II, dl, TII.get(XCore::LDAWF_l3r), Reg)
197 .addReg(ScratchBase, RegState::Kill)
198 .addReg(ScratchOffset, RegState::Kill);
199 break;
200 default:
201 llvm_unreachable("Unexpected Opcode");
205 bool XCoreRegisterInfo::needsFrameMoves(const MachineFunction &MF) {
206 return MF.needsFrameMoves();
209 const MCPhysReg *
210 XCoreRegisterInfo::getCalleeSavedRegs(const MachineFunction *MF) const {
211 // The callee saved registers LR & FP are explicitly handled during
212 // emitPrologue & emitEpilogue and related functions.
213 static const MCPhysReg CalleeSavedRegs[] = {
214 XCore::R4, XCore::R5, XCore::R6, XCore::R7,
215 XCore::R8, XCore::R9, XCore::R10,
218 static const MCPhysReg CalleeSavedRegsFP[] = {
219 XCore::R4, XCore::R5, XCore::R6, XCore::R7,
220 XCore::R8, XCore::R9,
223 const XCoreFrameLowering *TFI = getFrameLowering(*MF);
224 if (TFI->hasFP(*MF))
225 return CalleeSavedRegsFP;
226 return CalleeSavedRegs;
229 BitVector XCoreRegisterInfo::getReservedRegs(const MachineFunction &MF) const {
230 BitVector Reserved(getNumRegs());
231 const XCoreFrameLowering *TFI = getFrameLowering(MF);
233 Reserved.set(XCore::CP);
234 Reserved.set(XCore::DP);
235 Reserved.set(XCore::SP);
236 Reserved.set(XCore::LR);
237 if (TFI->hasFP(MF)) {
238 Reserved.set(XCore::R10);
240 return Reserved;
243 bool
244 XCoreRegisterInfo::requiresRegisterScavenging(const MachineFunction &MF) const {
245 return true;
248 bool
249 XCoreRegisterInfo::useFPForScavengingIndex(const MachineFunction &MF) const {
250 return false;
253 void
254 XCoreRegisterInfo::eliminateFrameIndex(MachineBasicBlock::iterator II,
255 int SPAdj, unsigned FIOperandNum,
256 RegScavenger *RS) const {
257 assert(SPAdj == 0 && "Unexpected");
258 MachineInstr &MI = *II;
259 MachineOperand &FrameOp = MI.getOperand(FIOperandNum);
260 int FrameIndex = FrameOp.getIndex();
262 MachineFunction &MF = *MI.getParent()->getParent();
263 const XCoreInstrInfo &TII =
264 *static_cast<const XCoreInstrInfo *>(MF.getSubtarget().getInstrInfo());
266 const XCoreFrameLowering *TFI = getFrameLowering(MF);
267 int Offset = MF.getFrameInfo().getObjectOffset(FrameIndex);
268 int StackSize = MF.getFrameInfo().getStackSize();
270 #ifndef NDEBUG
271 LLVM_DEBUG(errs() << "\nFunction : " << MF.getName() << "\n");
272 LLVM_DEBUG(errs() << "<--------->\n");
273 LLVM_DEBUG(MI.print(errs()));
274 LLVM_DEBUG(errs() << "FrameIndex : " << FrameIndex << "\n");
275 LLVM_DEBUG(errs() << "FrameOffset : " << Offset << "\n");
276 LLVM_DEBUG(errs() << "StackSize : " << StackSize << "\n");
277 #endif
279 Offset += StackSize;
281 Register FrameReg = getFrameRegister(MF);
283 // Special handling of DBG_VALUE instructions.
284 if (MI.isDebugValue()) {
285 MI.getOperand(FIOperandNum).ChangeToRegister(FrameReg, false /*isDef*/);
286 MI.getOperand(FIOperandNum + 1).ChangeToImmediate(Offset);
287 return;
290 // fold constant into offset.
291 Offset += MI.getOperand(FIOperandNum + 1).getImm();
292 MI.getOperand(FIOperandNum + 1).ChangeToImmediate(0);
294 assert(Offset%4 == 0 && "Misaligned stack offset");
295 LLVM_DEBUG(errs() << "Offset : " << Offset << "\n"
296 << "<--------->\n");
297 Offset/=4;
299 Register Reg = MI.getOperand(0).getReg();
300 assert(XCore::GRRegsRegClass.contains(Reg) && "Unexpected register operand");
302 if (TFI->hasFP(MF)) {
303 if (isImmUs(Offset))
304 InsertFPImmInst(II, TII, Reg, FrameReg, Offset);
305 else
306 InsertFPConstInst(II, TII, Reg, FrameReg, Offset, RS);
307 } else {
308 if (isImmU16(Offset))
309 InsertSPImmInst(II, TII, Reg, Offset);
310 else
311 InsertSPConstInst(II, TII, Reg, Offset, RS);
313 // Erase old instruction.
314 MachineBasicBlock &MBB = *MI.getParent();
315 MBB.erase(II);
319 Register XCoreRegisterInfo::getFrameRegister(const MachineFunction &MF) const {
320 const XCoreFrameLowering *TFI = getFrameLowering(MF);
322 return TFI->hasFP(MF) ? XCore::R10 : XCore::SP;