clang 23.0.0git
InterpFrame.h
Go to the documentation of this file.
1//===--- InterpFrame.h - Call Frame implementation for the VM ---*- 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// Defines the class storing information about stack frames in the interpreter.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_CLANG_AST_INTERP_INTERPFRAME_H
14#define LLVM_CLANG_AST_INTERP_INTERPFRAME_H
15
16#include "Frame.h"
17#include "InterpBlock.h"
18#include "Pointer.h"
19
20namespace clang {
21namespace interp {
22class Function;
23class InterpState;
24class Pointer;
25
26/// Frame storing local variables.
27class InterpFrame final : public Frame {
28public:
29 /// The frame of the previous function.
31
32 /// Bottom Frame.
34
35 /// Creates a new frame for a method call.
37 CodePtr RetPC, unsigned ArgSize);
38
39 /// Creates a new frame with the values that make sense.
40 /// I.e., the caller is the current frame of S,
41 /// the This() pointer is the current Pointer on the top of S's stack,
42 /// and the RVO pointer is before that.
43 InterpFrame(InterpState &S, const Function *Func, CodePtr RetPC,
44 unsigned VarArgSize = 0);
45
46 /// Destroys the frame, killing all live pointers to stack slots.
48
49 /// Returns the number of bytes needed to allocate an InterpFrame for the
50 /// given function.
51 static size_t allocSize(const Function *F) {
52 return sizeof(InterpFrame) + F->getFrameSize() +
53 (F->getArgSize() + (sizeof(Block) * F->getNumWrittenParams()));
54 }
55
56 std::string getName() const {
57 if (!Func)
58 return "Bottom frame";
59 return Func->getName();
60 }
61
62 static void free(InterpFrame *F) {
63 if (!F->isBottomFrame()) {
64 F->~InterpFrame();
65 delete[] reinterpret_cast<char *>(F);
66 } else {
67 F->~InterpFrame();
68 }
69 }
70
71 /// Invokes the destructors for a scope.
72 void destroy(unsigned Idx);
73 void initScope(unsigned Idx);
74 void destroyScopes();
75 void enableLocal(unsigned Idx);
76 bool isLocalEnabled(unsigned Idx) const {
77 return localInlineDesc(Idx)->IsActive;
78 }
79
80 /// Describes the frame with arguments for diagnostic purposes.
81 void describe(llvm::raw_ostream &OS) const override;
82
83 /// Returns the parent frame object.
84 Frame *getCaller() const override { return Caller; }
85
86 /// Returns the location of the call to the frame.
87 SourceRange getCallRange() const override;
88
89 /// Returns the caller.
90 const FunctionDecl *getCallee() const override;
91
92 /// Returns the current function.
93 const Function *getFunction() const { return Func; }
94
95 /// Returns the offset on the stack at which the frame starts.
96 size_t getFrameOffset() const { return FrameOffset; }
97
98 /// Returns the value of a local variable.
99 template <typename T> const T &getLocal(unsigned Offset) const {
100 return localRef<T>(Offset);
101 }
102
103 /// Mutates a local variable.
104 template <typename T> void setLocal(unsigned Offset, const T &Value) {
105 localRef<T>(Offset) = Value;
106 localInlineDesc(Offset)->IsInitialized = true;
107 localInlineDesc(Offset)->LifeState = Lifetime::Started;
108 }
109
110 /// Returns a pointer to a local variables.
111 Pointer getLocalPointer(unsigned Offset) const;
112 Block *getLocalBlock(unsigned Offset) const;
113
114 /// Returns the value of an argument.
115 template <typename T> const T &getParam(unsigned Index) const {
116 Block *ArgBlock = argBlock(Index);
117 if (!ArgBlock->isInitialized())
118 return stackRef<T>(Func->getParamDescriptor(Index).Offset);
119 return ArgBlock->deref<T>();
120 }
121
122 /// Mutates a local copy of a parameter.
123 template <typename T> void setParam(unsigned Index, const T &Value) {
124 argBlock(Index)->deref<T>() = Value;
125 }
126
127 /// Returns a pointer to an argument - lazily creates a block.
128 Pointer getParamPointer(unsigned Offset);
129
130 bool hasThisPointer() const { return Func && Func->hasThisPointer(); }
131
132 /// Returns the 'this' pointer.
133 const Pointer &getThis() const {
134 assert(hasThisPointer());
135 assert(!isBottomFrame());
136 return stackRef<Pointer>(ThisPointerOffset);
137 }
138
139 /// Returns the RVO pointer, if the Function has one.
140 const Pointer &getRVOPtr() const {
141 assert(Func);
142 assert(Func->hasRVO());
143 assert(!isBottomFrame());
144 return stackRef<Pointer>(0);
145 }
146
147 /// Checks if the frame is a root frame - return should quit the interpreter.
148 bool isRoot() const { return !Func; }
149
150 /// Returns the PC of the frame's code start.
151 CodePtr getPC() const { return Func->getCodeBegin(); }
152
153 /// Returns the return address of the frame.
154 CodePtr getRetPC() const { return RetPC; }
155
156 /// Map a location to a source.
157 SourceInfo getSource(CodePtr PC) const;
158 const Expr *getExpr(CodePtr PC) const;
160 SourceRange getRange(CodePtr PC) const;
161
162 unsigned getDepth() const { return Depth; }
163
164 bool isStdFunction() const;
165
166 bool isBottomFrame() const { return !Caller; }
167
168 void dump() const { dump(llvm::errs(), 0); }
169 void dump(llvm::raw_ostream &OS, unsigned Indent = 0) const;
170
171private:
172 /// Returns an original argument from the stack.
173 template <typename T> const T &stackRef(unsigned Offset) const {
174 assert(Args);
175 return *reinterpret_cast<const T *>(Args - ArgSize + Offset);
176 }
177
178 /// Returns an offset to a local.
179 template <typename T> T &localRef(unsigned Offset) const {
180 return localBlock(Offset)->deref<T>();
181 }
182
183 /// Pointer to local memory.
184 char *locals() const {
185 return (reinterpret_cast<char *>(const_cast<InterpFrame *>(this))) +
186 align(sizeof(InterpFrame));
187 }
188
189 /// Pointer to argument memory.
190 char *args() const {
191 return (reinterpret_cast<char *>(const_cast<InterpFrame *>(this))) +
192 sizeof(InterpFrame) + Func->getFrameSize();
193 }
194
195 /// Returns a pointer to a local's block.
196 Block *localBlock(unsigned Offset) const {
197 return reinterpret_cast<Block *>(locals() + Offset - sizeof(Block));
198 }
199
200 /// Returns a pointer to an argument block.
201 Block *argBlock(unsigned Index) const {
202 unsigned ByteOffset = Func->getParamDescriptor(Index).BlockOffset;
203 return reinterpret_cast<Block *>(args() + ByteOffset);
204 }
205
206 /// Returns the inline descriptor of the local.
207 InlineDescriptor *localInlineDesc(unsigned Offset) const {
208 return reinterpret_cast<InlineDescriptor *>(locals() + Offset);
209 }
210
211private:
212 /// Reference to the interpreter state.
213 InterpState &S;
214 /// Depth of this frame.
215 unsigned Depth;
216 /// Reference to the function being executed.
217 const Function *Func;
218 /// Offset of the instance pointer. Use with stackRef<>().
219 unsigned ThisPointerOffset;
220 /// Return address.
221 CodePtr RetPC;
222 /// The size of all the arguments.
223 const unsigned ArgSize;
224 /// Pointer to the arguments in the callee's frame.
225 char *Args = nullptr;
226 /// Offset on the stack at entry.
227 const size_t FrameOffset;
228
229public:
231};
232
233} // namespace interp
234} // namespace clang
235
236#endif
Expr * getExpr()
Get 'expr' part of the associated expression/statement.
This represents one expression.
Definition Expr.h:112
Represents a function declaration or definition.
Definition Decl.h:2018
Encodes a location in the source.
A trivial tuple used to represent a source range.
A memory block, either on the stack or in the heap.
Definition InterpBlock.h:44
const T & deref() const
bool isInitialized() const
Returns whether the data of this block has been initialized via invoking the Ctor func.
Definition InterpBlock.h:92
Pointer into the code segment.
Definition Source.h:30
Base class for stack frames, shared between VM and walker.
Definition Frame.h:25
Bytecode function.
Definition Function.h:99
unsigned getFrameSize() const
Returns the size of the function's local stack.
Definition Function.h:124
unsigned getNumWrittenParams() const
Returns the number of parameter this function takes when it's called, i.e excluding the instance poin...
Definition Function.h:242
unsigned getArgSize() const
Returns the size of the argument stack.
Definition Function.h:126
static void free(InterpFrame *F)
Definition InterpFrame.h:62
InterpFrame(InterpState &S)
Bottom Frame.
bool isLocalEnabled(unsigned Idx) const
Definition InterpFrame.h:76
void setParam(unsigned Index, const T &Value)
Mutates a local copy of a parameter.
InterpFrame * Caller
The frame of the previous function.
Definition InterpFrame.h:30
Frame * getCaller() const override
Returns the parent frame object.
Definition InterpFrame.h:84
SourceInfo getSource(CodePtr PC) const
Map a location to a source.
CodePtr getRetPC() const
Returns the return address of the frame.
void enableLocal(unsigned Idx)
Block * getLocalBlock(unsigned Offset) const
CodePtr getPC() const
Returns the PC of the frame's code start.
SourceLocation getLocation(CodePtr PC) const
~InterpFrame()
Destroys the frame, killing all live pointers to stack slots.
const Pointer & getThis() const
Returns the 'this' pointer.
const Function * getFunction() const
Returns the current function.
Definition InterpFrame.h:93
size_t getFrameOffset() const
Returns the offset on the stack at which the frame starts.
Definition InterpFrame.h:96
SourceRange getRange(CodePtr PC) const
void setLocal(unsigned Offset, const T &Value)
Mutates a local variable.
bool isRoot() const
Checks if the frame is a root frame - return should quit the interpreter.
Pointer getLocalPointer(unsigned Offset) const
Returns a pointer to a local variables.
unsigned getDepth() const
const T & getParam(unsigned Index) const
Returns the value of an argument.
void destroy(unsigned Idx)
Invokes the destructors for a scope.
const Pointer & getRVOPtr() const
Returns the RVO pointer, if the Function has one.
Pointer getParamPointer(unsigned Offset)
Returns a pointer to an argument - lazily creates a block.
const FunctionDecl * getCallee() const override
Returns the caller.
std::string getName() const
Definition InterpFrame.h:56
void initScope(unsigned Idx)
const T & getLocal(unsigned Offset) const
Returns the value of a local variable.
Definition InterpFrame.h:99
SourceRange getCallRange() const override
Returns the location of the call to the frame.
void describe(llvm::raw_ostream &OS) const override
Describes the frame with arguments for diagnostic purposes.
static size_t allocSize(const Function *F)
Returns the number of bytes needed to allocate an InterpFrame for the given function.
Definition InterpFrame.h:51
Interpreter context.
Definition InterpState.h:36
A pointer to a memory block, live or dead.
Definition Pointer.h:97
Describes the statement/declaration an opcode was generated from.
Definition Source.h:74
constexpr size_t align(size_t Size)
Aligns a size to the pointer alignment.
Definition PrimType.h:201
The JSON file list parser is used to communicate input to InstallAPI.
raw_ostream & Indent(raw_ostream &Out, const unsigned int Space, bool IsDot)
Definition JsonSupport.h:21