LLVM 20.0.0git
StackFrameLayoutAnalysisPass.cpp
Go to the documentation of this file.
1//===-- StackFrameLayoutAnalysisPass.cpp
2//------------------------------------===//
3//
4// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5// See https://llvm.org/LICENSE.txt for license information.
6// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7//
8//===----------------------------------------------------------------------===//
9//
10// StackFrameLayoutAnalysisPass implementation. Outputs information about the
11// layout of the stack frame, using the remarks interface. On the CLI it prints
12// a textual representation of the stack frame. When possible it prints the
13// values that occupy a stack slot using any available debug information. Since
14// output is remarks based, it is also available in a machine readable file
15// format, such as YAML.
16//
17//===----------------------------------------------------------------------===//
18
19#include "llvm/ADT/SetVector.h"
25#include "llvm/CodeGen/Passes.h"
31#include "llvm/IR/PrintPasses.h"
33#include "llvm/Support/Debug.h"
36
37#include <sstream>
38
39using namespace llvm;
40
41#define DEBUG_TYPE "stack-frame-layout"
42
43namespace {
44
45/// StackFrameLayoutAnalysisPass - This is a pass to dump the stack frame of a
46/// MachineFunction.
47///
48struct StackFrameLayoutAnalysisPass : public MachineFunctionPass {
50 static char ID;
51
52 enum SlotType {
53 Spill, // a Spill slot
54 Fixed, // a Fixed slot (e.g. arguments passed on the stack)
55 VariableSized, // a variable sized object
56 StackProtector, // Stack Protector slot
57 Variable, // a slot used to store a local data (could be a tmp)
58 Invalid // It's an error for a slot to have this type
59 };
60
61 struct SlotData {
62 int Slot;
63 int Size;
64 int Align;
65 StackOffset Offset;
66 SlotType SlotTy;
67 bool Scalable;
68
69 SlotData(const MachineFrameInfo &MFI, const StackOffset Offset,
70 const int Idx)
71 : Slot(Idx), Size(MFI.getObjectSize(Idx)),
72 Align(MFI.getObjectAlign(Idx).value()), Offset(Offset),
73 SlotTy(Invalid), Scalable(false) {
76 SlotTy = SlotType::Spill;
77 else if (MFI.isFixedObjectIndex(Idx))
78 SlotTy = SlotType::Fixed;
79 else if (MFI.isVariableSizedObjectIndex(Idx))
80 SlotTy = SlotType::VariableSized;
81 else if (MFI.hasStackProtectorIndex() &&
83 SlotTy = SlotType::StackProtector;
84 else
85 SlotTy = SlotType::Variable;
86 }
87
88 bool isVarSize() const { return SlotTy == SlotType::VariableSized; }
89
90 // We use this to sort in reverse order, so that the layout is displayed
91 // correctly. Variable sized slots are sorted to the end of the list, as
92 // offsets are currently incorrect for these but they reside at the end of
93 // the stack frame. The Slot index is used to ensure deterministic order
94 // when offsets are equal.
95 bool operator<(const SlotData &Rhs) const {
96 return std::make_tuple(!isVarSize(),
97 Offset.getFixed() + Offset.getScalable(), Slot) >
98 std::make_tuple(!Rhs.isVarSize(),
99 Rhs.Offset.getFixed() + Rhs.Offset.getScalable(),
100 Rhs.Slot);
101 }
102 };
103
104 StackFrameLayoutAnalysisPass() : MachineFunctionPass(ID) {}
105
106 StringRef getPassName() const override {
107 return "Stack Frame Layout Analysis";
108 }
109
110 void getAnalysisUsage(AnalysisUsage &AU) const override {
111 AU.setPreservesAll();
114 }
115
116 bool runOnMachineFunction(MachineFunction &MF) override {
117 // TODO: We should implement a similar filter for remarks:
118 // -Rpass-func-filter=<regex>
120 return false;
121
122 LLVMContext &Ctx = MF.getFunction().getContext();
124 return false;
125
128 &MF.front());
129 Rem << ("\nFunction: " + MF.getName()).str();
130 emitStackFrameLayoutRemarks(MF, Rem);
131 getAnalysis<MachineOptimizationRemarkEmitterPass>().getORE().emit(Rem);
132 return false;
133 }
134
135 std::string getTypeString(SlotType Ty) {
136 switch (Ty) {
137 case SlotType::Spill:
138 return "Spill";
139 case SlotType::Fixed:
140 return "Fixed";
141 case SlotType::VariableSized:
142 return "VariableSized";
143 case SlotType::StackProtector:
144 return "Protector";
145 case SlotType::Variable:
146 return "Variable";
147 default:
148 llvm_unreachable("bad slot type for stack layout");
149 }
150 }
151
152 void emitStackSlotRemark(const MachineFunction &MF, const SlotData &D,
154 // To make it easy to understand the stack layout from the CLI, we want to
155 // print each slot like the following:
156 //
157 // Offset: [SP+8], Type: Spill, Align: 8, Size: 16
158 // foo @ /path/to/file.c:25
159 // bar @ /path/to/file.c:35
160 //
161 // Which prints the size, alignment, and offset from the SP at function
162 // entry.
163 //
164 // But we also want the machine readable remarks data to be nicely
165 // organized. So we print some additional data as strings for the CLI
166 // output, but maintain more structured data for the YAML.
167 //
168 // For example we store the Offset in YAML as:
169 // ...
170 // - Offset: -8
171 // - ScalableOffset: -16
172 // Note: the ScalableOffset entries are added only for slots with non-zero
173 // scalable offsets.
174 //
175 // But we print it to the CLI as:
176 // Offset: [SP-8]
177 //
178 // Or with non-zero scalable offset:
179 // Offset: [SP-8-16 x vscale]
180
181 // Negative offsets will print a leading `-`, so only add `+`
182 std::string Prefix =
183 formatv("\nOffset: [SP{0}", (D.Offset.getFixed() < 0) ? "" : "+").str();
184 Rem << Prefix << ore::NV("Offset", D.Offset.getFixed());
185
186 if (D.Offset.getScalable()) {
187 Rem << ((D.Offset.getScalable() < 0) ? "" : "+")
188 << ore::NV("ScalableOffset", D.Offset.getScalable()) << " x vscale";
189 }
190
191 Rem << "], Type: " << ore::NV("Type", getTypeString(D.SlotTy))
192 << ", Align: " << ore::NV("Align", D.Align)
193 << ", Size: " << ore::NV("Size", ElementCount::get(D.Size, D.Scalable));
194 }
195
196 void emitSourceLocRemark(const MachineFunction &MF, const DILocalVariable *N,
198 std::string Loc =
199 formatv("{0} @ {1}:{2}", N->getName(), N->getFilename(), N->getLine())
200 .str();
201 Rem << "\n " << ore::NV("DataLoc", Loc);
202 }
203
205 const MachineFrameInfo &MFI,
206 const TargetFrameLowering *FI, int FrameIdx) {
207 if (!FI)
208 return StackOffset::getFixed(MFI.getObjectOffset(FrameIdx));
209
210 return FI->getFrameIndexReferenceFromSP(MF, FrameIdx);
211 }
212
213 void emitStackFrameLayoutRemarks(MachineFunction &MF,
215 const MachineFrameInfo &MFI = MF.getFrameInfo();
216 if (!MFI.hasStackObjects())
217 return;
218
220
221 LLVM_DEBUG(dbgs() << "getStackProtectorIndex =="
222 << MFI.getStackProtectorIndex() << "\n");
223
224 std::vector<SlotData> SlotInfo;
225
226 const unsigned int NumObj = MFI.getNumObjects();
227 SlotInfo.reserve(NumObj);
228 // initialize slot info
229 for (int Idx = MFI.getObjectIndexBegin(), EndIdx = MFI.getObjectIndexEnd();
230 Idx != EndIdx; ++Idx) {
231 if (MFI.isDeadObjectIndex(Idx))
232 continue;
233 SlotInfo.emplace_back(MFI, getStackOffset(MF, MFI, FI, Idx), Idx);
234 }
235
236 // sort the ordering, to match the actual layout in memory
237 llvm::sort(SlotInfo);
238
239 SlotDbgMap SlotMap = genSlotDbgMapping(MF);
240
241 for (const SlotData &Info : SlotInfo) {
242 emitStackSlotRemark(MF, Info, Rem);
243 for (const DILocalVariable *N : SlotMap[Info.Slot])
244 emitSourceLocRemark(MF, N, Rem);
245 }
246 }
247
248 // We need to generate a mapping of slots to the values that are stored to
249 // them. This information is lost by the time we need to print out the frame,
250 // so we reconstruct it here by walking the CFG, and generating the mapping.
251 SlotDbgMap genSlotDbgMapping(MachineFunction &MF) {
252 SlotDbgMap SlotDebugMap;
253
254 // add variables to the map
257 SlotDebugMap[DI.getStackSlot()].insert(DI.Var);
258
259 // Then add all the spills that have debug data
260 for (MachineBasicBlock &MBB : MF) {
261 for (MachineInstr &MI : MBB) {
262 for (MachineMemOperand *MO : MI.memoperands()) {
263 if (!MO->isStore())
264 continue;
265 auto *FI = dyn_cast_or_null<FixedStackPseudoSourceValue>(
266 MO->getPseudoValue());
267 if (!FI)
268 continue;
269 int FrameIdx = FI->getFrameIndex();
271 MI.collectDebugValues(Dbg);
272
273 for (MachineInstr *MI : Dbg)
274 SlotDebugMap[FrameIdx].insert(MI->getDebugVariable());
275 }
276 }
277 }
278
279 return SlotDebugMap;
280 }
281};
282
283char StackFrameLayoutAnalysisPass::ID = 0;
284} // namespace
285
286char &llvm::StackFrameLayoutAnalysisPassID = StackFrameLayoutAnalysisPass::ID;
287INITIALIZE_PASS(StackFrameLayoutAnalysisPass, "stack-frame-layout",
288 "Stack Frame Layout", false, false)
289
290namespace llvm {
291/// Returns a newly-created StackFrameLayout pass.
293 return new StackFrameLayoutAnalysisPass();
294}
295
296} // namespace llvm
static StackOffset getStackOffset(const MachineFunction &MF, int64_t ObjectOffset)
MachineBasicBlock & MBB
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
Analysis containing CSE Info
Definition: CSEInfo.cpp:27
Returns the sub type a function will return at a given Idx Should correspond to the result type of an ExtractValue instruction executed with just that one unsigned Idx
#define LLVM_DEBUG(X)
Definition: Debug.h:101
IRTranslator LLVM IR MI
static std::string getTypeString(Type *T)
Definition: LLParser.cpp:71
===- MachineOptimizationRemarkEmitter.h - Opt Diagnostics -*- C++ -*-—===//
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition: PassSupport.h:38
This file implements a set that has insertion order iteration characteristics.
#define DEBUG_TYPE
Represent the analysis usage information of a pass.
AnalysisUsage & addRequired()
void setPreservesAll()
Set by analyses that do not transform their input at all.
static constexpr ElementCount get(ScalarTy MinVal, bool Scalable)
Definition: TypeSize.h:317
DISubprogram * getSubprogram() const
Get the attached subprogram.
Definition: Metadata.cpp:1837
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
Definition: Function.cpp:380
This is an important class for using LLVM in a threaded context.
Definition: LLVMContext.h:67
const DiagnosticHandler * getDiagHandlerPtr() const
getDiagHandlerPtr - Returns const raw pointer of DiagnosticHandler set by setDiagnosticHandler.
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
int getStackProtectorIndex() const
Return the index for the stack protector object.
Align getObjectAlign(int ObjectIdx) const
Return the alignment of the specified stack object.
bool isSpillSlotObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a spill slot.
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
unsigned getNumObjects() const
Return the number of objects.
bool isVariableSizedObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a variable sized object.
int getObjectIndexEnd() const
Return one past the maximum frame object index.
bool hasStackProtectorIndex() const
bool hasStackObjects() const
Return true if there are any stack objects in this function.
uint8_t getStackID(int ObjectIdx) const
int64_t getObjectOffset(int ObjectIdx) const
Return the assigned stack offset of the specified object from the incoming stack pointer.
bool isFixedObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a fixed stack object.
int getObjectIndexBegin() const
Return the minimum frame object index.
bool isDeadObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a dead object.
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
virtual bool runOnMachineFunction(MachineFunction &MF)=0
runOnMachineFunction - This method must be overloaded to perform the desired machine code transformat...
Description of the location of a variable whose Address is valid and unchanging during function execu...
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
auto getInStackSlotVariableDbgInfo()
Returns the collection of variables for which we have debug info and that have been assigned a stack ...
Function & getFunction()
Return the LLVM function that this machine code represents.
const MachineBasicBlock & front() const
Representation of each machine instruction.
Definition: MachineInstr.h:69
A description of a memory reference used in the backend.
Diagnostic information for optimization analysis remarks.
virtual StringRef getPassName() const
getPassName - Return a nice clean name for a pass.
Definition: Pass.cpp:81
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Definition: SmallVector.h:1209
StackOffset holds a fixed and a scalable offset in bytes.
Definition: TypeSize.h:33
int64_t getFixed() const
Returns the fixed component of the stack.
Definition: TypeSize.h:49
StringRef - Represent a constant reference to a string, i.e.
Definition: StringRef.h:50
Information about stack frame layout on the target.
virtual StackOffset getFrameIndexReferenceFromSP(const MachineFunction &MF, int FI) const
getFrameIndexReferenceFromSP - This method returns the offset from the stack pointer to the slot of t...
virtual const TargetFrameLowering * getFrameLowering() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition: CallingConv.h:24
DiagnosticInfoOptimizationBase::Argument NV
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
bool operator<(int64_t V1, const APSInt &V2)
Definition: APSInt.h:361
auto formatv(const char *Fmt, Ts &&...Vals) -> formatv_object< decltype(std::make_tuple(support::detail::build_format_adapter(std::forward< Ts >(Vals))...))>
MachineFunctionPass * createStackFrameLayoutAnalysisPass()
StackFramePrinter pass - This pass prints out the machine function's stack frame to the given stream ...
void sort(IteratorTy Start, IteratorTy End)
Definition: STLExtras.h:1647
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition: Debug.cpp:163
bool isFunctionInPrintList(StringRef FunctionName)
char & StackFrameLayoutAnalysisPassID
StackFramePrinter - This pass prints the stack frame layout and variable mappings.
#define N
This struct is a compact representation of a valid (non-zero power of two) alignment.
Definition: Alignment.h:39
uint64_t value() const
This is a hole in the type system and should not be abused.
Definition: Alignment.h:85
virtual bool isAnalysisRemarkEnabled(StringRef PassName) const
Return true if analysis remarks are enabled, override to provide different implementation.