c++: Fix ICE with implicit operator== [PR94462]
authorJason Merrill <jason@redhat.com>
Mon, 6 Apr 2020 22:19:07 +0000 (18:19 -0400)
committerJason Merrill <jason@redhat.com>
Tue, 7 Apr 2020 04:11:44 +0000 (00:11 -0400)
duplicate_decls assumed that any TREE_ARTIFICIAL function at namespace scope
was a built-in function, but now in C++20 it's possible to have an
implicitly declared hidden friend operator==.  We just need to move the
assert into the if condition.

gcc/cp/ChangeLog
2020-04-06  Jason Merrill  <jason@redhat.com>

PR c++/94462
* decl.c (duplicate_decls): Fix handling of DECL_HIDDEN_FRIEND_P.

gcc/cp/ChangeLog
gcc/cp/decl.c
gcc/testsuite/g++.dg/cpp2a/spaceship-eq9.C [new file with mode: 0644]

index fc75879d14fdbf5d929bbd5a328b8e0924330fd5..02fd9cff79ddc87f397a90607ab40d881a360dc1 100644 (file)
@@ -1,3 +1,8 @@
+2020-04-06  Jason Merrill  <jason@redhat.com>
+
+       PR c++/94462
+       * decl.c (duplicate_decls): Fix handling of DECL_HIDDEN_FRIEND_P.
+
 2020-04-04  Marek Polacek  <polacek@redhat.com>
            Jason Merrill  <jason@redhat.com>
 
index 69a238997b467f9236878be1b0460637740626a0..a127734af6937adcb22db8e76900166230afa9d8 100644 (file)
@@ -1451,9 +1451,10 @@ duplicate_decls (tree newdecl, tree olddecl, bool newdecl_is_friend)
 
   /* Check for redeclaration and other discrepancies.  */
   if (TREE_CODE (olddecl) == FUNCTION_DECL
-      && DECL_ARTIFICIAL (olddecl))
+      && DECL_ARTIFICIAL (olddecl)
+      /* A C++20 implicit friend operator== uses the normal path (94462).  */
+      && !DECL_HIDDEN_FRIEND_P (olddecl))
     {
-      gcc_assert (!DECL_HIDDEN_FRIEND_P (olddecl));
       if (TREE_CODE (newdecl) != FUNCTION_DECL)
        {
          /* Avoid warnings redeclaring built-ins which have not been
diff --git a/gcc/testsuite/g++.dg/cpp2a/spaceship-eq9.C b/gcc/testsuite/g++.dg/cpp2a/spaceship-eq9.C
new file mode 100644 (file)
index 0000000..4f5df22
--- /dev/null
@@ -0,0 +1,17 @@
+// PR c++/94462
+// { dg-do compile { target c++2a } }
+
+namespace std {
+  struct strong_ordering { };
+}
+
+namespace Synth {
+  struct B {
+    friend std::strong_ordering operator<=>(B, B) = default;
+  };
+
+  struct C {
+    friend bool operator==(C, C);
+  };
+}
+