xref: /freebsd/contrib/llvm-project/llvm/lib/Target/X86/X86MachineFunctionInfo.h (revision 05427f4639bcf2703329a9be9d25ec09bb782742)
1 //===-- X86MachineFunctionInfo.h - X86 machine function info ----*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file declares X86-specific per-machine-function information.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #ifndef LLVM_LIB_TARGET_X86_X86MACHINEFUNCTIONINFO_H
14 #define LLVM_LIB_TARGET_X86_X86MACHINEFUNCTIONINFO_H
15 
16 #include "llvm/ADT/ArrayRef.h"
17 #include "llvm/ADT/SmallVector.h"
18 #include "llvm/CodeGen/CallingConvLower.h"
19 #include "llvm/CodeGen/MIRYamlMapping.h"
20 #include "llvm/CodeGen/MachineFunction.h"
21 #include "llvm/Support/YAMLTraits.h"
22 #include <set>
23 
24 namespace llvm {
25 
26 enum AMXProgModelEnum { None = 0, DirectReg = 1, ManagedRA = 2 };
27 
28 class X86MachineFunctionInfo;
29 
30 namespace yaml {
31 template <> struct ScalarEnumerationTraits<AMXProgModelEnum> {
32   static void enumeration(IO &YamlIO, AMXProgModelEnum &Value) {
33     YamlIO.enumCase(Value, "None", AMXProgModelEnum::None);
34     YamlIO.enumCase(Value, "DirectReg", AMXProgModelEnum::DirectReg);
35     YamlIO.enumCase(Value, "ManagedRA", AMXProgModelEnum::ManagedRA);
36   }
37 };
38 
39 struct X86MachineFunctionInfo final : public yaml::MachineFunctionInfo {
40   AMXProgModelEnum AMXProgModel;
41 
42   X86MachineFunctionInfo() = default;
43   X86MachineFunctionInfo(const llvm::X86MachineFunctionInfo &MFI);
44 
45   void mappingImpl(yaml::IO &YamlIO) override;
46   ~X86MachineFunctionInfo() = default;
47 };
48 
49 template <> struct MappingTraits<X86MachineFunctionInfo> {
50   static void mapping(IO &YamlIO, X86MachineFunctionInfo &MFI) {
51     YamlIO.mapOptional("amxProgModel", MFI.AMXProgModel);
52   }
53 };
54 } // end namespace yaml
55 
56 /// X86MachineFunctionInfo - This class is derived from MachineFunction and
57 /// contains private X86 target-specific information for each MachineFunction.
58 class X86MachineFunctionInfo : public MachineFunctionInfo {
59   virtual void anchor();
60 
61   /// ForceFramePointer - True if the function is required to use of frame
62   /// pointer for reasons other than it containing dynamic allocation or
63   /// that FP eliminatation is turned off. For example, Cygwin main function
64   /// contains stack pointer re-alignment code which requires FP.
65   bool ForceFramePointer = false;
66 
67   /// RestoreBasePointerOffset - Non-zero if the function has base pointer
68   /// and makes call to llvm.eh.sjlj.setjmp. When non-zero, the value is a
69   /// displacement from the frame pointer to a slot where the base pointer
70   /// is stashed.
71   signed char RestoreBasePointerOffset = 0;
72 
73   /// WinEHXMMSlotInfo - Slot information of XMM registers in the stack frame
74   /// in bytes.
75   DenseMap<int, unsigned> WinEHXMMSlotInfo;
76 
77   /// CalleeSavedFrameSize - Size of the callee-saved register portion of the
78   /// stack frame in bytes.
79   unsigned CalleeSavedFrameSize = 0;
80 
81   /// BytesToPopOnReturn - Number of bytes function pops on return (in addition
82   /// to the space used by the return address).
83   /// Used on windows platform for stdcall & fastcall name decoration
84   unsigned BytesToPopOnReturn = 0;
85 
86   /// ReturnAddrIndex - FrameIndex for return slot.
87   int ReturnAddrIndex = 0;
88 
89   /// FrameIndex for return slot.
90   int FrameAddrIndex = 0;
91 
92   /// TailCallReturnAddrDelta - The number of bytes by which return address
93   /// stack slot is moved as the result of tail call optimization.
94   int TailCallReturnAddrDelta = 0;
95 
96   /// SRetReturnReg - Some subtargets require that sret lowering includes
97   /// returning the value of the returned struct in a register. This field
98   /// holds the virtual register into which the sret argument is passed.
99   Register SRetReturnReg;
100 
101   /// GlobalBaseReg - keeps track of the virtual register initialized for
102   /// use as the global base register. This is used for PIC in some PIC
103   /// relocation models.
104   Register GlobalBaseReg;
105 
106   /// VarArgsFrameIndex - FrameIndex for start of varargs area.
107   int VarArgsFrameIndex = 0;
108   /// RegSaveFrameIndex - X86-64 vararg func register save area.
109   int RegSaveFrameIndex = 0;
110   /// VarArgsGPOffset - X86-64 vararg func int reg offset.
111   unsigned VarArgsGPOffset = 0;
112   /// VarArgsFPOffset - X86-64 vararg func fp reg offset.
113   unsigned VarArgsFPOffset = 0;
114   /// ArgumentStackSize - The number of bytes on stack consumed by the arguments
115   /// being passed on the stack.
116   unsigned ArgumentStackSize = 0;
117   /// NumLocalDynamics - Number of local-dynamic TLS accesses.
118   unsigned NumLocalDynamics = 0;
119   /// HasPushSequences - Keeps track of whether this function uses sequences
120   /// of pushes to pass function parameters.
121   bool HasPushSequences = false;
122 
123   /// True if the function recovers from an SEH exception, and therefore needs
124   /// to spill and restore the frame pointer.
125   bool HasSEHFramePtrSave = false;
126 
127   /// The frame index of a stack object containing the original frame pointer
128   /// used to address arguments in a function using a base pointer.
129   int SEHFramePtrSaveIndex = 0;
130 
131   /// The AMX programing model used in the function.
132   AMXProgModelEnum AMXProgModel = AMXProgModelEnum::None;
133 
134   /// True if this function has a subset of CSRs that is handled explicitly via
135   /// copies.
136   bool IsSplitCSR = false;
137 
138   /// True if this function uses the red zone.
139   bool UsesRedZone = false;
140 
141   /// True if this function has DYN_ALLOCA instructions.
142   bool HasDynAlloca = false;
143 
144   /// True if this function has any preallocated calls.
145   bool HasPreallocatedCall = false;
146 
147   /// Whether this function has an extended frame record [Ctx, RBP, Return
148   /// addr]. If so, bit 60 of the in-memory frame pointer will be 1 to enable
149   /// other tools to detect the extended record.
150   bool HasSwiftAsyncContext = false;
151 
152   /// Ajust stack for push2/pop2
153   bool PadForPush2Pop2 = false;
154 
155   /// Candidate registers for push2/pop2
156   std::set<Register> CandidatesForPush2Pop2;
157 
158   /// True if this function has CFI directives that adjust the CFA.
159   /// This is used to determine if we should direct the debugger to use
160   /// the CFA instead of the stack pointer.
161   bool HasCFIAdjustCfa = false;
162 
163   MachineInstr *StackPtrSaveMI = nullptr;
164 
165   std::optional<int> SwiftAsyncContextFrameIdx;
166 
167   // Preallocated fields are only used during isel.
168   // FIXME: Can we find somewhere else to store these?
169   DenseMap<const Value *, size_t> PreallocatedIds;
170   SmallVector<size_t, 0> PreallocatedStackSizes;
171   SmallVector<SmallVector<size_t, 4>, 0> PreallocatedArgOffsets;
172 
173 private:
174   /// ForwardedMustTailRegParms - A list of virtual and physical registers
175   /// that must be forwarded to every musttail call.
176   SmallVector<ForwardedRegister, 1> ForwardedMustTailRegParms;
177 
178 public:
179   X86MachineFunctionInfo() = default;
180   X86MachineFunctionInfo(const Function &F, const TargetSubtargetInfo *STI) {}
181 
182   X86MachineFunctionInfo(const X86MachineFunctionInfo &) = default;
183 
184   MachineFunctionInfo *
185   clone(BumpPtrAllocator &Allocator, MachineFunction &DestMF,
186         const DenseMap<MachineBasicBlock *, MachineBasicBlock *> &Src2DstMBB)
187       const override;
188 
189   void initializeBaseYamlFields(const yaml::X86MachineFunctionInfo &YamlMFI);
190 
191   bool getForceFramePointer() const { return ForceFramePointer;}
192   void setForceFramePointer(bool forceFP) { ForceFramePointer = forceFP; }
193 
194   bool getHasPushSequences() const { return HasPushSequences; }
195   void setHasPushSequences(bool HasPush) { HasPushSequences = HasPush; }
196 
197   bool getRestoreBasePointer() const { return RestoreBasePointerOffset!=0; }
198   void setRestoreBasePointer(const MachineFunction *MF);
199   void setRestoreBasePointer(unsigned CalleeSavedFrameSize) {
200     RestoreBasePointerOffset = -CalleeSavedFrameSize;
201   }
202   int getRestoreBasePointerOffset() const {return RestoreBasePointerOffset; }
203 
204   DenseMap<int, unsigned>& getWinEHXMMSlotInfo() { return WinEHXMMSlotInfo; }
205   const DenseMap<int, unsigned>& getWinEHXMMSlotInfo() const {
206     return WinEHXMMSlotInfo; }
207 
208   unsigned getCalleeSavedFrameSize() const {
209     return CalleeSavedFrameSize + 8 * padForPush2Pop2();
210   }
211   void setCalleeSavedFrameSize(unsigned bytes) { CalleeSavedFrameSize = bytes; }
212 
213   unsigned getBytesToPopOnReturn() const { return BytesToPopOnReturn; }
214   void setBytesToPopOnReturn (unsigned bytes) { BytesToPopOnReturn = bytes;}
215 
216   int getRAIndex() const { return ReturnAddrIndex; }
217   void setRAIndex(int Index) { ReturnAddrIndex = Index; }
218 
219   int getFAIndex() const { return FrameAddrIndex; }
220   void setFAIndex(int Index) { FrameAddrIndex = Index; }
221 
222   int getTCReturnAddrDelta() const { return TailCallReturnAddrDelta; }
223   void setTCReturnAddrDelta(int delta) {TailCallReturnAddrDelta = delta;}
224 
225   Register getSRetReturnReg() const { return SRetReturnReg; }
226   void setSRetReturnReg(Register Reg) { SRetReturnReg = Reg; }
227 
228   Register getGlobalBaseReg() const { return GlobalBaseReg; }
229   void setGlobalBaseReg(Register Reg) { GlobalBaseReg = Reg; }
230 
231   int getVarArgsFrameIndex() const { return VarArgsFrameIndex; }
232   void setVarArgsFrameIndex(int Idx) { VarArgsFrameIndex = Idx; }
233 
234   int getRegSaveFrameIndex() const { return RegSaveFrameIndex; }
235   void setRegSaveFrameIndex(int Idx) { RegSaveFrameIndex = Idx; }
236 
237   unsigned getVarArgsGPOffset() const { return VarArgsGPOffset; }
238   void setVarArgsGPOffset(unsigned Offset) { VarArgsGPOffset = Offset; }
239 
240   unsigned getVarArgsFPOffset() const { return VarArgsFPOffset; }
241   void setVarArgsFPOffset(unsigned Offset) { VarArgsFPOffset = Offset; }
242 
243   unsigned getArgumentStackSize() const { return ArgumentStackSize; }
244   void setArgumentStackSize(unsigned size) { ArgumentStackSize = size; }
245 
246   unsigned getNumLocalDynamicTLSAccesses() const { return NumLocalDynamics; }
247   void incNumLocalDynamicTLSAccesses() { ++NumLocalDynamics; }
248 
249   bool getHasSEHFramePtrSave() const { return HasSEHFramePtrSave; }
250   void setHasSEHFramePtrSave(bool V) { HasSEHFramePtrSave = V; }
251 
252   int getSEHFramePtrSaveIndex() const { return SEHFramePtrSaveIndex; }
253   void setSEHFramePtrSaveIndex(int Index) { SEHFramePtrSaveIndex = Index; }
254 
255   AMXProgModelEnum getAMXProgModel() const { return AMXProgModel; }
256   void setAMXProgModel(AMXProgModelEnum Model) {
257     assert((AMXProgModel == AMXProgModelEnum::None || AMXProgModel == Model) &&
258            "mixed model is not supported");
259     AMXProgModel = Model;
260   }
261 
262   SmallVectorImpl<ForwardedRegister> &getForwardedMustTailRegParms() {
263     return ForwardedMustTailRegParms;
264   }
265 
266   bool isSplitCSR() const { return IsSplitCSR; }
267   void setIsSplitCSR(bool s) { IsSplitCSR = s; }
268 
269   bool getUsesRedZone() const { return UsesRedZone; }
270   void setUsesRedZone(bool V) { UsesRedZone = V; }
271 
272   bool hasDynAlloca() const { return HasDynAlloca; }
273   void setHasDynAlloca(bool v) { HasDynAlloca = v; }
274 
275   bool hasPreallocatedCall() const { return HasPreallocatedCall; }
276   void setHasPreallocatedCall(bool v) { HasPreallocatedCall = v; }
277 
278   bool hasSwiftAsyncContext() const { return HasSwiftAsyncContext; }
279   void setHasSwiftAsyncContext(bool v) { HasSwiftAsyncContext = v; }
280 
281   bool padForPush2Pop2() const { return PadForPush2Pop2; }
282   void setPadForPush2Pop2(bool V) { PadForPush2Pop2 = V; }
283 
284   bool isCandidateForPush2Pop2(Register Reg) const {
285     return CandidatesForPush2Pop2.find(Reg) != CandidatesForPush2Pop2.end();
286   }
287   void addCandidateForPush2Pop2(Register Reg) {
288     CandidatesForPush2Pop2.insert(Reg);
289   }
290   size_t getNumCandidatesForPush2Pop2() const {
291     return CandidatesForPush2Pop2.size();
292   }
293 
294   bool hasCFIAdjustCfa() const { return HasCFIAdjustCfa; }
295   void setHasCFIAdjustCfa(bool v) { HasCFIAdjustCfa = v; }
296 
297   void setStackPtrSaveMI(MachineInstr *MI) { StackPtrSaveMI = MI; }
298   MachineInstr *getStackPtrSaveMI() const { return StackPtrSaveMI; }
299 
300   std::optional<int> getSwiftAsyncContextFrameIdx() const {
301     return SwiftAsyncContextFrameIdx;
302   }
303   void setSwiftAsyncContextFrameIdx(int v) { SwiftAsyncContextFrameIdx = v; }
304 
305   size_t getPreallocatedIdForCallSite(const Value *CS) {
306     auto Insert = PreallocatedIds.insert({CS, PreallocatedIds.size()});
307     if (Insert.second) {
308       PreallocatedStackSizes.push_back(0);
309       PreallocatedArgOffsets.emplace_back();
310     }
311     return Insert.first->second;
312   }
313 
314   void setPreallocatedStackSize(size_t Id, size_t StackSize) {
315     PreallocatedStackSizes[Id] = StackSize;
316   }
317 
318   size_t getPreallocatedStackSize(const size_t Id) {
319     assert(PreallocatedStackSizes[Id] != 0 && "stack size not set");
320     return PreallocatedStackSizes[Id];
321   }
322 
323   void setPreallocatedArgOffsets(size_t Id, ArrayRef<size_t> AO) {
324     PreallocatedArgOffsets[Id].assign(AO.begin(), AO.end());
325   }
326 
327   ArrayRef<size_t> getPreallocatedArgOffsets(const size_t Id) {
328     assert(!PreallocatedArgOffsets[Id].empty() && "arg offsets not set");
329     return PreallocatedArgOffsets[Id];
330   }
331 };
332 
333 } // End llvm namespace
334 
335 #endif
336