diff options
Diffstat (limited to 'clang/lib/AST/Interp/EvalEmitter.cpp')
-rw-r--r-- | clang/lib/AST/Interp/EvalEmitter.cpp | 253 |
1 files changed, 0 insertions, 253 deletions
diff --git a/clang/lib/AST/Interp/EvalEmitter.cpp b/clang/lib/AST/Interp/EvalEmitter.cpp deleted file mode 100644 index 22e8695b921..00000000000 --- a/clang/lib/AST/Interp/EvalEmitter.cpp +++ /dev/null @@ -1,253 +0,0 @@ -//===--- EvalEmitter.cpp - Instruction emitter for the VM -------*- C++ -*-===// -// -// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. -// See https://llvm.org/LICENSE.txt for license information. -// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception -// -//===----------------------------------------------------------------------===// - -#include "EvalEmitter.h" -#include "Context.h" -#include "Interp.h" -#include "Opcode.h" -#include "Program.h" -#include "clang/AST/DeclCXX.h" - -using namespace clang; -using namespace clang::interp; - -using APSInt = llvm::APSInt; -template <typename T> using Expected = llvm::Expected<T>; - -EvalEmitter::EvalEmitter(Context &Ctx, Program &P, State &Parent, - InterpStack &Stk, APValue &Result) - : Ctx(Ctx), P(P), S(Parent, P, Stk, Ctx, this), Result(Result) { - // Create a dummy frame for the interpreter which does not have locals. - S.Current = new InterpFrame(S, nullptr, nullptr, CodePtr(), Pointer()); -} - -llvm::Expected<bool> EvalEmitter::interpretExpr(const Expr *E) { - if (this->visitExpr(E)) - return true; - if (BailLocation) - return llvm::make_error<ByteCodeGenError>(*BailLocation); - return false; -} - -llvm::Expected<bool> EvalEmitter::interpretDecl(const VarDecl *VD) { - if (this->visitDecl(VD)) - return true; - if (BailLocation) - return llvm::make_error<ByteCodeGenError>(*BailLocation); - return false; -} - -void EvalEmitter::emitLabel(LabelTy Label) { - CurrentLabel = Label; -} - -EvalEmitter::LabelTy EvalEmitter::getLabel() { return NextLabel++; } - -Scope::Local EvalEmitter::createLocal(Descriptor *D) { - // Allocate memory for a local. - auto Memory = std::make_unique<char[]>(sizeof(Block) + D->getAllocSize()); - auto *B = new (Memory.get()) Block(D, /*isStatic=*/false); - B->invokeCtor(); - - // Register the local. - unsigned Off = Locals.size(); - Locals.insert({Off, std::move(Memory)}); - return {Off, D}; -} - -bool EvalEmitter::bail(const SourceLocation &Loc) { - if (!BailLocation) - BailLocation = Loc; - return false; -} - -bool EvalEmitter::jumpTrue(const LabelTy &Label) { - if (isActive()) { - if (S.Stk.pop<bool>()) - ActiveLabel = Label; - } - return true; -} - -bool EvalEmitter::jumpFalse(const LabelTy &Label) { - if (isActive()) { - if (!S.Stk.pop<bool>()) - ActiveLabel = Label; - } - return true; -} - -bool EvalEmitter::jump(const LabelTy &Label) { - if (isActive()) - CurrentLabel = ActiveLabel = Label; - return true; -} - -bool EvalEmitter::fallthrough(const LabelTy &Label) { - if (isActive()) - ActiveLabel = Label; - CurrentLabel = Label; - return true; -} - -template <PrimType OpType> bool EvalEmitter::emitRet(const SourceInfo &Info) { - if (!isActive()) - return true; - using T = typename PrimConv<OpType>::T; - return ReturnValue<T>(S.Stk.pop<T>(), Result); -} - -bool EvalEmitter::emitRetVoid(const SourceInfo &Info) { return true; } - -bool EvalEmitter::emitRetValue(const SourceInfo &Info) { - // Method to recursively traverse composites. - std::function<bool(QualType, const Pointer &, APValue &)> Composite; - Composite = [this, &Composite](QualType Ty, const Pointer &Ptr, APValue &R) { - if (auto *AT = Ty->getAs<AtomicType>()) - Ty = AT->getValueType(); - - if (auto *RT = Ty->getAs<RecordType>()) { - auto *Record = Ptr.getRecord(); - assert(Record && "Missing record descriptor"); - - bool Ok = true; - if (RT->getDecl()->isUnion()) { - const FieldDecl *ActiveField = nullptr; - APValue Value; - for (auto &F : Record->fields()) { - const Pointer &FP = Ptr.atField(F.Offset); - QualType FieldTy = F.Decl->getType(); - if (FP.isActive()) { - if (llvm::Optional<PrimType> T = Ctx.classify(FieldTy)) { - TYPE_SWITCH(*T, Ok &= ReturnValue<T>(FP.deref<T>(), Value)); - } else { - Ok &= Composite(FieldTy, FP, Value); - } - break; - } - } - R = APValue(ActiveField, Value); - } else { - unsigned NF = Record->getNumFields(); - unsigned NB = Record->getNumBases(); - unsigned NV = Ptr.isBaseClass() ? 0 : Record->getNumVirtualBases(); - - R = APValue(APValue::UninitStruct(), NB, NF); - - for (unsigned I = 0; I < NF; ++I) { - const Record::Field *FD = Record->getField(I); - QualType FieldTy = FD->Decl->getType(); - const Pointer &FP = Ptr.atField(FD->Offset); - APValue &Value = R.getStructField(I); - - if (llvm::Optional<PrimType> T = Ctx.classify(FieldTy)) { - TYPE_SWITCH(*T, Ok &= ReturnValue<T>(FP.deref<T>(), Value)); - } else { - Ok &= Composite(FieldTy, FP, Value); - } - } - - for (unsigned I = 0; I < NB; ++I) { - const Record::Base *BD = Record->getBase(I); - QualType BaseTy = Ctx.getASTContext().getRecordType(BD->Decl); - const Pointer &BP = Ptr.atField(BD->Offset); - Ok &= Composite(BaseTy, BP, R.getStructBase(I)); - } - - for (unsigned I = 0; I < NV; ++I) { - const Record::Base *VD = Record->getVirtualBase(I); - QualType VirtBaseTy = Ctx.getASTContext().getRecordType(VD->Decl); - const Pointer &VP = Ptr.atField(VD->Offset); - Ok &= Composite(VirtBaseTy, VP, R.getStructBase(NB + I)); - } - } - return Ok; - } - if (auto *AT = Ty->getAsArrayTypeUnsafe()) { - const size_t NumElems = Ptr.getNumElems(); - QualType ElemTy = AT->getElementType(); - R = APValue(APValue::UninitArray{}, NumElems, NumElems); - - bool Ok = true; - for (unsigned I = 0; I < NumElems; ++I) { - APValue &Slot = R.getArrayInitializedElt(I); - const Pointer &EP = Ptr.atIndex(I); - if (llvm::Optional<PrimType> T = Ctx.classify(ElemTy)) { - TYPE_SWITCH(*T, Ok &= ReturnValue<T>(EP.deref<T>(), Slot)); - } else { - Ok &= Composite(ElemTy, EP.narrow(), Slot); - } - } - return Ok; - } - llvm_unreachable("invalid value to return"); - }; - - // Return the composite type. - const auto &Ptr = S.Stk.pop<Pointer>(); - return Composite(Ptr.getType(), Ptr, Result); -} - -bool EvalEmitter::emitGetPtrLocal(uint32_t I, const SourceInfo &Info) { - if (!isActive()) - return true; - - auto It = Locals.find(I); - assert(It != Locals.end() && "Missing local variable"); - S.Stk.push<Pointer>(reinterpret_cast<Block *>(It->second.get())); - return true; -} - -template <PrimType OpType> -bool EvalEmitter::emitGetLocal(uint32_t I, const SourceInfo &Info) { - if (!isActive()) - return true; - - using T = typename PrimConv<OpType>::T; - - auto It = Locals.find(I); - assert(It != Locals.end() && "Missing local variable"); - auto *B = reinterpret_cast<Block *>(It->second.get()); - S.Stk.push<T>(*reinterpret_cast<T *>(B + 1)); - return true; -} - -template <PrimType OpType> -bool EvalEmitter::emitSetLocal(uint32_t I, const SourceInfo &Info) { - if (!isActive()) - return true; - - using T = typename PrimConv<OpType>::T; - - auto It = Locals.find(I); - assert(It != Locals.end() && "Missing local variable"); - auto *B = reinterpret_cast<Block *>(It->second.get()); - *reinterpret_cast<T *>(B + 1) = S.Stk.pop<T>(); - return true; -} - -bool EvalEmitter::emitDestroy(uint32_t I, const SourceInfo &Info) { - if (!isActive()) - return true; - - for (auto &Local : Descriptors[I]) { - auto It = Locals.find(Local.Offset); - assert(It != Locals.end() && "Missing local variable"); - S.deallocate(reinterpret_cast<Block *>(It->second.get())); - } - - return true; -} - -//===----------------------------------------------------------------------===// -// Opcode evaluators -//===----------------------------------------------------------------------===// - -#define GET_EVAL_IMPL -#include "Opcodes.inc" -#undef GET_EVAL_IMPL |