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"
13f1bd22dfSEric Christopher #include "SystemZInstrInfo.h"
145f613dfdSUlrich Weigand #include "SystemZMachineFunctionInfo.h"
15f1bd22dfSEric Christopher #include "SystemZRegisterInfo.h"
16d913448bSEric Christopher #include "SystemZSubtarget.h"
175f613dfdSUlrich Weigand #include "llvm/CodeGen/MachineModuleInfo.h"
185f613dfdSUlrich Weigand #include "llvm/CodeGen/MachineRegisterInfo.h"
195dd52f8cSRichard Sandiford #include "llvm/CodeGen/RegisterScavenging.h"
205f613dfdSUlrich Weigand #include "llvm/IR/Function.h"
215f613dfdSUlrich Weigand 
225f613dfdSUlrich Weigand using namespace llvm;
235f613dfdSUlrich Weigand 
24db39b4a2SRichard Sandiford namespace {
255f613dfdSUlrich Weigand // The ABI-defined register save slots, relative to the incoming stack
265f613dfdSUlrich Weigand // pointer.
27db39b4a2SRichard Sandiford static const TargetFrameLowering::SpillSlot SpillOffsetTable[] = {
285f613dfdSUlrich Weigand   { SystemZ::R2D,  0x10 },
295f613dfdSUlrich Weigand   { SystemZ::R3D,  0x18 },
305f613dfdSUlrich Weigand   { SystemZ::R4D,  0x20 },
315f613dfdSUlrich Weigand   { SystemZ::R5D,  0x28 },
325f613dfdSUlrich Weigand   { SystemZ::R6D,  0x30 },
335f613dfdSUlrich Weigand   { SystemZ::R7D,  0x38 },
345f613dfdSUlrich Weigand   { SystemZ::R8D,  0x40 },
355f613dfdSUlrich Weigand   { SystemZ::R9D,  0x48 },
365f613dfdSUlrich Weigand   { SystemZ::R10D, 0x50 },
375f613dfdSUlrich Weigand   { SystemZ::R11D, 0x58 },
385f613dfdSUlrich Weigand   { SystemZ::R12D, 0x60 },
395f613dfdSUlrich Weigand   { SystemZ::R13D, 0x68 },
405f613dfdSUlrich Weigand   { SystemZ::R14D, 0x70 },
415f613dfdSUlrich Weigand   { SystemZ::R15D, 0x78 },
425f613dfdSUlrich Weigand   { SystemZ::F0D,  0x80 },
435f613dfdSUlrich Weigand   { SystemZ::F2D,  0x88 },
445f613dfdSUlrich Weigand   { SystemZ::F4D,  0x90 },
455f613dfdSUlrich Weigand   { SystemZ::F6D,  0x98 }
465f613dfdSUlrich Weigand };
47c231269fSRichard Sandiford } // end anonymous namespace
485f613dfdSUlrich Weigand 
49f1bd22dfSEric Christopher SystemZFrameLowering::SystemZFrameLowering()
50db39b4a2SRichard Sandiford     : TargetFrameLowering(TargetFrameLowering::StackGrowsDown, 8,
51*f12b925bSJonas Paulsson                           -SystemZMC::CallFrameSize, 8,
52*f12b925bSJonas Paulsson                           false /* StackRealignable */) {
535f613dfdSUlrich Weigand   // Create a mapping from register number to save slot offset.
545f613dfdSUlrich Weigand   RegSpillOffsets.grow(SystemZ::NUM_TARGET_REGS);
555f613dfdSUlrich Weigand   for (unsigned I = 0, E = array_lengthof(SpillOffsetTable); I != E; ++I)
56db39b4a2SRichard Sandiford     RegSpillOffsets[SpillOffsetTable[I].Reg] = SpillOffsetTable[I].Offset;
57db39b4a2SRichard Sandiford }
58db39b4a2SRichard Sandiford 
59db39b4a2SRichard Sandiford const TargetFrameLowering::SpillSlot *
60db39b4a2SRichard Sandiford SystemZFrameLowering::getCalleeSavedSpillSlots(unsigned &NumEntries) const {
61db39b4a2SRichard Sandiford   NumEntries = array_lengthof(SpillOffsetTable);
62db39b4a2SRichard Sandiford   return SpillOffsetTable;
635f613dfdSUlrich Weigand }
645f613dfdSUlrich Weigand 
6502564865SMatthias Braun void SystemZFrameLowering::determineCalleeSaves(MachineFunction &MF,
6602564865SMatthias Braun                                                 BitVector &SavedRegs,
675f613dfdSUlrich Weigand                                                 RegScavenger *RS) const {
6802564865SMatthias Braun   TargetFrameLowering::determineCalleeSaves(MF, SavedRegs, RS);
6902564865SMatthias Braun 
705f613dfdSUlrich Weigand   MachineFrameInfo *MFFrame = MF.getFrameInfo();
71fc6de428SEric Christopher   const TargetRegisterInfo *TRI = MF.getSubtarget().getRegisterInfo();
725f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
735f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *MFI = MF.getInfo<SystemZMachineFunctionInfo>();
745f613dfdSUlrich Weigand   bool IsVarArg = MF.getFunction()->isVarArg();
755f613dfdSUlrich Weigand 
765f613dfdSUlrich Weigand   // va_start stores incoming FPR varargs in the normal way, but delegates
775f613dfdSUlrich Weigand   // the saving of incoming GPR varargs to spillCalleeSavedRegisters().
785f613dfdSUlrich Weigand   // Record these pending uses, which typically include the call-saved
795f613dfdSUlrich Weigand   // argument register R6D.
805f613dfdSUlrich Weigand   if (IsVarArg)
815f613dfdSUlrich Weigand     for (unsigned I = MFI->getVarArgsFirstGPR(); I < SystemZ::NumArgGPRs; ++I)
8202564865SMatthias Braun       SavedRegs.set(SystemZ::ArgGPRs[I]);
835f613dfdSUlrich Weigand 
845f613dfdSUlrich Weigand   // If the function requires a frame pointer, record that the hard
855f613dfdSUlrich Weigand   // frame pointer will be clobbered.
865f613dfdSUlrich Weigand   if (HasFP)
8702564865SMatthias Braun     SavedRegs.set(SystemZ::R11D);
885f613dfdSUlrich Weigand 
895f613dfdSUlrich Weigand   // If the function calls other functions, record that the return
905f613dfdSUlrich Weigand   // address register will be clobbered.
915f613dfdSUlrich Weigand   if (MFFrame->hasCalls())
9202564865SMatthias Braun     SavedRegs.set(SystemZ::R14D);
935f613dfdSUlrich Weigand 
945f613dfdSUlrich Weigand   // If we are saving GPRs other than the stack pointer, we might as well
955f613dfdSUlrich Weigand   // save and restore the stack pointer at the same time, via STMG and LMG.
965f613dfdSUlrich Weigand   // This allows the deallocation to be done by the LMG, rather than needing
975f613dfdSUlrich Weigand   // a separate %r15 addition.
98840beec2SCraig Topper   const MCPhysReg *CSRegs = TRI->getCalleeSavedRegs(&MF);
995f613dfdSUlrich Weigand   for (unsigned I = 0; CSRegs[I]; ++I) {
1005f613dfdSUlrich Weigand     unsigned Reg = CSRegs[I];
10102564865SMatthias Braun     if (SystemZ::GR64BitRegClass.contains(Reg) && SavedRegs.test(Reg)) {
10202564865SMatthias Braun       SavedRegs.set(SystemZ::R15D);
1035f613dfdSUlrich Weigand       break;
1045f613dfdSUlrich Weigand     }
1055f613dfdSUlrich Weigand   }
1065f613dfdSUlrich Weigand }
1075f613dfdSUlrich Weigand 
1085f613dfdSUlrich Weigand // Add GPR64 to the save instruction being built by MIB, which is in basic
1095f613dfdSUlrich Weigand // block MBB.  IsImplicit says whether this is an explicit operand to the
1105f613dfdSUlrich Weigand // instruction, or an implicit one that comes between the explicit start
1115f613dfdSUlrich Weigand // and end registers.
1125f613dfdSUlrich Weigand static void addSavedGPR(MachineBasicBlock &MBB, MachineInstrBuilder &MIB,
1135f613dfdSUlrich Weigand                         unsigned GPR64, bool IsImplicit) {
114d913448bSEric Christopher   const TargetRegisterInfo *RI =
115fc6de428SEric Christopher       MBB.getParent()->getSubtarget().getRegisterInfo();
11687a44364SRichard Sandiford   unsigned GPR32 = RI->getSubReg(GPR64, SystemZ::subreg_l32);
1175f613dfdSUlrich Weigand   bool IsLive = MBB.isLiveIn(GPR64) || MBB.isLiveIn(GPR32);
1185f613dfdSUlrich Weigand   if (!IsLive || !IsImplicit) {
1195f613dfdSUlrich Weigand     MIB.addReg(GPR64, getImplRegState(IsImplicit) | getKillRegState(!IsLive));
1205f613dfdSUlrich Weigand     if (!IsLive)
1215f613dfdSUlrich Weigand       MBB.addLiveIn(GPR64);
1225f613dfdSUlrich Weigand   }
1235f613dfdSUlrich Weigand }
1245f613dfdSUlrich Weigand 
1255f613dfdSUlrich Weigand bool SystemZFrameLowering::
1265f613dfdSUlrich Weigand spillCalleeSavedRegisters(MachineBasicBlock &MBB,
1275f613dfdSUlrich Weigand                           MachineBasicBlock::iterator MBBI,
1285f613dfdSUlrich Weigand                           const std::vector<CalleeSavedInfo> &CSI,
1295f613dfdSUlrich Weigand                           const TargetRegisterInfo *TRI) const {
1305f613dfdSUlrich Weigand   if (CSI.empty())
1315f613dfdSUlrich Weigand     return false;
1325f613dfdSUlrich Weigand 
1335f613dfdSUlrich Weigand   MachineFunction &MF = *MBB.getParent();
134fc6de428SEric Christopher   const TargetInstrInfo *TII = MF.getSubtarget().getInstrInfo();
1355f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
1365f613dfdSUlrich Weigand   bool IsVarArg = MF.getFunction()->isVarArg();
137057c5a6bSOleg Ranevskyy   DebugLoc DL;
1385f613dfdSUlrich Weigand 
1395f613dfdSUlrich Weigand   // Scan the call-saved GPRs and find the bounds of the register spill area.
1405f613dfdSUlrich Weigand   unsigned LowGPR = 0;
1415f613dfdSUlrich Weigand   unsigned HighGPR = SystemZ::R15D;
1425f613dfdSUlrich Weigand   unsigned StartOffset = -1U;
1435f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
1445f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
1455f613dfdSUlrich Weigand     if (SystemZ::GR64BitRegClass.contains(Reg)) {
1465f613dfdSUlrich Weigand       unsigned Offset = RegSpillOffsets[Reg];
1475f613dfdSUlrich Weigand       assert(Offset && "Unexpected GPR save");
1485f613dfdSUlrich Weigand       if (StartOffset > Offset) {
1495f613dfdSUlrich Weigand         LowGPR = Reg;
1505f613dfdSUlrich Weigand         StartOffset = Offset;
1515f613dfdSUlrich Weigand       }
1525f613dfdSUlrich Weigand     }
1535f613dfdSUlrich Weigand   }
1545f613dfdSUlrich Weigand 
155db39b4a2SRichard Sandiford   // Save the range of call-saved registers, for use by the epilogue inserter.
1565f613dfdSUlrich Weigand   ZFI->setLowSavedGPR(LowGPR);
1575f613dfdSUlrich Weigand   ZFI->setHighSavedGPR(HighGPR);
1585f613dfdSUlrich Weigand 
1595f613dfdSUlrich Weigand   // Include the GPR varargs, if any.  R6D is call-saved, so would
1605f613dfdSUlrich Weigand   // be included by the loop above, but we also need to handle the
1615f613dfdSUlrich Weigand   // call-clobbered argument registers.
1625f613dfdSUlrich Weigand   if (IsVarArg) {
1635f613dfdSUlrich Weigand     unsigned FirstGPR = ZFI->getVarArgsFirstGPR();
1645f613dfdSUlrich Weigand     if (FirstGPR < SystemZ::NumArgGPRs) {
1655f613dfdSUlrich Weigand       unsigned Reg = SystemZ::ArgGPRs[FirstGPR];
1665f613dfdSUlrich Weigand       unsigned Offset = RegSpillOffsets[Reg];
1675f613dfdSUlrich Weigand       if (StartOffset > Offset) {
1685f613dfdSUlrich Weigand         LowGPR = Reg; StartOffset = Offset;
1695f613dfdSUlrich Weigand       }
1705f613dfdSUlrich Weigand     }
1715f613dfdSUlrich Weigand   }
1725f613dfdSUlrich Weigand 
1735f613dfdSUlrich Weigand   // Save GPRs
1745f613dfdSUlrich Weigand   if (LowGPR) {
1755f613dfdSUlrich Weigand     assert(LowGPR != HighGPR && "Should be saving %r15 and something else");
1765f613dfdSUlrich Weigand 
1775f613dfdSUlrich Weigand     // Build an STMG instruction.
1785f613dfdSUlrich Weigand     MachineInstrBuilder MIB = BuildMI(MBB, MBBI, DL, TII->get(SystemZ::STMG));
1795f613dfdSUlrich Weigand 
1805f613dfdSUlrich Weigand     // Add the explicit register operands.
181f1bd22dfSEric Christopher     addSavedGPR(MBB, MIB, LowGPR, false);
182f1bd22dfSEric Christopher     addSavedGPR(MBB, MIB, HighGPR, false);
1835f613dfdSUlrich Weigand 
1845f613dfdSUlrich Weigand     // Add the address.
1855f613dfdSUlrich Weigand     MIB.addReg(SystemZ::R15D).addImm(StartOffset);
1865f613dfdSUlrich Weigand 
1875f613dfdSUlrich Weigand     // Make sure all call-saved GPRs are included as operands and are
1885f613dfdSUlrich Weigand     // marked as live on entry.
1895f613dfdSUlrich Weigand     for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
1905f613dfdSUlrich Weigand       unsigned Reg = CSI[I].getReg();
1915f613dfdSUlrich Weigand       if (SystemZ::GR64BitRegClass.contains(Reg))
192f1bd22dfSEric Christopher         addSavedGPR(MBB, MIB, Reg, true);
1935f613dfdSUlrich Weigand     }
1945f613dfdSUlrich Weigand 
1955f613dfdSUlrich Weigand     // ...likewise GPR varargs.
1965f613dfdSUlrich Weigand     if (IsVarArg)
1975f613dfdSUlrich Weigand       for (unsigned I = ZFI->getVarArgsFirstGPR(); I < SystemZ::NumArgGPRs; ++I)
198f1bd22dfSEric Christopher         addSavedGPR(MBB, MIB, SystemZ::ArgGPRs[I], true);
1995f613dfdSUlrich Weigand   }
2005f613dfdSUlrich Weigand 
2015f613dfdSUlrich Weigand   // Save FPRs in the normal TargetInstrInfo way.
2025f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
2035f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
2045f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg)) {
2055f613dfdSUlrich Weigand       MBB.addLiveIn(Reg);
2065f613dfdSUlrich Weigand       TII->storeRegToStackSlot(MBB, MBBI, Reg, true, CSI[I].getFrameIdx(),
2075f613dfdSUlrich Weigand                                &SystemZ::FP64BitRegClass, TRI);
2085f613dfdSUlrich Weigand     }
2095f613dfdSUlrich Weigand   }
2105f613dfdSUlrich Weigand 
2115f613dfdSUlrich Weigand   return true;
2125f613dfdSUlrich Weigand }
2135f613dfdSUlrich Weigand 
2145f613dfdSUlrich Weigand bool SystemZFrameLowering::
2155f613dfdSUlrich Weigand restoreCalleeSavedRegisters(MachineBasicBlock &MBB,
2165f613dfdSUlrich Weigand                             MachineBasicBlock::iterator MBBI,
2175f613dfdSUlrich Weigand                             const std::vector<CalleeSavedInfo> &CSI,
2185f613dfdSUlrich Weigand                             const TargetRegisterInfo *TRI) const {
2195f613dfdSUlrich Weigand   if (CSI.empty())
2205f613dfdSUlrich Weigand     return false;
2215f613dfdSUlrich Weigand 
2225f613dfdSUlrich Weigand   MachineFunction &MF = *MBB.getParent();
223fc6de428SEric Christopher   const TargetInstrInfo *TII = MF.getSubtarget().getInstrInfo();
2245f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
2255f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
2265f613dfdSUlrich Weigand   DebugLoc DL = MBBI != MBB.end() ? MBBI->getDebugLoc() : DebugLoc();
2275f613dfdSUlrich Weigand 
2285f613dfdSUlrich Weigand   // Restore FPRs in the normal TargetInstrInfo way.
2295f613dfdSUlrich Weigand   for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
2305f613dfdSUlrich Weigand     unsigned Reg = CSI[I].getReg();
2315f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg))
2325f613dfdSUlrich Weigand       TII->loadRegFromStackSlot(MBB, MBBI, Reg, CSI[I].getFrameIdx(),
2335f613dfdSUlrich Weigand                                 &SystemZ::FP64BitRegClass, TRI);
2345f613dfdSUlrich Weigand   }
2355f613dfdSUlrich Weigand 
2365f613dfdSUlrich Weigand   // Restore call-saved GPRs (but not call-clobbered varargs, which at
2375f613dfdSUlrich Weigand   // this point might hold return values).
2385f613dfdSUlrich Weigand   unsigned LowGPR = ZFI->getLowSavedGPR();
2395f613dfdSUlrich Weigand   unsigned HighGPR = ZFI->getHighSavedGPR();
2405f613dfdSUlrich Weigand   unsigned StartOffset = RegSpillOffsets[LowGPR];
2415f613dfdSUlrich Weigand   if (LowGPR) {
2425f613dfdSUlrich Weigand     // If we saved any of %r2-%r5 as varargs, we should also be saving
2435f613dfdSUlrich Weigand     // and restoring %r6.  If we're saving %r6 or above, we should be
2445f613dfdSUlrich Weigand     // restoring it too.
2455f613dfdSUlrich Weigand     assert(LowGPR != HighGPR && "Should be loading %r15 and something else");
2465f613dfdSUlrich Weigand 
2475f613dfdSUlrich Weigand     // Build an LMG instruction.
2485f613dfdSUlrich Weigand     MachineInstrBuilder MIB = BuildMI(MBB, MBBI, DL, TII->get(SystemZ::LMG));
2495f613dfdSUlrich Weigand 
2505f613dfdSUlrich Weigand     // Add the explicit register operands.
2515f613dfdSUlrich Weigand     MIB.addReg(LowGPR, RegState::Define);
2525f613dfdSUlrich Weigand     MIB.addReg(HighGPR, RegState::Define);
2535f613dfdSUlrich Weigand 
2545f613dfdSUlrich Weigand     // Add the address.
2555f613dfdSUlrich Weigand     MIB.addReg(HasFP ? SystemZ::R11D : SystemZ::R15D);
2565f613dfdSUlrich Weigand     MIB.addImm(StartOffset);
2575f613dfdSUlrich Weigand 
2585f613dfdSUlrich Weigand     // Do a second scan adding regs as being defined by instruction
2595f613dfdSUlrich Weigand     for (unsigned I = 0, E = CSI.size(); I != E; ++I) {
2605f613dfdSUlrich Weigand       unsigned Reg = CSI[I].getReg();
2615f613dfdSUlrich Weigand       if (Reg != LowGPR && Reg != HighGPR)
2625f613dfdSUlrich Weigand         MIB.addReg(Reg, RegState::ImplicitDefine);
2635f613dfdSUlrich Weigand     }
2645f613dfdSUlrich Weigand   }
2655f613dfdSUlrich Weigand 
2665f613dfdSUlrich Weigand   return true;
2675f613dfdSUlrich Weigand }
2685f613dfdSUlrich Weigand 
2695dd52f8cSRichard Sandiford void SystemZFrameLowering::
2705dd52f8cSRichard Sandiford processFunctionBeforeFrameFinalized(MachineFunction &MF,
2715dd52f8cSRichard Sandiford                                     RegScavenger *RS) const {
2725dd52f8cSRichard Sandiford   MachineFrameInfo *MFFrame = MF.getFrameInfo();
2735dd52f8cSRichard Sandiford   uint64_t MaxReach = (MFFrame->estimateStackSize(MF) +
2745dd52f8cSRichard Sandiford                        SystemZMC::CallFrameSize * 2);
27523943229SRichard Sandiford   if (!isUInt<12>(MaxReach)) {
27623943229SRichard Sandiford     // We may need register scavenging slots if some parts of the frame
2775dd52f8cSRichard Sandiford     // are outside the reach of an unsigned 12-bit displacement.
27823943229SRichard Sandiford     // Create 2 for the case where both addresses in an MVC are
27923943229SRichard Sandiford     // out of range.
2805dd52f8cSRichard Sandiford     RS->addScavengingFrameIndex(MFFrame->CreateStackObject(8, 8, false));
28123943229SRichard Sandiford     RS->addScavengingFrameIndex(MFFrame->CreateStackObject(8, 8, false));
28223943229SRichard Sandiford   }
2835dd52f8cSRichard Sandiford }
2845dd52f8cSRichard Sandiford 
2855f613dfdSUlrich Weigand // Emit instructions before MBBI (in MBB) to add NumBytes to Reg.
2865f613dfdSUlrich Weigand static void emitIncrement(MachineBasicBlock &MBB,
2875f613dfdSUlrich Weigand                           MachineBasicBlock::iterator &MBBI,
2885f613dfdSUlrich Weigand                           const DebugLoc &DL,
2895f613dfdSUlrich Weigand                           unsigned Reg, int64_t NumBytes,
2905f613dfdSUlrich Weigand                           const TargetInstrInfo *TII) {
2915f613dfdSUlrich Weigand   while (NumBytes) {
2925f613dfdSUlrich Weigand     unsigned Opcode;
2935f613dfdSUlrich Weigand     int64_t ThisVal = NumBytes;
2945f613dfdSUlrich Weigand     if (isInt<16>(NumBytes))
2955f613dfdSUlrich Weigand       Opcode = SystemZ::AGHI;
2965f613dfdSUlrich Weigand     else {
2975f613dfdSUlrich Weigand       Opcode = SystemZ::AGFI;
2985f613dfdSUlrich Weigand       // Make sure we maintain 8-byte stack alignment.
299fffd56ecSAlexey Samsonov       int64_t MinVal = -uint64_t(1) << 31;
3005f613dfdSUlrich Weigand       int64_t MaxVal = (int64_t(1) << 31) - 8;
3015f613dfdSUlrich Weigand       if (ThisVal < MinVal)
3025f613dfdSUlrich Weigand         ThisVal = MinVal;
3035f613dfdSUlrich Weigand       else if (ThisVal > MaxVal)
3045f613dfdSUlrich Weigand         ThisVal = MaxVal;
3055f613dfdSUlrich Weigand     }
3065f613dfdSUlrich Weigand     MachineInstr *MI = BuildMI(MBB, MBBI, DL, TII->get(Opcode), Reg)
3075f613dfdSUlrich Weigand       .addReg(Reg).addImm(ThisVal);
30814a44495SRichard Sandiford     // The CC implicit def is dead.
3095f613dfdSUlrich Weigand     MI->getOperand(3).setIsDead();
3105f613dfdSUlrich Weigand     NumBytes -= ThisVal;
3115f613dfdSUlrich Weigand   }
3125f613dfdSUlrich Weigand }
3135f613dfdSUlrich Weigand 
31461b305edSQuentin Colombet void SystemZFrameLowering::emitPrologue(MachineFunction &MF,
31561b305edSQuentin Colombet                                         MachineBasicBlock &MBB) const {
31661b305edSQuentin Colombet   assert(&MF.front() == &MBB && "Shrink-wrapping not yet supported");
3175f613dfdSUlrich Weigand   MachineFrameInfo *MFFrame = MF.getFrameInfo();
318fc6de428SEric Christopher   auto *ZII =
319fc6de428SEric Christopher       static_cast<const SystemZInstrInfo *>(MF.getSubtarget().getInstrInfo());
3205f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
3215f613dfdSUlrich Weigand   MachineBasicBlock::iterator MBBI = MBB.begin();
3225f613dfdSUlrich Weigand   MachineModuleInfo &MMI = MF.getMMI();
323bc07a890SBill Wendling   const MCRegisterInfo *MRI = MMI.getContext().getRegisterInfo();
3245f613dfdSUlrich Weigand   const std::vector<CalleeSavedInfo> &CSI = MFFrame->getCalleeSavedInfo();
3255f613dfdSUlrich Weigand   bool HasFP = hasFP(MF);
326057c5a6bSOleg Ranevskyy 
327057c5a6bSOleg Ranevskyy   // Debug location must be unknown since the first debug location is used
328057c5a6bSOleg Ranevskyy   // to determine the end of the prologue.
329057c5a6bSOleg Ranevskyy   DebugLoc DL;
3305f613dfdSUlrich Weigand 
3315f613dfdSUlrich Weigand   // The current offset of the stack pointer from the CFA.
3325f613dfdSUlrich Weigand   int64_t SPOffsetFromCFA = -SystemZMC::CFAOffsetFromInitialSP;
3335f613dfdSUlrich Weigand 
3345f613dfdSUlrich Weigand   if (ZFI->getLowSavedGPR()) {
3355f613dfdSUlrich Weigand     // Skip over the GPR saves.
3365f613dfdSUlrich Weigand     if (MBBI != MBB.end() && MBBI->getOpcode() == SystemZ::STMG)
3375f613dfdSUlrich Weigand       ++MBBI;
3385f613dfdSUlrich Weigand     else
3395f613dfdSUlrich Weigand       llvm_unreachable("Couldn't skip over GPR saves");
3405f613dfdSUlrich Weigand 
3415f613dfdSUlrich Weigand     // Add CFI for the GPR saves.
34228c111ecSRichard Sandiford     for (auto &Save : CSI) {
34328c111ecSRichard Sandiford       unsigned Reg = Save.getReg();
3445f613dfdSUlrich Weigand       if (SystemZ::GR64BitRegClass.contains(Reg)) {
3455f613dfdSUlrich Weigand         int64_t Offset = SPOffsetFromCFA + RegSpillOffsets[Reg];
346b1f25f1bSRafael Espindola         unsigned CFIIndex = MMI.addFrameInst(MCCFIInstruction::createOffset(
347b1f25f1bSRafael Espindola             nullptr, MRI->getDwarfRegNum(Reg, true), Offset));
348b1f25f1bSRafael Espindola         BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::CFI_INSTRUCTION))
349b1f25f1bSRafael Espindola             .addCFIIndex(CFIIndex);
3505f613dfdSUlrich Weigand       }
3515f613dfdSUlrich Weigand     }
3525f613dfdSUlrich Weigand   }
3535f613dfdSUlrich Weigand 
3545f613dfdSUlrich Weigand   uint64_t StackSize = getAllocatedStackSize(MF);
3555f613dfdSUlrich Weigand   if (StackSize) {
3565f613dfdSUlrich Weigand     // Allocate StackSize bytes.
3575f613dfdSUlrich Weigand     int64_t Delta = -int64_t(StackSize);
3585f613dfdSUlrich Weigand     emitIncrement(MBB, MBBI, DL, SystemZ::R15D, Delta, ZII);
3595f613dfdSUlrich Weigand 
3605f613dfdSUlrich Weigand     // Add CFI for the allocation.
361b1f25f1bSRafael Espindola     unsigned CFIIndex = MMI.addFrameInst(
362b1f25f1bSRafael Espindola         MCCFIInstruction::createDefCfaOffset(nullptr, SPOffsetFromCFA + Delta));
363b1f25f1bSRafael Espindola     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::CFI_INSTRUCTION))
364b1f25f1bSRafael Espindola         .addCFIIndex(CFIIndex);
3655f613dfdSUlrich Weigand     SPOffsetFromCFA += Delta;
3665f613dfdSUlrich Weigand   }
3675f613dfdSUlrich Weigand 
3685f613dfdSUlrich Weigand   if (HasFP) {
3695f613dfdSUlrich Weigand     // Copy the base of the frame to R11.
3705f613dfdSUlrich Weigand     BuildMI(MBB, MBBI, DL, ZII->get(SystemZ::LGR), SystemZ::R11D)
3715f613dfdSUlrich Weigand       .addReg(SystemZ::R15D);
3725f613dfdSUlrich Weigand 
3735f613dfdSUlrich Weigand     // Add CFI for the new frame location.
374bc07a890SBill Wendling     unsigned HardFP = MRI->getDwarfRegNum(SystemZ::R11D, true);
375b1f25f1bSRafael Espindola     unsigned CFIIndex = MMI.addFrameInst(
376b1f25f1bSRafael Espindola         MCCFIInstruction::createDefCfaRegister(nullptr, HardFP));
377b1f25f1bSRafael Espindola     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::CFI_INSTRUCTION))
378b1f25f1bSRafael Espindola         .addCFIIndex(CFIIndex);
3795f613dfdSUlrich Weigand 
3805f613dfdSUlrich Weigand     // Mark the FramePtr as live at the beginning of every block except
3815f613dfdSUlrich Weigand     // the entry block.  (We'll have marked R11 as live on entry when
3825f613dfdSUlrich Weigand     // saving the GPRs.)
38328c111ecSRichard Sandiford     for (auto I = std::next(MF.begin()), E = MF.end(); I != E; ++I)
3845f613dfdSUlrich Weigand       I->addLiveIn(SystemZ::R11D);
3855f613dfdSUlrich Weigand   }
3865f613dfdSUlrich Weigand 
3875f613dfdSUlrich Weigand   // Skip over the FPR saves.
388b1f25f1bSRafael Espindola   SmallVector<unsigned, 8> CFIIndexes;
38928c111ecSRichard Sandiford   for (auto &Save : CSI) {
39028c111ecSRichard Sandiford     unsigned Reg = Save.getReg();
3915f613dfdSUlrich Weigand     if (SystemZ::FP64BitRegClass.contains(Reg)) {
3925f613dfdSUlrich Weigand       if (MBBI != MBB.end() &&
3935f613dfdSUlrich Weigand           (MBBI->getOpcode() == SystemZ::STD ||
3945f613dfdSUlrich Weigand            MBBI->getOpcode() == SystemZ::STDY))
3955f613dfdSUlrich Weigand         ++MBBI;
3965f613dfdSUlrich Weigand       else
3975f613dfdSUlrich Weigand         llvm_unreachable("Couldn't skip over FPR save");
3985f613dfdSUlrich Weigand 
3995f613dfdSUlrich Weigand       // Add CFI for the this save.
40028c111ecSRichard Sandiford       unsigned DwarfReg = MRI->getDwarfRegNum(Reg, true);
4015567bafeSJames Y Knight       unsigned IgnoredFrameReg;
4025567bafeSJames Y Knight       int64_t Offset =
4035567bafeSJames Y Knight           getFrameIndexReference(MF, Save.getFrameIdx(), IgnoredFrameReg);
4045567bafeSJames Y Knight 
405b1f25f1bSRafael Espindola       unsigned CFIIndex = MMI.addFrameInst(MCCFIInstruction::createOffset(
406b1f25f1bSRafael Espindola           nullptr, DwarfReg, SPOffsetFromCFA + Offset));
407b1f25f1bSRafael Espindola       CFIIndexes.push_back(CFIIndex);
4085f613dfdSUlrich Weigand     }
4095f613dfdSUlrich Weigand   }
4105f613dfdSUlrich Weigand   // Complete the CFI for the FPR saves, modelling them as taking effect
4115f613dfdSUlrich Weigand   // after the last save.
412b1f25f1bSRafael Espindola   for (auto CFIIndex : CFIIndexes) {
413b1f25f1bSRafael Espindola     BuildMI(MBB, MBBI, DL, ZII->get(TargetOpcode::CFI_INSTRUCTION))
414b1f25f1bSRafael Espindola         .addCFIIndex(CFIIndex);
415b1f25f1bSRafael Espindola   }
4165f613dfdSUlrich Weigand }
4175f613dfdSUlrich Weigand 
4185f613dfdSUlrich Weigand void SystemZFrameLowering::emitEpilogue(MachineFunction &MF,
4195f613dfdSUlrich Weigand                                         MachineBasicBlock &MBB) const {
4205f613dfdSUlrich Weigand   MachineBasicBlock::iterator MBBI = MBB.getLastNonDebugInstr();
421fc6de428SEric Christopher   auto *ZII =
422fc6de428SEric Christopher       static_cast<const SystemZInstrInfo *>(MF.getSubtarget().getInstrInfo());
4235f613dfdSUlrich Weigand   SystemZMachineFunctionInfo *ZFI = MF.getInfo<SystemZMachineFunctionInfo>();
4245f613dfdSUlrich Weigand 
4255f613dfdSUlrich Weigand   // Skip the return instruction.
426709bda66SRichard Sandiford   assert(MBBI->isReturn() && "Can only insert epilogue into returning blocks");
4275f613dfdSUlrich Weigand 
4285f613dfdSUlrich Weigand   uint64_t StackSize = getAllocatedStackSize(MF);
4295f613dfdSUlrich Weigand   if (ZFI->getLowSavedGPR()) {
4305f613dfdSUlrich Weigand     --MBBI;
4315f613dfdSUlrich Weigand     unsigned Opcode = MBBI->getOpcode();
4325f613dfdSUlrich Weigand     if (Opcode != SystemZ::LMG)
4335f613dfdSUlrich Weigand       llvm_unreachable("Expected to see callee-save register restore code");
4345f613dfdSUlrich Weigand 
4355f613dfdSUlrich Weigand     unsigned AddrOpNo = 2;
4365f613dfdSUlrich Weigand     DebugLoc DL = MBBI->getDebugLoc();
4375f613dfdSUlrich Weigand     uint64_t Offset = StackSize + MBBI->getOperand(AddrOpNo + 1).getImm();
4385f613dfdSUlrich Weigand     unsigned NewOpcode = ZII->getOpcodeForOffset(Opcode, Offset);
4395f613dfdSUlrich Weigand 
4405f613dfdSUlrich Weigand     // If the offset is too large, use the largest stack-aligned offset
4415f613dfdSUlrich Weigand     // and add the rest to the base register (the stack or frame pointer).
4425f613dfdSUlrich Weigand     if (!NewOpcode) {
4435f613dfdSUlrich Weigand       uint64_t NumBytes = Offset - 0x7fff8;
4445f613dfdSUlrich Weigand       emitIncrement(MBB, MBBI, DL, MBBI->getOperand(AddrOpNo).getReg(),
4455f613dfdSUlrich Weigand                     NumBytes, ZII);
4465f613dfdSUlrich Weigand       Offset -= NumBytes;
4475f613dfdSUlrich Weigand       NewOpcode = ZII->getOpcodeForOffset(Opcode, Offset);
4485f613dfdSUlrich Weigand       assert(NewOpcode && "No restore instruction available");
4495f613dfdSUlrich Weigand     }
4505f613dfdSUlrich Weigand 
4515f613dfdSUlrich Weigand     MBBI->setDesc(ZII->get(NewOpcode));
4525f613dfdSUlrich Weigand     MBBI->getOperand(AddrOpNo + 1).ChangeToImmediate(Offset);
4535f613dfdSUlrich Weigand   } else if (StackSize) {
4545f613dfdSUlrich Weigand     DebugLoc DL = MBBI->getDebugLoc();
4555f613dfdSUlrich Weigand     emitIncrement(MBB, MBBI, DL, SystemZ::R15D, StackSize, ZII);
4565f613dfdSUlrich Weigand   }
4575f613dfdSUlrich Weigand }
4585f613dfdSUlrich Weigand 
4595f613dfdSUlrich Weigand bool SystemZFrameLowering::hasFP(const MachineFunction &MF) const {
4605f613dfdSUlrich Weigand   return (MF.getTarget().Options.DisableFramePointerElim(MF) ||
4615f613dfdSUlrich Weigand           MF.getFrameInfo()->hasVarSizedObjects() ||
4625f613dfdSUlrich Weigand           MF.getInfo<SystemZMachineFunctionInfo>()->getManipulatesSP());
4635f613dfdSUlrich Weigand }
4645f613dfdSUlrich Weigand 
4655567bafeSJames Y Knight int SystemZFrameLowering::getFrameIndexReference(const MachineFunction &MF,
4665567bafeSJames Y Knight                                                  int FI,
4675567bafeSJames Y Knight                                                  unsigned &FrameReg) const {
4685f613dfdSUlrich Weigand   const MachineFrameInfo *MFFrame = MF.getFrameInfo();
4695567bafeSJames Y Knight   const TargetRegisterInfo *RI = MF.getSubtarget().getRegisterInfo();
4705567bafeSJames Y Knight 
4715567bafeSJames Y Knight   // Fill in FrameReg output argument.
4725567bafeSJames Y Knight   FrameReg = RI->getFrameRegister(MF);
4735f613dfdSUlrich Weigand 
4745f613dfdSUlrich Weigand   // Start with the offset of FI from the top of the caller-allocated frame
4755f613dfdSUlrich Weigand   // (i.e. the top of the 160 bytes allocated by the caller).  This initial
4765f613dfdSUlrich Weigand   // offset is therefore negative.
4775f613dfdSUlrich Weigand   int64_t Offset = (MFFrame->getObjectOffset(FI) +
4785f613dfdSUlrich Weigand                     MFFrame->getOffsetAdjustment());
4795f613dfdSUlrich Weigand 
4805f613dfdSUlrich Weigand   // Make the offset relative to the incoming stack pointer.
4815f613dfdSUlrich Weigand   Offset -= getOffsetOfLocalArea();
4825f613dfdSUlrich Weigand 
4835f613dfdSUlrich Weigand   // Make the offset relative to the bottom of the frame.
4845f613dfdSUlrich Weigand   Offset += getAllocatedStackSize(MF);
4855f613dfdSUlrich Weigand 
4865f613dfdSUlrich Weigand   return Offset;
4875f613dfdSUlrich Weigand }
4885f613dfdSUlrich Weigand 
4895f613dfdSUlrich Weigand uint64_t SystemZFrameLowering::
4905f613dfdSUlrich Weigand getAllocatedStackSize(const MachineFunction &MF) const {
4915f613dfdSUlrich Weigand   const MachineFrameInfo *MFFrame = MF.getFrameInfo();
4925f613dfdSUlrich Weigand 
4935f613dfdSUlrich Weigand   // Start with the size of the local variables and spill slots.
4945f613dfdSUlrich Weigand   uint64_t StackSize = MFFrame->getStackSize();
4955f613dfdSUlrich Weigand 
4965f613dfdSUlrich Weigand   // We need to allocate the ABI-defined 160-byte base area whenever
4975f613dfdSUlrich Weigand   // we allocate stack space for our own use and whenever we call another
4985f613dfdSUlrich Weigand   // function.
4995f613dfdSUlrich Weigand   if (StackSize || MFFrame->hasVarSizedObjects() || MFFrame->hasCalls())
5005f613dfdSUlrich Weigand     StackSize += SystemZMC::CallFrameSize;
5015f613dfdSUlrich Weigand 
5025f613dfdSUlrich Weigand   return StackSize;
5035f613dfdSUlrich Weigand }
5045f613dfdSUlrich Weigand 
5055f613dfdSUlrich Weigand bool
5065f613dfdSUlrich Weigand SystemZFrameLowering::hasReservedCallFrame(const MachineFunction &MF) const {
5075f613dfdSUlrich Weigand   // The ABI requires us to allocate 160 bytes of stack space for the callee,
5085f613dfdSUlrich Weigand   // with any outgoing stack arguments being placed above that.  It seems
5095f613dfdSUlrich Weigand   // better to make that area a permanent feature of the frame even if
5105f613dfdSUlrich Weigand   // we're using a frame pointer.
5115f613dfdSUlrich Weigand   return true;
5125f613dfdSUlrich Weigand }
5135f613dfdSUlrich Weigand 
5145f613dfdSUlrich Weigand void SystemZFrameLowering::
5155f613dfdSUlrich Weigand eliminateCallFramePseudoInstr(MachineFunction &MF,
5165f613dfdSUlrich Weigand                               MachineBasicBlock &MBB,
5175f613dfdSUlrich Weigand                               MachineBasicBlock::iterator MI) const {
5185f613dfdSUlrich Weigand   switch (MI->getOpcode()) {
5195f613dfdSUlrich Weigand   case SystemZ::ADJCALLSTACKDOWN:
5205f613dfdSUlrich Weigand   case SystemZ::ADJCALLSTACKUP:
5215f613dfdSUlrich Weigand     assert(hasReservedCallFrame(MF) &&
5225f613dfdSUlrich Weigand            "ADJSTACKDOWN and ADJSTACKUP should be no-ops");
5235f613dfdSUlrich Weigand     MBB.erase(MI);
5245f613dfdSUlrich Weigand     break;
5255f613dfdSUlrich Weigand 
5265f613dfdSUlrich Weigand   default:
5275f613dfdSUlrich Weigand     llvm_unreachable("Unexpected call frame instruction");
5285f613dfdSUlrich Weigand   }
5295f613dfdSUlrich Weigand }
530