From: Richard Smith Date: Wed, 26 Sep 2018 04:36:55 +0000 (+0000) Subject: P0859R0: List-initialization is potentially-constant-evaluated and X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=471124bcd92370793440cd7290e3adb1f8006227;p=clang P0859R0: List-initialization is potentially-constant-evaluated and triggers instantiation of constexpr functions. We mostly implemented this since Clang 6, but missed the template instantiation case. We do not implement the '&cast-expression' special case. It appears to be a mistake / oversight. I've mailed CWG to see if we can remove it. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@343064 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/TreeTransform.h b/lib/Sema/TreeTransform.h index 2bee5e7ec8..3ec467a39c 100644 --- a/lib/Sema/TreeTransform.h +++ b/lib/Sema/TreeTransform.h @@ -3386,6 +3386,11 @@ ExprResult TreeTransform::TransformInitializer(Expr *Init, if (Construct && Construct->isStdInitListInitialization()) return TransformInitializer(Construct->getArg(0), NotCopyInit); + // Enter a list-init context if this was list initialization. + EnterExpressionEvaluationContext Context( + getSema(), EnterExpressionEvaluationContext::InitList, + Construct->isListInitialization()); + SmallVector NewArgs; bool ArgChanged = false; if (getDerived().TransformExprs(Construct->getArgs(), Construct->getNumArgs(), @@ -9549,6 +9554,9 @@ TreeTransform::TransformInitListExpr(InitListExpr *E) { bool InitChanged = false; + EnterExpressionEvaluationContext Context( + getSema(), EnterExpressionEvaluationContext::InitList); + SmallVector Inits; if (getDerived().TransformExprs(E->getInits(), E->getNumInits(), false, Inits, &InitChanged)) @@ -10780,9 +10788,14 @@ TreeTransform::TransformCXXConstructExpr(CXXConstructExpr *E) { bool ArgumentChanged = false; SmallVector Args; - if (getDerived().TransformExprs(E->getArgs(), E->getNumArgs(), true, Args, - &ArgumentChanged)) - return ExprError(); + { + EnterExpressionEvaluationContext Context( + getSema(), EnterExpressionEvaluationContext::InitList, + E->isListInitialization()); + if (getDerived().TransformExprs(E->getArgs(), E->getNumArgs(), true, Args, + &ArgumentChanged)) + return ExprError(); + } if (!getDerived().AlwaysRebuild() && T == E->getType() && @@ -10865,9 +10878,14 @@ TreeTransform::TransformCXXTemporaryObjectExpr( bool ArgumentChanged = false; SmallVector Args; Args.reserve(E->getNumArgs()); - if (TransformExprs(E->getArgs(), E->getNumArgs(), true, Args, - &ArgumentChanged)) - return ExprError(); + { + EnterExpressionEvaluationContext Context( + getSema(), EnterExpressionEvaluationContext::InitList, + E->isListInitialization()); + if (TransformExprs(E->getArgs(), E->getNumArgs(), true, Args, + &ArgumentChanged)) + return ExprError(); + } if (!getDerived().AlwaysRebuild() && T == E->getTypeSourceInfo() && @@ -11159,9 +11177,14 @@ TreeTransform::TransformCXXUnresolvedConstructExpr( bool ArgumentChanged = false; SmallVector Args; Args.reserve(E->arg_size()); - if (getDerived().TransformExprs(E->arg_begin(), E->arg_size(), true, Args, - &ArgumentChanged)) - return ExprError(); + { + EnterExpressionEvaluationContext Context( + getSema(), EnterExpressionEvaluationContext::InitList, + E->isListInitialization()); + if (getDerived().TransformExprs(E->arg_begin(), E->arg_size(), true, Args, + &ArgumentChanged)) + return ExprError(); + } if (!getDerived().AlwaysRebuild() && T == E->getTypeSourceInfo() && diff --git a/test/CXX/expr/expr.const/p6.cpp b/test/CXX/expr/expr.const/p6.cpp new file mode 100644 index 0000000000..a8fc7754e7 --- /dev/null +++ b/test/CXX/expr/expr.const/p6.cpp @@ -0,0 +1,39 @@ +// RUN: %clang_cc1 -std=c++17 -verify %s + +template int not_constexpr() { return T::error; } +template constexpr int is_constexpr() { return T::error; } // expected-error {{'::'}} + +template int not_constexpr_var = T::error; +template constexpr int is_constexpr_var = T::error; // expected-error {{'::'}} +template const int is_const_var = T::error; // expected-error {{'::'}} +template const volatile int is_const_volatile_var = T::error; +template T is_dependent_var = T::error; // expected-error {{'::'}} +template int &is_reference_var = T::error; // expected-error {{'::'}} +template float is_float_var = T::error; + +void test() { + // Do not instantiate functions referenced in unevaluated operands... + (void)sizeof(not_constexpr()); + (void)sizeof(is_constexpr()); + (void)sizeof(not_constexpr_var); + (void)sizeof(is_constexpr_var); + (void)sizeof(is_const_var); + (void)sizeof(is_const_volatile_var); + (void)sizeof(is_dependent_var); + (void)sizeof(is_dependent_var); + (void)sizeof(is_reference_var); + (void)sizeof(is_float_var); + + // ... but do if they are potentially constant evaluated, and refer to + // constexpr functions or to variables usable in constant expressions. + (void)sizeof(int{not_constexpr()}); + (void)sizeof(int{is_constexpr()}); // expected-note {{instantiation of}} + (void)sizeof(int{not_constexpr_var}); + (void)sizeof(int{is_constexpr_var}); // expected-note {{instantiation of}} + (void)sizeof(int{is_const_var}); // expected-note {{instantiation of}} + (void)sizeof(int{is_const_volatile_var}); + (void)sizeof(int{is_dependent_var}); + (void)sizeof(int{is_dependent_var}); // expected-note {{instantiation of}} + (void)sizeof(int{is_reference_var}); // expected-note {{instantiation of}} + (void)sizeof(int{is_float_var}); // expected-error {{cannot be narrowed}} expected-note {{cast}} +} diff --git a/test/CXX/temp/temp.spec/temp.inst/p7.cpp b/test/CXX/temp/temp.spec/temp.inst/p7.cpp new file mode 100644 index 0000000000..5145dafdc5 --- /dev/null +++ b/test/CXX/temp/temp.spec/temp.inst/p7.cpp @@ -0,0 +1,12 @@ +// RUN: %clang_cc1 -verify -std=c++17 %s + +template constexpr int f() { return T::value; } // expected-error {{'::'}} +template void g(decltype(B ? f() : 0)); +template void g(...); +template void h(decltype(int{B ? f() : 0})); // expected-note {{instantiation of}} +template void h(...); +void x() { + g(0); // ok + g(0); // ok + h(0); // expected-note {{while substituting}} +} diff --git a/www/cxx_status.html b/www/cxx_status.html index 0036f424f7..0d0add1584 100755 --- a/www/cxx_status.html +++ b/www/cxx_status.html @@ -207,7 +207,7 @@ with libc++ or with gcc's libstdc++. P0859R0 (DR) - No + SVN Alignment support