LLVM 20.0.0git
Mips16FrameLowering.cpp
Go to the documentation of this file.
1//===- Mips16FrameLowering.cpp - Mips16 Frame 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 Mips16 implementation of TargetFrameLowering class.
10//
11//===----------------------------------------------------------------------===//
12
13#include "Mips16FrameLowering.h"
14#include "Mips16InstrInfo.h"
15#include "MipsInstrInfo.h"
16#include "MipsRegisterInfo.h"
17#include "MipsSubtarget.h"
18#include "llvm/ADT/BitVector.h"
26#include "llvm/IR/DebugLoc.h"
27#include "llvm/MC/MCContext.h"
28#include "llvm/MC/MCDwarf.h"
31#include <cstdint>
32#include <vector>
33
34using namespace llvm;
35
37 : MipsFrameLowering(STI, STI.getStackAlignment()) {}
38
40 MachineBasicBlock &MBB) const {
42 const Mips16InstrInfo &TII =
43 *static_cast<const Mips16InstrInfo *>(STI.getInstrInfo());
45
46 // Debug location must be unknown since the first debug location is used
47 // to determine the end of the prologue.
48 DebugLoc dl;
49
50 uint64_t StackSize = MFI.getStackSize();
51
52 // No need to allocate space on the stack.
53 if (StackSize == 0 && !MFI.adjustsStack()) return;
54
56
57 // Adjust stack.
58 TII.makeFrame(Mips::SP, StackSize, MBB, MBBI);
59
60 // emit ".cfi_def_cfa_offset StackSize"
61 unsigned CFIIndex =
62 MF.addFrameInst(MCCFIInstruction::cfiDefCfaOffset(nullptr, StackSize));
63 BuildMI(MBB, MBBI, dl, TII.get(TargetOpcode::CFI_INSTRUCTION))
64 .addCFIIndex(CFIIndex);
65
66 const std::vector<CalleeSavedInfo> &CSI = MFI.getCalleeSavedInfo();
67
68 if (!CSI.empty()) {
69 const std::vector<CalleeSavedInfo> &CSI = MFI.getCalleeSavedInfo();
70
71 for (const CalleeSavedInfo &I : CSI) {
72 int64_t Offset = MFI.getObjectOffset(I.getFrameIdx());
73 Register Reg = I.getReg();
74 unsigned DReg = MRI->getDwarfRegNum(Reg, true);
75 unsigned CFIIndex = MF.addFrameInst(
77 BuildMI(MBB, MBBI, dl, TII.get(TargetOpcode::CFI_INSTRUCTION))
78 .addCFIIndex(CFIIndex);
79 }
80 }
81 if (hasFP(MF))
82 BuildMI(MBB, MBBI, dl, TII.get(Mips::MoveR3216), Mips::S0)
84}
85
87 MachineBasicBlock &MBB) const {
90 const Mips16InstrInfo &TII =
91 *static_cast<const Mips16InstrInfo *>(STI.getInstrInfo());
92 DebugLoc dl = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
93 uint64_t StackSize = MFI.getStackSize();
94
95 if (!StackSize)
96 return;
97
98 if (hasFP(MF))
99 BuildMI(MBB, MBBI, dl, TII.get(Mips::Move32R16), Mips::SP)
100 .addReg(Mips::S0);
101
102 // Adjust stack.
103 // assumes stacksize multiple of 8
104 TII.restoreFrame(Mips::SP, StackSize, MBB, MBBI);
105}
106
111
112 //
113 // Registers RA, S0,S1 are the callee saved registers and they
114 // will be saved with the "save" instruction
115 // during emitPrologue
116 //
117 for (const CalleeSavedInfo &I : CSI) {
118 // Add the callee-saved register as live-in. Do not add if the register is
119 // RA and return address is taken, because it has already been added in
120 // method MipsTargetLowering::lowerRETURNADDR.
121 // It's killed at the spill, unless the register is RA and return address
122 // is taken.
123 Register Reg = I.getReg();
124 bool IsRAAndRetAddrIsTaken = (Reg == Mips::RA)
126 if (!IsRAAndRetAddrIsTaken)
127 MBB.addLiveIn(Reg);
128 }
129
130 return true;
131}
132
136 //
137 // Registers RA,S0,S1 are the callee saved registers and they will be restored
138 // with the restore instruction during emitEpilogue.
139 // We need to override this virtual function, otherwise llvm will try and
140 // restore the registers on it's on from the stack.
141 //
142
143 return true;
144}
145
146bool
148 const MachineFrameInfo &MFI = MF.getFrameInfo();
149 // Reserve call frame if the size of the maximum call frame fits into 15-bit
150 // immediate field and there are no variable sized objects on the stack.
151 return isInt<15>(MFI.getMaxCallFrameSize()) && !MFI.hasVarSizedObjects();
152}
153
155 BitVector &SavedRegs,
156 RegScavenger *RS) const {
158 const Mips16InstrInfo &TII =
159 *static_cast<const Mips16InstrInfo *>(STI.getInstrInfo());
160 const MipsRegisterInfo &RI = TII.getRegisterInfo();
161 const BitVector Reserved = RI.getReservedRegs(MF);
162 bool SaveS2 = Reserved[Mips::S2];
163 if (SaveS2)
164 SavedRegs.set(Mips::S2);
165 if (hasFP(MF))
166 SavedRegs.set(Mips::S0);
167}
168
169const MipsFrameLowering *
171 return new Mips16FrameLowering(ST);
172}
unsigned const MachineRegisterInfo * MRI
MachineBasicBlock & MBB
MachineBasicBlock MachineBasicBlock::iterator MBBI
This file implements the BitVector class.
const HexagonInstrInfo * TII
IRTranslator LLVM IR MI
#define I(x, y, z)
Definition: MD5.cpp:58
unsigned const TargetRegisterInfo * TRI
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition: ArrayRef.h:41
BitVector & set()
Definition: BitVector.h:351
The CalleeSavedInfo class tracks the information need to locate where a callee saved register is in t...
A debug info location.
Definition: DebugLoc.h:33
static MCCFIInstruction createOffset(MCSymbol *L, unsigned Register, int64_t Offset, SMLoc Loc={})
.cfi_offset Previous value of Register is saved at offset Offset from CFA.
Definition: MCDwarf.h:617
static MCCFIInstruction cfiDefCfaOffset(MCSymbol *L, int64_t Offset, SMLoc Loc={})
.cfi_def_cfa_offset modifies a rule for computing CFA.
Definition: MCDwarf.h:590
const MCRegisterInfo * getRegisterInfo() const
Definition: MCContext.h:414
MCRegisterInfo base class - We assume that the target defines a static array of MCRegisterDesc object...
iterator getFirstTerminator()
Returns an iterator to the first terminator instruction of this basic block.
void addLiveIn(MCRegister PhysReg, LaneBitmask LaneMask=LaneBitmask::getAll())
Adds the specified register as a live in.
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
bool hasVarSizedObjects() const
This method may be called any time after instruction selection is complete to determine if the stack ...
uint64_t getStackSize() const
Return the number of bytes that must be allocated to hold all of the fixed size frame objects.
bool adjustsStack() const
Return true if this function adjusts the stack – e.g., when calling another function.
bool isReturnAddressTaken() const
This method may be called any time after instruction selection is complete to determine if there is a...
uint64_t getMaxCallFrameSize() const
Return the maximum size of a call frame that must be allocated for an outgoing function call.
const std::vector< CalleeSavedInfo > & getCalleeSavedInfo() const
Returns a reference to call saved info vector for the current function.
int64_t getObjectOffset(int ObjectIdx) const
Return the assigned stack offset of the specified object from the incoming stack pointer.
unsigned addFrameInst(const MCCFIInstruction &Inst)
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MCContext & getContext() const
const MachineInstrBuilder & addCFIIndex(unsigned CFIIndex) const
const MachineInstrBuilder & setMIFlag(MachineInstr::MIFlag Flag) const
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, RegScavenger *RS) const override
This method determines which of the registers reported by TargetRegisterInfo::getCalleeSavedRegs() sh...
void emitEpilogue(MachineFunction &MF, MachineBasicBlock &MBB) const override
bool spillCalleeSavedRegisters(MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, ArrayRef< CalleeSavedInfo > CSI, const TargetRegisterInfo *TRI) const override
spillCalleeSavedRegisters - Issues instruction(s) to spill all callee saved registers and returns tru...
bool hasReservedCallFrame(const MachineFunction &MF) const override
hasReservedCallFrame - Under normal circumstances, when a frame pointer is not required,...
Mips16FrameLowering(const MipsSubtarget &STI)
bool restoreCalleeSavedRegisters(MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, MutableArrayRef< CalleeSavedInfo > CSI, const TargetRegisterInfo *TRI) const override
restoreCalleeSavedRegisters - Issues instruction(s) to restore all callee saved registers and returns...
void emitPrologue(MachineFunction &MF, MachineBasicBlock &MBB) const override
emitProlog/emitEpilog - These methods insert prolog and epilog code into the function.
const MipsSubtarget & STI
BitVector getReservedRegs(const MachineFunction &MF) const override
const MipsInstrInfo * getInstrInfo() const override
MutableArrayRef - Represent a mutable reference to an array (0 or more elements consecutively in memo...
Definition: ArrayRef.h:310
Wrapper class representing virtual and physical registers.
Definition: Register.h:19
bool hasFP(const MachineFunction &MF) const
hasFP - Return true if the specified function should have a dedicated frame pointer register.
virtual void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, RegScavenger *RS=nullptr) const
This method determines which of the registers reported by TargetRegisterInfo::getCalleeSavedRegs() sh...
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
@ Offset
Definition: DWP.cpp:480
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
const MipsFrameLowering * createMips16FrameLowering(const MipsSubtarget &ST)
Create MipsFrameLowering objects.