libstdc++: Support arrays in std::is_nothrow_constructible (PR 94149)
authorJonathan Wakely <jwakely@redhat.com>
Tue, 21 Apr 2020 21:18:51 +0000 (22:18 +0100)
committerJonathan Wakely <jwakely@redhat.com>
Tue, 21 Apr 2020 21:18:51 +0000 (22:18 +0100)
The front end now supports parenthesized initialization for arrays in
C++20, so extend std::is_nothrow_constructible to support them too.

gcc/testsuite:

PR c++/94149
* g++.dg/cpp2a/paren-init24.C: Fix FIXMEs.

libstdc++-v3:

PR c++/94149
* include/std/type_traits (__is_nt_constructible_impl): Add partial
specializations for bounded arrays with non-empty initializers.
* testsuite/20_util/is_nothrow_constructible/value_c++20.cc: New test.

gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp2a/paren-init24.C
libstdc++-v3/ChangeLog
libstdc++-v3/include/std/type_traits
libstdc++-v3/testsuite/20_util/is_nothrow_constructible/value_c++20.cc [new file with mode: 0644]

index bd196966e6cec790e3c0c521d7a002ce1e300edf..91fe0587a9575aba7c6d55e7918d35545f088c54 100644 (file)
        * g++.dg/coroutines/ramp-return-b.C: New test.
        * g++.dg/coroutines/ramp-return-c.C: New test.
 
+2020-04-21  Jonathan Wakely  <jwakely@redhat.com>
+
+       PR c++/94149
+       * g++.dg/cpp2a/paren-init24.C: Fix FIXMEs.
+
 2020-04-17  Marek Polacek  <polacek@redhat.com>
 
        PR c++/94592
index a636a28ee6dfb5cb2f1234322c2c4e9955e1c731..4e97bbc5b561e9a5199d6f289bb6dd4be99d4e1e 100644 (file)
@@ -11,8 +11,7 @@ int main()
   static_assert(__is_constructible(T, int));
   static_assert(!__is_constructible(T, int, int));
   static_assert(std::is_constructible_v<T, int>);
-  //FIXME: libstdc++ problem?
-  //static_assert(std::is_nothrow_constructible_v<T, int>);
+  static_assert(std::is_nothrow_constructible_v<T, int>);
 
   using T2 = int[2];
   T2 t2(1);
@@ -21,6 +20,5 @@ int main()
   static_assert(__is_constructible(T2, int));
   static_assert(__is_constructible(T2, int, int));
   static_assert(std::is_constructible_v<T2, int, int>);
-  // FIXME libstdc++ problem?
-  //static_assert(std::is_nothrow_constructible_v<T2, int, int>);
+  static_assert(std::is_nothrow_constructible_v<T2, int, int>);
 }
index d09d114d948d6bfa11b7a4b2a38d2417f0545db4..80c14a7f45b0d2a96e1f0ebf3acd0992738c58a3 100644 (file)
@@ -1,3 +1,10 @@
+2020-04-21  Jonathan Wakely  <jwakely@redhat.com>
+
+       PR c++/94149
+       * include/std/type_traits (__is_nt_constructible_impl): Add partial
+       specializations for bounded arrays with non-empty initializers.
+       * testsuite/20_util/is_nothrow_constructible/value_c++20.cc: New test.
+
 2020-04-20  Thomas Rodgers  <trodgers@redhat.com>
 
        * testsuite/lib/libstdc++.exp: Add additional_flags=
index 65b9902b56d745683c212b7d23c9c4c09045cf4f..f96b5297b83af75822546b2f98f9d58446ba51ca 100644 (file)
@@ -986,6 +986,18 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
     : public __bool_constant<noexcept(typename remove_all_extents<_Tp>::type())>
     { };
 
+#if __cpp_aggregate_paren_init
+  template<typename _Tp, size_t _Num, typename _Arg>
+    struct __is_nt_constructible_impl<true, _Tp[_Num], _Arg>
+    : public __is_nt_constructible_impl<true, _Tp, _Arg>
+    { };
+
+  template<typename _Tp, size_t _Num, typename... _Args>
+    struct __is_nt_constructible_impl<true, _Tp[_Num], _Args...>
+    : public __and_<__is_nt_constructible_impl<true, _Tp, _Args>...>
+    { };
+#endif
+
   template<typename _Tp, typename... _Args>
     using __is_nothrow_constructible_impl
       = __is_nt_constructible_impl<__is_constructible(_Tp, _Args...),
diff --git a/libstdc++-v3/testsuite/20_util/is_nothrow_constructible/value_c++20.cc b/libstdc++-v3/testsuite/20_util/is_nothrow_constructible/value_c++20.cc
new file mode 100644 (file)
index 0000000..6bf0a51
--- /dev/null
@@ -0,0 +1,69 @@
+// { dg-options "-std=gnu++2a" }
+// { dg-do compile { target c++2a } }
+
+// Copyright (C) 2020 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 <type_traits>
+
+static_assert( std::is_nothrow_constructible_v<int[1]> );
+static_assert( std::is_nothrow_constructible_v<int[1], int> );
+static_assert( std::is_nothrow_constructible_v<int[2], int> );
+static_assert( std::is_nothrow_constructible_v<int[2], int, int> );
+static_assert( ! std::is_nothrow_constructible_v<int[1], int, int> );
+static_assert( ! std::is_nothrow_constructible_v<int[]> );
+static_assert( ! std::is_nothrow_constructible_v<int[], int> );
+static_assert( ! std::is_nothrow_constructible_v<int[], int, int> );
+
+struct X
+{
+  X() = default;
+  X(int) noexcept { }
+  X(double) { }
+};
+
+static_assert( std::is_nothrow_constructible_v<X[2]> );
+static_assert( std::is_nothrow_constructible_v<X[1], X> );
+static_assert( std::is_nothrow_constructible_v<X[1], int> );
+static_assert( ! std::is_nothrow_constructible_v<X[1], double> );
+static_assert( ! std::is_nothrow_constructible_v<X[2], int, double> );
+
+struct Y
+{
+  int i;
+  X x;
+};
+
+static_assert( std::is_nothrow_constructible_v<Y> );
+static_assert( std::is_nothrow_constructible_v<Y, Y> );
+static_assert( std::is_nothrow_constructible_v<Y, int> );
+static_assert( ! std::is_nothrow_constructible_v<Y, X> );
+static_assert( std::is_nothrow_constructible_v<Y, int, X> );
+static_assert( std::is_nothrow_constructible_v<Y, int, int> );
+static_assert( ! std::is_nothrow_constructible_v<Y, int, double> );
+
+struct Z : Y { };
+
+static_assert( std::is_nothrow_constructible_v<Z> );
+static_assert( std::is_nothrow_constructible_v<Z, Z> );
+static_assert( std::is_nothrow_constructible_v<Z, Y> );
+static_assert( ! std::is_nothrow_constructible_v<Z, int> );
+static_assert( ! std::is_nothrow_constructible_v<Z, int, X> );
+static_assert( ! std::is_nothrow_constructible_v<Z, int, int> );
+static_assert( ! std::is_nothrow_constructible_v<Z, Y, double> );
+static_assert( ! std::is_nothrow_constructible_v<Z, int, double> );
+static_assert( ! std::is_nothrow_constructible_v<Z, X> );