]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
re PR c++/62101 (deleted definitions of friend functions are rejected)
authorVille Voutilainen <ville.voutilainen@gmail.com>
Thu, 14 Aug 2014 17:11:26 +0000 (20:11 +0300)
committerJason Merrill <jason@gcc.gnu.org>
Thu, 14 Aug 2014 17:11:26 +0000 (13:11 -0400)
PR c++/62101
* decl.c (grokdeclarator): Move the check for friend initializers..
* decl2.c (grokfield) ..here. Postpone early return for friends
until after the initializer check.

From-SVN: r213974

gcc/cp/ChangeLog
gcc/cp/decl.c
gcc/cp/decl2.c
gcc/testsuite/g++.dg/cpp0x/pr62101.C [new file with mode: 0644]

index 06b004bae35d3e74fb46b652599c28505fb4b540..a256953912452a51e7ae9ab16ee6c0d950d1b83d 100644 (file)
@@ -1,3 +1,10 @@
+2014-08-14  Ville Voutilainen  <ville.voutilainen@gmail.com>
+
+       PR c++/62101
+       * decl.c (grokdeclarator): Move the check for friend initializers..
+       * decl2.c (grokfield) ..here. Postpone early return for friends
+       until after the initializer check.
+
 2014-08-14  Paolo Carlini  <paolo.carlini@oracle.com>
 
        PR c++/54377
index 79e7362497358bf32890aaa9f912186e7f3b7285..92a6dbc96c7b591183ddc53d288d1987bf5b9b42 100644 (file)
@@ -9765,8 +9765,6 @@ grokdeclarator (const cp_declarator *declarator,
              }
            else if (friendp)
              {
-               if (initialized)
-                 error ("can%'t initialize friend function %qs", name);
                if (virtualp)
                  {
                    /* Cannot be both friend and virtual.  */
index 1740a2e7340acf474e8c25c32decb0ebc24c899a..de28cb712c3aa968b363834ee2ddec1c26b747db 100644 (file)
@@ -870,11 +870,6 @@ grokfield (const cp_declarator *declarator,
   if (value == void_type_node)
     return value;
 
-  /* Pass friend decls back.  */
-  if ((TREE_CODE (value) == FUNCTION_DECL
-       || TREE_CODE (value) == TEMPLATE_DECL)
-      && DECL_CONTEXT (value) != current_class_type)
-    return value;
 
   name = DECL_NAME (value);
 
@@ -926,7 +921,9 @@ grokfield (const cp_declarator *declarator,
       return value;
     }
 
-  if (DECL_IN_AGGR_P (value))
+  int friendp = decl_spec_seq_has_spec_p (declspecs, ds_friend);
+
+  if (!friendp && DECL_IN_AGGR_P (value))
     {
       error ("%qD is already defined in %qT", value, DECL_CONTEXT (value));
       return void_type_node;
@@ -939,8 +936,6 @@ grokfield (const cp_declarator *declarator,
     {
       if (TREE_CODE (value) == FUNCTION_DECL)
        {
-         /* Initializers for functions are rejected early in the parser.
-            If we get here, it must be a pure specifier for a method.  */
          if (init == ridpointers[(int)RID_DELETE])
            {
              DECL_DELETED_FN (value) = 1;
@@ -971,8 +966,12 @@ grokfield (const cp_declarator *declarator,
          else
            {
              gcc_assert (TREE_CODE (TREE_TYPE (value)) == FUNCTION_TYPE);
-             error ("initializer specified for static member function %qD",
-                    value);
+             if (friendp)
+               error ("initializer specified for friend function %qD",
+                      value);
+             else
+               error ("initializer specified for static member function %qD",
+                      value);
            }
        }
       else if (TREE_CODE (value) == FIELD_DECL)
@@ -981,6 +980,12 @@ grokfield (const cp_declarator *declarator,
        gcc_unreachable ();
     }
 
+  /* Pass friend decls back.  */
+  if ((TREE_CODE (value) == FUNCTION_DECL
+       || TREE_CODE (value) == TEMPLATE_DECL)
+      && DECL_CONTEXT (value) != current_class_type)
+    return value;
+
   if (processing_template_decl && VAR_OR_FUNCTION_DECL_P (value))
     {
       value = push_template_decl (value);
diff --git a/gcc/testsuite/g++.dg/cpp0x/pr62101.C b/gcc/testsuite/g++.dg/cpp0x/pr62101.C
new file mode 100644 (file)
index 0000000..2c05dd5
--- /dev/null
@@ -0,0 +1,32 @@
+// PR c++/62101
+// { dg-do compile { target c++11 } }
+
+struct X
+{
+  friend void g(X, int) = 0; // { dg-error "initializer specified for friend function" }
+  friend void g(X, int) = default; // { dg-error "cannot be defaulted" }
+  // { dg-prune-output "note" }
+  friend void f(X, int) = delete;
+  friend void f(X, double) {}
+};
+
+struct Y;
+void g(Y, int);
+void g(Y, double);
+
+struct Y
+{
+  // { dg-prune-output "note" }
+  friend void g(Y, int) = delete;
+  friend void g(Y, double) {}
+};
+
+int main()
+{
+  X x;
+  f(x, 5.0);
+  f(x, 5); // { dg-error "use of deleted function" }
+  Y y;
+  g(y, 5.0);
+  g(y, 5); // { dg-error "use of deleted function" }
+}