rsmith added inline comments.

================
Comment at: clang/include/clang/AST/Expr.h:1033-1035
+  bool IsCausedByImmediateInvocation() const {
+    return ConstantExprBits.IsCausedByImmediateInvocation;
+  }
----------------
I'd remove the "CausedBy" here -- the `ConstantExpr` is our representation of 
the immediate invocation.

Also, please start this function name with a lowercase `i` for local 
consistency.


================
Comment at: clang/include/clang/Sema/Sema.h:1027
 
+  using IICandidate = llvm::PointerIntPair<ConstantExpr *, 1>;
+
----------------
`II` doesn't mean anything to a reader here; please expand the abbreviation.


================
Comment at: clang/include/clang/Sema/Sema.h:5438-5439
 
+  /// Wrap the expression in a ConstantExpr if it is a potention immediate
+  /// invocation
+  ExprResult CheckForImmediateInvocation(ExprResult E, FunctionDecl *Decl);
----------------
potention -> potential
Missing period at end of sentence.


================
Comment at: clang/lib/AST/ExprConstant.cpp:2012-2013
 
+  if (auto *VD = LVal.getLValueBase().dyn_cast<const ValueDecl *>())
+    if (auto *FD = dyn_cast<FunctionDecl>(VD))
+      if (FD->isConsteval()) {
----------------
Please add braces to these `if` statements (their controlled statement is too 
long to omit the braces). As a general rule: if an inner construct needs 
braces, put braces on the outer constructs too.


================
Comment at: clang/lib/AST/ExprConstant.cpp:13618
+  if (InPlace) {
+    LValue LVal;
+    if (!::EvaluateInPlace(Result.Val, Info, LVal, this) ||
----------------
This isn't sufficient: the evaluation process can refer back to the object 
under construction (eg, via `this`), and we need an lvalue that actually names 
the result in order for this to work.

I think you'll need to do something like creating a suitable object (perhaps a 
`LifetimeExtendedTemporaryDecl`) in the caller to represent the result of the 
immediate invocation, and passing that into here.


================
Comment at: clang/lib/Sema/Sema.cpp:164
   isConstantEvaluatedOverride = false;
+  RebuildingImmediateInvocation = false;
 
----------------
Consider using a default member initializer instead.


================
Comment at: clang/lib/Sema/SemaDecl.cpp:8835-8836
+      // specifier.
+      if (ConstexprKind == CSK_consteval)
+        if (CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(NewFD))
+          if (MD->getOverloadedOperator() == OO_New ||
----------------
Please add braces here.


================
Comment at: clang/lib/Sema/SemaDecl.cpp:8836
+      if (ConstexprKind == CSK_consteval)
+        if (CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(NewFD))
+          if (MD->getOverloadedOperator() == OO_New ||
----------------
This rule also applies to non-member allocator functions. (Don't check for a 
`CXXMethodDecl` here.)


================
Comment at: clang/lib/Sema/SemaDeclCXX.cpp:11188-11189
 
+  ConstexprSpecKind ConstexprKind = DetermineSpecialMemberConstexprKind(
+      Constexpr, ClassDecl->hasDefaultedConstevalDefaultCtor());
+
----------------
Tyker wrote:
> rsmith wrote:
> > I don't think this is necessary: implicit special members are never 
> > `consteval`. (We use `DeclareImplicitDefaultConstructor` when there is no 
> > user-declared default constructor, in which case there can't possibly be a 
> > defaulted consteval default constructor.) I don't think you need any of the 
> > `DetermineSpecialMemberConstexprKind` machinery, nor the tracking of 
> > `DefaultedSpecialMemberIsConsteval` in `CXXRecordDecl`.
> all the code you mention is to fixe an issue i saw while working on consteval.
> the AST of
> ```
> struct A {
>     consteval A() = default;
>     consteval A(int) { }
> };
> ```
> is
> ```
> TranslationUnitDecl
> `-CXXRecordDecl <line:1:1, line:4:1> line:1:8 struct A definition
>   |-DefinitionData pass_in_registers empty standard_layout trivially_copyable 
> trivial literal has_user_declared_ctor has_constexpr_non_copy_move_ctor 
> can_const_default_init
>   | |-DefaultConstructor exists trivial constexpr defaulted_is_constexpr
>   | |-CopyConstructor simple trivial has_const_param needs_implicit 
> implicit_has_const_param
>   | |-MoveConstructor exists simple trivial needs_implicit
>   | |-CopyAssignment trivial has_const_param needs_implicit 
> implicit_has_const_param
>   | |-MoveAssignment exists simple trivial needs_implicit
>   | `-Destructor simple irrelevant trivial needs_implicit
>   |-CXXRecordDecl <col:1, col:8> col:8 implicit referenced struct A
>   |-CXXConstructorDecl <line:2:5, col:27> col:15 constexpr A 'void ()' 
> default trivial noexcept-unevaluated 0x55585ae25160
>   `-CXXConstructorDecl <line:3:5, col:24> col:15 consteval A 'void (int)'
>     |-ParmVarDecl <col:17> col:20 'int'
>     `-CompoundStmt <col:22, col:24>
> ```
> [[ https://godbolt.org/z/oRx0Ss | godbolt ]]
> notice that `A::A()` is marked as constexpr instead of consteval.
> and `A::A(int)` is marked correctly.
> 
> this is because consteval defaulted special members are not marked as 
> consteval.
> those code changes are intended to fix that. with this patch both constructor 
> are marked as consteval
> 
This change is incorrect: an implicit special member is never `consteval`. 
(Note that implicit special members are the ones the compiler declares for you, 
not the ones you get from `= default`.) We don't need this, and as a 
consequence we don't need `DefaultedSpecialMemberIsConsteval` on 
`DefinitionData`.


================
Comment at: clang/lib/Sema/SemaExpr.cpp:15192
+      FD = cast<FunctionDecl>(Call->getCalleeDecl());
+    else if (auto *Call = dyn_cast<CXXMemberCallExpr>(InnerExpr))
+      FD = Call->getMethodDecl();
----------------
This is unreachable; a `CXXMemberCallExpr` is a `CallExpr` so this was already 
handled above. Can you just delete this case?


================
Comment at: clang/lib/Sema/SemaExpr.cpp:15197
+    else
+      llvm_unreachable("unhandeld decl kind");
+    assert(FD->isConsteval());
----------------
Typo unhandeld -> unhandled


================
Comment at: clang/lib/Sema/SemaExpr.cpp:15248
+    bool ReplacingOriginal() { return true; }
+  } Transfromer(SemaRef, Rec.ReferenceToConsteval,
+                Rec.ImmediateInvocationCandidates, It);
----------------
Transfromer -> Transformer


================
Comment at: clang/lib/Sema/TreeTransform.h:3451-3452
 
-  if (auto *FE = dyn_cast<FullExpr>(Init))
-    Init = FE->getSubExpr();
-
----------------
I'm surprised that we can remove this; I'd expect this to be necessary to 
properly handle transforming cases such as

```
struct X { X(int); ~X(); int n; };
struct Y { explicit Y(int); };
template<typename T> void f() {
  Y y(X(sizeof(sizeof(T))).n);
}
template void f<int>();
```

... where we need to strip off an `ExprWithCleanups` to find the syntactic 
initializer inside it.

If you want to special-case this from `ComplexRemove`, to make sure you see all 
`ConstantExpr`s, can you override `TransformInitializer` there instead of 
changing it globally?


CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D63960/new/

https://reviews.llvm.org/D63960



_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits
  • [PATCH] D63960: [... Richard Smith - zygoloid via Phabricator via cfe-commits

Reply via email to