c++: ->template and implicit typedef [PR104608]
authorMarek Polacek <polacek@redhat.com>
Tue, 22 Feb 2022 20:03:34 +0000 (15:03 -0500)
committerMarek Polacek <polacek@redhat.com>
Thu, 10 Mar 2022 16:23:41 +0000 (11:23 -0500)
Here we have a forward declaration of Parameter for which we create
an implicit typedef, which is a TYPE_DECL.  Then, when looking it up
at template definition time, cp_parser_template_id gets (since r12-6754)
this TYPE_DECL which it can't handle.

This patch defers lookup for TYPE_DECLs that cp_parser_template_id can't
handle, a la r12-6879.

PR c++/104608

gcc/cp/ChangeLog:

* parser.cc (cp_parser_template_name): Repeat lookup of
TYPE_DECLs.

gcc/testsuite/ChangeLog:

* g++.dg/parse/template-keyword3.C: New test.
* g++.dg/parse/template-keyword4.C: New test.

gcc/cp/parser.cc
gcc/testsuite/g++.dg/parse/template-keyword3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/parse/template-keyword4.C [new file with mode: 0644]

index 18db9d4..87b9781 100644 (file)
@@ -18674,7 +18674,10 @@ cp_parser_template_name (cp_parser* parser,
          return error_mark_node;
        }
       else if ((!DECL_P (decl) && !is_overloaded_fn (decl))
-              || TREE_CODE (decl) == USING_DECL)
+              || TREE_CODE (decl) == USING_DECL
+              /* cp_parser_template_id can only handle some TYPE_DECLs.  */
+              || (TREE_CODE (decl) == TYPE_DECL
+                  && TREE_CODE (TREE_TYPE (decl)) != TYPENAME_TYPE))
        /* Repeat the lookup at instantiation time.  */
        decl = identifier;
     }
diff --git a/gcc/testsuite/g++.dg/parse/template-keyword3.C b/gcc/testsuite/g++.dg/parse/template-keyword3.C
new file mode 100644 (file)
index 0000000..59fe0fc
--- /dev/null
@@ -0,0 +1,12 @@
+// PR c++/104608
+
+class Parameter;
+template <typename R> class Function 
+: public R  
+{
+    Function();
+};
+template <typename R>
+Function<R>::Function() {
+    this->template Parameter<R>();
+}
diff --git a/gcc/testsuite/g++.dg/parse/template-keyword4.C b/gcc/testsuite/g++.dg/parse/template-keyword4.C
new file mode 100644 (file)
index 0000000..c688094
--- /dev/null
@@ -0,0 +1,17 @@
+// PR c++/104608
+// { dg-do compile { target c++11 } }
+
+class S;
+using Parameter = S;
+typedef S Parameter2;
+
+template <typename R> class Function 
+: public R  
+{
+    Function();
+};
+template <typename R>
+Function<R>::Function() {
+    this->template Parameter<R>();
+    this->template Parameter2<R>();
+}