[clang] [clang][Interp] Only evaluate the source array initialization of an `ArrayInitLoopExpr` once (PR #68039)

via cfe-commits cfe-commits at lists.llvm.org
Thu Oct 12 17:20:02 PDT 2023


================
@@ -478,6 +480,43 @@ template <class Emitter> class SourceLocScope final {
   bool Enabled = false;
 };
 
+template <class Emitter> class StoredOpaqueValueScope final {
+public:
+  StoredOpaqueValueScope(ByteCodeExprGen<Emitter> *Ctx) : Ctx(Ctx) {}
+
+  bool VisitAndStoreOpaqueValue(const OpaqueValueExpr *Ove) {
+    assert(Ove && "OpaqueValueExpr is a nullptr!");
+    assert(!Ctx->OpaqueExprs.contains(Ove) &&
+           "OpaqueValueExpr already stored!");
+
+    std::optional<PrimType> CommonTy = Ctx->classify(Ove);
+    std::optional<unsigned> LocalIndex = Ctx->allocateLocalPrimitive(
+        Ove, *CommonTy, Ove->getType().isConstQualified());
+    if (!LocalIndex)
+      return false;
+
+    if (!Ctx->visit(Ove))
+      return false;
+
+    if (!Ctx->emitSetLocal(*CommonTy, *LocalIndex, Ove))
+      return false;
+
+    Ctx->OpaqueExprs.insert({Ove, *LocalIndex});
+    StoredValues.emplace_back(Ove);
+
+    return true;
+  }
+
+  ~StoredOpaqueValueScope() {
+    for (const auto *SV : StoredValues)
+      Ctx->OpaqueExprs.erase(SV);
+  }
+
+private:
+  ByteCodeExprGen<Emitter> *Ctx;
+  std::vector<const OpaqueValueExpr *> StoredValues;
----------------
isuckatcs wrote:

I kept in mind the scenario, when we might have nested `StoredOpaqueValueScope`s. Somehow we need to keep track of which `OpaqueValueExpr` should be removed from `Ctx->OpaqueExprs`, when we exit the scope.

Do you suggest keeping everything in `Ctx->OpaqueExprs` during the whole lifetime of the bytecode generator?

https://github.com/llvm/llvm-project/pull/68039


More information about the cfe-commits mailing list