* parser.c (synthesize_implicit_template_parm): Inject new template
        argument list appropriately when a generic member function
        of a class template is declared out-of-line.

        * g++.dg/cpp1y/fn-generic-member-ool.C: New testcase.
---
 gcc/cp/parser.c                                    |  3 +-
 gcc/testsuite/g++.dg/cpp1y/fn-generic-member-ool.C | 35 ++++++++++++++++++++++
 2 files changed, 37 insertions(+), 1 deletion(-)
 create mode 100644 gcc/testsuite/g++.dg/cpp1y/fn-generic-member-ool.C

diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c
index 7e0531b..85b2f4b 100644
--- a/gcc/cp/parser.c
+++ b/gcc/cp/parser.c
@@ -31906,7 +31906,8 @@ synthesize_implicit_template_parm  (cp_parser *parser)
 
       current_binding_level = scope;
 
-      if (scope->kind != sk_template_parms)
+      if (scope->kind != sk_template_parms
+         || !function_being_declared_is_template_p (parser))
        {
          /* Introduce a new template parameter list for implicit template
             parameters.  */
diff --git a/gcc/testsuite/g++.dg/cpp1y/fn-generic-member-ool.C 
b/gcc/testsuite/g++.dg/cpp1y/fn-generic-member-ool.C
new file mode 100644
index 0000000..f9c814e
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp1y/fn-generic-member-ool.C
@@ -0,0 +1,35 @@
+// Out-of-line generic member function definitions.
+// { dg-options "-std=c++1y" }
+
+struct A {
+  void f(auto x);
+};
+
+void A::f(auto x) {}  // injects a new list
+
+template <typename T>
+struct B {
+  void f(auto x);
+};
+
+template <typename T>
+void B<T>::f(auto x) {}  // injects a new list
+
+struct C {
+  template <int N>
+  void f(auto x);
+};
+
+template <int N>
+void C::f(auto x) {}  // extends existing inner list
+
+template <typename T>
+struct D
+{
+  template <int N>
+  void f(auto x);
+};
+
+template <typename T>
+template <int N>
+void D<T>::f(auto x) {}  // extends existing inner list
-- 
1.9.0

Reply via email to