re PR libstdc++/70437 (Instantiation loop with pair and is_constructible)
authorVille Voutilainen <ville.voutilainen@gmail.com>
Tue, 5 Apr 2016 11:31:30 +0000 (14:31 +0300)
committerVille Voutilainen <ville@gcc.gnu.org>
Tue, 5 Apr 2016 11:31:30 +0000 (14:31 +0300)
  PR libstdc++/70437
  * include/bits/stl_pair.h (_ConstructiblePair,
_ImplicitlyConvertiblePair, _MoveConstructiblePair,
_ImplicitlyMoveConvertiblePair): Add shortcut conditions
for same-type cases.
* testsuite/20_util/pair/70437.cc: New.

From-SVN: r234743

libstdc++-v3/ChangeLog
libstdc++-v3/include/bits/stl_pair.h
libstdc++-v3/testsuite/20_util/pair/70437.cc [new file with mode: 0644]

index 4e6db2ab3db8bc3238ddfdb2cfad03b9a38b7d3d..2977bd0938a75c01a1c5cf9c5d346308149edc00 100644 (file)
@@ -1,3 +1,12 @@
+2016-04-05  Ville Voutilainen  <ville.voutilainen@gmail.com>
+
+       PR libstdc++/70437
+       * include/bits/stl_pair.h (_ConstructiblePair,
+       _ImplicitlyConvertiblePair, _MoveConstructiblePair,
+       _ImplicitlyMoveConvertiblePair): Add shortcut conditions
+       for same-type cases.
+       * testsuite/20_util/pair/70437.cc: New.
+
 2016-03-24  Jonathan Wakely  <jwakely@redhat.com>
 
        PR libstdc++/69945
index 7057030d9d7f0c327199d48c6947ffdc3ba76844..37ee5cc405328a5df727deb5a69418b65ae3cdcd 100644 (file)
@@ -87,32 +87,51 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
 
   // Concept utility functions, reused in conditionally-explicit
   // constructors.
+  // See PR 70437, don't look at is_constructible or
+  // is_convertible if the decayed types are the same to
+  // avoid querying those properties for incomplete types.
   template <typename _T1, typename _T2, typename _U1, typename _U2>
   constexpr bool _ConstructiblePair()
   {
-    return __and_<is_constructible<_T1, const _U1&>,
-                 is_constructible<_T2, const _U2&>>::value;
+    return __and_<__or_<is_same<typename decay<_T1>::type,
+                               typename decay<_U1>::type>,
+                       is_constructible<_T1, const _U1&>>,
+                 __or_<is_same<typename decay<_T2>::type,
+                               typename decay<_U2>::type>,
+                       is_constructible<_T2, const _U2&>>>::value;
   }
 
   template <typename _T1, typename _T2, typename _U1, typename _U2>
   constexpr bool _ImplicitlyConvertiblePair()
   {
-    return __and_<is_convertible<const _U1&, _T1>,
-                 is_convertible<const _U2&, _T2>>::value;
+    return __and_<__or_<is_same<typename decay<_T1>::type,
+                               typename decay<_U1>::type>,
+                       is_convertible<const _U1&, _T1>>,
+                 __or_<is_same<typename decay<_T2>::type,
+                               typename decay<_U2>::type>,
+                      is_convertible<const _U2&, _T2>>>::value;
   }
 
   template <typename _T1, typename _T2, typename _U1, typename _U2>
   constexpr bool _MoveConstructiblePair()
   {
-    return __and_<is_constructible<_T1, _U1&&>,
-                 is_constructible<_T2, _U2&&>>::value;
+    return __and_<__or_<is_same<typename decay<_T1>::type,
+                               typename decay<_U1>::type>,
+                       is_constructible<_T1, _U1&&>>,
+                 __or_<is_same<typename decay<_T2>::type,
+                               typename decay<_U2>::type>,
+                       is_constructible<_T2, _U2&&>>>::value;
   }
 
   template <typename _T1, typename _T2, typename _U1, typename _U2>
   constexpr bool _ImplicitlyMoveConvertiblePair()
   {
-    return __and_<is_convertible<_U1&&, _T1>,
-                 is_convertible<_U2&&, _T2>>::value;
+    return __and_<__or_<is_same<typename decay<_T1>::type,
+                               typename decay<_U1>::type>,
+                       is_convertible<_U1&&, _T1>>,
+                 __or_<is_same<typename decay<_T2>::type,
+                               typename decay<_U2>::type>,
+                      is_convertible<_U2&&, _T2>>>::value;
   }
 
 
diff --git a/libstdc++-v3/testsuite/20_util/pair/70437.cc b/libstdc++-v3/testsuite/20_util/pair/70437.cc
new file mode 100644 (file)
index 0000000..37e6fb7
--- /dev/null
@@ -0,0 +1,37 @@
+// { dg-options "-std=gnu++11" }
+// { dg-do compile }
+
+// Copyright (C) 2016 Free Software Foundation, Inc.
+//
+// This file is part of the GNU ISO C++ Library.  This library is free
+// software; you can redistribute it and/or modify it under the
+// terms of the GNU General Public License as published by the
+// Free Software Foundation; either version 3, or (at your option)
+// any later version.
+
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License along
+// with this library; see the file COPYING3.  If not see
+// <http://www.gnu.org/licenses/>.
+
+#include <utility>
+
+template <class T> struct B;
+
+template <class T> struct A
+{
+  A(A&&) = default;
+  A(const B<T> &);
+};
+
+template <class T> struct B
+{
+  std::pair<A<T>,int> a;
+  B(B&&) = default;
+};
+
+bool b = std::is_move_constructible<A<int> >::value;