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 160b57cec5SDimitry Andric #include "llvm/CodeGen/TargetFrameLowering.h" 170b57cec5SDimitry Andric 180b57cec5SDimitry Andric namespace llvm { 190b57cec5SDimitry Andric 200b57cec5SDimitry Andric class MachineInstrBuilder; 210b57cec5SDimitry Andric class MCCFIInstruction; 220b57cec5SDimitry Andric class X86InstrInfo; 230b57cec5SDimitry Andric class X86Subtarget; 240b57cec5SDimitry Andric class X86RegisterInfo; 250b57cec5SDimitry Andric 260b57cec5SDimitry Andric class X86FrameLowering : public TargetFrameLowering { 270b57cec5SDimitry Andric public: 28*8bcb0991SDimitry Andric X86FrameLowering(const X86Subtarget &STI, MaybeAlign StackAlignOverride); 290b57cec5SDimitry Andric 300b57cec5SDimitry Andric // Cached subtarget predicates. 310b57cec5SDimitry Andric 320b57cec5SDimitry Andric const X86Subtarget &STI; 330b57cec5SDimitry Andric const X86InstrInfo &TII; 340b57cec5SDimitry Andric const X86RegisterInfo *TRI; 350b57cec5SDimitry Andric 360b57cec5SDimitry Andric unsigned SlotSize; 370b57cec5SDimitry Andric 380b57cec5SDimitry Andric /// Is64Bit implies that x86_64 instructions are available. 390b57cec5SDimitry Andric bool Is64Bit; 400b57cec5SDimitry Andric 410b57cec5SDimitry Andric bool IsLP64; 420b57cec5SDimitry Andric 430b57cec5SDimitry Andric /// True if the 64-bit frame or stack pointer should be used. True for most 440b57cec5SDimitry Andric /// 64-bit targets with the exception of x32. If this is false, 32-bit 450b57cec5SDimitry Andric /// instruction operands should be used to manipulate StackPtr and FramePtr. 460b57cec5SDimitry Andric bool Uses64BitFramePtr; 470b57cec5SDimitry Andric 480b57cec5SDimitry Andric unsigned StackPtr; 490b57cec5SDimitry Andric 500b57cec5SDimitry Andric /// Emit target stack probe code. This is required for all 510b57cec5SDimitry Andric /// large stack allocations on Windows. The caller is required to materialize 520b57cec5SDimitry Andric /// the number of bytes to probe in RAX/EAX. 530b57cec5SDimitry Andric void emitStackProbe(MachineFunction &MF, MachineBasicBlock &MBB, 540b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 550b57cec5SDimitry Andric bool InProlog) const; 560b57cec5SDimitry Andric 570b57cec5SDimitry Andric /// Replace a StackProbe inline-stub with the actual probe code inline. 580b57cec5SDimitry Andric void inlineStackProbe(MachineFunction &MF, 590b57cec5SDimitry Andric MachineBasicBlock &PrologMBB) const override; 600b57cec5SDimitry Andric 610b57cec5SDimitry Andric void emitCalleeSavedFrameMoves(MachineBasicBlock &MBB, 620b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 630b57cec5SDimitry Andric const DebugLoc &DL) const; 640b57cec5SDimitry Andric 650b57cec5SDimitry Andric /// emitProlog/emitEpilog - These methods insert prolog and epilog code into 660b57cec5SDimitry Andric /// the function. 670b57cec5SDimitry Andric void emitPrologue(MachineFunction &MF, MachineBasicBlock &MBB) const override; 680b57cec5SDimitry Andric void emitEpilogue(MachineFunction &MF, MachineBasicBlock &MBB) const override; 690b57cec5SDimitry Andric 700b57cec5SDimitry Andric void adjustForSegmentedStacks(MachineFunction &MF, 710b57cec5SDimitry Andric MachineBasicBlock &PrologueMBB) const override; 720b57cec5SDimitry Andric 730b57cec5SDimitry Andric void adjustForHiPEPrologue(MachineFunction &MF, 740b57cec5SDimitry Andric MachineBasicBlock &PrologueMBB) const override; 750b57cec5SDimitry Andric 760b57cec5SDimitry Andric void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, 770b57cec5SDimitry Andric RegScavenger *RS = nullptr) const override; 780b57cec5SDimitry Andric 790b57cec5SDimitry Andric bool 800b57cec5SDimitry Andric assignCalleeSavedSpillSlots(MachineFunction &MF, 810b57cec5SDimitry Andric const TargetRegisterInfo *TRI, 820b57cec5SDimitry Andric std::vector<CalleeSavedInfo> &CSI) const override; 830b57cec5SDimitry Andric 840b57cec5SDimitry Andric bool spillCalleeSavedRegisters(MachineBasicBlock &MBB, 850b57cec5SDimitry Andric MachineBasicBlock::iterator MI, 860b57cec5SDimitry Andric const std::vector<CalleeSavedInfo> &CSI, 870b57cec5SDimitry Andric const TargetRegisterInfo *TRI) const override; 880b57cec5SDimitry Andric 890b57cec5SDimitry Andric bool restoreCalleeSavedRegisters(MachineBasicBlock &MBB, 900b57cec5SDimitry Andric MachineBasicBlock::iterator MI, 910b57cec5SDimitry Andric std::vector<CalleeSavedInfo> &CSI, 920b57cec5SDimitry Andric const TargetRegisterInfo *TRI) const override; 930b57cec5SDimitry Andric 940b57cec5SDimitry Andric bool hasFP(const MachineFunction &MF) const override; 950b57cec5SDimitry Andric bool hasReservedCallFrame(const MachineFunction &MF) const override; 960b57cec5SDimitry Andric bool canSimplifyCallFramePseudos(const MachineFunction &MF) const override; 970b57cec5SDimitry Andric bool needsFrameIndexResolution(const MachineFunction &MF) const override; 980b57cec5SDimitry Andric 990b57cec5SDimitry Andric int getFrameIndexReference(const MachineFunction &MF, int FI, 1000b57cec5SDimitry Andric unsigned &FrameReg) const override; 1010b57cec5SDimitry Andric 102c14a5a88SDimitry Andric int getWin64EHFrameIndexRef(const MachineFunction &MF, 103c14a5a88SDimitry Andric int FI, unsigned &SPReg) const; 1040b57cec5SDimitry Andric int getFrameIndexReferenceSP(const MachineFunction &MF, 1050b57cec5SDimitry Andric int FI, unsigned &SPReg, int Adjustment) const; 1060b57cec5SDimitry Andric int getFrameIndexReferencePreferSP(const MachineFunction &MF, int FI, 1070b57cec5SDimitry Andric unsigned &FrameReg, 1080b57cec5SDimitry Andric bool IgnoreSPUpdates) const override; 1090b57cec5SDimitry Andric 1100b57cec5SDimitry Andric MachineBasicBlock::iterator 1110b57cec5SDimitry Andric eliminateCallFramePseudoInstr(MachineFunction &MF, MachineBasicBlock &MBB, 1120b57cec5SDimitry Andric MachineBasicBlock::iterator MI) const override; 1130b57cec5SDimitry Andric 1140b57cec5SDimitry Andric unsigned getWinEHParentFrameOffset(const MachineFunction &MF) const override; 1150b57cec5SDimitry Andric 1160b57cec5SDimitry Andric void processFunctionBeforeFrameFinalized(MachineFunction &MF, 1170b57cec5SDimitry Andric RegScavenger *RS) const override; 1180b57cec5SDimitry Andric 1190b57cec5SDimitry Andric /// Check the instruction before/after the passed instruction. If 1200b57cec5SDimitry Andric /// it is an ADD/SUB/LEA instruction it is deleted argument and the 1210b57cec5SDimitry Andric /// stack adjustment is returned as a positive value for ADD/LEA and 1220b57cec5SDimitry Andric /// a negative for SUB. 1230b57cec5SDimitry Andric int mergeSPUpdates(MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI, 1240b57cec5SDimitry Andric bool doMergeWithPrevious) const; 1250b57cec5SDimitry Andric 1260b57cec5SDimitry Andric /// Emit a series of instructions to increment / decrement the stack 1270b57cec5SDimitry Andric /// pointer by a constant value. 1280b57cec5SDimitry Andric void emitSPUpdate(MachineBasicBlock &MBB, MachineBasicBlock::iterator &MBBI, 1290b57cec5SDimitry Andric const DebugLoc &DL, int64_t NumBytes, bool InEpilogue) const; 1300b57cec5SDimitry Andric 1310b57cec5SDimitry Andric /// Check that LEA can be used on SP in an epilogue sequence for \p MF. 1320b57cec5SDimitry Andric bool canUseLEAForSPInEpilogue(const MachineFunction &MF) const; 1330b57cec5SDimitry Andric 1340b57cec5SDimitry Andric /// Check whether or not the given \p MBB can be used as a prologue 1350b57cec5SDimitry Andric /// for the target. 1360b57cec5SDimitry Andric /// The prologue will be inserted first in this basic block. 1370b57cec5SDimitry Andric /// This method is used by the shrink-wrapping pass to decide if 1380b57cec5SDimitry Andric /// \p MBB will be correctly handled by the target. 1390b57cec5SDimitry Andric /// As soon as the target enable shrink-wrapping without overriding 1400b57cec5SDimitry Andric /// this method, we assume that each basic block is a valid 1410b57cec5SDimitry Andric /// prologue. 1420b57cec5SDimitry Andric bool canUseAsPrologue(const MachineBasicBlock &MBB) const override; 1430b57cec5SDimitry Andric 1440b57cec5SDimitry Andric /// Check whether or not the given \p MBB can be used as a epilogue 1450b57cec5SDimitry Andric /// for the target. 1460b57cec5SDimitry Andric /// The epilogue will be inserted before the first terminator of that block. 1470b57cec5SDimitry Andric /// This method is used by the shrink-wrapping pass to decide if 1480b57cec5SDimitry Andric /// \p MBB will be correctly handled by the target. 1490b57cec5SDimitry Andric bool canUseAsEpilogue(const MachineBasicBlock &MBB) const override; 1500b57cec5SDimitry Andric 1510b57cec5SDimitry Andric /// Returns true if the target will correctly handle shrink wrapping. 1520b57cec5SDimitry Andric bool enableShrinkWrapping(const MachineFunction &MF) const override; 1530b57cec5SDimitry Andric 1540b57cec5SDimitry Andric /// Order the symbols in the local stack. 1550b57cec5SDimitry Andric /// We want to place the local stack objects in some sort of sensible order. 1560b57cec5SDimitry Andric /// The heuristic we use is to try and pack them according to static number 1570b57cec5SDimitry Andric /// of uses and size in order to minimize code size. 1580b57cec5SDimitry Andric void orderFrameObjects(const MachineFunction &MF, 1590b57cec5SDimitry Andric SmallVectorImpl<int> &ObjectsToAllocate) const override; 1600b57cec5SDimitry Andric 1610b57cec5SDimitry Andric /// Wraps up getting a CFI index and building a MachineInstr for it. 1620b57cec5SDimitry Andric void BuildCFI(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, 1630b57cec5SDimitry Andric const DebugLoc &DL, const MCCFIInstruction &CFIInst) const; 1640b57cec5SDimitry Andric 1650b57cec5SDimitry Andric /// Sets up EBP and optionally ESI based on the incoming EBP value. Only 1660b57cec5SDimitry Andric /// needed for 32-bit. Used in funclet prologues and at catchret destinations. 1670b57cec5SDimitry Andric MachineBasicBlock::iterator 1680b57cec5SDimitry Andric restoreWin32EHStackPointers(MachineBasicBlock &MBB, 1690b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 1700b57cec5SDimitry Andric const DebugLoc &DL, bool RestoreSP = false) const; 1710b57cec5SDimitry Andric 1720b57cec5SDimitry Andric int getInitialCFAOffset(const MachineFunction &MF) const override; 1730b57cec5SDimitry Andric 1740b57cec5SDimitry Andric unsigned getInitialCFARegister(const MachineFunction &MF) const override; 1750b57cec5SDimitry Andric 1760b57cec5SDimitry Andric /// Return true if the function has a redzone (accessible bytes past the 1770b57cec5SDimitry Andric /// frame of the top of stack function) as part of it's ABI. 1780b57cec5SDimitry Andric bool has128ByteRedZone(const MachineFunction& MF) const; 1790b57cec5SDimitry Andric 1800b57cec5SDimitry Andric private: 1810b57cec5SDimitry Andric uint64_t calculateMaxStackAlign(const MachineFunction &MF) const; 1820b57cec5SDimitry Andric 1830b57cec5SDimitry Andric /// Emit target stack probe as a call to a helper function 1840b57cec5SDimitry Andric void emitStackProbeCall(MachineFunction &MF, MachineBasicBlock &MBB, 1850b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 1860b57cec5SDimitry Andric bool InProlog) const; 1870b57cec5SDimitry Andric 1880b57cec5SDimitry Andric /// Emit target stack probe as an inline sequence. 1890b57cec5SDimitry Andric void emitStackProbeInline(MachineFunction &MF, MachineBasicBlock &MBB, 1900b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 1910b57cec5SDimitry Andric const DebugLoc &DL, bool InProlog) const; 1920b57cec5SDimitry Andric 1930b57cec5SDimitry Andric /// Emit a stub to later inline the target stack probe. 1940b57cec5SDimitry Andric void emitStackProbeInlineStub(MachineFunction &MF, MachineBasicBlock &MBB, 1950b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 1960b57cec5SDimitry Andric const DebugLoc &DL, bool InProlog) const; 1970b57cec5SDimitry Andric 1980b57cec5SDimitry Andric /// Aligns the stack pointer by ANDing it with -MaxAlign. 1990b57cec5SDimitry Andric void BuildStackAlignAND(MachineBasicBlock &MBB, 2000b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 2010b57cec5SDimitry Andric unsigned Reg, uint64_t MaxAlign) const; 2020b57cec5SDimitry Andric 2030b57cec5SDimitry Andric /// Make small positive stack adjustments using POPs. 2040b57cec5SDimitry Andric bool adjustStackWithPops(MachineBasicBlock &MBB, 2050b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, const DebugLoc &DL, 2060b57cec5SDimitry Andric int Offset) const; 2070b57cec5SDimitry Andric 2080b57cec5SDimitry Andric /// Adjusts the stack pointer using LEA, SUB, or ADD. 2090b57cec5SDimitry Andric MachineInstrBuilder BuildStackAdjustment(MachineBasicBlock &MBB, 2100b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2110b57cec5SDimitry Andric const DebugLoc &DL, int64_t Offset, 2120b57cec5SDimitry Andric bool InEpilogue) const; 2130b57cec5SDimitry Andric 2140b57cec5SDimitry Andric unsigned getPSPSlotOffsetFromSP(const MachineFunction &MF) const; 2150b57cec5SDimitry Andric 2160b57cec5SDimitry Andric unsigned getWinEHFuncletFrameSize(const MachineFunction &MF) const; 2170b57cec5SDimitry Andric 2180b57cec5SDimitry Andric /// Materialize the catchret target MBB in RAX. 2190b57cec5SDimitry Andric void emitCatchRetReturnValue(MachineBasicBlock &MBB, 2200b57cec5SDimitry Andric MachineBasicBlock::iterator MBBI, 2210b57cec5SDimitry Andric MachineInstr *CatchRet) const; 2220b57cec5SDimitry Andric }; 2230b57cec5SDimitry Andric 2240b57cec5SDimitry Andric } // End llvm namespace 2250b57cec5SDimitry Andric 2260b57cec5SDimitry Andric #endif 227