@@ -426,6 +426,10 @@ do_friend (tree ctype, tree declarator, tree decl,
/* Every decl that gets here is a friend of something. */
DECL_FRIEND_P (decl) = 1;
+
+ if (DECL_OVERRIDE_P (decl) || DECL_FINAL_P (decl))
+ error ("friend declaration %qD may not have virt-specifiers",
+ decl);
/* Unfortunately, we have to handle attributes here. Normally we would
handle them in start_decl_1, but since this is a friend decl start_decl_1
@@ -4737,6 +4737,11 @@ push_template_decl_real (tree decl, bool is_friend)
}
else if (TREE_CODE (decl) == FUNCTION_DECL)
{
+ if (member_template_p)
+ {
+ if (DECL_OVERRIDE_P (decl) || DECL_FINAL_P (decl))
+ error ("member template %qD may not have virt-specifiers", decl);
+ }
if (DECL_DESTRUCTOR_P (decl))
{
/* [temp.mem]
@@ -52,6 +52,14 @@ struct D5 : B
void D5::g() override {} // { dg-error "not allowed outside a class definition" }
void g() override {} // { dg-error "not allowed outside a class definition" }
+struct B5
+{
+ friend void f() final; // { dg-error "may not have virt-specifiers" }
+ friend void g() override; // { dg-error "may not have virt-specifiers" }
+ template <class T> void h() final; // { dg-error "may not have virt-specifiers" }
+ template <class T> void i() override; // { dg-error "may not have virt-specifiers" }
+};
+
int main()
{
D2<B> d;