#ifndef LLVM_CLANG_AST_INTERP_INTERPFRAME_H
#define LLVM_CLANG_AST_INTERP_INTERPFRAME_H
#include "Frame.h"
#include "Program.h"
namespace clang {
namespace interp {
class Function;
class InterpState;
class Pointer;
class InterpFrame final : public Frame {
public:
InterpFrame *Caller;
InterpFrame(InterpState &S, const Function *Func, InterpFrame *Caller,
CodePtr RetPC, unsigned ArgSize);
InterpFrame(InterpState &S, const Function *Func, CodePtr RetPC,
unsigned VarArgSize = 0);
~InterpFrame();
void destroy(unsigned Idx);
void popArgs();
void describe(llvm::raw_ostream &OS) const override;
Frame *getCaller() const override;
SourceRange getCallRange() const override;
const FunctionDecl *getCallee() const override;
const Function *getFunction() const { return Func; }
size_t getFrameOffset() const { return FrameOffset; }
template <typename T> const T &getLocal(unsigned Offset) const {
return localRef<T>(Offset);
}
template <typename T> void setLocal(unsigned Offset, const T &Value) {
localRef<T>(Offset) = Value;
localInlineDesc(Offset)->IsInitialized = true;
}
Pointer getLocalPointer(unsigned Offset) const;
template <typename T> const T &getParam(unsigned Offset) const {
auto Pt = Params.find(Offset);
if (Pt == Params.end())
return stackRef<T>(Offset);
return Pointer(reinterpret_cast<Block *>(Pt->second.get())).deref<T>();
}
template <typename T> void setParam(unsigned Offset, const T &Value) {
getParamPointer(Offset).deref<T>() = Value;
}
Pointer getParamPointer(unsigned Offset);
const Pointer &getThis() const { return This; }
const Pointer &getRVOPtr() const { return RVOPtr; }
bool isRoot() const { return !Func; }
CodePtr getPC() const { return Func->getCodeBegin(); }
CodePtr getRetPC() const { return RetPC; }
virtual SourceInfo getSource(CodePtr PC) const;
const Expr *getExpr(CodePtr PC) const;
SourceLocation getLocation(CodePtr PC) const;
SourceRange getRange(CodePtr PC) const;
unsigned getDepth() const { return Depth; }
void dump() const { dump(llvm::errs(), 0); }
void dump(llvm::raw_ostream &OS, unsigned Indent = 0) const;
private:
template <typename T> const T &stackRef(unsigned Offset) const {
assert(Args);
return *reinterpret_cast<const T *>(Args - ArgSize + Offset);
}
template <typename T> T &localRef(unsigned Offset) const {
return getLocalPointer(Offset).deref<T>();
}
Block *localBlock(unsigned Offset) const {
return reinterpret_cast<Block *>(Locals.get() + Offset - sizeof(Block));
}
InlineDescriptor *localInlineDesc(unsigned Offset) const {
return reinterpret_cast<InlineDescriptor *>(Locals.get() + Offset);
}
private:
InterpState &S;
unsigned Depth;
const Function *Func;
Pointer This;
Pointer RVOPtr;
CodePtr RetPC;
const unsigned ArgSize;
char *Args = nullptr;
std::unique_ptr<char[]> Locals;
const size_t FrameOffset;
llvm::DenseMap<unsigned, std::unique_ptr<char[]>> Params;
};
}
}
#endif