15f613dfdSUlrich Weigand //===-- SystemZFrameLowering.cpp - Frame lowering for SystemZ -------------===//
25f613dfdSUlrich Weigand //
35f613dfdSUlrich Weigand //                     The LLVM Compiler Infrastructure
45f613dfdSUlrich Weigand //
55f613dfdSUlrich Weigand // This file is distributed under the University of Illinois Open Source
65f613dfdSUlrich Weigand // License. See LICENSE.TXT for details.
75f613dfdSUlrich Weigand //
85f613dfdSUlrich Weigand //===----------------------------------------------------------------------===//
95f613dfdSUlrich Weigand 
105f613dfdSUlrich Weigand #include "SystemZFrameLowering.h"
115f613dfdSUlrich Weigand #include "SystemZCallingConv.h"
125f613dfdSUlrich Weigand #include "SystemZInstrBuilder.h"
135f613dfdSUlrich Weigand #include "SystemZMachineFunctionInfo.h"
145f613dfdSUlrich Weigand #include "SystemZTargetMachine.h"
155f613dfdSUlrich Weigand #include "llvm/CodeGen/MachineModuleInfo.h"
165f613dfdSUlrich Weigand #include "llvm/CodeGen/MachineRegisterInfo.h"
175f613dfdSUlrich Weigand #include "llvm/IR/Function.h"
185f613dfdSUlrich Weigand 
195f613dfdSUlrich Weigand using namespace llvm;
205f613dfdSUlrich Weigand 
215f613dfdSUlrich Weigand SystemZFrameLowering::SystemZFrameLowering(const SystemZTargetMachine &tm,
225f613dfdSUlrich Weigand                                            const SystemZSubtarget &sti)
235f613dfdSUlrich Weigand   : TargetFrameLowering(TargetFrameLowering::StackGrowsDown, 8,
245f613dfdSUlrich Weigand                         -SystemZMC::CallFrameSize),
255f613dfdSUlrich Weigand     TM(tm),
265f613dfdSUlrich Weigand     STI(sti) {
275f613dfdSUlrich Weigand   // The ABI-defined register save slots, relative to the incoming stack
285f613dfdSUlrich Weigand   // pointer.
295f613dfdSUlrich Weigand   static const unsigned SpillOffsetTable[][2] = {
305f613dfdSUlrich Weigand     { SystemZ::R2D,  0x10 },
315f613dfdSUlrich Weigand     { SystemZ::R3D,  0x18 },
325f613dfdSUlrich Weigand     { SystemZ::R4D,  0x20 },
335f613dfdSUlrich Weigand     { SystemZ::R5D,  0x28 },
345f613dfdSUlrich Weigand     { SystemZ::R6D,  0x30 },
355f613dfdSUlrich Weigand     { SystemZ::R7D,  0x38 },
365f613dfdSUlrich Weigand     { SystemZ::R8D,  0x40 },
375f613dfdSUlrich Weigand     { SystemZ::R9D,  0x48 },
385f613dfdSUlrich Weigand     { SystemZ::R10D, 0x50 },
395f613dfdSUlrich Weigand     { SystemZ::R11D, 0x58 },
405f613dfdSUlrich Weigand     { SystemZ::R12D, 0x60 },
415f613dfdSUlrich Weigand     { SystemZ::R13D, 0x68 },
425f613dfdSUlrich Weigand     { SystemZ::R14D, 0x70 },
435f613dfdSUlrich Weigand     { SystemZ::R15D, 0x78 },
445f613dfdSUlrich Weigand     { SystemZ::F0D,  0x80 },
455f613dfdSUlrich Weigand     { SystemZ::F2D,  0x88 },
465f613dfdSUlrich Weigand     { SystemZ::F4D,  0x90 },
475f613dfdSUlrich Weigand     { SystemZ::F6D,  0x98 }
485f613dfdSUlrich Weigand   };
495f613dfdSUlrich Weigand 
505f613dfdSUlrich Weigand   // Create a mapping from register number to save slot offset.
515f613dfdSUlrich Weigand   RegSpillOffsets.grow(SystemZ::NUM_TARGET_REGS);
525f613dfdSUlrich Weigand   for (unsigned I = 0, E = array_lengthof(SpillOffsetTable); I != E; ++I)
535f613dfdSUlrich Weigand     RegSpillOffsets[SpillOffsetTable[I][0]] = SpillOffsetTable[I][1];
545f613dfdSUlrich Weigand }
555f613dfdSUlrich Weigand 
565f613dfdSUlrich Weigand void SystemZFrameLowering::
575f613dfdSUlrich Weigand processFunctionBeforeCalleeSavedScan(MachineFunction &MF,
585f613dfdSUlrich Weigand                                      RegScavenger *RS) const {
595f613dfdSUlrich Weigand   MachineFrameInfo *MFFrame = MF.getFrameInfo();
605f613dfdSUlrich Weigand   MachineRegisterInfo &MRI = MF.getRegInfo();
615f613dfdSUlrich Weigand   const TargetRegisterInfo *TRI = MF.getTarget().getRegisterInfo();
625f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
635f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *MFI = MF.getInfo<SystemZMachineFunctionInfo>();
645f613dfdSUlrich Weigand   bool IsVarArg = MF.getFunction()->isVarArg();
655f613dfdSUlrich Weigand 
665f613dfdSUlrich Weigand   // va_start stores incoming FPR varargs in the normal way, but delegates
675f613dfdSUlrich Weigand   // the saving of incoming GPR varargs to spillCalleeSavedRegisters().
685f613dfdSUlrich Weigand   // Record these pending uses, which typically include the call-saved
695f613dfdSUlrich Weigand   // argument register R6D.
705f613dfdSUlrich Weigand   if (IsVarArg)
715f613dfdSUlrich Weigand     for (unsigned I = MFI->getVarArgsFirstGPR(); I < SystemZ::NumArgGPRs; ++I)
725f613dfdSUlrich Weigand       MRI.setPhysRegUsed(SystemZ::ArgGPRs[I]);
735f613dfdSUlrich Weigand 
745f613dfdSUlrich Weigand   // If the function requires a frame pointer, record that the hard
755f613dfdSUlrich Weigand   // frame pointer will be clobbered.
765f613dfdSUlrich Weigand   if (HasFP)
775f613dfdSUlrich Weigand     MRI.setPhysRegUsed(SystemZ::R11D);
785f613dfdSUlrich Weigand 
795f613dfdSUlrich Weigand   // If the function calls other functions, record that the return
805f613dfdSUlrich Weigand   // address register will be clobbered.
815f613dfdSUlrich Weigand   if (MFFrame->hasCalls())
825f613dfdSUlrich Weigand     MRI.setPhysRegUsed(SystemZ::R14D);
835f613dfdSUlrich Weigand 
845f613dfdSUlrich Weigand   // If we are saving GPRs other than the stack pointer, we might as well
855f613dfdSUlrich Weigand   // save and restore the stack pointer at the same time, via STMG and LMG.
865f613dfdSUlrich Weigand   // This allows the deallocation to be done by the LMG, rather than needing
875f613dfdSUlrich Weigand   // a separate %r15 addition.
885f613dfdSUlrich Weigand   const uint16_t *CSRegs = TRI->getCalleeSavedRegs(&MF);
895f613dfdSUlrich Weigand   for (unsigned I = 0; CSRegs[I]; ++I) {
905f613dfdSUlrich Weigand     unsigned Reg = CSRegs[I];
915f613dfdSUlrich Weigand     if (SystemZ::GR64BitRegClass.contains(Reg) && MRI.isPhysRegUsed(Reg)) {
925f613dfdSUlrich Weigand       MRI.setPhysRegUsed(SystemZ::R15D);
935f613dfdSUlrich Weigand       break;
945f613dfdSUlrich Weigand     }
955f613dfdSUlrich Weigand   }
965f613dfdSUlrich Weigand }
975f613dfdSUlrich Weigand 
985f613dfdSUlrich Weigand // Add GPR64 to the save instruction being built by MIB, which is in basic
995f613dfdSUlrich Weigand // block MBB.  IsImplicit says whether this is an explicit operand to the
1005f613dfdSUlrich Weigand // instruction, or an implicit one that comes between the explicit start
1015f613dfdSUlrich Weigand // and end registers.
1025f613dfdSUlrich Weigand static void addSavedGPR(MachineBasicBlock &MBB, MachineInstrBuilder &MIB,
1035f613dfdSUlrich Weigand                         const SystemZTargetMachine &TM,
1045f613dfdSUlrich Weigand                         unsigned GPR64, bool IsImplicit) {
1055f613dfdSUlrich Weigand   const SystemZRegisterInfo *RI = TM.getRegisterInfo();
1065f613dfdSUlrich Weigand   unsigned GPR32 = RI->getSubReg(GPR64, SystemZ::subreg_32bit);
1075f613dfdSUlrich Weigand   bool IsLive = MBB.isLiveIn(GPR64) || MBB.isLiveIn(GPR32);
1085f613dfdSUlrich Weigand   if (!IsLive || !IsImplicit) {
1095f613dfdSUlrich Weigand     MIB.addReg(GPR64, getImplRegState(IsImplicit) | getKillRegState(!IsLive));
1105f613dfdSUlrich Weigand     if (!IsLive)
1115f613dfdSUlrich Weigand       MBB.addLiveIn(GPR64);
1125f613dfdSUlrich Weigand   }
1135f613dfdSUlrich Weigand }
1145f613dfdSUlrich Weigand 
1155f613dfdSUlrich Weigand bool SystemZFrameLowering::
1165f613dfdSUlrich Weigand spillCalleeSavedRegisters(MachineBasicBlock &MBB,
1175f613dfdSUlrich Weigand                           MachineBasicBlock::iterator MBBI,
1185f613dfdSUlrich Weigand                           const std::vector<CalleeSavedInfo> &CSI,
1195f613dfdSUlrich Weigand                           const TargetRegisterInfo *TRI) const {
1205f613dfdSUlrich Weigand   if (CSI.empty())
1215f613dfdSUlrich Weigand     return false;
1225f613dfdSUlrich Weigand 
1235f613dfdSUlrich Weigand   MachineFunction &MF = *MBB.getParent();
1245f613dfdSUlrich Weigand   const TargetInstrInfo *TII = MF.getTarget().getInstrInfo();
1255f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
1265f613dfdSUlrich Weigand   bool IsVarArg = MF.getFunction()->isVarArg();
1275f613dfdSUlrich Weigand   DebugLoc DL = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
1285f613dfdSUlrich Weigand 
1295f613dfdSUlrich Weigand   // Scan the call-saved GPRs and find the bounds of the register spill area.
1305f613dfdSUlrich Weigand   unsigned SavedGPRFrameSize = 0;
1315f613dfdSUlrich Weigand   unsigned LowGPR = 0;
1325f613dfdSUlrich Weigand   unsigned HighGPR = SystemZ::R15D;
1335f613dfdSUlrich Weigand   unsigned StartOffset = -1U;
1345f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
1355f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
1365f613dfdSUlrich Weigand     if (SystemZ::GR64BitRegClass.contains(Reg)) {
1375f613dfdSUlrich Weigand       SavedGPRFrameSize += 8;
1385f613dfdSUlrich Weigand       unsigned Offset = RegSpillOffsets[Reg];
1395f613dfdSUlrich Weigand       assert(Offset && "Unexpected GPR save");
1405f613dfdSUlrich Weigand       if (StartOffset > Offset) {
1415f613dfdSUlrich Weigand         LowGPR = Reg;
1425f613dfdSUlrich Weigand         StartOffset = Offset;
1435f613dfdSUlrich Weigand       }
1445f613dfdSUlrich Weigand     }
1455f613dfdSUlrich Weigand   }
1465f613dfdSUlrich Weigand 
1475f613dfdSUlrich Weigand   // Save information about the range and location of the call-saved
1485f613dfdSUlrich Weigand   // registers, for use by the epilogue inserter.
1495f613dfdSUlrich Weigand   ZFI->setSavedGPRFrameSize(SavedGPRFrameSize);
1505f613dfdSUlrich Weigand   ZFI->setLowSavedGPR(LowGPR);
1515f613dfdSUlrich Weigand   ZFI->setHighSavedGPR(HighGPR);
1525f613dfdSUlrich Weigand 
1535f613dfdSUlrich Weigand   // Include the GPR varargs, if any.  R6D is call-saved, so would
1545f613dfdSUlrich Weigand   // be included by the loop above, but we also need to handle the
1555f613dfdSUlrich Weigand   // call-clobbered argument registers.
1565f613dfdSUlrich Weigand   if (IsVarArg) {
1575f613dfdSUlrich Weigand     unsigned FirstGPR = ZFI->getVarArgsFirstGPR();
1585f613dfdSUlrich Weigand     if (FirstGPR < SystemZ::NumArgGPRs) {
1595f613dfdSUlrich Weigand       unsigned Reg = SystemZ::ArgGPRs[FirstGPR];
1605f613dfdSUlrich Weigand       unsigned Offset = RegSpillOffsets[Reg];
1615f613dfdSUlrich Weigand       if (StartOffset > Offset) {
1625f613dfdSUlrich Weigand         LowGPR = Reg; StartOffset = Offset;
1635f613dfdSUlrich Weigand       }
1645f613dfdSUlrich Weigand     }
1655f613dfdSUlrich Weigand   }
1665f613dfdSUlrich Weigand 
1675f613dfdSUlrich Weigand   // Save GPRs
1685f613dfdSUlrich Weigand   if (LowGPR) {
1695f613dfdSUlrich Weigand     assert(LowGPR != HighGPR && "Should be saving %r15 and something else");
1705f613dfdSUlrich Weigand 
1715f613dfdSUlrich Weigand     // Build an STMG instruction.
1725f613dfdSUlrich Weigand     MachineInstrBuilder MIB = BuildMI(MBB, MBBI, DL, TII->get(SystemZ::STMG));
1735f613dfdSUlrich Weigand 
1745f613dfdSUlrich Weigand     // Add the explicit register operands.
1755f613dfdSUlrich Weigand     addSavedGPR(MBB, MIB, TM, LowGPR, false);
1765f613dfdSUlrich Weigand     addSavedGPR(MBB, MIB, TM, HighGPR, false);
1775f613dfdSUlrich Weigand 
1785f613dfdSUlrich Weigand     // Add the address.
1795f613dfdSUlrich Weigand     MIB.addReg(SystemZ::R15D).addImm(StartOffset);
1805f613dfdSUlrich Weigand 
1815f613dfdSUlrich Weigand     // Make sure all call-saved GPRs are included as operands and are
1825f613dfdSUlrich Weigand     // marked as live on entry.
1835f613dfdSUlrich Weigand     for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
1845f613dfdSUlrich Weigand       unsigned Reg = CSI[I].getReg();
1855f613dfdSUlrich Weigand       if (SystemZ::GR64BitRegClass.contains(Reg))
1865f613dfdSUlrich Weigand         addSavedGPR(MBB, MIB, TM, Reg, true);
1875f613dfdSUlrich Weigand     }
1885f613dfdSUlrich Weigand 
1895f613dfdSUlrich Weigand     // ...likewise GPR varargs.
1905f613dfdSUlrich Weigand     if (IsVarArg)
1915f613dfdSUlrich Weigand       for (unsigned I = ZFI->getVarArgsFirstGPR(); I < SystemZ::NumArgGPRs; ++I)
1925f613dfdSUlrich Weigand         addSavedGPR(MBB, MIB, TM, SystemZ::ArgGPRs[I], true);
1935f613dfdSUlrich Weigand   }
1945f613dfdSUlrich Weigand 
1955f613dfdSUlrich Weigand   // Save FPRs in the normal TargetInstrInfo way.
1965f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
1975f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
1985f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg)) {
1995f613dfdSUlrich Weigand       MBB.addLiveIn(Reg);
2005f613dfdSUlrich Weigand       TII->storeRegToStackSlot(MBB, MBBI, Reg, true, CSI[I].getFrameIdx(),
2015f613dfdSUlrich Weigand                                &SystemZ::FP64BitRegClass, TRI);
2025f613dfdSUlrich Weigand     }
2035f613dfdSUlrich Weigand   }
2045f613dfdSUlrich Weigand 
2055f613dfdSUlrich Weigand   return true;
2065f613dfdSUlrich Weigand }
2075f613dfdSUlrich Weigand 
2085f613dfdSUlrich Weigand bool SystemZFrameLowering::
2095f613dfdSUlrich Weigand restoreCalleeSavedRegisters(MachineBasicBlock &MBB,
2105f613dfdSUlrich Weigand                             MachineBasicBlock::iterator MBBI,
2115f613dfdSUlrich Weigand                             const std::vector<CalleeSavedInfo> &CSI,
2125f613dfdSUlrich Weigand                             const TargetRegisterInfo *TRI) const {
2135f613dfdSUlrich Weigand   if (CSI.empty())
2145f613dfdSUlrich Weigand     return false;
2155f613dfdSUlrich Weigand 
2165f613dfdSUlrich Weigand   MachineFunction &MF = *MBB.getParent();
2175f613dfdSUlrich Weigand   const TargetInstrInfo *TII = MF.getTarget().getInstrInfo();
2185f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
2195f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
2205f613dfdSUlrich Weigand   DebugLoc DL = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
2215f613dfdSUlrich Weigand 
2225f613dfdSUlrich Weigand   // Restore FPRs in the normal TargetInstrInfo way.
2235f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
2245f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
2255f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg))
2265f613dfdSUlrich Weigand       TII->loadRegFromStackSlot(MBB, MBBI, Reg, CSI[I].getFrameIdx(),
2275f613dfdSUlrich Weigand                                 &SystemZ::FP64BitRegClass, TRI);
2285f613dfdSUlrich Weigand   }
2295f613dfdSUlrich Weigand 
2305f613dfdSUlrich Weigand   // Restore call-saved GPRs (but not call-clobbered varargs, which at
2315f613dfdSUlrich Weigand   // this point might hold return values).
2325f613dfdSUlrich Weigand   unsigned LowGPR = ZFI->getLowSavedGPR();
2335f613dfdSUlrich Weigand   unsigned HighGPR = ZFI->getHighSavedGPR();
2345f613dfdSUlrich Weigand   unsigned StartOffset = RegSpillOffsets[LowGPR];
2355f613dfdSUlrich Weigand   if (LowGPR) {
2365f613dfdSUlrich Weigand     // If we saved any of %r2-%r5 as varargs, we should also be saving
2375f613dfdSUlrich Weigand     // and restoring %r6.  If we're saving %r6 or above, we should be
2385f613dfdSUlrich Weigand     // restoring it too.
2395f613dfdSUlrich Weigand     assert(LowGPR != HighGPR && "Should be loading %r15 and something else");
2405f613dfdSUlrich Weigand 
2415f613dfdSUlrich Weigand     // Build an LMG instruction.
2425f613dfdSUlrich Weigand     MachineInstrBuilder MIB = BuildMI(MBB, MBBI, DL, TII->get(SystemZ::LMG));
2435f613dfdSUlrich Weigand 
2445f613dfdSUlrich Weigand     // Add the explicit register operands.
2455f613dfdSUlrich Weigand     MIB.addReg(LowGPR, RegState::Define);
2465f613dfdSUlrich Weigand     MIB.addReg(HighGPR, RegState::Define);
2475f613dfdSUlrich Weigand 
2485f613dfdSUlrich Weigand     // Add the address.
2495f613dfdSUlrich Weigand     MIB.addReg(HasFP ? SystemZ::R11D : SystemZ::R15D);
2505f613dfdSUlrich Weigand     MIB.addImm(StartOffset);
2515f613dfdSUlrich Weigand 
2525f613dfdSUlrich Weigand     // Do a second scan adding regs as being defined by instruction
2535f613dfdSUlrich Weigand     for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
2545f613dfdSUlrich Weigand       unsigned Reg = CSI[I].getReg();
2555f613dfdSUlrich Weigand       if (Reg != LowGPR && Reg != HighGPR)
2565f613dfdSUlrich Weigand         MIB.addReg(Reg, RegState::ImplicitDefine);
2575f613dfdSUlrich Weigand     }
2585f613dfdSUlrich Weigand   }
2595f613dfdSUlrich Weigand 
2605f613dfdSUlrich Weigand   return true;
2615f613dfdSUlrich Weigand }
2625f613dfdSUlrich Weigand 
2635f613dfdSUlrich Weigand // Emit instructions before MBBI (in MBB) to add NumBytes to Reg.
2645f613dfdSUlrich Weigand static void emitIncrement(MachineBasicBlock &MBB,
2655f613dfdSUlrich Weigand                           MachineBasicBlock::iterator &MBBI,
2665f613dfdSUlrich Weigand                           const DebugLoc &DL,
2675f613dfdSUlrich Weigand                           unsigned Reg, int64_t NumBytes,
2685f613dfdSUlrich Weigand                           const TargetInstrInfo *TII) {
2695f613dfdSUlrich Weigand   while (NumBytes) {
2705f613dfdSUlrich Weigand     unsigned Opcode;
2715f613dfdSUlrich Weigand     int64_t ThisVal = NumBytes;
2725f613dfdSUlrich Weigand     if (isInt<16>(NumBytes))
2735f613dfdSUlrich Weigand       Opcode = SystemZ::AGHI;
2745f613dfdSUlrich Weigand     else {
2755f613dfdSUlrich Weigand       Opcode = SystemZ::AGFI;
2765f613dfdSUlrich Weigand       // Make sure we maintain 8-byte stack alignment.
2775f613dfdSUlrich Weigand       int64_t MinVal = -int64_t(1) << 31;
2785f613dfdSUlrich Weigand       int64_t MaxVal = (int64_t(1) << 31) - 8;
2795f613dfdSUlrich Weigand       if (ThisVal < MinVal)
2805f613dfdSUlrich Weigand         ThisVal = MinVal;
2815f613dfdSUlrich Weigand       else if (ThisVal > MaxVal)
2825f613dfdSUlrich Weigand         ThisVal = MaxVal;
2835f613dfdSUlrich Weigand     }
2845f613dfdSUlrich Weigand     MachineInstr *MI = BuildMI(MBB, MBBI, DL, TII->get(Opcode), Reg)
2855f613dfdSUlrich Weigand       .addReg(Reg).addImm(ThisVal);
2865f613dfdSUlrich Weigand     // The PSW implicit def is dead.
2875f613dfdSUlrich Weigand     MI->getOperand(3).setIsDead();
2885f613dfdSUlrich Weigand     NumBytes -= ThisVal;
2895f613dfdSUlrich Weigand   }
2905f613dfdSUlrich Weigand }
2915f613dfdSUlrich Weigand 
2925f613dfdSUlrich Weigand void SystemZFrameLowering::emitPrologue(MachineFunction &MF) const {
2935f613dfdSUlrich Weigand   MachineBasicBlock &MBB = MF.front();
2945f613dfdSUlrich Weigand   MachineFrameInfo *MFFrame = MF.getFrameInfo();
2955f613dfdSUlrich Weigand   const SystemZInstrInfo *ZII =
2965f613dfdSUlrich Weigand     static_cast<const SystemZInstrInfo*>(MF.getTarget().getInstrInfo());
2975f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
2985f613dfdSUlrich Weigand   MachineBasicBlock::iterator MBBI = MBB.begin();
2995f613dfdSUlrich Weigand   MachineModuleInfo &MMI = MF.getMMI();
300*b08d2c2dSRafael Espindola   const MCRegisterInfo &MRI = MMI.getContext().getRegisterInfo();
3015f613dfdSUlrich Weigand   const std::vector<CalleeSavedInfo> &CSI = MFFrame->getCalleeSavedInfo();
3025f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
3035f613dfdSUlrich Weigand   DebugLoc DL = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
3045f613dfdSUlrich Weigand 
3055f613dfdSUlrich Weigand   // The current offset of the stack pointer from the CFA.
3065f613dfdSUlrich Weigand   int64_t SPOffsetFromCFA = -SystemZMC::CFAOffsetFromInitialSP;
3075f613dfdSUlrich Weigand 
3085f613dfdSUlrich Weigand   if (ZFI->getLowSavedGPR()) {
3095f613dfdSUlrich Weigand     // Skip over the GPR saves.
3105f613dfdSUlrich Weigand     if (MBBI != MBB.end() && MBBI->getOpcode() == SystemZ::STMG)
3115f613dfdSUlrich Weigand       ++MBBI;
3125f613dfdSUlrich Weigand     else
3135f613dfdSUlrich Weigand       llvm_unreachable("Couldn't skip over GPR saves");
3145f613dfdSUlrich Weigand 
3155f613dfdSUlrich Weigand     // Add CFI for the GPR saves.
3165f613dfdSUlrich Weigand     MCSymbol *GPRSaveLabel = MMI.getContext().CreateTempSymbol();
3175f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL,
3185f613dfdSUlrich Weigand             ZII->get(TargetOpcode::PROLOG_LABEL)).addSym(GPRSaveLabel);
3195f613dfdSUlrich Weigand     for (std::vector<CalleeSavedInfo>::const_iterator
3205f613dfdSUlrich Weigand            I = CSI.begin(), E = CSI.end(); I != E; ++I) {
3215f613dfdSUlrich Weigand       unsigned Reg = I->getReg();
3225f613dfdSUlrich Weigand       if (SystemZ::GR64BitRegClass.contains(Reg)) {
3235f613dfdSUlrich Weigand         int64_t Offset = SPOffsetFromCFA + RegSpillOffsets[Reg];
324*b08d2c2dSRafael Espindola         MMI.addFrameInst(MCCFIInstruction::createOffset(
325*b08d2c2dSRafael Espindola             GPRSaveLabel, MRI.getDwarfRegNum(Reg, true), Offset));
3265f613dfdSUlrich Weigand       }
3275f613dfdSUlrich Weigand     }
3285f613dfdSUlrich Weigand   }
3295f613dfdSUlrich Weigand 
3305f613dfdSUlrich Weigand   uint64_t StackSize = getAllocatedStackSize(MF);
3315f613dfdSUlrich Weigand   if (StackSize) {
3325f613dfdSUlrich Weigand     // Allocate StackSize bytes.
3335f613dfdSUlrich Weigand     int64_t Delta = -int64_t(StackSize);
3345f613dfdSUlrich Weigand     emitIncrement(MBB, MBBI, DL, SystemZ::R15D, Delta, ZII);
3355f613dfdSUlrich Weigand 
3365f613dfdSUlrich Weigand     // Add CFI for the allocation.
3375f613dfdSUlrich Weigand     MCSymbol *AdjustSPLabel = MMI.getContext().CreateTempSymbol();
3385f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::PROLOG_LABEL))
3395f613dfdSUlrich Weigand       .addSym(AdjustSPLabel);
340*b08d2c2dSRafael Espindola     MMI.addFrameInst(MCCFIInstruction::createDefCfaOffset(
341*b08d2c2dSRafael Espindola         AdjustSPLabel, SPOffsetFromCFA + Delta));
3425f613dfdSUlrich Weigand     SPOffsetFromCFA += Delta;
3435f613dfdSUlrich Weigand   }
3445f613dfdSUlrich Weigand 
3455f613dfdSUlrich Weigand   if (HasFP) {
3465f613dfdSUlrich Weigand     // Copy the base of the frame to R11.
3475f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL, ZII->get(SystemZ::LGR), SystemZ::R11D)
3485f613dfdSUlrich Weigand       .addReg(SystemZ::R15D);
3495f613dfdSUlrich Weigand 
3505f613dfdSUlrich Weigand     // Add CFI for the new frame location.
3515f613dfdSUlrich Weigand     MCSymbol *SetFPLabel = MMI.getContext().CreateTempSymbol();
3525f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::PROLOG_LABEL))
3535f613dfdSUlrich Weigand       .addSym(SetFPLabel);
354*b08d2c2dSRafael Espindola     unsigned HardFP = MRI.getDwarfRegNum(SystemZ::R11D, true);
355*b08d2c2dSRafael Espindola     MMI.addFrameInst(
356*b08d2c2dSRafael Espindola         MCCFIInstruction::createDefCfaRegister(SetFPLabel, HardFP));
3575f613dfdSUlrich Weigand 
3585f613dfdSUlrich Weigand     // Mark the FramePtr as live at the beginning of every block except
3595f613dfdSUlrich Weigand     // the entry block.  (We'll have marked R11 as live on entry when
3605f613dfdSUlrich Weigand     // saving the GPRs.)
3615f613dfdSUlrich Weigand     for (MachineFunction::iterator
3625f613dfdSUlrich Weigand            I = llvm::next(MF.begin()), E = MF.end(); I != E; ++I)
3635f613dfdSUlrich Weigand       I->addLiveIn(SystemZ::R11D);
3645f613dfdSUlrich Weigand   }
3655f613dfdSUlrich Weigand 
3665f613dfdSUlrich Weigand   // Skip over the FPR saves.
3675f613dfdSUlrich Weigand   MCSymbol *FPRSaveLabel = 0;
3685f613dfdSUlrich Weigand   for (std::vector<CalleeSavedInfo>::const_iterator
3695f613dfdSUlrich Weigand          I = CSI.begin(), E = CSI.end(); I != E; ++I) {
3705f613dfdSUlrich Weigand     unsigned Reg = I->getReg();
3715f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg)) {
3725f613dfdSUlrich Weigand       if (MBBI != MBB.end() &&
3735f613dfdSUlrich Weigand           (MBBI->getOpcode() == SystemZ::STD ||
3745f613dfdSUlrich Weigand            MBBI->getOpcode() == SystemZ::STDY))
3755f613dfdSUlrich Weigand         ++MBBI;
3765f613dfdSUlrich Weigand       else
3775f613dfdSUlrich Weigand         llvm_unreachable("Couldn't skip over FPR save");
3785f613dfdSUlrich Weigand 
3795f613dfdSUlrich Weigand       // Add CFI for the this save.
3805f613dfdSUlrich Weigand       if (!FPRSaveLabel)
3815f613dfdSUlrich Weigand         FPRSaveLabel = MMI.getContext().CreateTempSymbol();
382*b08d2c2dSRafael Espindola       unsigned Reg = MRI.getDwarfRegNum(I->getReg(), true);
3835f613dfdSUlrich Weigand       int64_t Offset = getFrameIndexOffset(MF, I->getFrameIdx());
384*b08d2c2dSRafael Espindola       MMI.addFrameInst(MCCFIInstruction::createOffset(
385*b08d2c2dSRafael Espindola           FPRSaveLabel, Reg, SPOffsetFromCFA + Offset));
3865f613dfdSUlrich Weigand     }
3875f613dfdSUlrich Weigand   }
3885f613dfdSUlrich Weigand   // Complete the CFI for the FPR saves, modelling them as taking effect
3895f613dfdSUlrich Weigand   // after the last save.
3905f613dfdSUlrich Weigand   if (FPRSaveLabel)
3915f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::PROLOG_LABEL))
3925f613dfdSUlrich Weigand       .addSym(FPRSaveLabel);
3935f613dfdSUlrich Weigand }
3945f613dfdSUlrich Weigand 
3955f613dfdSUlrich Weigand void SystemZFrameLowering::emitEpilogue(MachineFunction &MF,
3965f613dfdSUlrich Weigand                                         MachineBasicBlock &MBB) const {
3975f613dfdSUlrich Weigand   MachineBasicBlock::iterator MBBI = MBB.getLastNonDebugInstr();
3985f613dfdSUlrich Weigand   const SystemZInstrInfo *ZII =
3995f613dfdSUlrich Weigand     static_cast<const SystemZInstrInfo*>(MF.getTarget().getInstrInfo());
4005f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
4015f613dfdSUlrich Weigand 
4025f613dfdSUlrich Weigand   // Skip the return instruction.
4035f613dfdSUlrich Weigand   assert(MBBI->getOpcode() == SystemZ::RET &&
4045f613dfdSUlrich Weigand          "Can only insert epilogue into returning blocks");
4055f613dfdSUlrich Weigand 
4065f613dfdSUlrich Weigand   uint64_t StackSize = getAllocatedStackSize(MF);
4075f613dfdSUlrich Weigand   if (ZFI->getLowSavedGPR()) {
4085f613dfdSUlrich Weigand     --MBBI;
4095f613dfdSUlrich Weigand     unsigned Opcode = MBBI->getOpcode();
4105f613dfdSUlrich Weigand     if (Opcode != SystemZ::LMG)
4115f613dfdSUlrich Weigand       llvm_unreachable("Expected to see callee-save register restore code");
4125f613dfdSUlrich Weigand 
4135f613dfdSUlrich Weigand     unsigned AddrOpNo = 2;
4145f613dfdSUlrich Weigand     DebugLoc DL = MBBI->getDebugLoc();
4155f613dfdSUlrich Weigand     uint64_t Offset = StackSize + MBBI->getOperand(AddrOpNo + 1).getImm();
4165f613dfdSUlrich Weigand     unsigned NewOpcode = ZII->getOpcodeForOffset(Opcode, Offset);
4175f613dfdSUlrich Weigand 
4185f613dfdSUlrich Weigand     // If the offset is too large, use the largest stack-aligned offset
4195f613dfdSUlrich Weigand     // and add the rest to the base register (the stack or frame pointer).
4205f613dfdSUlrich Weigand     if (!NewOpcode) {
4215f613dfdSUlrich Weigand       uint64_t NumBytes = Offset - 0x7fff8;
4225f613dfdSUlrich Weigand       emitIncrement(MBB, MBBI, DL, MBBI->getOperand(AddrOpNo).getReg(),
4235f613dfdSUlrich Weigand                     NumBytes, ZII);
4245f613dfdSUlrich Weigand       Offset -= NumBytes;
4255f613dfdSUlrich Weigand       NewOpcode = ZII->getOpcodeForOffset(Opcode, Offset);
4265f613dfdSUlrich Weigand       assert(NewOpcode && "No restore instruction available");
4275f613dfdSUlrich Weigand     }
4285f613dfdSUlrich Weigand 
4295f613dfdSUlrich Weigand     MBBI->setDesc(ZII->get(NewOpcode));
4305f613dfdSUlrich Weigand     MBBI->getOperand(AddrOpNo + 1).ChangeToImmediate(Offset);
4315f613dfdSUlrich Weigand   } else if (StackSize) {
4325f613dfdSUlrich Weigand     DebugLoc DL = MBBI->getDebugLoc();
4335f613dfdSUlrich Weigand     emitIncrement(MBB, MBBI, DL, SystemZ::R15D, StackSize, ZII);
4345f613dfdSUlrich Weigand   }
4355f613dfdSUlrich Weigand }
4365f613dfdSUlrich Weigand 
4375f613dfdSUlrich Weigand bool SystemZFrameLowering::hasFP(const MachineFunction &MF) const {
4385f613dfdSUlrich Weigand   return (MF.getTarget().Options.DisableFramePointerElim(MF) ||
4395f613dfdSUlrich Weigand           MF.getFrameInfo()->hasVarSizedObjects() ||
4405f613dfdSUlrich Weigand           MF.getInfo<SystemZMachineFunctionInfo>()->getManipulatesSP());
4415f613dfdSUlrich Weigand }
4425f613dfdSUlrich Weigand 
4435f613dfdSUlrich Weigand int SystemZFrameLowering::getFrameIndexOffset(const MachineFunction &MF,
4445f613dfdSUlrich Weigand                                               int FI) const {
4455f613dfdSUlrich Weigand   const MachineFrameInfo *MFFrame = MF.getFrameInfo();
4465f613dfdSUlrich Weigand 
4475f613dfdSUlrich Weigand   // Start with the offset of FI from the top of the caller-allocated frame
4485f613dfdSUlrich Weigand   // (i.e. the top of the 160 bytes allocated by the caller).  This initial
4495f613dfdSUlrich Weigand   // offset is therefore negative.
4505f613dfdSUlrich Weigand   int64_t Offset = (MFFrame->getObjectOffset(FI) +
4515f613dfdSUlrich Weigand                     MFFrame->getOffsetAdjustment());
4525f613dfdSUlrich Weigand   if (FI >= 0)
4535f613dfdSUlrich Weigand     // Non-fixed objects are allocated below the incoming stack pointer.
4545f613dfdSUlrich Weigand     // Account for the space at the top of the frame that we choose not
4555f613dfdSUlrich Weigand     // to allocate.
4565f613dfdSUlrich Weigand     Offset += getUnallocatedTopBytes(MF);
4575f613dfdSUlrich Weigand 
4585f613dfdSUlrich Weigand   // Make the offset relative to the incoming stack pointer.
4595f613dfdSUlrich Weigand   Offset -= getOffsetOfLocalArea();
4605f613dfdSUlrich Weigand 
4615f613dfdSUlrich Weigand   // Make the offset relative to the bottom of the frame.
4625f613dfdSUlrich Weigand   Offset += getAllocatedStackSize(MF);
4635f613dfdSUlrich Weigand 
4645f613dfdSUlrich Weigand   return Offset;
4655f613dfdSUlrich Weigand }
4665f613dfdSUlrich Weigand 
4675f613dfdSUlrich Weigand uint64_t SystemZFrameLowering::
4685f613dfdSUlrich Weigand getUnallocatedTopBytes(const MachineFunction &MF) const {
4695f613dfdSUlrich Weigand   return MF.getInfo<SystemZMachineFunctionInfo>()->getSavedGPRFrameSize();
4705f613dfdSUlrich Weigand }
4715f613dfdSUlrich Weigand 
4725f613dfdSUlrich Weigand uint64_t SystemZFrameLowering::
4735f613dfdSUlrich Weigand getAllocatedStackSize(const MachineFunction &MF) const {
4745f613dfdSUlrich Weigand   const MachineFrameInfo *MFFrame = MF.getFrameInfo();
4755f613dfdSUlrich Weigand 
4765f613dfdSUlrich Weigand   // Start with the size of the local variables and spill slots.
4775f613dfdSUlrich Weigand   uint64_t StackSize = MFFrame->getStackSize();
4785f613dfdSUlrich Weigand 
4795f613dfdSUlrich Weigand   // Remove any bytes that we choose not to allocate.
4805f613dfdSUlrich Weigand   StackSize -= getUnallocatedTopBytes(MF);
4815f613dfdSUlrich Weigand 
4825f613dfdSUlrich Weigand   // Include space for an emergency spill slot, if one might be needed.
4835f613dfdSUlrich Weigand   StackSize += getEmergencySpillSlotSize(MF);
4845f613dfdSUlrich Weigand 
4855f613dfdSUlrich Weigand   // We need to allocate the ABI-defined 160-byte base area whenever
4865f613dfdSUlrich Weigand   // we allocate stack space for our own use and whenever we call another
4875f613dfdSUlrich Weigand   // function.
4885f613dfdSUlrich Weigand   if (StackSize || MFFrame->hasVarSizedObjects() || MFFrame->hasCalls())
4895f613dfdSUlrich Weigand     StackSize += SystemZMC::CallFrameSize;
4905f613dfdSUlrich Weigand 
4915f613dfdSUlrich Weigand   return StackSize;
4925f613dfdSUlrich Weigand }
4935f613dfdSUlrich Weigand 
4945f613dfdSUlrich Weigand unsigned SystemZFrameLowering::
4955f613dfdSUlrich Weigand getEmergencySpillSlotSize(const MachineFunction &MF) const {
4965f613dfdSUlrich Weigand   const MachineFrameInfo *MFFrame = MF.getFrameInfo();
4975f613dfdSUlrich Weigand   uint64_t MaxReach = MFFrame->getStackSize() + SystemZMC::CallFrameSize * 2;
4985f613dfdSUlrich Weigand   return isUInt<12>(MaxReach) ? 0 : 8;
4995f613dfdSUlrich Weigand }
5005f613dfdSUlrich Weigand 
5015f613dfdSUlrich Weigand unsigned SystemZFrameLowering::
5025f613dfdSUlrich Weigand getEmergencySpillSlotOffset(const MachineFunction &MF) const {
5035f613dfdSUlrich Weigand   assert(getEmergencySpillSlotSize(MF) && "No emergency spill slot");
5045f613dfdSUlrich Weigand   return SystemZMC::CallFrameSize;
5055f613dfdSUlrich Weigand }
5065f613dfdSUlrich Weigand 
5075f613dfdSUlrich Weigand bool
5085f613dfdSUlrich Weigand SystemZFrameLowering::hasReservedCallFrame(const MachineFunction &MF) const {
5095f613dfdSUlrich Weigand   // The ABI requires us to allocate 160 bytes of stack space for the callee,
5105f613dfdSUlrich Weigand   // with any outgoing stack arguments being placed above that.  It seems
5115f613dfdSUlrich Weigand   // better to make that area a permanent feature of the frame even if
5125f613dfdSUlrich Weigand   // we're using a frame pointer.
5135f613dfdSUlrich Weigand   return true;
5145f613dfdSUlrich Weigand }
5155f613dfdSUlrich Weigand 
5165f613dfdSUlrich Weigand void SystemZFrameLowering::
5175f613dfdSUlrich Weigand eliminateCallFramePseudoInstr(MachineFunction &MF,
5185f613dfdSUlrich Weigand                               MachineBasicBlock &MBB,
5195f613dfdSUlrich Weigand                               MachineBasicBlock::iterator MI) const {
5205f613dfdSUlrich Weigand   switch (MI->getOpcode()) {
5215f613dfdSUlrich Weigand   case SystemZ::ADJCALLSTACKDOWN:
5225f613dfdSUlrich Weigand   case SystemZ::ADJCALLSTACKUP:
5235f613dfdSUlrich Weigand     assert(hasReservedCallFrame(MF) &&
5245f613dfdSUlrich Weigand            "ADJSTACKDOWN and ADJSTACKUP should be no-ops");
5255f613dfdSUlrich Weigand     MBB.erase(MI);
5265f613dfdSUlrich Weigand     break;
5275f613dfdSUlrich Weigand 
5285f613dfdSUlrich Weigand   default:
5295f613dfdSUlrich Weigand     llvm_unreachable("Unexpected call frame instruction");
5305f613dfdSUlrich Weigand   }
5315f613dfdSUlrich Weigand }
532