// -*- C++ -*- //===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14 // // template struct hash>; // template <> struct hash; #include #include #include #include "test_macros.h" #include "variant_test_helpers.h" #include "poisoned_hash_helper.h" #ifndef TEST_HAS_NO_EXCEPTIONS namespace std { template <> struct hash<::MakeEmptyT> { size_t operator()(const ::MakeEmptyT &) const { assert(false); return 0; } }; } #endif void test_hash_variant() { { using V = std::variant; using H = std::hash; const V v(std::in_place_index<0>, 42); const V v_copy = v; V v2(std::in_place_index<0>, 100); const H h{}; assert(h(v) == h(v)); assert(h(v) != h(v2)); assert(h(v) == h(v_copy)); { ASSERT_SAME_TYPE(decltype(h(v)), std::size_t); static_assert(std::is_copy_constructible::value, ""); } } { using V = std::variant; using H = std::hash; const char *str = "hello"; const V v0; const V v0_other; const V v1(42); const V v1_other(100); V v2(100l); V v2_other(999l); V v3(str); V v3_other("not hello"); const H h{}; assert(h(v0) == h(v0)); assert(h(v0) == h(v0_other)); assert(h(v1) == h(v1)); assert(h(v1) != h(v1_other)); assert(h(v2) == h(v2)); assert(h(v2) != h(v2_other)); assert(h(v3) == h(v3)); assert(h(v3) != h(v3_other)); assert(h(v0) != h(v1)); assert(h(v0) != h(v2)); assert(h(v0) != h(v3)); assert(h(v1) != h(v2)); assert(h(v1) != h(v3)); assert(h(v2) != h(v3)); } #ifndef TEST_HAS_NO_EXCEPTIONS { using V = std::variant; using H = std::hash; V v; makeEmpty(v); V v2; makeEmpty(v2); const H h{}; assert(h(v) == h(v2)); } #endif } void test_hash_monostate() { using H = std::hash; const H h{}; std::monostate m1{}; const std::monostate m2{}; assert(h(m1) == h(m1)); assert(h(m2) == h(m2)); assert(h(m1) == h(m2)); { ASSERT_SAME_TYPE(decltype(h(m1)), std::size_t); ASSERT_NOEXCEPT(h(m1)); static_assert(std::is_copy_constructible::value, ""); } { test_hash_enabled_for_type(); } } void test_hash_variant_duplicate_elements() { // Test that the index of the alternative participates in the hash value. using V = std::variant; using H = std::hash; H h{}; const V v1(std::in_place_index<0>); const V v2(std::in_place_index<1>); assert(h(v1) == h(v1)); assert(h(v2) == h(v2)); LIBCPP_ASSERT(h(v1) != h(v2)); } struct A {}; struct B {}; namespace std { template <> struct hash { size_t operator()(B const&) const { return 0; } }; } void test_hash_variant_enabled() { { test_hash_enabled_for_type >(); test_hash_enabled_for_type >(); } { test_hash_disabled_for_type>(); test_hash_disabled_for_type>(); } { test_hash_enabled_for_type>(); test_hash_enabled_for_type>(); } } int main(int, char**) { test_hash_variant(); test_hash_variant_duplicate_elements(); test_hash_monostate(); test_hash_variant_enabled(); return 0; }