10b57cec5SDimitry Andric //===-- X86TargetFrameLowering.h - Define frame lowering for X86 -*- C++ -*-==// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric // 90b57cec5SDimitry Andric // This class implements X86-specific bits of TargetFrameLowering class. 100b57cec5SDimitry Andric // 110b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 120b57cec5SDimitry Andric 130b57cec5SDimitry Andric #ifndef LLVM_LIB_TARGET_X86_X86FRAMELOWERING_H 140b57cec5SDimitry Andric #define LLVM_LIB_TARGET_X86_X86FRAMELOWERING_H 150b57cec5SDimitry Andric 164824e7fdSDimitry Andric #include "llvm/CodeGen/MachineFunction.h" 170b57cec5SDimitry Andric #include "llvm/CodeGen/TargetFrameLowering.h" 18e8d8bef9SDimitry Andric #include "llvm/Support/TypeSize.h" 190b57cec5SDimitry Andric 200b57cec5SDimitry Andric namespace llvm { 210b57cec5SDimitry Andric 220b57cec5SDimitry Andric class MachineInstrBuilder; 230b57cec5SDimitry Andric class MCCFIInstruction; 240b57cec5SDimitry Andric class X86InstrInfo; 250b57cec5SDimitry Andric class X86Subtarget; 260b57cec5SDimitry Andric class X86RegisterInfo; 270b57cec5SDimitry Andric 280b57cec5SDimitry Andric class X86FrameLowering : public TargetFrameLowering { 290b57cec5SDimitry Andric public: 308bcb0991SDimitry Andric X86FrameLowering(const X86Subtarget &STI, MaybeAlign StackAlignOverride); 310b57cec5SDimitry Andric 320b57cec5SDimitry Andric // Cached subtarget predicates. 330b57cec5SDimitry Andric 340b57cec5SDimitry Andric const X86Subtarget &STI; 350b57cec5SDimitry Andric const X86InstrInfo &TII; 360b57cec5SDimitry Andric const X86RegisterInfo *TRI; 370b57cec5SDimitry Andric 380b57cec5SDimitry Andric unsigned SlotSize; 390b57cec5SDimitry Andric 400b57cec5SDimitry Andric /// Is64Bit implies that x86_64 instructions are available. 410b57cec5SDimitry Andric bool Is64Bit; 420b57cec5SDimitry Andric 430b57cec5SDimitry Andric bool IsLP64; 440b57cec5SDimitry Andric 450b57cec5SDimitry Andric /// True if the 64-bit frame or stack pointer should be used. True for most 460b57cec5SDimitry Andric /// 64-bit targets with the exception of x32. If this is false, 32-bit 470b57cec5SDimitry Andric /// instruction operands should be used to manipulate StackPtr and FramePtr. 480b57cec5SDimitry Andric bool Uses64BitFramePtr; 490b57cec5SDimitry Andric 500b57cec5SDimitry Andric unsigned StackPtr; 510b57cec5SDimitry Andric 520b57cec5SDimitry Andric /// Emit target stack probe code. This is required for all 530b57cec5SDimitry Andric /// large stack allocations on Windows. The caller is required to materialize 540b57cec5SDimitry Andric /// the number of bytes to probe in RAX/EAX. 554824e7fdSDimitry Andric /// \p InstrNum optionally contains a debug-info instruction number for the 564824e7fdSDimitry Andric /// new stack pointer. 57bdd1243dSDimitry Andric void emitStackProbe(MachineFunction &MF, MachineBasicBlock &MBB, 58bdd1243dSDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 59bdd1243dSDimitry Andric bool InProlog, 60bdd1243dSDimitry Andric std::optional<MachineFunction::DebugInstrOperandPair> 61bdd1243dSDimitry Andric InstrNum = std::nullopt) const; 624824e7fdSDimitry Andric 634824e7fdSDimitry Andric bool stackProbeFunctionModifiesSP() const override; 640b57cec5SDimitry Andric 650b57cec5SDimitry Andric /// Replace a StackProbe inline-stub with the actual probe code inline. 660b57cec5SDimitry Andric void inlineStackProbe(MachineFunction &MF, 670b57cec5SDimitry Andric MachineBasicBlock &PrologMBB) const override; 680b57cec5SDimitry Andric 6904eeddc0SDimitry Andric void emitCalleeSavedFrameMovesFullCFA( 7004eeddc0SDimitry Andric MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI) const override; 715ffd83dbSDimitry Andric 720b57cec5SDimitry Andric void emitCalleeSavedFrameMoves(MachineBasicBlock &MBB, 730b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 74fe6060f1SDimitry Andric const DebugLoc &DL, bool IsPrologue) const; 750b57cec5SDimitry Andric 760b57cec5SDimitry Andric /// emitProlog/emitEpilog - These methods insert prolog and epilog code into 770b57cec5SDimitry Andric /// the function. 780b57cec5SDimitry Andric void emitPrologue(MachineFunction &MF, MachineBasicBlock &MBB) const override; 790b57cec5SDimitry Andric void emitEpilogue(MachineFunction &MF, MachineBasicBlock &MBB) const override; 800b57cec5SDimitry Andric 810b57cec5SDimitry Andric void adjustForSegmentedStacks(MachineFunction &MF, 820b57cec5SDimitry Andric MachineBasicBlock &PrologueMBB) const override; 830b57cec5SDimitry Andric 840b57cec5SDimitry Andric void adjustForHiPEPrologue(MachineFunction &MF, 850b57cec5SDimitry Andric MachineBasicBlock &PrologueMBB) const override; 860b57cec5SDimitry Andric 870b57cec5SDimitry Andric void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, 880b57cec5SDimitry Andric RegScavenger *RS = nullptr) const override; 890b57cec5SDimitry Andric 900b57cec5SDimitry Andric bool 910b57cec5SDimitry Andric assignCalleeSavedSpillSlots(MachineFunction &MF, 920b57cec5SDimitry Andric const TargetRegisterInfo *TRI, 930b57cec5SDimitry Andric std::vector<CalleeSavedInfo> &CSI) const override; 940b57cec5SDimitry Andric 950b57cec5SDimitry Andric bool spillCalleeSavedRegisters(MachineBasicBlock &MBB, 960b57cec5SDimitry Andric MachineBasicBlock::iterator MI, 975ffd83dbSDimitry Andric ArrayRef<CalleeSavedInfo> CSI, 980b57cec5SDimitry Andric const TargetRegisterInfo *TRI) const override; 990b57cec5SDimitry Andric 1005ffd83dbSDimitry Andric bool 1015ffd83dbSDimitry Andric restoreCalleeSavedRegisters(MachineBasicBlock &MBB, 1020b57cec5SDimitry Andric MachineBasicBlock::iterator MI, 1035ffd83dbSDimitry Andric MutableArrayRef<CalleeSavedInfo> CSI, 1040b57cec5SDimitry Andric const TargetRegisterInfo *TRI) const override; 1050b57cec5SDimitry Andric 1060b57cec5SDimitry Andric bool hasFP(const MachineFunction &MF) const override; 1070b57cec5SDimitry Andric bool hasReservedCallFrame(const MachineFunction &MF) const override; 1080b57cec5SDimitry Andric bool canSimplifyCallFramePseudos(const MachineFunction &MF) const override; 1090b57cec5SDimitry Andric bool needsFrameIndexResolution(const MachineFunction &MF) const override; 1100b57cec5SDimitry Andric 111e8d8bef9SDimitry Andric StackOffset getFrameIndexReference(const MachineFunction &MF, int FI, 1125ffd83dbSDimitry Andric Register &FrameReg) const override; 1130b57cec5SDimitry Andric 1145ffd83dbSDimitry Andric int getWin64EHFrameIndexRef(const MachineFunction &MF, int FI, 1155ffd83dbSDimitry Andric Register &SPReg) const; 116e8d8bef9SDimitry Andric StackOffset getFrameIndexReferenceSP(const MachineFunction &MF, int FI, 1175ffd83dbSDimitry Andric Register &SPReg, int Adjustment) const; 118e8d8bef9SDimitry Andric StackOffset 119e8d8bef9SDimitry Andric getFrameIndexReferencePreferSP(const MachineFunction &MF, int FI, 1205ffd83dbSDimitry Andric Register &FrameReg, 1210b57cec5SDimitry Andric bool IgnoreSPUpdates) const override; 1220b57cec5SDimitry Andric 1230b57cec5SDimitry Andric MachineBasicBlock::iterator 1240b57cec5SDimitry Andric eliminateCallFramePseudoInstr(MachineFunction &MF, MachineBasicBlock &MBB, 1250b57cec5SDimitry Andric MachineBasicBlock::iterator MI) const override; 1260b57cec5SDimitry Andric 1270b57cec5SDimitry Andric unsigned getWinEHParentFrameOffset(const MachineFunction &MF) const override; 1280b57cec5SDimitry Andric 1290b57cec5SDimitry Andric void processFunctionBeforeFrameFinalized(MachineFunction &MF, 1300b57cec5SDimitry Andric RegScavenger *RS) const override; 1310b57cec5SDimitry Andric 1325ffd83dbSDimitry Andric void 1335ffd83dbSDimitry Andric processFunctionBeforeFrameIndicesReplaced(MachineFunction &MF, 1345ffd83dbSDimitry Andric RegScavenger *RS) const override; 1355ffd83dbSDimitry Andric 1360b57cec5SDimitry Andric /// Check the instruction before/after the passed instruction. If 1370b57cec5SDimitry Andric /// it is an ADD/SUB/LEA instruction it is deleted argument and the 1380b57cec5SDimitry Andric /// stack adjustment is returned as a positive value for ADD/LEA and 1390b57cec5SDimitry Andric /// a negative for SUB. 1400b57cec5SDimitry Andric int mergeSPUpdates(MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI, 1410b57cec5SDimitry Andric bool doMergeWithPrevious) const; 1420b57cec5SDimitry Andric 1430b57cec5SDimitry Andric /// Emit a series of instructions to increment / decrement the stack 1440b57cec5SDimitry Andric /// pointer by a constant value. 1450b57cec5SDimitry Andric void emitSPUpdate(MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI, 1460b57cec5SDimitry Andric const DebugLoc &DL, int64_t NumBytes, bool InEpilogue) const; 1470b57cec5SDimitry Andric 1480b57cec5SDimitry Andric /// Check that LEA can be used on SP in an epilogue sequence for \p MF. 1490b57cec5SDimitry Andric bool canUseLEAForSPInEpilogue(const MachineFunction &MF) const; 1500b57cec5SDimitry Andric 1510b57cec5SDimitry Andric /// Check whether or not the given \p MBB can be used as a prologue 1520b57cec5SDimitry Andric /// for the target. 1530b57cec5SDimitry Andric /// The prologue will be inserted first in this basic block. 1540b57cec5SDimitry Andric /// This method is used by the shrink-wrapping pass to decide if 1550b57cec5SDimitry Andric /// \p MBB will be correctly handled by the target. 1560b57cec5SDimitry Andric /// As soon as the target enable shrink-wrapping without overriding 1570b57cec5SDimitry Andric /// this method, we assume that each basic block is a valid 1580b57cec5SDimitry Andric /// prologue. 1590b57cec5SDimitry Andric bool canUseAsPrologue(const MachineBasicBlock &MBB) const override; 1600b57cec5SDimitry Andric 1610b57cec5SDimitry Andric /// Check whether or not the given \p MBB can be used as a epilogue 1620b57cec5SDimitry Andric /// for the target. 1630b57cec5SDimitry Andric /// The epilogue will be inserted before the first terminator of that block. 1640b57cec5SDimitry Andric /// This method is used by the shrink-wrapping pass to decide if 1650b57cec5SDimitry Andric /// \p MBB will be correctly handled by the target. 1660b57cec5SDimitry Andric bool canUseAsEpilogue(const MachineBasicBlock &MBB) const override; 1670b57cec5SDimitry Andric 1680b57cec5SDimitry Andric /// Returns true if the target will correctly handle shrink wrapping. 1690b57cec5SDimitry Andric bool enableShrinkWrapping(const MachineFunction &MF) const override; 1700b57cec5SDimitry Andric 1710b57cec5SDimitry Andric /// Order the symbols in the local stack. 1720b57cec5SDimitry Andric /// We want to place the local stack objects in some sort of sensible order. 1730b57cec5SDimitry Andric /// The heuristic we use is to try and pack them according to static number 1740b57cec5SDimitry Andric /// of uses and size in order to minimize code size. 1750b57cec5SDimitry Andric void orderFrameObjects(const MachineFunction &MF, 1760b57cec5SDimitry Andric SmallVectorImpl<int> &ObjectsToAllocate) const override; 1770b57cec5SDimitry Andric 1780b57cec5SDimitry Andric /// Wraps up getting a CFI index and building a MachineInstr for it. 1790b57cec5SDimitry Andric void BuildCFI(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, 18081ad6265SDimitry Andric const DebugLoc &DL, const MCCFIInstruction &CFIInst, 18181ad6265SDimitry Andric MachineInstr::MIFlag Flag = MachineInstr::NoFlags) const; 1820b57cec5SDimitry Andric 1830b57cec5SDimitry Andric /// Sets up EBP and optionally ESI based on the incoming EBP value. Only 1840b57cec5SDimitry Andric /// needed for 32-bit. Used in funclet prologues and at catchret destinations. 1850b57cec5SDimitry Andric MachineBasicBlock::iterator 1860b57cec5SDimitry Andric restoreWin32EHStackPointers(MachineBasicBlock &MBB, 1870b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 1880b57cec5SDimitry Andric const DebugLoc &DL, bool RestoreSP = false) const; 1890b57cec5SDimitry Andric 1905ffd83dbSDimitry Andric void restoreWinEHStackPointersInParent(MachineFunction &MF) const; 1915ffd83dbSDimitry Andric 1920b57cec5SDimitry Andric int getInitialCFAOffset(const MachineFunction &MF) const override; 1930b57cec5SDimitry Andric 1945ffd83dbSDimitry Andric Register getInitialCFARegister(const MachineFunction &MF) const override; 1950b57cec5SDimitry Andric 196*06c3fb27SDimitry Andric DwarfFrameBase getDwarfFrameBase(const MachineFunction &MF) const override; 197*06c3fb27SDimitry Andric 1980b57cec5SDimitry Andric /// Return true if the function has a redzone (accessible bytes past the 1990b57cec5SDimitry Andric /// frame of the top of stack function) as part of it's ABI. 2000b57cec5SDimitry Andric bool has128ByteRedZone(const MachineFunction& MF) const; 2010b57cec5SDimitry Andric 2020b57cec5SDimitry Andric private: 203fe6060f1SDimitry Andric bool isWin64Prologue(const MachineFunction &MF) const; 204fe6060f1SDimitry Andric 205fe6060f1SDimitry Andric bool needsDwarfCFI(const MachineFunction &MF) const; 206fe6060f1SDimitry Andric 2070b57cec5SDimitry Andric uint64_t calculateMaxStackAlign(const MachineFunction &MF) const; 2080b57cec5SDimitry Andric 2090b57cec5SDimitry Andric /// Emit target stack probe as a call to a helper function 2104824e7fdSDimitry Andric void emitStackProbeCall( 2114824e7fdSDimitry Andric MachineFunction &MF, MachineBasicBlock &MBB, 2124824e7fdSDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, bool InProlog, 213bdd1243dSDimitry Andric std::optional<MachineFunction::DebugInstrOperandPair> InstrNum) const; 2140b57cec5SDimitry Andric 2150b57cec5SDimitry Andric /// Emit target stack probe as an inline sequence. 2160b57cec5SDimitry Andric void emitStackProbeInline(MachineFunction &MF, MachineBasicBlock &MBB, 2170b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2180b57cec5SDimitry Andric const DebugLoc &DL, bool InProlog) const; 2195ffd83dbSDimitry Andric void emitStackProbeInlineWindowsCoreCLR64(MachineFunction &MF, 2205ffd83dbSDimitry Andric MachineBasicBlock &MBB, 2215ffd83dbSDimitry Andric MachineBasicBlock::iterator MBBI, 2225ffd83dbSDimitry Andric const DebugLoc &DL, 2235ffd83dbSDimitry Andric bool InProlog) const; 2245ffd83dbSDimitry Andric void emitStackProbeInlineGeneric(MachineFunction &MF, MachineBasicBlock &MBB, 2250b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2260b57cec5SDimitry Andric const DebugLoc &DL, bool InProlog) const; 2270b57cec5SDimitry Andric 2285ffd83dbSDimitry Andric void emitStackProbeInlineGenericBlock(MachineFunction &MF, 2295ffd83dbSDimitry Andric MachineBasicBlock &MBB, 2305ffd83dbSDimitry Andric MachineBasicBlock::iterator MBBI, 231eaeb601bSDimitry Andric const DebugLoc &DL, uint64_t Offset, 232eaeb601bSDimitry Andric uint64_t Align) const; 2335ffd83dbSDimitry Andric 2345ffd83dbSDimitry Andric void emitStackProbeInlineGenericLoop(MachineFunction &MF, 2355ffd83dbSDimitry Andric MachineBasicBlock &MBB, 2365ffd83dbSDimitry Andric MachineBasicBlock::iterator MBBI, 237eaeb601bSDimitry Andric const DebugLoc &DL, uint64_t Offset, 238eaeb601bSDimitry Andric uint64_t Align) const; 2395ffd83dbSDimitry Andric 24081ad6265SDimitry Andric /// Emit target zero call-used regs. 24181ad6265SDimitry Andric void emitZeroCallUsedRegs(BitVector RegsToZero, 24281ad6265SDimitry Andric MachineBasicBlock &MBB) const override; 24381ad6265SDimitry Andric 244e8d8bef9SDimitry Andric void adjustFrameForMsvcCxxEh(MachineFunction &MF) const; 2455ffd83dbSDimitry Andric 2460b57cec5SDimitry Andric /// Aligns the stack pointer by ANDing it with -MaxAlign. 2470b57cec5SDimitry Andric void BuildStackAlignAND(MachineBasicBlock &MBB, 2480b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 2490b57cec5SDimitry Andric unsigned Reg, uint64_t MaxAlign) const; 2500b57cec5SDimitry Andric 2510b57cec5SDimitry Andric /// Make small positive stack adjustments using POPs. 2520b57cec5SDimitry Andric bool adjustStackWithPops(MachineBasicBlock &MBB, 2530b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 2540b57cec5SDimitry Andric int Offset) const; 2550b57cec5SDimitry Andric 2560b57cec5SDimitry Andric /// Adjusts the stack pointer using LEA, SUB, or ADD. 2570b57cec5SDimitry Andric MachineInstrBuilder BuildStackAdjustment(MachineBasicBlock &MBB, 2580b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2590b57cec5SDimitry Andric const DebugLoc &DL, int64_t Offset, 2600b57cec5SDimitry Andric bool InEpilogue) const; 2610b57cec5SDimitry Andric 2620b57cec5SDimitry Andric unsigned getPSPSlotOffsetFromSP(const MachineFunction &MF) const; 2630b57cec5SDimitry Andric 2640b57cec5SDimitry Andric unsigned getWinEHFuncletFrameSize(const MachineFunction &MF) const; 2650b57cec5SDimitry Andric 2660b57cec5SDimitry Andric /// Materialize the catchret target MBB in RAX. 2670b57cec5SDimitry Andric void emitCatchRetReturnValue(MachineBasicBlock &MBB, 2680b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2690b57cec5SDimitry Andric MachineInstr *CatchRet) const; 2700b57cec5SDimitry Andric }; 2710b57cec5SDimitry Andric 2720b57cec5SDimitry Andric } // End llvm namespace 2730b57cec5SDimitry Andric 2740b57cec5SDimitry Andric #endif 275