diff options
Diffstat (limited to 'libcxx')
3 files changed, 54 insertions, 9 deletions
diff --git a/libcxx/include/functional b/libcxx/include/functional index 2056ffe90ba..db137d1eaaf 100644 --- a/libcxx/include/functional +++ b/libcxx/include/functional @@ -2574,12 +2574,11 @@ struct _LIBCPP_TYPE_VIS_ONLY hash<long double>  };  #if _LIBCPP_STD_VER > 11 -template <class _Tp> -struct _LIBCPP_TYPE_VIS_ONLY hash + +template <class _Tp, bool = is_enum<_Tp>::value> +struct _LIBCPP_TYPE_VIS_ONLY __enum_hash      : public unary_function<_Tp, size_t>  { -    static_assert(is_enum<_Tp>::value, "This hash only works for enumeration types"); -      _LIBCPP_INLINE_VISIBILITY      size_t operator()(_Tp __v) const _NOEXCEPT      { @@ -2587,6 +2586,17 @@ struct _LIBCPP_TYPE_VIS_ONLY hash          return hash<type>{}(static_cast<type>(__v));      }  }; +template <class _Tp> +struct _LIBCPP_TYPE_VIS_ONLY __enum_hash<_Tp, false> { +    __enum_hash() = delete; +    __enum_hash(__enum_hash const&) = delete; +    __enum_hash& operator=(__enum_hash const&) = delete; +}; + +template <class _Tp> +struct _LIBCPP_TYPE_VIS_ONLY hash : public __enum_hash<_Tp> +{ +};  #endif diff --git a/libcxx/test/std/utilities/function.objects/unord.hash/enum.pass.cpp b/libcxx/test/std/utilities/function.objects/unord.hash/enum.pass.cpp index 8aa2c1df893..af367789a10 100644 --- a/libcxx/test/std/utilities/function.objects/unord.hash/enum.pass.cpp +++ b/libcxx/test/std/utilities/function.objects/unord.hash/enum.pass.cpp @@ -7,6 +7,8 @@  //  //===----------------------------------------------------------------------===// +// UNSUPPORTED: c++98, c++03, c++11 +  // <functional>  // make sure that we can hash enumeration values @@ -14,8 +16,6 @@  #include "test_macros.h" -#if TEST_STD_VER >= 14 -  #include <functional>  #include <cassert>  #include <type_traits> @@ -59,6 +59,3 @@ int main()      test<Fruits>();  } -#else -int main () {} -#endif diff --git a/libcxx/test/std/utilities/function.objects/unord.hash/non_enum.pass.cpp b/libcxx/test/std/utilities/function.objects/unord.hash/non_enum.pass.cpp new file mode 100644 index 00000000000..ed173f280d0 --- /dev/null +++ b/libcxx/test/std/utilities/function.objects/unord.hash/non_enum.pass.cpp @@ -0,0 +1,38 @@ +//===----------------------------------------------------------------------===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03, c++11 + +// <functional> + +//  Hashing a struct w/o a defined hash should *not* fail, but it should +// create a type that is not constructible and not callable. +// See also: http://cplusplus.github.io/LWG/lwg-active.html#2543 + +#include <functional> +#include <cassert> +#include <type_traits> + +#include "test_macros.h" + +struct X {}; + +int main() +{ +    using H = std::hash<X>; +    static_assert(!std::is_default_constructible<H>::value, ""); +    static_assert(!std::is_copy_constructible<H>::value, ""); +    static_assert(!std::is_move_constructible<H>::value, ""); +    static_assert(!std::is_copy_assignable<H>::value, ""); +    static_assert(!std::is_move_assignable<H>::value, ""); +#if TEST_STD_VER > 14 +    static_assert(!std::is_callable<H(X&)>::value, ""); +    static_assert(!std::is_callable<H(X const&)>::value, ""); +#endif +}  | 

