1 //===--- InterpFrame.h - Call Frame implementation for the VM ---*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 // Defines the class storing information about stack frames in the interpreter.
11 //===----------------------------------------------------------------------===//
13 #ifndef LLVM_CLANG_AST_INTERP_INTERPFRAME_H
14 #define LLVM_CLANG_AST_INTERP_INTERPFRAME_H
27 /// Frame storing local variables.
28 class InterpFrame final
: public Frame
{
30 /// The frame of the previous function.
33 /// Creates a new frame for a method call.
34 InterpFrame(InterpState
&S
, const Function
*Func
, InterpFrame
*Caller
,
37 /// Creates a new frame with the values that make sense.
38 /// I.e., the caller is the current frame of S,
39 /// the This() pointer is the current Pointer on the top of S's stack,
40 /// and the RVO pointer is before that.
41 InterpFrame(InterpState
&S
, const Function
*Func
, CodePtr RetPC
);
43 /// Destroys the frame, killing all live pointers to stack slots.
46 /// Invokes the destructors for a scope.
47 void destroy(unsigned Idx
);
49 /// Pops the arguments off the stack.
52 /// Describes the frame with arguments for diagnostic purposes.
53 void describe(llvm::raw_ostream
&OS
) const override
;
55 /// Returns the parent frame object.
56 Frame
*getCaller() const override
;
58 /// Returns the location of the call to the frame.
59 SourceRange
getCallRange() const override
;
61 /// Returns the caller.
62 const FunctionDecl
*getCallee() const override
;
64 /// Returns the current function.
65 const Function
*getFunction() const { return Func
; }
67 /// Returns the offset on the stack at which the frame starts.
68 size_t getFrameOffset() const { return FrameOffset
; }
70 /// Returns the value of a local variable.
71 template <typename T
> const T
&getLocal(unsigned Offset
) const {
72 return localRef
<T
>(Offset
);
75 /// Mutates a local variable.
76 template <typename T
> void setLocal(unsigned Offset
, const T
&Value
) {
77 localRef
<T
>(Offset
) = Value
;
78 localInlineDesc(Offset
)->IsInitialized
= true;
81 /// Returns a pointer to a local variables.
82 Pointer
getLocalPointer(unsigned Offset
) const;
84 /// Returns the value of an argument.
85 template <typename T
> const T
&getParam(unsigned Offset
) const {
86 auto Pt
= Params
.find(Offset
);
87 if (Pt
== Params
.end()) {
88 return stackRef
<T
>(Offset
);
90 return Pointer(reinterpret_cast<Block
*>(Pt
->second
.get())).deref
<T
>();
94 /// Mutates a local copy of a parameter.
95 template <typename T
> void setParam(unsigned Offset
, const T
&Value
) {
96 getParamPointer(Offset
).deref
<T
>() = Value
;
99 /// Returns a pointer to an argument - lazily creates a block.
100 Pointer
getParamPointer(unsigned Offset
);
102 /// Returns the 'this' pointer.
103 const Pointer
&getThis() const { return This
; }
105 /// Returns the RVO pointer, if the Function has one.
106 const Pointer
&getRVOPtr() const { return RVOPtr
; }
108 /// Checks if the frame is a root frame - return should quit the interpreter.
109 bool isRoot() const { return !Func
; }
111 /// Returns the PC of the frame's code start.
112 CodePtr
getPC() const { return Func
->getCodeBegin(); }
114 /// Returns the return address of the frame.
115 CodePtr
getRetPC() const { return RetPC
; }
117 /// Map a location to a source.
118 virtual SourceInfo
getSource(CodePtr PC
) const;
119 const Expr
*getExpr(CodePtr PC
) const;
120 SourceLocation
getLocation(CodePtr PC
) const;
121 SourceRange
getRange(CodePtr PC
) const;
123 unsigned getDepth() const { return Depth
; }
126 /// Returns an original argument from the stack.
127 template <typename T
> const T
&stackRef(unsigned Offset
) const {
129 return *reinterpret_cast<const T
*>(Args
- ArgSize
+ Offset
);
132 /// Returns an offset to a local.
133 template <typename T
> T
&localRef(unsigned Offset
) const {
134 return getLocalPointer(Offset
).deref
<T
>();
137 /// Returns a pointer to a local's block.
138 Block
*localBlock(unsigned Offset
) const {
139 return reinterpret_cast<Block
*>(Locals
.get() + Offset
- sizeof(Block
));
142 /// Returns the inline descriptor of the local.
143 InlineDescriptor
*localInlineDesc(unsigned Offset
) const {
144 return reinterpret_cast<InlineDescriptor
*>(Locals
.get() + Offset
);
148 /// Reference to the interpreter state.
150 /// Depth of this frame.
152 /// Reference to the function being executed.
153 const Function
*Func
;
154 /// Current object pointer for methods.
156 /// Pointer the non-primitive return value gets constructed in.
160 /// The size of all the arguments.
161 const unsigned ArgSize
;
162 /// Pointer to the arguments in the callee's frame.
163 char *Args
= nullptr;
164 /// Fixed, initial storage for known local variables.
165 std::unique_ptr
<char[]> Locals
;
166 /// Offset on the stack at entry.
167 const size_t FrameOffset
;
168 /// Mapping from arg offsets to their argument blocks.
169 llvm::DenseMap
<unsigned, std::unique_ptr
<char[]>> Params
;
172 } // namespace interp