@@ -855,6 +855,7 @@ check_classfn (tree ctype, tree function, tree template_parms)
So tell check_explicit_specialization to look for a match. */
SET_DECL_IMPLICIT_INSTANTIATION (function);
+ DECL_TEMPLATE_INFO (function) = build_template_info (fns, NULL_TREE);
matched = function;
}
@@ -661,7 +661,8 @@ do_friend (tree scope, tree declarator, tree decl,
if (decl == error_mark_node)
return error_mark_node;
- if (!class_template_depth && DECL_IMPLICIT_INSTANTIATION (decl))
+ if (!class_template_depth && DECL_IMPLICIT_INSTANTIATION (decl)
+ && TREE_CODE (DECL_TI_TEMPLATE (decl)) != TEMPLATE_DECL)
/* "[if no non-template match is found,] each remaining function template
is replaced with the specialization chosen by deduction from the
friend declaration or discarded if deduction fails."
@@ -5938,6 +5938,7 @@ set_decl_namespace (tree decl, tree scope, bool friendp)
So tell check_explicit_specialization to look for a match. */
SET_DECL_IMPLICIT_INSTANTIATION (decl);
+ DECL_TEMPLATE_INFO (decl) = build_template_info (old, NULL_TREE);
return;
}
new file mode 100644
@@ -0,0 +1,19 @@
+// PR c++/109649
+
+template <typename>
+class X
+{
+ void f(){}
+};
+
+class Y
+{
+ friend void X<int>::f(); // { dg-error "private" }
+};
+
+int main()
+{
+ X<int> t;
+ t.f(); // { dg-error "private" }
+ Y b;
+}