From: Jason Merrill Date: Sun, 24 Jul 2016 02:39:41 +0000 (-0400) Subject: PR c++/71738 - nested template friend X-Git-Url: https://git.libre-soc.org/?a=commitdiff_plain;h=6970223f93ce9102cd95044f7c4d10a4ce695a77;p=gcc.git PR c++/71738 - nested template friend * pt.c (lookup_template_class_1): Handle getting template from tsubst. From-SVN: r238685 --- diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index e1e4506690e..c9a28aed253 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,5 +1,8 @@ 2016-07-23 Jason Merrill + PR c++/71738 + * pt.c (lookup_template_class_1): Handle getting template from tsubst. + PR c++/71350 * decl.c (reshape_init_r): Check complain for missing braces warning. diff --git a/gcc/cp/pt.c b/gcc/cp/pt.c index 3ee53d1b2f0..a44beadd8af 100644 --- a/gcc/cp/pt.c +++ b/gcc/cp/pt.c @@ -8601,7 +8601,9 @@ lookup_template_class_1 (tree d1, tree arglist, tree in_decl, tree context, for parameters in the TYPE_DECL of the alias template done earlier. So be careful while getting the template of FOUND. */ - found = TREE_CODE (found) == TYPE_DECL + found = TREE_CODE (found) == TEMPLATE_DECL + ? found + : TREE_CODE (found) == TYPE_DECL ? TYPE_TI_TEMPLATE (TREE_TYPE (found)) : CLASSTYPE_TI_TEMPLATE (found); } diff --git a/gcc/testsuite/g++.dg/template/friend63.C b/gcc/testsuite/g++.dg/template/friend63.C new file mode 100644 index 00000000000..f3a292c3fd3 --- /dev/null +++ b/gcc/testsuite/g++.dg/template/friend63.C @@ -0,0 +1,29 @@ +// PR c++/71738 + +template < class > struct S +{ + template < class > struct A + { + template < class > struct B + { + template + void operator=(Z) { S::i = 0; } + }; + }; + + // Note that this friend declaration is useless, since nested classes are + // already friends of their enclosing class. + template < class X > + template < class Y > + template < class Z > + friend void A < X >::B < Y >::operator= (Z); + +private: + static int i; +}; + +int main() +{ + S::A::B b; + b = 0; +}