mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-01-01 08:28:19 +00:00
Modify MipsFrameLowering::emitPrologue and emitEpilogue.
- Use MipsAnalyzeImmediate to expand immediates that do not fit in 16-bit. - Change the types of variables so that they are sufficiently large to handle 64-bit pointers. - Emit instructions to set register $28 in a function prologue after instructions which store callee-saved registers have been emitted. llvm-svn: 148917
This commit is contained in:
parent
9a829c1762
commit
175341c860
@ -11,6 +11,7 @@
|
|||||||
//
|
//
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#include "MipsAnalyzeImmediate.h"
|
||||||
#include "MipsFrameLowering.h"
|
#include "MipsFrameLowering.h"
|
||||||
#include "MipsInstrInfo.h"
|
#include "MipsInstrInfo.h"
|
||||||
#include "MipsMachineFunction.h"
|
#include "MipsMachineFunction.h"
|
||||||
@ -93,47 +94,40 @@ bool MipsFrameLowering::targetHandlesStackFrameRounding() const {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned AlignOffset(unsigned Offset, unsigned Align) {
|
// Build an instruction sequence to load an immediate that is too large to fit
|
||||||
return (Offset + Align - 1) / Align * Align;
|
// in 16-bit and add the result to Reg.
|
||||||
}
|
static void expandLargeImm(unsigned Reg, int64_t Imm, bool IsN64,
|
||||||
|
const MipsInstrInfo &TII, MachineBasicBlock& MBB,
|
||||||
// expand pair of register and immediate if the immediate doesn't fit in the
|
MachineBasicBlock::iterator II, DebugLoc DL) {
|
||||||
// 16-bit offset field.
|
unsigned LUi = IsN64 ? Mips::LUi64 : Mips::LUi;
|
||||||
// e.g.
|
unsigned ADDu = IsN64 ? Mips::DADDu : Mips::ADDu;
|
||||||
// if OrigImm = 0x10000, OrigReg = $sp:
|
unsigned ZEROReg = IsN64 ? Mips::ZERO_64 : Mips::ZERO;
|
||||||
// generate the following sequence of instrs:
|
unsigned ATReg = IsN64 ? Mips::AT_64 : Mips::AT;
|
||||||
// lui $at, hi(0x10000)
|
MipsAnalyzeImmediate AnalyzeImm;
|
||||||
// addu $at, $sp, $at
|
const MipsAnalyzeImmediate::InstSeq &Seq =
|
||||||
//
|
AnalyzeImm.Analyze(Imm, IsN64 ? 64 : 32, false /* LastInstrIsADDiu */);
|
||||||
// (NewReg, NewImm) = ($at, lo(Ox10000))
|
MipsAnalyzeImmediate::InstSeq::const_iterator Inst = Seq.begin();
|
||||||
// return true
|
|
||||||
static bool expandRegLargeImmPair(unsigned OrigReg, int OrigImm,
|
|
||||||
unsigned& NewReg, int& NewImm,
|
|
||||||
MachineBasicBlock& MBB,
|
|
||||||
MachineBasicBlock::iterator I) {
|
|
||||||
// OrigImm fits in the 16-bit field
|
|
||||||
if (OrigImm < 0x8000 && OrigImm >= -0x8000) {
|
|
||||||
NewReg = OrigReg;
|
|
||||||
NewImm = OrigImm;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
MachineFunction* MF = MBB.getParent();
|
|
||||||
const TargetInstrInfo *TII = MF->getTarget().getInstrInfo();
|
|
||||||
DebugLoc DL = I->getDebugLoc();
|
|
||||||
int ImmLo = (short)(OrigImm & 0xffff);
|
|
||||||
int ImmHi = (((unsigned)OrigImm & 0xffff0000) >> 16) +
|
|
||||||
((OrigImm & 0x8000) != 0);
|
|
||||||
|
|
||||||
// FIXME: change this when mips goes MC".
|
// FIXME: change this when mips goes MC".
|
||||||
BuildMI(MBB, I, DL, TII->get(Mips::NOAT));
|
BuildMI(MBB, II, DL, TII.get(Mips::NOAT));
|
||||||
BuildMI(MBB, I, DL, TII->get(Mips::LUi), Mips::AT).addImm(ImmHi);
|
|
||||||
BuildMI(MBB, I, DL, TII->get(Mips::ADDu), Mips::AT).addReg(OrigReg)
|
|
||||||
.addReg(Mips::AT);
|
|
||||||
NewReg = Mips::AT;
|
|
||||||
NewImm = ImmLo;
|
|
||||||
|
|
||||||
return true;
|
// The first instruction can be a LUi, which is different from other
|
||||||
|
// instructions (ADDiu, ORI and SLL) in that it does not have a register
|
||||||
|
// operand.
|
||||||
|
if (Inst->Opc == LUi)
|
||||||
|
BuildMI(MBB, II, DL, TII.get(LUi), ATReg)
|
||||||
|
.addImm(SignExtend64<16>(Inst->ImmOpnd));
|
||||||
|
else
|
||||||
|
BuildMI(MBB, II, DL, TII.get(Inst->Opc), ATReg).addReg(ZEROReg)
|
||||||
|
.addImm(SignExtend64<16>(Inst->ImmOpnd));
|
||||||
|
|
||||||
|
// Build the remaining instructions in Seq.
|
||||||
|
for (++Inst; Inst != Seq.end(); ++Inst)
|
||||||
|
BuildMI(MBB, II, DL, TII.get(Inst->Opc), ATReg).addReg(ATReg)
|
||||||
|
.addImm(SignExtend64<16>(Inst->ImmOpnd));
|
||||||
|
|
||||||
|
BuildMI(MBB, II, DL, TII.get(ADDu), Reg).addReg(Reg).addReg(ATReg);
|
||||||
|
BuildMI(MBB, II, DL, TII.get(Mips::ATMACRO));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
||||||
@ -142,14 +136,12 @@ void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
|||||||
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
||||||
const MipsRegisterInfo *RegInfo =
|
const MipsRegisterInfo *RegInfo =
|
||||||
static_cast<const MipsRegisterInfo*>(MF.getTarget().getRegisterInfo());
|
static_cast<const MipsRegisterInfo*>(MF.getTarget().getRegisterInfo());
|
||||||
|
MachineRegisterInfo& MRI = MF.getRegInfo();
|
||||||
const MipsInstrInfo &TII =
|
const MipsInstrInfo &TII =
|
||||||
*static_cast<const MipsInstrInfo*>(MF.getTarget().getInstrInfo());
|
*static_cast<const MipsInstrInfo*>(MF.getTarget().getInstrInfo());
|
||||||
MachineBasicBlock::iterator MBBI = MBB.begin();
|
MachineBasicBlock::iterator MBBI = MBB.begin();
|
||||||
DebugLoc dl = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
|
DebugLoc dl = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
|
||||||
bool isPIC = (MF.getTarget().getRelocationModel() == Reloc::PIC_);
|
bool isPIC = (MF.getTarget().getRelocationModel() == Reloc::PIC_);
|
||||||
unsigned NewReg = 0;
|
|
||||||
int NewImm = 0;
|
|
||||||
bool ATUsed;
|
|
||||||
unsigned GP = STI.isABI_N64() ? Mips::GP_64 : Mips::GP;
|
unsigned GP = STI.isABI_N64() ? Mips::GP_64 : Mips::GP;
|
||||||
unsigned T9 = STI.isABI_N64() ? Mips::T9_64 : Mips::T9;
|
unsigned T9 = STI.isABI_N64() ? Mips::T9_64 : Mips::T9;
|
||||||
unsigned SP = STI.isABI_N64() ? Mips::SP_64 : Mips::SP;
|
unsigned SP = STI.isABI_N64() ? Mips::SP_64 : Mips::SP;
|
||||||
@ -165,33 +157,23 @@ void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
|||||||
unsigned LocalVarAreaOffset = MipsFI->needGPSaveRestore() ?
|
unsigned LocalVarAreaOffset = MipsFI->needGPSaveRestore() ?
|
||||||
(MFI->getObjectOffset(MipsFI->getGPFI()) + RegSize) :
|
(MFI->getObjectOffset(MipsFI->getGPFI()) + RegSize) :
|
||||||
MipsFI->getMaxCallFrameSize();
|
MipsFI->getMaxCallFrameSize();
|
||||||
unsigned StackSize = AlignOffset(LocalVarAreaOffset, StackAlign) +
|
uint64_t StackSize = RoundUpToAlignment(LocalVarAreaOffset, StackAlign) +
|
||||||
AlignOffset(MFI->getStackSize(), StackAlign);
|
RoundUpToAlignment(MFI->getStackSize(), StackAlign);
|
||||||
|
|
||||||
// Update stack size
|
// Update stack size
|
||||||
MFI->setStackSize(StackSize);
|
MFI->setStackSize(StackSize);
|
||||||
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(Mips::NOREORDER));
|
BuildMI(MBB, MBBI, dl, TII.get(Mips::NOREORDER));
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(Mips::NOMACRO));
|
|
||||||
|
|
||||||
// Emit instructions that set $gp using the the value of $t9.
|
// Emit instructions that set $gp using the the value of $t9.
|
||||||
// O32 uses the directive .cpload while N32/64 requires three instructions to
|
// O32 uses the directive .cpload while N32/64 requires three instructions to
|
||||||
// do this.
|
// do this.
|
||||||
// TODO: Do not emit these instructions if no instructions use $gp.
|
// TODO: Do not emit these instructions if no instructions use $gp.
|
||||||
if (isPIC && STI.isABI_O32())
|
if (isPIC && STI.isABI_O32())
|
||||||
BuildMI(MBB, llvm::prior(MBBI), dl, TII.get(Mips::CPLOAD))
|
BuildMI(MBB, MBBI, dl, TII.get(Mips::CPLOAD))
|
||||||
.addReg(RegInfo->getPICCallReg());
|
.addReg(RegInfo->getPICCallReg());
|
||||||
else if (STI.isABI_N64() || (isPIC && STI.isABI_N32())) {
|
|
||||||
// lui $28,%hi(%neg(%gp_rel(fname)))
|
BuildMI(MBB, MBBI, dl, TII.get(Mips::NOMACRO));
|
||||||
// addu $28,$28,$25
|
|
||||||
// addiu $28,$28,%lo(%neg(%gp_rel(fname)))
|
|
||||||
const GlobalValue *FName = MF.getFunction();
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(LUi), GP)
|
|
||||||
.addGlobalAddress(FName, 0, MipsII::MO_GPOFF_HI);
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(ADDu), GP).addReg(GP).addReg(T9);
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), GP).addReg(GP)
|
|
||||||
.addGlobalAddress(FName, 0, MipsII::MO_GPOFF_LO);
|
|
||||||
}
|
|
||||||
|
|
||||||
// No need to allocate space on the stack.
|
// No need to allocate space on the stack.
|
||||||
if (StackSize == 0 && !MFI->adjustsStack()) return;
|
if (StackSize == 0 && !MFI->adjustsStack()) return;
|
||||||
@ -200,13 +182,11 @@ void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
|||||||
std::vector<MachineMove> &Moves = MMI.getFrameMoves();
|
std::vector<MachineMove> &Moves = MMI.getFrameMoves();
|
||||||
MachineLocation DstML, SrcML;
|
MachineLocation DstML, SrcML;
|
||||||
|
|
||||||
// Adjust stack : addi sp, sp, (-imm)
|
// Adjust stack.
|
||||||
ATUsed = expandRegLargeImmPair(SP, -StackSize, NewReg, NewImm, MBB, MBBI);
|
if (isInt<16>(-StackSize)) // addi sp, sp, (-stacksize)
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), SP).addReg(NewReg).addImm(NewImm);
|
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), SP).addReg(SP).addImm(-StackSize);
|
||||||
|
else // Expand immediate that doesn't fit in 16-bit.
|
||||||
// FIXME: change this when mips goes MC".
|
expandLargeImm(SP, -StackSize, STI.isABI_N64(), TII, MBB, MBBI, dl);
|
||||||
if (ATUsed)
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(Mips::ATMACRO));
|
|
||||||
|
|
||||||
// emit ".cfi_def_cfa_offset StackSize"
|
// emit ".cfi_def_cfa_offset StackSize"
|
||||||
MCSymbol *AdjustSPLabel = MMI.getContext().CreateTempSymbol();
|
MCSymbol *AdjustSPLabel = MMI.getContext().CreateTempSymbol();
|
||||||
@ -259,6 +239,21 @@ void MipsFrameLowering::emitPrologue(MachineFunction &MF) const {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((STI.isABI_N64() || (isPIC && STI.isABI_N32())) &&
|
||||||
|
MRI.isPhysRegUsed(GP)) {
|
||||||
|
// lui $28,%hi(%neg(%gp_rel(fname)))
|
||||||
|
// addu $28,$28,$25
|
||||||
|
// addiu $28,$28,%lo(%neg(%gp_rel(fname)))
|
||||||
|
MachineBasicBlock::iterator InsPos = llvm::prior(MBBI);
|
||||||
|
const GlobalValue *FName = MF.getFunction();
|
||||||
|
BuildMI(MBB, MBBI, dl, TII.get(LUi), GP)
|
||||||
|
.addGlobalAddress(FName, 0, MipsII::MO_GPOFF_HI);
|
||||||
|
BuildMI(MBB, MBBI, dl, TII.get(ADDu), GP).addReg(GP).addReg(T9);
|
||||||
|
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), GP).addReg(GP)
|
||||||
|
.addGlobalAddress(FName, 0, MipsII::MO_GPOFF_LO);
|
||||||
|
MBBI = ++InsPos;
|
||||||
|
}
|
||||||
|
|
||||||
// if framepointer enabled, set it to point to the stack pointer.
|
// if framepointer enabled, set it to point to the stack pointer.
|
||||||
if (hasFP(MF)) {
|
if (hasFP(MF)) {
|
||||||
// Insert instruction "move $fp, $sp" at this location.
|
// Insert instruction "move $fp, $sp" at this location.
|
||||||
@ -298,13 +293,6 @@ void MipsFrameLowering::emitEpilogue(MachineFunction &MF,
|
|||||||
unsigned ADDu = STI.isABI_N64() ? Mips::DADDu : Mips::ADDu;
|
unsigned ADDu = STI.isABI_N64() ? Mips::DADDu : Mips::ADDu;
|
||||||
unsigned ADDiu = STI.isABI_N64() ? Mips::DADDiu : Mips::ADDiu;
|
unsigned ADDiu = STI.isABI_N64() ? Mips::DADDiu : Mips::ADDiu;
|
||||||
|
|
||||||
// Get the number of bytes from FrameInfo
|
|
||||||
unsigned StackSize = MFI->getStackSize();
|
|
||||||
|
|
||||||
unsigned NewReg = 0;
|
|
||||||
int NewImm = 0;
|
|
||||||
bool ATUsed = false;
|
|
||||||
|
|
||||||
// if framepointer enabled, restore the stack pointer.
|
// if framepointer enabled, restore the stack pointer.
|
||||||
if (hasFP(MF)) {
|
if (hasFP(MF)) {
|
||||||
// Find the first instruction that restores a callee-saved register.
|
// Find the first instruction that restores a callee-saved register.
|
||||||
@ -317,15 +305,17 @@ void MipsFrameLowering::emitEpilogue(MachineFunction &MF,
|
|||||||
BuildMI(MBB, I, dl, TII.get(ADDu), SP).addReg(FP).addReg(ZERO);
|
BuildMI(MBB, I, dl, TII.get(ADDu), SP).addReg(FP).addReg(ZERO);
|
||||||
}
|
}
|
||||||
|
|
||||||
// adjust stack : insert addi sp, sp, (imm)
|
// Get the number of bytes from FrameInfo
|
||||||
if (StackSize) {
|
uint64_t StackSize = MFI->getStackSize();
|
||||||
ATUsed = expandRegLargeImmPair(SP, StackSize, NewReg, NewImm, MBB, MBBI);
|
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), SP).addReg(NewReg).addImm(NewImm);
|
|
||||||
|
|
||||||
// FIXME: change this when mips goes MC".
|
if (!StackSize)
|
||||||
if (ATUsed)
|
return;
|
||||||
BuildMI(MBB, MBBI, dl, TII.get(Mips::ATMACRO));
|
|
||||||
}
|
// Adjust stack.
|
||||||
|
if (isInt<16>(StackSize)) // addi sp, sp, (-stacksize)
|
||||||
|
BuildMI(MBB, MBBI, dl, TII.get(ADDiu), SP).addReg(SP).addImm(StackSize);
|
||||||
|
else // Expand immediate that doesn't fit in 16-bit.
|
||||||
|
expandLargeImm(SP, StackSize, STI.isABI_N64(), TII, MBB, MBBI, dl);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MipsFrameLowering::
|
void MipsFrameLowering::
|
||||||
|
@ -7,8 +7,8 @@
|
|||||||
define void @f() nounwind {
|
define void @f() nounwind {
|
||||||
entry:
|
entry:
|
||||||
; CHECK: lui $at, 65534
|
; CHECK: lui $at, 65534
|
||||||
; CHECK: addu $at, $sp, $at
|
; CHECK: addiu $at, $at, -24
|
||||||
; CHECK: addiu $sp, $at, -24
|
; CHECK: addu $sp, $sp, $at
|
||||||
; CHECK: .cprestore 65536
|
; CHECK: .cprestore 65536
|
||||||
|
|
||||||
%agg.tmp = alloca %struct.S1, align 1
|
%agg.tmp = alloca %struct.S1, align 1
|
||||||
|
Loading…
Reference in New Issue
Block a user