ilya-biryukov added inline comments.
================ Comment at: clang/lib/Sema/SemaExprCXX.cpp:7374 auto *E = ExprWithCleanups::Create( Context, SubExpr, Cleanup.cleanupsHaveSideEffects(), Cleanups); ---------------- Because we may potentially pass some cleanups here (which are outside `ConstantExpr`) and won't discard them later, we end up adding the unrelated blocks to this cleanup expression. Here's an example. If we start with this code: ```cpp struct P { consteval P() {} }; struct A { A(int v) { this->data = new int(v); } const int& get() const { return *this->data; } ~A() { delete data; } private: int *data; }; void foo() { A a(1); for (; ^{ ptr = &a.get(); }(), P(), false;); } ``` And run `clang++ -std=c++20 -fblocks -Xclang=-ast-dump`, we'll get: (notice same Block in two cleanups) ``` `-ForStmt 0x5592888b1318 <line:17:3, col:46> |-<<<NULL>>> |-<<<NULL>>> |-ExprWithCleanups 0x5592888b12f0 <col:10, col:39> 'bool' | |-cleanup Block 0x5592888ad728 | `-BinaryOperator 0x5592888b12d0 <col:10, col:39> 'bool' ',' | |-BinaryOperator 0x5592888b12a0 <col:10, col:36> 'P':'P' ',' | | |-CallExpr 0x5592888adb48 <col:10, col:31> 'void' | | | `-BlockExpr 0x5592888adb30 <col:10, col:29> 'void (^)()' | | | `-BlockDecl 0x5592888ad728 <col:10, col:29> col:10 | | | |-capture Var 0x5592888ad318 'a' 'A':'A' | | | `-CompoundStmt 0x5592888ad930 <col:11, col:29> | | `-CXXFunctionalCastExpr 0x5592888b1278 <col:34, col:36> 'P':'P' functional cast to P <NoOp> | | `-ConstantExpr 0x5592888b1108 <col:34, col:36> 'P':'P' | | |-value: Struct | | `-ExprWithCleanups 0x5592888b10e8 <col:34, col:36> 'P':'P' | | |-cleanup Block 0x5592888ad728 | | `-CXXTemporaryObjectExpr 0x5592888b10b8 <col:34, col:36> 'P':'P' 'void ()' | `-CXXBoolLiteralExpr 0x5592888b12c0 <col:39> 'bool' false ``` I didn't check if this particular error can lead to codegen bug, but even having a misplaced cleanup in the AST is already a red flag. I think in case of immediate invocations we could simply create `ExprWithCleanups` and not do anything else: - `CleanupVarDeclMarking` will be handled when containing evaluation context is popped from the stack. - `ExprCleanupObjects` may only contain blocks at this point (as it's C++ and we have assert for this). One can't use blocks inside a constant expression (Clang will produce an error) so in correct code any cleanups we see inside the current context **must handled by the containing evaluation context** and not this `ExprWithCleanups`. Repository: rG LLVM Github Monorepo CHANGES SINCE LAST ACTION https://reviews.llvm.org/D153962/new/ https://reviews.llvm.org/D153962 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits