2017-10-20 05:37:38 +08:00
|
|
|
//===-- RISCVFrameLowering.cpp - RISCV Frame Information ------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains the RISCV implementation of TargetFrameLowering class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "RISCVFrameLowering.h"
|
2018-01-11 03:41:03 +08:00
|
|
|
#include "RISCVMachineFunctionInfo.h"
|
2017-10-20 05:37:38 +08:00
|
|
|
#include "RISCVSubtarget.h"
|
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
bool RISCVFrameLowering::hasFP(const MachineFunction &MF) const { return true; }
|
|
|
|
|
2017-12-11 20:34:11 +08:00
|
|
|
// Determines the size of the frame and maximum call frame size.
|
|
|
|
void RISCVFrameLowering::determineFrameLayout(MachineFunction &MF) const {
|
|
|
|
MachineFrameInfo &MFI = MF.getFrameInfo();
|
|
|
|
const RISCVRegisterInfo *RI = STI.getRegisterInfo();
|
|
|
|
|
|
|
|
// Get the number of bytes to allocate from the FrameInfo.
|
|
|
|
uint64_t FrameSize = MFI.getStackSize();
|
|
|
|
|
|
|
|
// Get the alignment.
|
|
|
|
uint64_t StackAlign = RI->needsStackRealignment(MF) ? MFI.getMaxAlignment()
|
|
|
|
: getStackAlignment();
|
|
|
|
|
|
|
|
// Get the maximum call frame size of all the calls.
|
|
|
|
uint64_t MaxCallFrameSize = MFI.getMaxCallFrameSize();
|
|
|
|
|
|
|
|
// If we have dynamic alloca then MaxCallFrameSize needs to be aligned so
|
|
|
|
// that allocations will be aligned.
|
|
|
|
if (MFI.hasVarSizedObjects())
|
|
|
|
MaxCallFrameSize = alignTo(MaxCallFrameSize, StackAlign);
|
|
|
|
|
|
|
|
// Update maximum call frame size.
|
|
|
|
MFI.setMaxCallFrameSize(MaxCallFrameSize);
|
|
|
|
|
|
|
|
// Include call frame size in total.
|
|
|
|
if (!(hasReservedCallFrame(MF) && MFI.adjustsStack()))
|
|
|
|
FrameSize += MaxCallFrameSize;
|
|
|
|
|
|
|
|
// Make sure the frame is aligned.
|
|
|
|
FrameSize = alignTo(FrameSize, StackAlign);
|
|
|
|
|
|
|
|
// Update frame info.
|
|
|
|
MFI.setStackSize(FrameSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RISCVFrameLowering::adjustReg(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator MBBI,
|
|
|
|
const DebugLoc &DL, unsigned DestReg,
|
|
|
|
unsigned SrcReg, int64_t Val,
|
|
|
|
MachineInstr::MIFlag Flag) const {
|
2018-01-11 03:53:46 +08:00
|
|
|
MachineRegisterInfo &MRI = MBB.getParent()->getRegInfo();
|
2017-12-11 20:34:11 +08:00
|
|
|
const RISCVInstrInfo *TII = STI.getInstrInfo();
|
|
|
|
|
|
|
|
if (DestReg == SrcReg && Val == 0)
|
|
|
|
return;
|
|
|
|
|
2018-01-11 03:53:46 +08:00
|
|
|
if (isInt<12>(Val)) {
|
|
|
|
BuildMI(MBB, MBBI, DL, TII->get(RISCV::ADDI), DestReg)
|
|
|
|
.addReg(SrcReg)
|
|
|
|
.addImm(Val)
|
|
|
|
.setMIFlag(Flag);
|
|
|
|
} else if (isInt<32>(Val)) {
|
|
|
|
unsigned Opc = RISCV::ADD;
|
|
|
|
bool isSub = Val < 0;
|
|
|
|
if (isSub) {
|
|
|
|
Val = -Val;
|
|
|
|
Opc = RISCV::SUB;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ScratchReg = MRI.createVirtualRegister(&RISCV::GPRRegClass);
|
|
|
|
TII->movImm32(MBB, MBBI, DL, ScratchReg, Val, Flag);
|
|
|
|
BuildMI(MBB, MBBI, DL, TII->get(Opc), DestReg)
|
|
|
|
.addReg(SrcReg)
|
|
|
|
.addReg(ScratchReg, RegState::Kill)
|
|
|
|
.setMIFlag(Flag);
|
|
|
|
} else {
|
|
|
|
report_fatal_error("adjustReg cannot yet handle adjustments >32 bits");
|
|
|
|
}
|
2017-12-11 20:34:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the register used to hold the frame pointer.
|
|
|
|
static unsigned getFPReg(const RISCVSubtarget &STI) { return RISCV::X8; }
|
|
|
|
|
|
|
|
// Returns the register used to hold the stack pointer.
|
|
|
|
static unsigned getSPReg(const RISCVSubtarget &STI) { return RISCV::X2; }
|
|
|
|
|
2017-10-20 05:37:38 +08:00
|
|
|
void RISCVFrameLowering::emitPrologue(MachineFunction &MF,
|
2017-12-11 20:34:11 +08:00
|
|
|
MachineBasicBlock &MBB) const {
|
|
|
|
assert(&MF.front() == &MBB && "Shrink-wrapping not yet supported");
|
|
|
|
|
|
|
|
if (!hasFP(MF)) {
|
|
|
|
report_fatal_error(
|
|
|
|
"emitPrologue doesn't support framepointer-less functions");
|
|
|
|
}
|
|
|
|
|
|
|
|
MachineFrameInfo &MFI = MF.getFrameInfo();
|
2018-01-11 03:41:03 +08:00
|
|
|
auto *RVFI = MF.getInfo<RISCVMachineFunctionInfo>();
|
2017-12-11 20:34:11 +08:00
|
|
|
MachineBasicBlock::iterator MBBI = MBB.begin();
|
|
|
|
|
|
|
|
unsigned FPReg = getFPReg(STI);
|
|
|
|
unsigned SPReg = getSPReg(STI);
|
|
|
|
|
|
|
|
// Debug location must be unknown since the first debug location is used
|
|
|
|
// to determine the end of the prologue.
|
|
|
|
DebugLoc DL;
|
|
|
|
|
|
|
|
// Determine the correct frame layout
|
|
|
|
determineFrameLayout(MF);
|
|
|
|
|
|
|
|
// FIXME (note copied from Lanai): This appears to be overallocating. Needs
|
|
|
|
// investigation. Get the number of bytes to allocate from the FrameInfo.
|
|
|
|
uint64_t StackSize = MFI.getStackSize();
|
|
|
|
|
|
|
|
// Early exit if there is no need to allocate on the stack
|
|
|
|
if (StackSize == 0 && !MFI.adjustsStack())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Allocate space on the stack if necessary.
|
|
|
|
adjustReg(MBB, MBBI, DL, SPReg, SPReg, -StackSize, MachineInstr::FrameSetup);
|
|
|
|
|
|
|
|
// The frame pointer is callee-saved, and code has been generated for us to
|
|
|
|
// save it to the stack. We need to skip over the storing of callee-saved
|
|
|
|
// registers as the frame pointer must be modified after it has been saved
|
|
|
|
// to the stack, not before.
|
|
|
|
// FIXME: assumes exactly one instruction is used to save each callee-saved
|
|
|
|
// register.
|
|
|
|
const std::vector<CalleeSavedInfo> &CSI = MFI.getCalleeSavedInfo();
|
|
|
|
std::advance(MBBI, CSI.size());
|
|
|
|
|
|
|
|
// Generate new FP.
|
2018-01-11 03:41:03 +08:00
|
|
|
adjustReg(MBB, MBBI, DL, FPReg, SPReg, StackSize - RVFI->getVarArgsSaveSize(),
|
|
|
|
MachineInstr::FrameSetup);
|
2017-12-11 20:34:11 +08:00
|
|
|
}
|
2017-10-20 05:37:38 +08:00
|
|
|
|
|
|
|
void RISCVFrameLowering::emitEpilogue(MachineFunction &MF,
|
2017-12-11 20:34:11 +08:00
|
|
|
MachineBasicBlock &MBB) const {
|
|
|
|
if (!hasFP(MF)) {
|
|
|
|
report_fatal_error(
|
|
|
|
"emitEpilogue doesn't support framepointer-less functions");
|
|
|
|
}
|
|
|
|
|
|
|
|
MachineBasicBlock::iterator MBBI = MBB.getLastNonDebugInstr();
|
|
|
|
const RISCVRegisterInfo *RI = STI.getRegisterInfo();
|
|
|
|
MachineFrameInfo &MFI = MF.getFrameInfo();
|
2018-01-11 03:41:03 +08:00
|
|
|
auto *RVFI = MF.getInfo<RISCVMachineFunctionInfo>();
|
2017-12-11 20:34:11 +08:00
|
|
|
DebugLoc DL = MBBI->getDebugLoc();
|
|
|
|
unsigned FPReg = getFPReg(STI);
|
|
|
|
unsigned SPReg = getSPReg(STI);
|
|
|
|
|
|
|
|
// Skip to before the restores of callee-saved registers
|
|
|
|
// FIXME: assumes exactly one instruction is used to restore each
|
|
|
|
// callee-saved register.
|
|
|
|
MachineBasicBlock::iterator LastFrameDestroy = MBBI;
|
|
|
|
std::advance(LastFrameDestroy, -MFI.getCalleeSavedInfo().size());
|
|
|
|
|
|
|
|
uint64_t StackSize = MFI.getStackSize();
|
|
|
|
|
|
|
|
// Restore the stack pointer using the value of the frame pointer. Only
|
|
|
|
// necessary if the stack pointer was modified, meaning the stack size is
|
|
|
|
// unknown.
|
|
|
|
if (RI->needsStackRealignment(MF) || MFI.hasVarSizedObjects()) {
|
2018-01-11 03:41:03 +08:00
|
|
|
adjustReg(MBB, LastFrameDestroy, DL, SPReg, FPReg,
|
|
|
|
-StackSize + RVFI->getVarArgsSaveSize(),
|
2017-12-11 20:34:11 +08:00
|
|
|
MachineInstr::FrameDestroy);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deallocate stack
|
|
|
|
adjustReg(MBB, MBBI, DL, SPReg, SPReg, StackSize, MachineInstr::FrameDestroy);
|
|
|
|
}
|
2017-12-11 19:53:54 +08:00
|
|
|
|
|
|
|
int RISCVFrameLowering::getFrameIndexReference(const MachineFunction &MF,
|
|
|
|
int FI,
|
|
|
|
unsigned &FrameReg) const {
|
|
|
|
const MachineFrameInfo &MFI = MF.getFrameInfo();
|
|
|
|
const TargetRegisterInfo *RI = MF.getSubtarget().getRegisterInfo();
|
2018-01-11 03:41:03 +08:00
|
|
|
const auto *RVFI = MF.getInfo<RISCVMachineFunctionInfo>();
|
2017-12-11 19:53:54 +08:00
|
|
|
|
|
|
|
// Callee-saved registers should be referenced relative to the stack
|
|
|
|
// pointer (positive offset), otherwise use the frame pointer (negative
|
|
|
|
// offset).
|
|
|
|
const std::vector<CalleeSavedInfo> &CSI = MFI.getCalleeSavedInfo();
|
|
|
|
int MinCSFI = 0;
|
|
|
|
int MaxCSFI = -1;
|
|
|
|
|
|
|
|
int Offset = MFI.getObjectOffset(FI) - getOffsetOfLocalArea() +
|
|
|
|
MFI.getOffsetAdjustment();
|
|
|
|
|
|
|
|
if (CSI.size()) {
|
|
|
|
MinCSFI = CSI[0].getFrameIdx();
|
|
|
|
MaxCSFI = CSI[CSI.size() - 1].getFrameIdx();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FI >= MinCSFI && FI <= MaxCSFI) {
|
|
|
|
FrameReg = RISCV::X2;
|
|
|
|
Offset += MF.getFrameInfo().getStackSize();
|
2018-01-11 03:41:03 +08:00
|
|
|
} else {
|
|
|
|
FrameReg = RI->getFrameRegister(MF);
|
|
|
|
assert(hasFP(MF) && "Offset calculation incorrect if !hasFP");
|
|
|
|
Offset += RVFI->getVarArgsSaveSize();
|
2017-12-11 19:53:54 +08:00
|
|
|
}
|
|
|
|
return Offset;
|
|
|
|
}
|
2017-12-11 20:34:11 +08:00
|
|
|
|
|
|
|
void RISCVFrameLowering::determineCalleeSaves(MachineFunction &MF,
|
|
|
|
BitVector &SavedRegs,
|
|
|
|
RegScavenger *RS) const {
|
|
|
|
TargetFrameLowering::determineCalleeSaves(MF, SavedRegs, RS);
|
|
|
|
// TODO: Once frame pointer elimination is implemented, don't
|
|
|
|
// unconditionally spill the frame pointer and return address.
|
|
|
|
SavedRegs.set(RISCV::X1);
|
|
|
|
SavedRegs.set(RISCV::X8);
|
|
|
|
}
|