re PR c++/36871 (__has_nothrow_copy(T) false for T with a template ctor)
authorPaolo Carlini <paolo.carlini@oracle.com>
Mon, 21 Jul 2008 16:18:38 +0000 (16:18 +0000)
committerPaolo Carlini <paolo@gcc.gnu.org>
Mon, 21 Jul 2008 16:18:38 +0000 (16:18 +0000)
/cp
2008-07-21  Paolo Carlini  <paolo.carlini@oracle.com>

PR c++/36871
PR c++/36872
* semantics.c (classtype_has_nothrow_assign_or_copy_p): Only check
copy constructors and copy assignment operators proper.

/testsuite
2008-07-21  Paolo Carlini  <paolo.carlini@oracle.com>

        PR c++/36871
PR c++/36872
        * g++.dg/ext/has_nothrow_copy.C: Rename to...
        * g++.dg/ext/has_nothrow_copy-1.C: ... this.
        * g++.dg/ext/has_nothrow_copy-2.C: New.
        * g++.dg/ext/has_nothrow_copy-3.C: Likewise.
        * g++.dg/ext/has_nothrow_copy-4.C: Likewise.
        * g++.dg/ext/has_nothrow_copy-5.C: Likewise.
        * g++.dg/ext/has_nothrow_copy-6.C: Likewise.
        * g++.dg/ext/has_nothrow_copy-7.C: Likewise.

From-SVN: r138034

gcc/cp/ChangeLog
gcc/cp/semantics.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/ext/has_nothrow_copy-1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-6.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy-7.C [new file with mode: 0644]
gcc/testsuite/g++.dg/ext/has_nothrow_copy.C [deleted file]

index 4e3bb4cd8984697047abb3e46808accaff5d9d9e..4f9e10c709d0e3159cd322d84c826391b2ddb134 100644 (file)
@@ -1,3 +1,10 @@
+2008-07-21  Paolo Carlini  <paolo.carlini@oracle.com>
+
+       PR c++/36871
+       PR c++/36872
+       * semantics.c (classtype_has_nothrow_assign_or_copy_p): Only check
+       copy constructors and copy assignment operators proper.
+
 2008-07-21  Rafael Avila de Espindola  <espindola@google.com>
 
        * parser.c (cp_token): Remove in_system_header.
index 3b91ddb0f9015cd7ec9f0794470603b1acc73362..ffa6493ced494f07bcdea074693b9ba033c9ff76 100644 (file)
@@ -4677,8 +4677,20 @@ classtype_has_nothrow_assign_or_copy_p (tree type, bool assign_p)
     return false;
 
   for (; fns; fns = OVL_NEXT (fns))
-    if (!TYPE_NOTHROW_P (TREE_TYPE (OVL_CURRENT (fns))))
-      return false;
+    {
+      tree fn = OVL_CURRENT (fns);
+      if (assign_p)
+       {
+         if (copy_fn_p (fn) == 0)
+           continue;
+       }
+      else if (copy_fn_p (fn) <= 0)
+       continue;
+
+      if (!TYPE_NOTHROW_P (TREE_TYPE (fn)))
+       return false;
+    }
 
   return true;
 }
index d02787b284afc7a5010efe3a8b42ae0e059cdc9f..3f15e3890966a8ac7ecac6df0f09385d0c4cf31f 100644 (file)
@@ -1,3 +1,16 @@
+2008-07-21  Paolo Carlini  <paolo.carlini@oracle.com>
+
+        PR c++/36871
+       PR c++/36872
+        * g++.dg/ext/has_nothrow_copy.C: Rename to...
+        * g++.dg/ext/has_nothrow_copy-1.C: ... this.
+        * g++.dg/ext/has_nothrow_copy-2.C: New.
+        * g++.dg/ext/has_nothrow_copy-3.C: Likewise.
+        * g++.dg/ext/has_nothrow_copy-4.C: Likewise.
+        * g++.dg/ext/has_nothrow_copy-5.C: Likewise.
+        * g++.dg/ext/has_nothrow_copy-6.C: Likewise.
+        * g++.dg/ext/has_nothrow_copy-7.C: Likewise.
+
 2008-07-21  Thomas Koenig  <tkoenig@gcc.gnu.org>
 
        PR libfortran/36773
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-1.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-1.C
new file mode 100644 (file)
index 0000000..e8507cf
--- /dev/null
@@ -0,0 +1,140 @@
+// { dg-do "run" }
+#include <cassert>
+
+struct A
+{
+  double a;
+  double b;
+};
+
+struct B
+{
+  A a;
+};
+
+struct C
+: public A { };
+
+struct D
+{
+  D(const D&) throw() { }
+};
+
+struct E
+{
+  E(const E&) throw(int) { }
+};
+
+struct E1
+{
+  E1(const E1&) throw(int) { throw int(); }
+};
+
+struct F
+{
+  F() throw() { }
+};
+
+struct G
+{
+  G() throw(int) { throw int(); }
+};
+
+struct H
+{
+  H(H&) throw(int) { }
+};
+
+struct H1
+{
+  H1(H1&) throw(int) { throw int(); }
+};
+
+struct I
+{
+  I(I&) throw(int) { }
+  I(const I&) throw() { }
+};
+
+struct I1
+{
+  I1(I1&) throw(int) { throw int(); }
+  I1(const I1&) throw() { }
+};
+
+struct J
+{
+  J(J&) throw() { }
+  J(const J&) throw() { }
+  J(volatile J&) throw() { }
+  J(const volatile J&) throw() { }
+};
+
+template<typename T>
+  bool
+  f()
+  { return __has_nothrow_copy(T); } 
+
+template<typename T>
+  class My
+  {
+  public:
+    bool
+    f()
+    { return !!__has_nothrow_copy(T); }
+  };
+
+template<typename T>
+  class My2
+  {
+  public:
+    static const bool trait = __has_nothrow_copy(T);
+  };
+
+template<typename T>
+  const bool My2<T>::trait;
+
+template<typename T, bool b = __has_nothrow_copy(T)>
+  struct My3_help
+  { static const bool trait = b; };
+
+template<typename T, bool b>
+  const bool My3_help<T, b>::trait;
+
+template<typename T>
+  class My3
+  {
+  public:
+    bool
+    f()
+    { return My3_help<T>::trait; }
+  };
+
+#define PTEST(T) (__has_nothrow_copy(T) && f<T>() \
+                  && My<T>().f() && My2<T>::trait && My3<T>().f())
+
+#define NTEST(T) (!__has_nothrow_copy(T) && !f<T>() \
+                  && !My<T>().f() && !My2<T>::trait && !My3<T>().f())
+
+int main()
+{
+  assert (PTEST (int));
+  assert (NTEST (int (int)));
+  assert (NTEST (void));
+  assert (PTEST (A));
+  assert (PTEST (B));
+  assert (PTEST (C));
+  assert (NTEST (C[]));
+  assert (PTEST (D));
+  assert (NTEST (E));
+  assert (NTEST (E1));
+  assert (PTEST (F));
+  assert (PTEST (G));
+  assert (NTEST (H));
+  assert (NTEST (H1));
+  assert (NTEST (I));
+  assert (NTEST (I1));  
+  assert (PTEST (J));
+
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-2.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-2.C
new file mode 100644 (file)
index 0000000..276b11d
--- /dev/null
@@ -0,0 +1,12 @@
+// PR c++/36871
+// { dg-do "run" }
+#include <cassert>
+
+struct A { template <class T> A (T) throw (int); };
+struct B { B (B&) throw (); template <class T> B (T) throw (int); };
+
+int main ()
+{
+  assert (__has_nothrow_copy (A));
+  assert (__has_nothrow_copy (B));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-3.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-3.C
new file mode 100644 (file)
index 0000000..2fbcf8c
--- /dev/null
@@ -0,0 +1,13 @@
+// PR c++/36871
+// { dg-do "run" }
+#include <cassert>
+
+struct F {
+    F (const F&) throw () { }
+    template <class T> F (T) throw () { }
+};
+
+int main ()
+{
+  assert (__has_nothrow_copy (F));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-4.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-4.C
new file mode 100644 (file)
index 0000000..4bd7475
--- /dev/null
@@ -0,0 +1,13 @@
+// PR c++/36872
+// { dg-do "run" }
+#include <cassert>
+
+struct S {
+    S (const S&) throw ();
+    S (...) throw (int);
+};
+
+int main ()
+{
+  assert (__has_nothrow_copy (S));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-5.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-5.C
new file mode 100644 (file)
index 0000000..051675c
--- /dev/null
@@ -0,0 +1,13 @@
+// PR c++/36872
+// { dg-do "run" }
+#include <cassert>
+
+struct S {
+    S (const S&) throw ();
+    S (int) throw (int);
+};
+
+int main ()
+{
+  assert (__has_nothrow_copy (S));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-6.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-6.C
new file mode 100644 (file)
index 0000000..4330edd
--- /dev/null
@@ -0,0 +1,12 @@
+// { dg-do "run" }
+#include <cassert>
+
+struct S {
+    S (S&) throw ();
+    S (const S&, int) throw (int);
+};
+
+int main ()
+{
+  assert (__has_nothrow_copy (S));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy-7.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy-7.C
new file mode 100644 (file)
index 0000000..a85224c
--- /dev/null
@@ -0,0 +1,13 @@
+// { dg-do "run" }
+// { dg-options "-std=c++0x" }
+#include <cassert>
+
+struct S {
+    S (const S&) throw ();
+    S (S&&) throw (int);
+};
+
+int main ()
+{
+  assert (__has_nothrow_copy (S));
+}
diff --git a/gcc/testsuite/g++.dg/ext/has_nothrow_copy.C b/gcc/testsuite/g++.dg/ext/has_nothrow_copy.C
deleted file mode 100644 (file)
index e8507cf..0000000
+++ /dev/null
@@ -1,140 +0,0 @@
-// { dg-do "run" }
-#include <cassert>
-
-struct A
-{
-  double a;
-  double b;
-};
-
-struct B
-{
-  A a;
-};
-
-struct C
-: public A { };
-
-struct D
-{
-  D(const D&) throw() { }
-};
-
-struct E
-{
-  E(const E&) throw(int) { }
-};
-
-struct E1
-{
-  E1(const E1&) throw(int) { throw int(); }
-};
-
-struct F
-{
-  F() throw() { }
-};
-
-struct G
-{
-  G() throw(int) { throw int(); }
-};
-
-struct H
-{
-  H(H&) throw(int) { }
-};
-
-struct H1
-{
-  H1(H1&) throw(int) { throw int(); }
-};
-
-struct I
-{
-  I(I&) throw(int) { }
-  I(const I&) throw() { }
-};
-
-struct I1
-{
-  I1(I1&) throw(int) { throw int(); }
-  I1(const I1&) throw() { }
-};
-
-struct J
-{
-  J(J&) throw() { }
-  J(const J&) throw() { }
-  J(volatile J&) throw() { }
-  J(const volatile J&) throw() { }
-};
-
-template<typename T>
-  bool
-  f()
-  { return __has_nothrow_copy(T); } 
-
-template<typename T>
-  class My
-  {
-  public:
-    bool
-    f()
-    { return !!__has_nothrow_copy(T); }
-  };
-
-template<typename T>
-  class My2
-  {
-  public:
-    static const bool trait = __has_nothrow_copy(T);
-  };
-
-template<typename T>
-  const bool My2<T>::trait;
-
-template<typename T, bool b = __has_nothrow_copy(T)>
-  struct My3_help
-  { static const bool trait = b; };
-
-template<typename T, bool b>
-  const bool My3_help<T, b>::trait;
-
-template<typename T>
-  class My3
-  {
-  public:
-    bool
-    f()
-    { return My3_help<T>::trait; }
-  };
-
-#define PTEST(T) (__has_nothrow_copy(T) && f<T>() \
-                  && My<T>().f() && My2<T>::trait && My3<T>().f())
-
-#define NTEST(T) (!__has_nothrow_copy(T) && !f<T>() \
-                  && !My<T>().f() && !My2<T>::trait && !My3<T>().f())
-
-int main()
-{
-  assert (PTEST (int));
-  assert (NTEST (int (int)));
-  assert (NTEST (void));
-  assert (PTEST (A));
-  assert (PTEST (B));
-  assert (PTEST (C));
-  assert (NTEST (C[]));
-  assert (PTEST (D));
-  assert (NTEST (E));
-  assert (NTEST (E1));
-  assert (PTEST (F));
-  assert (PTEST (G));
-  assert (NTEST (H));
-  assert (NTEST (H1));
-  assert (NTEST (I));
-  assert (NTEST (I1));  
-  assert (PTEST (J));
-
-  return 0;
-}