Correctly implement LWG 2049; std::is_destructible.
git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@213163 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
7ef03b79f9
commit
2b44e3dd49
@ -1537,42 +1537,56 @@ template <class _Tp> struct _LIBCPP_TYPE_VIS_ONLY is_move_assignable
|
|||||||
|
|
||||||
// is_destructible
|
// is_destructible
|
||||||
|
|
||||||
template <class _Tp>
|
// if it's a reference, return true
|
||||||
struct __destructible_test
|
// if it's a function, return false
|
||||||
{
|
// if it's void, return false
|
||||||
_Tp __t;
|
// if it's an array of unknown bound, return false
|
||||||
};
|
// Otherwise, return "std::declval<_Up&>().~_Up()" is well-formed
|
||||||
|
// where _Up is remove_all_extents<_Tp>::type
|
||||||
|
|
||||||
|
template <typename _Tp>
|
||||||
|
struct __is_destructor_wellformed {
|
||||||
|
template <typename _Tp1, typename _Tp2 = decltype(_VSTD::declval<_Tp1&>().~_Tp1())>
|
||||||
|
static char __test (int);
|
||||||
|
|
||||||
|
template <typename _Tp1>
|
||||||
|
static __two __test (...);
|
||||||
|
|
||||||
|
static const bool value = sizeof(__test<_Tp>(12)) == sizeof(char);
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class _Tp, bool>
|
||||||
|
struct __destructible_imp;
|
||||||
|
|
||||||
template <class _Tp>
|
template <class _Tp>
|
||||||
decltype((_VSTD::declval<__destructible_test<_Tp> >().~__destructible_test<_Tp>(), true_type()))
|
struct __destructible_imp<_Tp, false>
|
||||||
#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
|
: public _VSTD::integral_constant<bool,
|
||||||
__is_destructible_test(_Tp&&);
|
__is_destructor_wellformed<typename _VSTD::remove_all_extents<_Tp>::type>::value> {};
|
||||||
#else
|
|
||||||
__is_destructible_test(_Tp&);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
false_type
|
|
||||||
__is_destructible_test(__any);
|
|
||||||
|
|
||||||
template <class _Tp, bool = is_void<_Tp>::value || is_abstract<_Tp>::value
|
|
||||||
|| is_function<_Tp>::value>
|
|
||||||
struct __destructible_imp
|
|
||||||
: public common_type
|
|
||||||
<
|
|
||||||
decltype(__is_destructible_test(declval<_Tp>()))
|
|
||||||
>::type {};
|
|
||||||
|
|
||||||
template <class _Tp>
|
template <class _Tp>
|
||||||
struct __destructible_imp<_Tp, true>
|
struct __destructible_imp<_Tp, true>
|
||||||
: public false_type {};
|
: public _VSTD::true_type {};
|
||||||
|
|
||||||
|
template <class _Tp, bool>
|
||||||
|
struct __destructible_false;
|
||||||
|
|
||||||
|
template <class _Tp>
|
||||||
|
struct __destructible_false<_Tp, false> : public __destructible_imp<_Tp, _VSTD::is_reference<_Tp>::value> {};
|
||||||
|
|
||||||
|
template <class _Tp>
|
||||||
|
struct __destructible_false<_Tp, true> : public _VSTD::false_type {};
|
||||||
|
|
||||||
template <class _Tp>
|
template <class _Tp>
|
||||||
struct is_destructible
|
struct is_destructible
|
||||||
: public __destructible_imp<_Tp> {};
|
: public __destructible_false<_Tp, _VSTD::is_function<_Tp>::value> {};
|
||||||
|
|
||||||
template <class _Tp>
|
template <class _Tp>
|
||||||
struct is_destructible<_Tp[]>
|
struct is_destructible<_Tp[]>
|
||||||
: public false_type {};
|
: public _VSTD::false_type {};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct is_destructible<void>
|
||||||
|
: public _VSTD::false_type {};
|
||||||
|
|
||||||
// move
|
// move
|
||||||
|
|
||||||
|
@ -24,6 +24,16 @@ private:
|
|||||||
A(char);
|
A(char);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Abstract
|
||||||
|
{
|
||||||
|
virtual void foo() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class AbstractDestructor
|
||||||
|
{
|
||||||
|
virtual ~AbstractDestructor() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
template <class T>
|
template <class T>
|
||||||
void test_is_constructible()
|
void test_is_constructible()
|
||||||
{
|
{
|
||||||
@ -71,4 +81,6 @@ int main()
|
|||||||
test_is_not_constructible<A, void> ();
|
test_is_not_constructible<A, void> ();
|
||||||
test_is_not_constructible<void> ();
|
test_is_not_constructible<void> ();
|
||||||
test_is_not_constructible<int&> ();
|
test_is_not_constructible<int&> ();
|
||||||
|
test_is_not_constructible<Abstract> ();
|
||||||
|
test_is_not_constructible<AbstractDestructor> ();
|
||||||
}
|
}
|
||||||
|
@ -31,9 +31,7 @@ void test_is_not_destructible()
|
|||||||
static_assert(!std::is_destructible<const volatile T>::value, "");
|
static_assert(!std::is_destructible<const volatile T>::value, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
class Empty
|
class Empty {};
|
||||||
{
|
|
||||||
};
|
|
||||||
|
|
||||||
class NotEmpty
|
class NotEmpty
|
||||||
{
|
{
|
||||||
@ -47,11 +45,6 @@ struct bit_zero
|
|||||||
int : 0;
|
int : 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class Abstract
|
|
||||||
{
|
|
||||||
virtual ~Abstract() = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct A
|
struct A
|
||||||
{
|
{
|
||||||
~A();
|
~A();
|
||||||
@ -59,6 +52,31 @@ struct A
|
|||||||
|
|
||||||
typedef void (Function) ();
|
typedef void (Function) ();
|
||||||
|
|
||||||
|
struct PublicAbstract { public: virtual void foo() = 0; };
|
||||||
|
struct ProtectedAbstract { protected: virtual void foo() = 0; };
|
||||||
|
struct PrivateAbstract { private: virtual void foo() = 0; };
|
||||||
|
|
||||||
|
struct PublicDestructor { public: ~PublicDestructor() {}};
|
||||||
|
struct ProtectedDestructor { protected: ~ProtectedDestructor() {}};
|
||||||
|
struct PrivateDestructor { private: ~PrivateDestructor() {}};
|
||||||
|
|
||||||
|
struct VirtualPublicDestructor { public: virtual ~VirtualPublicDestructor() {}};
|
||||||
|
struct VirtualProtectedDestructor { protected: virtual ~VirtualProtectedDestructor() {}};
|
||||||
|
struct VirtualPrivateDestructor { private: virtual ~VirtualPrivateDestructor() {}};
|
||||||
|
|
||||||
|
struct PurePublicDestructor { public: virtual ~PurePublicDestructor() = 0; };
|
||||||
|
struct PureProtectedDestructor { protected: virtual ~PureProtectedDestructor() = 0; };
|
||||||
|
struct PurePrivateDestructor { private: virtual ~PurePrivateDestructor() = 0; };
|
||||||
|
|
||||||
|
struct DeletedPublicDestructor { public: ~DeletedPublicDestructor() = delete; };
|
||||||
|
struct DeletedProtectedDestructor { protected: ~DeletedProtectedDestructor() = delete; };
|
||||||
|
struct DeletedPrivateDestructor { private: ~DeletedPrivateDestructor() = delete; };
|
||||||
|
|
||||||
|
struct DeletedVirtualPublicDestructor { public: virtual ~DeletedVirtualPublicDestructor() = delete; };
|
||||||
|
struct DeletedVirtualProtectedDestructor { protected: virtual ~DeletedVirtualProtectedDestructor() = delete; };
|
||||||
|
struct DeletedVirtualPrivateDestructor { private: virtual ~DeletedVirtualPrivateDestructor() = delete; };
|
||||||
|
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
test_is_destructible<A>();
|
test_is_destructible<A>();
|
||||||
@ -72,10 +90,30 @@ int main()
|
|||||||
test_is_destructible<char[3]>();
|
test_is_destructible<char[3]>();
|
||||||
test_is_destructible<bit_zero>();
|
test_is_destructible<bit_zero>();
|
||||||
test_is_destructible<int[3]>();
|
test_is_destructible<int[3]>();
|
||||||
|
test_is_destructible<ProtectedAbstract>();
|
||||||
|
test_is_destructible<PublicAbstract>();
|
||||||
|
test_is_destructible<PrivateAbstract>();
|
||||||
|
test_is_destructible<PublicDestructor>();
|
||||||
|
test_is_destructible<VirtualPublicDestructor>();
|
||||||
|
test_is_destructible<PurePublicDestructor>();
|
||||||
|
|
||||||
test_is_not_destructible<int[]>();
|
test_is_not_destructible<int[]>();
|
||||||
test_is_not_destructible<void>();
|
test_is_not_destructible<void>();
|
||||||
test_is_not_destructible<Abstract>();
|
|
||||||
|
test_is_not_destructible<ProtectedDestructor>();
|
||||||
|
test_is_not_destructible<PrivateDestructor>();
|
||||||
|
test_is_not_destructible<VirtualProtectedDestructor>();
|
||||||
|
test_is_not_destructible<VirtualPrivateDestructor>();
|
||||||
|
test_is_not_destructible<PureProtectedDestructor>();
|
||||||
|
test_is_not_destructible<PurePrivateDestructor>();
|
||||||
|
test_is_not_destructible<DeletedPublicDestructor>();
|
||||||
|
test_is_not_destructible<DeletedProtectedDestructor>();
|
||||||
|
test_is_not_destructible<DeletedPrivateDestructor>();
|
||||||
|
|
||||||
|
// test_is_not_destructible<DeletedVirtualPublicDestructor>(); // currently fails due to clang bug #20268
|
||||||
|
test_is_not_destructible<DeletedVirtualProtectedDestructor>();
|
||||||
|
test_is_not_destructible<DeletedVirtualPrivateDestructor>();
|
||||||
|
|
||||||
#if __has_feature(cxx_access_control_sfinae)
|
#if __has_feature(cxx_access_control_sfinae)
|
||||||
test_is_not_destructible<NotEmpty>();
|
test_is_not_destructible<NotEmpty>();
|
||||||
#endif
|
#endif
|
||||||
|
@ -49,7 +49,12 @@ struct bit_zero
|
|||||||
|
|
||||||
class Abstract
|
class Abstract
|
||||||
{
|
{
|
||||||
virtual ~Abstract() = 0;
|
virtual void foo() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class AbstractDestructor
|
||||||
|
{
|
||||||
|
virtual ~AbstractDestructor() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct A
|
struct A
|
||||||
@ -60,7 +65,7 @@ struct A
|
|||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
test_has_not_nothrow_destructor<void>();
|
test_has_not_nothrow_destructor<void>();
|
||||||
test_has_not_nothrow_destructor<Abstract>();
|
test_has_not_nothrow_destructor<AbstractDestructor>();
|
||||||
test_has_not_nothrow_destructor<NotEmpty>();
|
test_has_not_nothrow_destructor<NotEmpty>();
|
||||||
|
|
||||||
#if __has_feature(cxx_noexcept)
|
#if __has_feature(cxx_noexcept)
|
||||||
@ -79,6 +84,7 @@ int main()
|
|||||||
test_is_nothrow_destructible<const int*>();
|
test_is_nothrow_destructible<const int*>();
|
||||||
test_is_nothrow_destructible<char[3]>();
|
test_is_nothrow_destructible<char[3]>();
|
||||||
test_is_nothrow_destructible<char[3]>();
|
test_is_nothrow_destructible<char[3]>();
|
||||||
|
test_is_nothrow_destructible<Abstract>();
|
||||||
#if __has_feature(cxx_noexcept)
|
#if __has_feature(cxx_noexcept)
|
||||||
test_is_nothrow_destructible<bit_zero>();
|
test_is_nothrow_destructible<bit_zero>();
|
||||||
#endif
|
#endif
|
||||||
|
@ -49,7 +49,12 @@ struct bit_zero
|
|||||||
|
|
||||||
class Abstract
|
class Abstract
|
||||||
{
|
{
|
||||||
virtual ~Abstract() = 0;
|
virtual void foo() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class AbstractDestructor
|
||||||
|
{
|
||||||
|
virtual ~AbstractDestructor() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct A
|
struct A
|
||||||
@ -61,9 +66,10 @@ int main()
|
|||||||
{
|
{
|
||||||
test_has_not_trivial_destructor<void>();
|
test_has_not_trivial_destructor<void>();
|
||||||
test_has_not_trivial_destructor<A>();
|
test_has_not_trivial_destructor<A>();
|
||||||
test_has_not_trivial_destructor<Abstract>();
|
test_has_not_trivial_destructor<AbstractDestructor>();
|
||||||
test_has_not_trivial_destructor<NotEmpty>();
|
test_has_not_trivial_destructor<NotEmpty>();
|
||||||
|
|
||||||
|
test_is_trivially_destructible<Abstract>();
|
||||||
test_is_trivially_destructible<int&>();
|
test_is_trivially_destructible<int&>();
|
||||||
test_is_trivially_destructible<Union>();
|
test_is_trivially_destructible<Union>();
|
||||||
test_is_trivially_destructible<Empty>();
|
test_is_trivially_destructible<Empty>();
|
||||||
|
Loading…
Reference in New Issue
Block a user