mirror of
https://github.com/RPCS3/llvm.git
synced 2025-01-18 16:03:17 +00:00
Added support for fp callee saved registers.
Added fp register clobbering during calls. Added AsmPrinter support for "fmask", a bitmask that indicates where on the stack the fp callee saved registers are. Fixed the stack frame layout for Mips, now the callee saved regs are in the right stack location (a little documentation about how this stack frame must look like is present in MipsRegisterInfo.cpp). This was done using the method MipsRegisterInfo::adjustMipsStackFrame To be more clear, these are examples of what is solves : 1) FP and RA are also callee saved, and despite they aren't in CSI they must be saved before the fp callee saved registers. 2) The ABI requires that local varibles are allocated before the callee saved register area, the opposite behavior from the default allocation. 3) CPU and FPU saved register area must be aligned independent of each other. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@54403 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
01924d00c7
commit
bbe51362d5
@ -67,15 +67,13 @@ namespace {
|
||||
void printFCCOperand(const MachineInstr *MI, int opNum,
|
||||
const char *Modifier = 0);
|
||||
void printModuleLevelGV(const GlobalVariable* GVar);
|
||||
unsigned int getSavedRegsBitmask(bool isFloat, MachineFunction &MF);
|
||||
void printSavedRegsBitmask(MachineFunction &MF);
|
||||
void printHex32(unsigned int Value);
|
||||
|
||||
const char *emitCurrentABIString(void);
|
||||
void emitFunctionStart(MachineFunction &MF);
|
||||
void emitFunctionEnd(MachineFunction &MF);
|
||||
void emitFrameDirective(MachineFunction &MF);
|
||||
void emitMaskDirective(MachineFunction &MF);
|
||||
void emitFMaskDirective(MachineFunction &MF);
|
||||
|
||||
bool printInstruction(const MachineInstr *MI); // autogenerated.
|
||||
bool runOnMachineFunction(MachineFunction &F);
|
||||
@ -131,69 +129,45 @@ FunctionPass *llvm::createMipsCodePrinterPass(std::ostream &o,
|
||||
// Mask directives
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
/// Mask directive for GPR
|
||||
// Create a bitmask with all callee saved registers for CPU or Floating Point
|
||||
// registers. For CPU registers consider RA, GP and FP for saving if necessary.
|
||||
void MipsAsmPrinter::
|
||||
emitMaskDirective(MachineFunction &MF)
|
||||
{
|
||||
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
||||
|
||||
int StackSize = MF.getFrameInfo()->getStackSize();
|
||||
int Offset = (!MipsFI->getTopSavedRegOffset()) ? 0 :
|
||||
(-(StackSize-MipsFI->getTopSavedRegOffset()));
|
||||
|
||||
#ifndef NDEBUG
|
||||
DOUT << "--> emitMaskDirective" << '\n';
|
||||
DOUT << "StackSize : " << StackSize << '\n';
|
||||
DOUT << "getTopSavedReg : " << MipsFI->getTopSavedRegOffset() << '\n';
|
||||
DOUT << "Offset : " << Offset << "\n\n";
|
||||
#endif
|
||||
|
||||
unsigned int Bitmask = getSavedRegsBitmask(false, MF);
|
||||
O << "\t.mask \t";
|
||||
printHex32(Bitmask);
|
||||
O << ',' << Offset << '\n';
|
||||
}
|
||||
|
||||
/// TODO: Mask Directive for Floating Point
|
||||
void MipsAsmPrinter::
|
||||
emitFMaskDirective(MachineFunction &MF)
|
||||
{
|
||||
unsigned int Bitmask = getSavedRegsBitmask(true, MF);
|
||||
|
||||
O << "\t.fmask\t";
|
||||
printHex32(Bitmask);
|
||||
O << ",0" << '\n';
|
||||
}
|
||||
|
||||
// Create a bitmask with all callee saved registers for CPU
|
||||
// or Floating Point registers. For CPU registers consider RA,
|
||||
// GP and FP for saving if necessary.
|
||||
unsigned int MipsAsmPrinter::
|
||||
getSavedRegsBitmask(bool isFloat, MachineFunction &MF)
|
||||
printSavedRegsBitmask(MachineFunction &MF)
|
||||
{
|
||||
const TargetRegisterInfo &RI = *TM.getRegisterInfo();
|
||||
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
||||
|
||||
// Floating Point Registers, TODO
|
||||
if (isFloat)
|
||||
return 0;
|
||||
|
||||
// CPU Registers
|
||||
unsigned int Bitmask = 0;
|
||||
// CPU and FPU Saved Registers Bitmasks
|
||||
unsigned int CPUBitmask = 0;
|
||||
unsigned int FPUBitmask = 0;
|
||||
|
||||
// Set the CPU and FPU Bitmasks
|
||||
MachineFrameInfo *MFI = MF.getFrameInfo();
|
||||
const std::vector<CalleeSavedInfo> &CSI = MFI->getCalleeSavedInfo();
|
||||
for (unsigned i = 0, e = CSI.size(); i != e; ++i)
|
||||
Bitmask |= (1 << MipsRegisterInfo::getRegisterNumbering(CSI[i].getReg()));
|
||||
for (unsigned i = 0, e = CSI.size(); i != e; ++i) {
|
||||
unsigned RegNum = MipsRegisterInfo::getRegisterNumbering(CSI[i].getReg());
|
||||
if (CSI[i].getRegClass() == Mips::CPURegsRegisterClass)
|
||||
CPUBitmask |= (1 << RegNum);
|
||||
else
|
||||
FPUBitmask |= (1 << RegNum);
|
||||
}
|
||||
|
||||
// Return Address and Frame registers must also be set in CPUBitmask.
|
||||
if (RI.hasFP(MF))
|
||||
Bitmask |= (1 << MipsRegisterInfo::
|
||||
CPUBitmask |= (1 << MipsRegisterInfo::
|
||||
getRegisterNumbering(RI.getFrameRegister(MF)));
|
||||
|
||||
if (MF.getFrameInfo()->hasCalls())
|
||||
Bitmask |= (1 << MipsRegisterInfo::
|
||||
CPUBitmask |= (1 << MipsRegisterInfo::
|
||||
getRegisterNumbering(RI.getRARegister()));
|
||||
|
||||
return Bitmask;
|
||||
// Print CPUBitmask
|
||||
O << "\t.mask \t"; printHex32(CPUBitmask); O << ','
|
||||
<< MipsFI->getCPUTopSavedRegOff() << '\n';
|
||||
|
||||
// Print FPUBitmask
|
||||
O << "\t.fmask\t"; printHex32(FPUBitmask); O << ","
|
||||
<< MipsFI->getFPUTopSavedRegOff() << '\n';
|
||||
}
|
||||
|
||||
// Print a 32 bit hex number with all numbers.
|
||||
@ -269,8 +243,7 @@ emitFunctionStart(MachineFunction &MF)
|
||||
O << CurrentFnName << ":\n";
|
||||
|
||||
emitFrameDirective(MF);
|
||||
emitMaskDirective(MF);
|
||||
emitFMaskDirective(MF);
|
||||
printSavedRegsBitmask(MF);
|
||||
|
||||
O << '\n';
|
||||
}
|
||||
|
@ -279,8 +279,9 @@ class JumpFR<bits<6> op, bits<6> func, string instr_asm>:
|
||||
// Jump and Link (Call)
|
||||
let isCall=1, hasDelaySlot=1,
|
||||
// All calls clobber the non-callee saved registers...
|
||||
Defs = [AT, V0, V1, A0, A1, A2, A3, T0, T1, T2,
|
||||
T3, T4, T5, T6, T7, T8, T9, K0, K1], Uses = [GP] in {
|
||||
Defs = [AT, V0, V1, A0, A1, A2, A3, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9,
|
||||
K0, K1, F0, F1, F2, F3, F4, F5, F6, F7, F8, F9, F10, F11, F12, F13,
|
||||
F14, F15, F16, F17, F18, F19], Uses = [GP] in {
|
||||
class JumpLink<bits<6> op, string instr_asm>:
|
||||
FJ< op,
|
||||
(outs),
|
||||
|
@ -26,13 +26,20 @@ class MipsFunctionInfo : public MachineFunctionInfo {
|
||||
|
||||
private:
|
||||
/// Holds for each function where on the stack the Frame Pointer must be
|
||||
/// saved.
|
||||
/// saved. This is used on Prologue and Epilogue to emit FP save/restore
|
||||
int FPStackOffset;
|
||||
|
||||
/// Holds for each function where on the stack the Return Address must be
|
||||
/// saved.
|
||||
/// saved. This is used on Prologue and Epilogue to emit RA save/restore
|
||||
int RAStackOffset;
|
||||
|
||||
/// At each function entry, two special bitmask directives must be emitted
|
||||
/// to help debugging, for CPU and FPU callee saved registers. Both need
|
||||
/// the negative offset from the final stack size and its higher registers
|
||||
/// location on the stack.
|
||||
int CPUTopSavedRegOff;
|
||||
int FPUTopSavedRegOff;
|
||||
|
||||
/// MipsFIHolder - Holds a FrameIndex and it's Stack Pointer Offset
|
||||
struct MipsFIHolder {
|
||||
|
||||
@ -49,9 +56,9 @@ private:
|
||||
/// to be used on emitPrologue and processFunctionBeforeFrameFinalized.
|
||||
MipsFIHolder GPHolder;
|
||||
|
||||
// On LowerFORMAL_ARGUMENTS the stack size is unknown, so the Stack
|
||||
// Pointer Offset calculation of "not in register arguments" must be
|
||||
// postponed to emitPrologue.
|
||||
/// On LowerFORMAL_ARGUMENTS the stack size is unknown, so the Stack
|
||||
/// Pointer Offset calculation of "not in register arguments" must be
|
||||
/// postponed to emitPrologue.
|
||||
SmallVector<MipsFIHolder, 16> FnLoadArgs;
|
||||
bool HasLoadArgs;
|
||||
|
||||
@ -69,7 +76,8 @@ private:
|
||||
|
||||
public:
|
||||
MipsFunctionInfo(MachineFunction& MF)
|
||||
: FPStackOffset(0), RAStackOffset(0), GPHolder(-1,-1), HasLoadArgs(false),
|
||||
: FPStackOffset(0), RAStackOffset(0), CPUTopSavedRegOff(0),
|
||||
FPUTopSavedRegOff(0), GPHolder(-1,-1), HasLoadArgs(false),
|
||||
HasStoreVarArgs(false), SRetReturnReg(0)
|
||||
{}
|
||||
|
||||
@ -79,16 +87,17 @@ public:
|
||||
int getRAStackOffset() const { return RAStackOffset; }
|
||||
void setRAStackOffset(int Off) { RAStackOffset = Off; }
|
||||
|
||||
int getCPUTopSavedRegOff() const { return CPUTopSavedRegOff; }
|
||||
void setCPUTopSavedRegOff(int Off) { CPUTopSavedRegOff = Off; }
|
||||
|
||||
int getFPUTopSavedRegOff() const { return FPUTopSavedRegOff; }
|
||||
void setFPUTopSavedRegOff(int Off) { FPUTopSavedRegOff = Off; }
|
||||
|
||||
int getGPStackOffset() const { return GPHolder.SPOffset; }
|
||||
int getGPFI() const { return GPHolder.FI; }
|
||||
void setGPStackOffset(int Off) { GPHolder.SPOffset = Off; }
|
||||
void setGPFI(int FI) { GPHolder.FI = FI; }
|
||||
|
||||
int getTopSavedRegOffset() const {
|
||||
return (RAStackOffset > FPStackOffset) ?
|
||||
(RAStackOffset) : (FPStackOffset);
|
||||
}
|
||||
|
||||
bool hasLoadArgs() const { return HasLoadArgs; }
|
||||
bool hasStoreVarArgs() const { return HasStoreVarArgs; }
|
||||
|
||||
|
@ -47,37 +47,37 @@ unsigned MipsRegisterInfo::
|
||||
getRegisterNumbering(unsigned RegEnum)
|
||||
{
|
||||
switch (RegEnum) {
|
||||
case Mips::ZERO : case Mips::F0 : return 0;
|
||||
case Mips::ZERO : case Mips::F0 : case Mips::D0 : return 0;
|
||||
case Mips::AT : case Mips::F1 : return 1;
|
||||
case Mips::V0 : case Mips::F2 : return 2;
|
||||
case Mips::V0 : case Mips::F2 : case Mips::D1 : return 2;
|
||||
case Mips::V1 : case Mips::F3 : return 3;
|
||||
case Mips::A0 : case Mips::F4 : return 4;
|
||||
case Mips::A0 : case Mips::F4 : case Mips::D2 : return 4;
|
||||
case Mips::A1 : case Mips::F5 : return 5;
|
||||
case Mips::A2 : case Mips::F6 : return 6;
|
||||
case Mips::A2 : case Mips::F6 : case Mips::D3 : return 6;
|
||||
case Mips::A3 : case Mips::F7 : return 7;
|
||||
case Mips::T0 : case Mips::F8 : return 8;
|
||||
case Mips::T0 : case Mips::F8 : case Mips::D4 : return 8;
|
||||
case Mips::T1 : case Mips::F9 : return 9;
|
||||
case Mips::T2 : case Mips::F10: return 10;
|
||||
case Mips::T2 : case Mips::F10: case Mips::D5: return 10;
|
||||
case Mips::T3 : case Mips::F11: return 11;
|
||||
case Mips::T4 : case Mips::F12: return 12;
|
||||
case Mips::T4 : case Mips::F12: case Mips::D6: return 12;
|
||||
case Mips::T5 : case Mips::F13: return 13;
|
||||
case Mips::T6 : case Mips::F14: return 14;
|
||||
case Mips::T6 : case Mips::F14: case Mips::D7: return 14;
|
||||
case Mips::T7 : case Mips::F15: return 15;
|
||||
case Mips::T8 : case Mips::F16: return 16;
|
||||
case Mips::T8 : case Mips::F16: case Mips::D8: return 16;
|
||||
case Mips::T9 : case Mips::F17: return 17;
|
||||
case Mips::S0 : case Mips::F18: return 18;
|
||||
case Mips::S0 : case Mips::F18: case Mips::D9: return 18;
|
||||
case Mips::S1 : case Mips::F19: return 19;
|
||||
case Mips::S2 : case Mips::F20: return 20;
|
||||
case Mips::S2 : case Mips::F20: case Mips::D10: return 20;
|
||||
case Mips::S3 : case Mips::F21: return 21;
|
||||
case Mips::S4 : case Mips::F22: return 22;
|
||||
case Mips::S4 : case Mips::F22: case Mips::D11: return 22;
|
||||
case Mips::S5 : case Mips::F23: return 23;
|
||||
case Mips::S6 : case Mips::F24: return 24;
|
||||
case Mips::S6 : case Mips::F24: case Mips::D12: return 24;
|
||||
case Mips::S7 : case Mips::F25: return 25;
|
||||
case Mips::K0 : case Mips::F26: return 26;
|
||||
case Mips::K0 : case Mips::F26: case Mips::D13: return 26;
|
||||
case Mips::K1 : case Mips::F27: return 27;
|
||||
case Mips::GP : case Mips::F28: return 28;
|
||||
case Mips::GP : case Mips::F28: case Mips::D14: return 28;
|
||||
case Mips::SP : case Mips::F29: return 29;
|
||||
case Mips::FP : case Mips::F30: return 30;
|
||||
case Mips::FP : case Mips::F30: case Mips::D15: return 30;
|
||||
case Mips::RA : case Mips::F31: return 31;
|
||||
default: assert(0 && "Unknown register number!");
|
||||
}
|
||||
@ -94,26 +94,55 @@ unsigned MipsRegisterInfo::getPICCallReg(void) { return Mips::T9; }
|
||||
const unsigned* MipsRegisterInfo::
|
||||
getCalleeSavedRegs(const MachineFunction *MF) const
|
||||
{
|
||||
// Mips callee-save register range is $16-$23(s0-s7)
|
||||
static const unsigned CalleeSavedRegs[] = {
|
||||
// Mips callee-save register range is $16-$23, $f20-$f30
|
||||
static const unsigned SingleFloatOnlyCalleeSavedRegs[] = {
|
||||
Mips::S0, Mips::S1, Mips::S2, Mips::S3,
|
||||
Mips::S4, Mips::S5, Mips::S6, Mips::S7, 0
|
||||
Mips::S4, Mips::S5, Mips::S6, Mips::S7,
|
||||
Mips::F20, Mips::F21, Mips::F22, Mips::F23, Mips::F24, Mips::F25,
|
||||
Mips::F26, Mips::F27, Mips::F28, Mips::F29, Mips::F30, 0
|
||||
};
|
||||
|
||||
return CalleeSavedRegs;
|
||||
static const unsigned BitMode32CalleeSavedRegs[] = {
|
||||
Mips::S0, Mips::S1, Mips::S2, Mips::S3,
|
||||
Mips::S4, Mips::S5, Mips::S6, Mips::S7,
|
||||
Mips::F20, Mips::F22, Mips::F24, Mips::F26, Mips::F28, Mips::F30,
|
||||
Mips::D10, Mips::D11, Mips::D12, Mips::D13, Mips::D14, Mips::D15,0
|
||||
};
|
||||
|
||||
if (Subtarget.isSingleFloat())
|
||||
return SingleFloatOnlyCalleeSavedRegs;
|
||||
else
|
||||
return BitMode32CalleeSavedRegs;
|
||||
}
|
||||
|
||||
/// Mips Callee Saved Register Classes
|
||||
const TargetRegisterClass* const*
|
||||
MipsRegisterInfo::getCalleeSavedRegClasses(const MachineFunction *MF) const
|
||||
{
|
||||
static const TargetRegisterClass * const CalleeSavedRegClasses[] = {
|
||||
static const TargetRegisterClass * const SingleFloatOnlyCalleeSavedRC[] = {
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass, 0
|
||||
&Mips::FGR32RegClass, &Mips::FGR32RegClass, &Mips::FGR32RegClass,
|
||||
&Mips::FGR32RegClass, &Mips::FGR32RegClass, &Mips::FGR32RegClass,
|
||||
&Mips::FGR32RegClass, &Mips::FGR32RegClass, &Mips::FGR32RegClass,
|
||||
&Mips::FGR32RegClass, &Mips::FGR32RegClass, 0
|
||||
};
|
||||
return CalleeSavedRegClasses;
|
||||
|
||||
static const TargetRegisterClass * const BitMode32CalleeSavedRC[] = {
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::CPURegsRegClass, &Mips::CPURegsRegClass,
|
||||
&Mips::AFGR32RegClass, &Mips::AFGR32RegClass, &Mips::AFGR32RegClass,
|
||||
&Mips::AFGR32RegClass, &Mips::AFGR32RegClass, &Mips::AFGR32RegClass,
|
||||
&Mips::AFGR64RegClass, &Mips::AFGR64RegClass, &Mips::AFGR64RegClass,
|
||||
&Mips::AFGR64RegClass, &Mips::AFGR64RegClass, &Mips::AFGR64RegClass, 0
|
||||
};
|
||||
|
||||
if (Subtarget.isSingleFloat())
|
||||
return SingleFloatOnlyCalleeSavedRC;
|
||||
else
|
||||
return BitMode32CalleeSavedRC;
|
||||
}
|
||||
|
||||
BitVector MipsRegisterInfo::
|
||||
@ -143,16 +172,18 @@ getReservedRegs(const MachineFunction &MF) const
|
||||
// to grow up! Otherwise terrible hacks would have to be made
|
||||
// to get this stack ABI compliant :)
|
||||
//
|
||||
// The stack frame required by the ABI:
|
||||
// The stack frame required by the ABI (after call):
|
||||
// Offset
|
||||
//
|
||||
// 0 ----------
|
||||
// 4 Args to pass
|
||||
// 4 Args to pass
|
||||
// . saved $GP (used in PIC)
|
||||
// . Alloca allocations
|
||||
// . Local Area
|
||||
// . saved "Callee Saved" Registers
|
||||
// . CPU "Callee Saved" Registers
|
||||
// . saved FP
|
||||
// . saved RA
|
||||
// . FPU "Callee Saved" Registers
|
||||
// StackSize -----------
|
||||
//
|
||||
// Offset - offset from sp after stack allocation on function prologue
|
||||
@ -179,10 +210,112 @@ getReservedRegs(const MachineFunction &MF) const
|
||||
// possible to detect those references and the offsets are adjusted to
|
||||
// their real location.
|
||||
//
|
||||
//
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
void MipsRegisterInfo::adjustMipsStackFrame(MachineFunction &MF) const
|
||||
{
|
||||
MachineFrameInfo *MFI = MF.getFrameInfo();
|
||||
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
||||
const std::vector<CalleeSavedInfo> &CSI = MFI->getCalleeSavedInfo();
|
||||
unsigned StackAlign = MF.getTarget().getFrameInfo()->getStackAlignment();
|
||||
|
||||
// Min and Max CSI FrameIndex.
|
||||
int MinCSFI = -1, MaxCSFI = -1;
|
||||
|
||||
// See the description at MipsMachineFunction.h
|
||||
int TopCPUSavedRegOff = -1, TopFPUSavedRegOff = -1;
|
||||
|
||||
// Replace the dummy '0' SPOffset by the negative offsets, as explained on
|
||||
// LowerFORMAL_ARGUMENTS. Leaving '0' for while is necessary to avoid
|
||||
// the approach done by calculateFrameObjectOffsets to the stack frame.
|
||||
MipsFI->adjustLoadArgsFI(MFI);
|
||||
MipsFI->adjustStoreVarArgsFI(MFI);
|
||||
|
||||
// It happens that the default stack frame allocation order does not directly
|
||||
// map to the convention used for mips. So we must fix it. We move the callee
|
||||
// save register slots after the local variables area, as described in the
|
||||
// stack frame above.
|
||||
unsigned CalleeSavedAreaSize = 0;
|
||||
if (!CSI.empty()) {
|
||||
MinCSFI = CSI[0].getFrameIdx();
|
||||
MaxCSFI = CSI[CSI.size()-1].getFrameIdx();
|
||||
}
|
||||
for (unsigned i = 0, e = CSI.size(); i != e; ++i)
|
||||
CalleeSavedAreaSize += MFI->getObjectAlignment(CSI[i].getFrameIdx());
|
||||
|
||||
// Adjust local variables. They should come on the stack right
|
||||
// after the arguments.
|
||||
int LastOffsetFI = -1;
|
||||
for (int i = 0, e = MFI->getObjectIndexEnd(); i != e; ++i) {
|
||||
if (i >= MinCSFI && i <= MaxCSFI)
|
||||
continue;
|
||||
unsigned Offset = MFI->getObjectOffset(i) - CalleeSavedAreaSize;
|
||||
if (LastOffsetFI == -1)
|
||||
LastOffsetFI = i;
|
||||
if (Offset > MFI->getObjectOffset(LastOffsetFI))
|
||||
LastOffsetFI = i;
|
||||
MFI->setObjectOffset(i, Offset);
|
||||
}
|
||||
|
||||
// Adjust CPU Callee Saved Registers Area. Registers RA and FP must
|
||||
// be saved in this CPU Area there is the need. This whole Area must
|
||||
// be aligned to the default Stack Alignment requirements.
|
||||
unsigned StackOffset = 0;
|
||||
unsigned RegSize = Subtarget.isGP32bit() ? 4 : 8;
|
||||
|
||||
if (LastOffsetFI >= 0)
|
||||
StackOffset = MFI->getObjectOffset(LastOffsetFI)+
|
||||
MFI->getObjectAlignment(LastOffsetFI);
|
||||
|
||||
for (unsigned i = 0, e = CSI.size(); i != e ; ++i) {
|
||||
if (CSI[i].getRegClass() != Mips::CPURegsRegisterClass)
|
||||
break;
|
||||
MFI->setObjectOffset(CSI[i].getFrameIdx(), StackOffset);
|
||||
TopCPUSavedRegOff = StackOffset;
|
||||
StackOffset += MFI->getObjectAlignment(CSI[i].getFrameIdx());
|
||||
}
|
||||
|
||||
if (hasFP(MF)) {
|
||||
MFI->setObjectOffset(MFI->CreateStackObject(RegSize, RegSize),
|
||||
StackOffset);
|
||||
MipsFI->setFPStackOffset(StackOffset);
|
||||
TopCPUSavedRegOff = StackOffset;
|
||||
StackOffset += RegSize;
|
||||
}
|
||||
|
||||
if (MFI->hasCalls()) {
|
||||
MFI->setObjectOffset(MFI->CreateStackObject(RegSize, RegSize),
|
||||
StackOffset);
|
||||
MipsFI->setRAStackOffset(StackOffset);
|
||||
TopCPUSavedRegOff = StackOffset;
|
||||
StackOffset += RegSize;
|
||||
}
|
||||
StackOffset = ((StackOffset+StackAlign-1)/StackAlign*StackAlign);
|
||||
|
||||
// Adjust FPU Callee Saved Registers Area. This Area must be
|
||||
// aligned to the default Stack Alignment requirements.
|
||||
for (unsigned i = 0, e = CSI.size(); i != e; ++i) {
|
||||
if (CSI[i].getRegClass() == Mips::CPURegsRegisterClass)
|
||||
continue;
|
||||
MFI->setObjectOffset(CSI[i].getFrameIdx(), StackOffset);
|
||||
TopFPUSavedRegOff = StackOffset;
|
||||
StackOffset += MFI->getObjectAlignment(CSI[i].getFrameIdx());
|
||||
}
|
||||
StackOffset = ((StackOffset+StackAlign-1)/StackAlign*StackAlign);
|
||||
|
||||
// Update frame info
|
||||
MFI->setStackSize(StackOffset);
|
||||
|
||||
// Recalculate the final tops offset. The final values must be '0'
|
||||
// if there isn't a callee saved register for CPU or FPU, otherwise
|
||||
// a negative offset is needed.
|
||||
if (TopCPUSavedRegOff >= 0)
|
||||
MipsFI->setCPUTopSavedRegOff(TopCPUSavedRegOff-StackOffset);
|
||||
|
||||
if (TopFPUSavedRegOff >= 0)
|
||||
MipsFI->setFPUTopSavedRegOff(TopFPUSavedRegOff-StackOffset);
|
||||
}
|
||||
|
||||
// hasFP - Return true if the specified function should have a dedicated frame
|
||||
// pointer register. This is true if the function has variable sized allocas or
|
||||
// if frame pointer elimination is disabled.
|
||||
@ -256,60 +389,17 @@ emitPrologue(MachineFunction &MF) const
|
||||
MachineBasicBlock::iterator MBBI = MBB.begin();
|
||||
bool isPIC = (MF.getTarget().getRelocationModel() == Reloc::PIC_);
|
||||
|
||||
// Replace the dummy '0' SPOffset by the negative
|
||||
// offsets, as explained on LowerFORMAL_ARGUMENTS
|
||||
MipsFI->adjustLoadArgsFI(MFI);
|
||||
MipsFI->adjustStoreVarArgsFI(MFI);
|
||||
// Get the right frame order for Mips.
|
||||
adjustMipsStackFrame(MF);
|
||||
|
||||
// Get the number of bytes to allocate from the FrameInfo.
|
||||
int NumBytes = (int) MFI->getStackSize();
|
||||
|
||||
#ifndef NDEBUG
|
||||
DOUT << "\n<--- EMIT PROLOGUE --->\n";
|
||||
DOUT << "Actual Stack size :" << NumBytes << "\n";
|
||||
#endif
|
||||
unsigned StackSize = MFI->getStackSize();
|
||||
|
||||
// No need to allocate space on the stack.
|
||||
if (NumBytes == 0 && !MFI->hasCalls()) return;
|
||||
if (StackSize == 0 && !MFI->hasCalls()) return;
|
||||
|
||||
int FPOffset, RAOffset;
|
||||
|
||||
// Allocate space for saved RA and FP when needed
|
||||
// FIXME: within 64-bit registers, change hardcoded
|
||||
// sizes for RA and FP offsets.
|
||||
if ((hasFP(MF)) && (MFI->hasCalls())) {
|
||||
FPOffset = NumBytes;
|
||||
RAOffset = (NumBytes+4);
|
||||
NumBytes += 8;
|
||||
} else if ((!hasFP(MF)) && (MFI->hasCalls())) {
|
||||
FPOffset = 0;
|
||||
RAOffset = NumBytes;
|
||||
NumBytes += 4;
|
||||
} else if ((hasFP(MF)) && (!MFI->hasCalls())) {
|
||||
FPOffset = NumBytes;
|
||||
RAOffset = 0;
|
||||
NumBytes += 4;
|
||||
} else { // No calls and no fp.
|
||||
RAOffset = FPOffset = 0;
|
||||
}
|
||||
|
||||
MFI->setObjectOffset(MFI->CreateStackObject(4,4), FPOffset);
|
||||
MFI->setObjectOffset(MFI->CreateStackObject(4,4), RAOffset);
|
||||
MipsFI->setFPStackOffset(FPOffset);
|
||||
MipsFI->setRAStackOffset(RAOffset);
|
||||
|
||||
// Align stack.
|
||||
unsigned Align = MF.getTarget().getFrameInfo()->getStackAlignment();
|
||||
NumBytes = ((NumBytes+Align-1)/Align*Align);
|
||||
|
||||
#ifndef NDEBUG
|
||||
DOUT << "FPOffset :" << FPOffset << "\n";
|
||||
DOUT << "RAOffset :" << RAOffset << "\n";
|
||||
DOUT << "New stack size :" << NumBytes << "\n\n";
|
||||
#endif
|
||||
|
||||
// Update frame info
|
||||
MFI->setStackSize(NumBytes);
|
||||
int FPOffset = MipsFI->getFPStackOffset();
|
||||
int RAOffset = MipsFI->getRAStackOffset();
|
||||
|
||||
BuildMI(MBB, MBBI, TII.get(Mips::NOREORDER));
|
||||
|
||||
@ -320,7 +410,7 @@ emitPrologue(MachineFunction &MF) const
|
||||
|
||||
// Adjust stack : addi sp, sp, (-imm)
|
||||
BuildMI(MBB, MBBI, TII.get(Mips::ADDiu), Mips::SP)
|
||||
.addReg(Mips::SP).addImm(-NumBytes);
|
||||
.addReg(Mips::SP).addImm(-StackSize);
|
||||
|
||||
// Save the return address only if the function isnt a leaf one.
|
||||
// sw $ra, stack_loc($sp)
|
||||
@ -388,6 +478,7 @@ emitEpilogue(MachineFunction &MF, MachineBasicBlock &MBB) const
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void MipsRegisterInfo::
|
||||
processFunctionBeforeFrameFinalized(MachineFunction &MF) const {
|
||||
// Set the SPOffset on the FI where GP must be saved/loaded.
|
||||
@ -395,11 +486,6 @@ processFunctionBeforeFrameFinalized(MachineFunction &MF) const {
|
||||
bool isPIC = (MF.getTarget().getRelocationModel() == Reloc::PIC_);
|
||||
if (MFI->hasCalls() && isPIC) {
|
||||
MipsFunctionInfo *MipsFI = MF.getInfo<MipsFunctionInfo>();
|
||||
#ifndef NDEBUG
|
||||
DOUT << "processFunctionBeforeFrameFinalized\n";
|
||||
DOUT << "GPOffset :" << MipsFI->getGPStackOffset() << "\n";
|
||||
DOUT << "FI :" << MipsFI->getGPFI() << "\n";
|
||||
#endif
|
||||
MFI->setObjectOffset(MipsFI->getGPFI(), MipsFI->getGPStackOffset());
|
||||
}
|
||||
}
|
||||
|
@ -36,6 +36,9 @@ struct MipsRegisterInfo : public MipsGenRegisterInfo {
|
||||
/// Get PIC indirect call register
|
||||
static unsigned getPICCallReg(void);
|
||||
|
||||
/// Adjust the Mips stack frame.
|
||||
void adjustMipsStackFrame(MachineFunction &MF) const;
|
||||
|
||||
/// Code Generation virtual methods...
|
||||
const unsigned *getCalleeSavedRegs(const MachineFunction* MF = 0) const;
|
||||
|
||||
|
@ -33,7 +33,8 @@ createTargetAsmInfo() const
|
||||
// The stack is always 8 byte aligned
|
||||
// On function prologue, the stack is created by decrementing
|
||||
// its pointer. Once decremented, all references are done with positive
|
||||
// offset from the stack/frame pointer, so StackGrowsUp is used.
|
||||
// offset from the stack/frame pointer, using StackGrowsUp enables
|
||||
// an easier handling.
|
||||
// Using CodeModel::Large enables different CALL behavior.
|
||||
MipsTargetMachine::
|
||||
MipsTargetMachine(const Module &M, const std::string &FS, bool isLittle=false):
|
||||
|
Loading…
x
Reference in New Issue
Block a user