diff options
author | Jonathan Wakely <jwakely@redhat.com> | 2019-02-23 21:19:00 +0000 |
---|---|---|
committer | Jonathan Wakely <redi@gcc.gnu.org> | 2019-02-23 21:19:00 +0000 |
commit | 3c26b7598c6d800dcc037a66f2dacf80728fb480 (patch) | |
tree | 9465862e1aa509de999cfe72cfbc0d183f666ab3 | |
parent | 46d496b9d93f7a647c9f5d58fed94f9e7fa299ab (diff) | |
download | gcc-3c26b7598c6d800dcc037a66f2dacf80728fb480.zip gcc-3c26b7598c6d800dcc037a66f2dacf80728fb480.tar.gz gcc-3c26b7598c6d800dcc037a66f2dacf80728fb480.tar.bz2 |
P0340R2 Making std::underlying_type SFINAE-friendly
* include/std/type_traits (__underlying_type_impl): New helper to
make underlying_type SFINAE-friendly.
(underlying_type): Derive from __underlying_type_impl.
* testsuite/20_util/underlying_type/requirements/typedefs-3.cc: New
test.
From-SVN: r269168
-rw-r--r-- | libstdc++-v3/ChangeLog | 6 | ||||
-rw-r--r-- | libstdc++-v3/include/std/type_traits | 15 | ||||
-rw-r--r-- | libstdc++-v3/testsuite/20_util/underlying_type/requirements/typedefs-3.cc | 52 |
3 files changed, 70 insertions, 3 deletions
diff --git a/libstdc++-v3/ChangeLog b/libstdc++-v3/ChangeLog index 4596af8..0b6b050 100644 --- a/libstdc++-v3/ChangeLog +++ b/libstdc++-v3/ChangeLog @@ -1,5 +1,11 @@ 2019-02-23 Jonathan Wakely <jwakely@redhat.com> + * include/std/type_traits (__underlying_type_impl): New helper to + make underlying_type SFINAE-friendly. + (underlying_type): Derive from __underlying_type_impl. + * testsuite/20_util/underlying_type/requirements/typedefs-3.cc: New + test. + PR libstdc++/89446 * include/bits/char_traits.h (__constant_char_array): Check index is in range before dereferencing. diff --git a/libstdc++-v3/include/std/type_traits b/libstdc++-v3/include/std/type_traits index 9939672..977ca0e 100644 --- a/libstdc++-v3/include/std/type_traits +++ b/libstdc++-v3/include/std/type_traits @@ -2176,12 +2176,21 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION common_type<_Tp, _Up>>::type, _Vp...>::type { }; + template<typename _Tp, bool = is_enum<_Tp>::value> + struct __underlying_type_impl + { + using type = __underlying_type(_Tp); + }; + + template<typename _Tp> + struct __underlying_type_impl<_Tp, false> + { }; + /// The underlying type of an enum. template<typename _Tp> struct underlying_type - { - typedef __underlying_type(_Tp) type; - }; + : public __underlying_type_impl<_Tp> + { }; template<typename _Tp> struct __declval_protector diff --git a/libstdc++-v3/testsuite/20_util/underlying_type/requirements/typedefs-3.cc b/libstdc++-v3/testsuite/20_util/underlying_type/requirements/typedefs-3.cc new file mode 100644 index 0000000..5b9a4af --- /dev/null +++ b/libstdc++-v3/testsuite/20_util/underlying_type/requirements/typedefs-3.cc @@ -0,0 +1,52 @@ +// Copyright (C) 2019 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/>. + +// { dg-do compile { target c++11 } } + +#include <type_traits> + +// Test for SFINAE-friendly underlying_type + +template<typename T, typename = typename std::underlying_type<T>::type> + constexpr bool is_enum(int) { return true; } + +template<typename T> + constexpr bool is_enum(...) { return false; } + +void +test01() +{ + enum E { }; + static_assert( is_enum<E>(0), ""); + + static_assert( !is_enum<void>(0), ""); + static_assert( !is_enum<int>(0), ""); + static_assert( !is_enum<long>(0), ""); + static_assert( !is_enum<int*>(0), ""); + static_assert( !is_enum<int[]>(0), ""); + static_assert( !is_enum<const int*>(0), ""); + static_assert( !is_enum<const int&>(0), ""); + static_assert( !is_enum<int()>(0), ""); + static_assert( !is_enum<int(&)()>(0), ""); + static_assert( !is_enum<int(*)()>(0), ""); + struct S { }; + static_assert( !is_enum<S>(0), ""); + static_assert( !is_enum<S&>(0), ""); + static_assert( !is_enum<S*>(0), ""); + static_assert( !is_enum<int S::*>(0), ""); + static_assert( !is_enum<int (S::*)()>(0), ""); +} |