// Copyright 2018 The Abseil Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef THIS_HASH_MAP #define THIS_HASH_MAP flat_hash_map #define THIS_TEST_NAME FlatHashMap #define ORIG_FLAT_HASH_MAP 1 #endif #ifndef THIS_EXTRA_TPL_PARAMS #define THIS_EXTRA_TPL_PARAMS #endif #ifndef THIS_EXTRA_TPL_PARAMS_NULLMUTEX #define THIS_EXTRA_TPL_PARAMS_NULLMUTEX #endif #include "parallel_hashmap/phmap.h" #if defined(PHMAP_HAVE_STD_ANY) #include #endif #ifdef _MSC_VER #pragma warning(push) #pragma warning(disable: 4710 4711) #endif #include "hash_generator_testing.h" #include "unordered_map_constructor_test.h" #include "unordered_map_lookup_test.h" #include "unordered_map_members_test.h" #include "unordered_map_modifiers_test.h" #ifdef _MSC_VER #pragma warning(pop) #endif namespace phmap { namespace priv { namespace { using ::phmap::priv::hash_internal::Enum; using ::phmap::priv::hash_internal::EnumClass; using ::testing::_; using ::testing::Pair; using ::testing::UnorderedElementsAre; template using Map = THIS_HASH_MAP> THIS_EXTRA_TPL_PARAMS>; template , class Eq = phmap::priv::hash_default_eq, class Alloc = phmap::priv::Allocator< phmap::priv::Pair>> using ThisMap = THIS_HASH_MAP; template , class Eq = phmap::priv::hash_default_eq, class Alloc = phmap::priv::Allocator< phmap::priv::Pair>> using ThisMap_NullMutex = THIS_HASH_MAP; static_assert(!std::is_standard_layout(), ""); using MapTypes = ::testing::Types, Map, Map, Map, Map, Map>; INSTANTIATE_TYPED_TEST_SUITE_P(THIS_TEST_NAME, ConstructorTest, MapTypes); INSTANTIATE_TYPED_TEST_SUITE_P(THIS_TEST_NAME, LookupTest, MapTypes); INSTANTIATE_TYPED_TEST_SUITE_P(THIS_TEST_NAME, MembersTest, MapTypes); INSTANTIATE_TYPED_TEST_SUITE_P(THIS_TEST_NAME, ModifiersTest, MapTypes); TEST(THIS_TEST_NAME, StandardLayout) { struct Int { explicit Int(size_t val) : value(val) {} Int() : value(0) { ADD_FAILURE(); } Int(const Int& other) : value(other.value) { ADD_FAILURE(); } Int(Int&&) = default; bool operator==(const Int& other) const { return value == other.value; } size_t value; }; static_assert(std::is_standard_layout(), ""); struct Hash { size_t operator()(const Int& obj) const { return obj.value; } }; // Verify that neither the key nor the value get default-constructed or // copy-constructed. { ThisMap m; m.try_emplace(Int(1), Int(2)); m.try_emplace(Int(3), Int(4)); m.erase(Int(1)); m.rehash(2 * m.bucket_count()); } { ThisMap m; m.try_emplace(Int(1), Int(2)); m.try_emplace(Int(3), Int(4)); m.erase(Int(1)); m.clear(); } } // gcc becomes unhappy if this is inside the method, so pull it out here. struct balast {}; TEST(THIS_TEST_NAME, IteratesMsan) { // Because SwissTable randomizes on pointer addresses, we keep old tables // around to ensure we don't reuse old memory. std::vector> garbage; for (int i = 0; i < 100; ++i) { ThisMap t; for (int j = 0; j < 100; ++j) { t[j]; for (const auto& p : t) EXPECT_THAT(p, Pair(_, _)); } garbage.push_back(std::move(t)); } } // Demonstration of the "Lazy Key" pattern. This uses heterogeneous insert to // avoid creating expensive key elements when the item is already present in the // map. struct LazyInt { explicit LazyInt(size_t val, int* tracker_) : value(val), tracker(tracker_) {} explicit operator size_t() const { ++*tracker; return value; } size_t value; int* tracker; }; struct Hash { using is_transparent = void; int* tracker; size_t operator()(size_t obj) const { ++*tracker; return obj; } size_t operator()(const LazyInt& obj) const { ++*tracker; return obj.value; } }; struct Eq { using is_transparent = void; bool operator()(size_t lhs, size_t rhs) const { return lhs == rhs; } bool operator()(size_t lhs, const LazyInt& rhs) const { return lhs == rhs.value; } }; TEST(THIS_TEST_NAME, PtrKet) { using H = ThisMap; H hash; int a, b; hash.insert(H::value_type(&a, true)); hash.insert(H::value_type(&b, false)); } TEST(THIS_TEST_NAME, LazyKeyPattern) { // hashes are only guaranteed in opt mode, we use assertions to track internal // state that can cause extra calls to hash. int conversions = 0; int hashes = 0; ThisMap m(0, Hash{&hashes}); m.reserve(3); m[LazyInt(1, &conversions)] = 1; EXPECT_THAT(m, UnorderedElementsAre(Pair(1, 1))); EXPECT_EQ(conversions, 1); #ifdef NDEBUG EXPECT_EQ(hashes, 1); #endif m[LazyInt(1, &conversions)] = 2; EXPECT_THAT(m, UnorderedElementsAre(Pair(1, 2))); EXPECT_EQ(conversions, 1); #ifdef NDEBUG EXPECT_EQ(hashes, 2); #endif m.try_emplace(LazyInt(2, &conversions), 3); EXPECT_THAT(m, UnorderedElementsAre(Pair(1, 2), Pair(2, 3))); EXPECT_EQ(conversions, 2); #if defined(NDEBUG) && ORIG_FLAT_HASH_MAP // for parallel maps, the reserve(3) above is not sufficient to guarantee that a submap will not resize and therefore rehash EXPECT_EQ(hashes, 3); #endif m.try_emplace(LazyInt(2, &conversions), 4); EXPECT_THAT(m, UnorderedElementsAre(Pair(1, 2), Pair(2, 3))); EXPECT_EQ(conversions, 2); #if defined(NDEBUG) && ORIG_FLAT_HASH_MAP // for parallel maps, the reserve(3) above is not sufficient to guarantee that a submap will not resize and therefore rehash EXPECT_EQ(hashes, 4); #endif } TEST(THIS_TEST_NAME, BitfieldArgument) { union { int n : 1; }; n = 0; ThisMap m; m.erase(n); m.count(n); m.prefetch(n); m.find(n); m.contains(n); m.equal_range(n); m.insert_or_assign(n, n); m.insert_or_assign(m.end(), n, n); m.try_emplace(n); m.try_emplace(m.end(), n); m.at(n); m[n]; } TEST(THIS_TEST_NAME, MergeExtractInsert) { // We can't test mutable keys, or non-copyable keys with ThisMap. // Test that the nodes have the proper API. ThisMap m = {{1, 7}, {2, 9}}; auto node = m.extract(1); EXPECT_TRUE(node); EXPECT_EQ(node.key(), 1); EXPECT_EQ(node.mapped(), 7); EXPECT_THAT(m, UnorderedElementsAre(Pair(2, 9))); node.mapped() = 17; m.insert(std::move(node)); EXPECT_THAT(m, UnorderedElementsAre(Pair(1, 17), Pair(2, 9))); } #if 0 && !defined(__ANDROID__) && !defined(__APPLE__) && !defined(__EMSCRIPTEN__) && defined(PHMAP_HAVE_STD_ANY) TEST(THIS_TEST_NAME, Any) { ThisMap m; m.emplace(1, 7); auto it = m.find(1); ASSERT_NE(it, m.end()); EXPECT_EQ(7, std::any_cast(it->second)); m.emplace(std::piecewise_construct, std::make_tuple(2), std::make_tuple(8)); it = m.find(2); ASSERT_NE(it, m.end()); EXPECT_EQ(8, std::any_cast(it->second)); m.emplace(std::piecewise_construct, std::make_tuple(3), std::make_tuple(std::any(9))); it = m.find(3); ASSERT_NE(it, m.end()); EXPECT_EQ(9, std::any_cast(it->second)); struct H { size_t operator()(const std::any&) const { return 0; } }; struct E { bool operator()(const std::any&, const std::any&) const { return true; } }; ThisMap m2; m2.emplace(1, 7); auto it2 = m2.find(1); ASSERT_NE(it2, m2.end()); EXPECT_EQ(7, it2->second); } #endif // __ANDROID__ } // namespace } // namespace priv } // namespace phmap