c++: premature requires-expression folding [PR95020]
authorPatrick Palka <ppalka@redhat.com>
Wed, 13 May 2020 20:40:10 +0000 (16:40 -0400)
committerPatrick Palka <ppalka@redhat.com>
Wed, 13 May 2020 20:40:10 +0000 (16:40 -0400)
In the testcase below we're prematurely folding away the
requires-expression to 'true' after substituting in the function's
template arguments, but before substituting in the lambda's deduced
template arguments.

This patch removes the uses_template_parms check when deciding in
tsubst_requires_expr whether to keep around a new requires-expression.
Regardless of whether the template arguments are dependent, there still
might be more template parameters to later substitute in (as in the
below testcase) and even if not, tsubst_expr doesn't perform full
semantic processing unless !processing_template_decl, so we should still
wait until then to fold away the requires-expression.

gcc/cp/ChangeLog:

PR c++/95020
* constraint.c (tsubst_requires_expr): Produce a new
requires-expression when processing_template_decl, even if
template arguments are not dependent.

gcc/testsuite/ChangeLog:

PR c++/95020
* g++/cpp2a/concepts-lambda7.C: New test.

gcc/cp/ChangeLog
gcc/cp/constraint.cc
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp2a/concepts-lambda7.C [new file with mode: 0644]

index 145a5d577bbf038b77fb18eba37cc91953fdc12a..73a6500c8736376f3f5744d8d30fc5b0498b4a7f 100644 (file)
@@ -1,3 +1,10 @@
+2020-05-13  Patrick Palka  <ppalka@redhat.com>
+
+       PR c++/95020
+       * constraint.c (tsubst_requires_expr): Produce a new
+       requires-expression when processing_template_decl, even if
+       template arguments are not dependent.
+
 2020-05-13  Marek Polacek  <polacek@redhat.com>
 
        PR c++/95066
index 3a17005232e5ed84d1c1ec92651f9369040b6bfa..eb72bfe59368e9c74923e1685d6c603d312b50df 100644 (file)
@@ -2173,9 +2173,7 @@ tsubst_requires_expr (tree t, tree args,
   if (reqs == error_mark_node)
     return boolean_false_node;
 
-  /* In certain cases, produce a new requires-expression.
-     Otherwise the value of the expression is true.  */
-  if (processing_template_decl && uses_template_parms (args))
+  if (processing_template_decl)
     return finish_requires_expr (cp_expr_location (t), parms, reqs);
 
   return boolean_true_node;
index e2dea4dfecb87bf9824dd1bcab82e3c90dfce9cd..b88e1823c0c7f36cb7be87feaed7e2c194050e21 100644 (file)
@@ -1,3 +1,8 @@
+2020-05-13  Patrick Palka  <ppalka@redhat.com>
+
+       PR c++/95020
+       * g++/cpp2a/concepts-lambda7.C: New test.
+
 2020-05-13  Marek Polacek  <polacek@redhat.com>
 
        PR c++/95066
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-lambda7.C b/gcc/testsuite/g++.dg/cpp2a/concepts-lambda7.C
new file mode 100644 (file)
index 0000000..50746b7
--- /dev/null
@@ -0,0 +1,14 @@
+// PR c++/95020
+// { dg-do compile { target c++2a } }
+
+template<typename>
+void foo() {
+  auto t = [](auto v) {
+    static_assert(requires { *v; }); // { dg-error "static assertion failed" }
+  };
+  t(0);
+}
+
+void bar() {
+  foo<void>();
+}