aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Merrill <jason@redhat.com>2022-03-27 22:31:51 -0400
committerJason Merrill <jason@redhat.com>2022-04-05 12:28:59 -0400
commit1de6612d994ada8edaab18bbc6afd8e9a57413aa (patch)
tree89462116fe7189eb55b70b0692b66519217668f9
parent5c8d22b00adedc21f8b697dd6b990f4821a06634 (diff)
downloadgcc-1de6612d994ada8edaab18bbc6afd8e9a57413aa.zip
gcc-1de6612d994ada8edaab18bbc6afd8e9a57413aa.tar.gz
gcc-1de6612d994ada8edaab18bbc6afd8e9a57413aa.tar.bz2
c++: elaborated-type-spec in requires-expr [PR101677]
We were failing to declare class S in the global namespace because we were treating the requires-expression parameter scope as a normal block scope, so the implicit declaration went there. It seems to me that the requires parameter scope is more like a function parameter scope (not least in the use of the word "parameter"), so let's change the scope kind. But then we need to adjust the prohibition on placeholders declaring implicit template parameters. PR c++/101677 gcc/cp/ChangeLog: * name-lookup.h (struct cp_binding_level): Add requires_expression bit-field. * parser.cc (cp_parser_requires_expression): Set it. (synthesize_implicit_template_parm): Check it. gcc/testsuite/ChangeLog: * g++.dg/cpp2a/concepts-pr67178.C: Adjust error. * g++.dg/cpp2a/concepts-requires28.C: New test.
-rw-r--r--gcc/cp/name-lookup.h5
-rw-r--r--gcc/cp/parser.cc5
-rw-r--r--gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C2
-rw-r--r--gcc/testsuite/g++.dg/cpp2a/concepts-requires28.C13
4 files changed, 21 insertions, 4 deletions
diff --git a/gcc/cp/name-lookup.h b/gcc/cp/name-lookup.h
index 68d0872..fa03902 100644
--- a/gcc/cp/name-lookup.h
+++ b/gcc/cp/name-lookup.h
@@ -309,7 +309,10 @@ struct GTY(()) cp_binding_level {
/* true for SK_FUNCTION_PARMS of immediate functions. */
unsigned immediate_fn_ctx_p : 1;
- /* 22 bits left to fill a 32-bit word. */
+ /* True for SK_FUNCTION_PARMS of a requires-expression. */
+ unsigned requires_expression: 1;
+
+ /* 21 bits left to fill a 32-bit word. */
};
/* The binding level currently in effect. */
diff --git a/gcc/cp/parser.cc b/gcc/cp/parser.cc
index 7e1c777..bfd16e1 100644
--- a/gcc/cp/parser.cc
+++ b/gcc/cp/parser.cc
@@ -29964,7 +29964,8 @@ cp_parser_requires_expression (cp_parser *parser)
scope_sentinel ()
{
++cp_unevaluated_operand;
- begin_scope (sk_block, NULL_TREE);
+ begin_scope (sk_function_parms, NULL_TREE);
+ current_binding_level->requires_expression = true;
}
~scope_sentinel ()
@@ -48082,7 +48083,7 @@ static tree
synthesize_implicit_template_parm (cp_parser *parser, tree constr)
{
/* A requires-clause is not a function and cannot have placeholders. */
- if (current_binding_level->kind == sk_block)
+ if (current_binding_level->requires_expression)
{
error ("placeholder type not allowed in this context");
return error_mark_node;
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C
index c74f6f0..bdd5f96 100644
--- a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C
+++ b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C
@@ -12,7 +12,7 @@ concept C0 = requires (auto x) { // { dg-error "placeholder type" }
template<typename T>
concept C1 = requires (C1 auto x) { // { dg-error "not been declared|placeholder|two or more|in requirements" }
x; // { dg-error "not declared" }
- { x } -> c; // { dg-message "is invalid" }
+ { x } -> c; // { dg-message "is invalid|not declared" }
};
template<typename T>
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires28.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires28.C
new file mode 100644
index 0000000..e632f01
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp2a/concepts-requires28.C
@@ -0,0 +1,13 @@
+// PR c++/101677
+// { dg-do compile { target c++20 } }
+
+template<class T>
+concept C_bug_with_forward_decl = requires(T& t){
+ t.template f<class S>();
+};
+
+struct good {
+ template<class T> void f() {}
+};
+
+static_assert(C_bug_with_forward_decl<good>);