/** * MIT License * * Copyright (c) 2017 Tessil * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #define BOOST_TEST_DYN_LINK #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "utils.h" BOOST_AUTO_TEST_SUITE(test_robin_map) using test_types = boost::mpl::list< tsl::robin_map, tsl::robin_map, // Test with hash having a lot of collisions tsl::robin_map>, tsl::robin_map>, tsl::robin_map>, tsl::robin_map>, tsl::robin_map>, // other GrowthPolicy tsl::robin_map, std::equal_to, std::allocator>, true, tsl::rh::power_of_two_growth_policy<4>>, tsl::robin_pg_map>, tsl::robin_map, std::equal_to, std::allocator>, false, tsl::rh::mod_growth_policy<>>, tsl::robin_map, std::equal_to, std::allocator>, false, tsl::rh::power_of_two_growth_policy<4>>, tsl::robin_pg_map>, tsl::robin_map, std::equal_to, std::allocator>, true, tsl::rh::mod_growth_policy<>> >; /** * insert */ BOOST_AUTO_TEST_CASE_TEMPLATE(test_insert, HMap, test_types) { // insert x values, insert them again, check values using key_t = typename HMap::key_type; using value_t = typename HMap:: mapped_type; const std::size_t nb_values = 1000; HMap map; typename HMap::iterator it; bool inserted; for(std::size_t i = 0; i < nb_values; i++) { std::tie(it, inserted) = map.insert({utils::get_key(i), utils::get_value(i)}); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); BOOST_CHECK(inserted); } BOOST_CHECK_EQUAL(map.size(), nb_values); for(std::size_t i = 0; i < nb_values; i++) { std::tie(it, inserted) = map.insert({utils::get_key(i), utils::get_value(i + 1)}); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); BOOST_CHECK(!inserted); } for(std::size_t i = 0; i < nb_values; i++) { it = map.find(utils::get_key(i)); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); } } BOOST_AUTO_TEST_CASE(test_range_insert) { const int nb_values = 1000; std::vector> values; for(int i = 0; i < nb_values; i++) { values.push_back(std::make_pair(i, i+1)); } tsl::robin_map map = {{-1, 1}, {-2, 2}}; map.insert(std::next(values.begin(), 10), values.end() - 5); BOOST_CHECK_EQUAL(map.size(), 987); BOOST_CHECK_EQUAL(map[-1], 1); BOOST_CHECK_EQUAL(map[-2], 2); for(int i = 10; i < nb_values - 5; i++) { BOOST_CHECK_EQUAL(map[i], i+1); } } BOOST_AUTO_TEST_CASE(test_insert_with_hint) { tsl::robin_map map{{1, 0}, {2, 1}, {3, 2}}; // Wrong hint BOOST_CHECK(map.insert(map.find(2), std::make_pair(3, 4)) == map.find(3)); // Good hint BOOST_CHECK(map.insert(map.find(2), std::make_pair(2, 4)) == map.find(2)); // end() hint BOOST_CHECK(map.insert(map.find(10), std::make_pair(2, 4)) == map.find(2)); BOOST_CHECK_EQUAL(map.size(), 3); // end() hint, new value BOOST_CHECK_EQUAL(map.insert(map.find(10), std::make_pair(4, 3))->first, 4); // Wrong hint, new value BOOST_CHECK_EQUAL(map.insert(map.find(2), std::make_pair(5, 4))->first, 5); BOOST_CHECK_EQUAL(map.size(), 5); } /** * emplace */ BOOST_AUTO_TEST_CASE(test_emplace) { tsl::robin_map map; tsl::robin_map::iterator it; bool inserted; std::tie(it, inserted) = map.emplace(std::piecewise_construct, std::forward_as_tuple(10), std::forward_as_tuple(1)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); BOOST_CHECK(inserted); std::tie(it, inserted) = map.emplace(std::piecewise_construct, std::forward_as_tuple(10), std::forward_as_tuple(3)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); BOOST_CHECK(!inserted); } /** * try_emplace */ BOOST_AUTO_TEST_CASE(test_try_emplace) { tsl::robin_map map; tsl::robin_map::iterator it; bool inserted; std::tie(it, inserted) = map.try_emplace(10, 1); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); BOOST_CHECK(inserted); std::tie(it, inserted) = map.try_emplace(10, 3); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); BOOST_CHECK(!inserted); } BOOST_AUTO_TEST_CASE(test_try_emplace_2) { tsl::robin_map map; tsl::robin_map::iterator it; bool inserted; const std::size_t nb_values = 1000; for(std::size_t i = 0; i < nb_values; i++) { std::tie(it, inserted) = map.try_emplace(utils::get_key(i), i); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, move_only_test(i)); BOOST_CHECK(inserted); } BOOST_CHECK_EQUAL(map.size(), nb_values); for(std::size_t i = 0; i < nb_values; i++) { std::tie(it, inserted) = map.try_emplace(utils::get_key(i), i + 1); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, move_only_test(i)); BOOST_CHECK(!inserted); } for(std::size_t i = 0; i < nb_values; i++) { it = map.find(utils::get_key(i)); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, move_only_test(i)); } } BOOST_AUTO_TEST_CASE(test_try_emplace_hint) { tsl::robin_map map(0); // end() hint, new value auto it = map.try_emplace(map.find(10), 10, 1); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); // Good hint it = map.try_emplace(map.find(10), 10, 3); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); // Wrong hint, new value it = map.try_emplace(map.find(10), 1, 3); BOOST_CHECK_EQUAL(it->first, 1); BOOST_CHECK_EQUAL(it->second, move_only_test(3)); } /** * insert_or_assign */ BOOST_AUTO_TEST_CASE(test_insert_or_assign) { tsl::robin_map map; tsl::robin_map::iterator it; bool inserted; std::tie(it, inserted) = map.insert_or_assign(10, move_only_test(1)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); BOOST_CHECK(inserted); std::tie(it, inserted) = map.insert_or_assign(10, move_only_test(3)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(3)); BOOST_CHECK(!inserted); } BOOST_AUTO_TEST_CASE(test_insert_or_assign_hint) { tsl::robin_map map(0); // end() hint, new value auto it = map.insert_or_assign(map.find(10), 10, move_only_test(1)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(1)); // Good hint it = map.insert_or_assign(map.find(10), 10, move_only_test(3)); BOOST_CHECK_EQUAL(it->first, 10); BOOST_CHECK_EQUAL(it->second, move_only_test(3)); // Bad hint, new value it = map.insert_or_assign(map.find(10), 1, move_only_test(3)); BOOST_CHECK_EQUAL(it->first, 1); BOOST_CHECK_EQUAL(it->second, move_only_test(3)); } /** * erase */ BOOST_AUTO_TEST_CASE(test_range_erase_all) { // insert x values, delete all using HMap = tsl::robin_map; const std::size_t nb_values = 1000; HMap map = utils::get_filled_hash_map(nb_values); auto it = map.erase(map.begin(), map.end()); BOOST_CHECK(it == map.end()); BOOST_CHECK(map.empty()); } BOOST_AUTO_TEST_CASE(test_range_erase) { // insert x values, delete all except 10 first and 780 last values using HMap = tsl::robin_map; const std::size_t nb_values = 1000; HMap map = utils::get_filled_hash_map(nb_values); auto it_first = std::next(map.begin(), 10); auto it_last = std::next(map.begin(), 220); auto it = map.erase(it_first, it_last); BOOST_CHECK_EQUAL(std::distance(it, map.end()), 780); BOOST_CHECK_EQUAL(map.size(), 790); BOOST_CHECK_EQUAL(std::distance(map.begin(), map.end()), 790); for(auto& val: map) { BOOST_CHECK_EQUAL(map.count(val.first), 1); } } BOOST_AUTO_TEST_CASE_TEMPLATE(test_erase_loop, HMap, test_types) { // insert x values, delete all one by one std::size_t nb_values = 1000; HMap map = utils::get_filled_hash_map(nb_values); HMap map2 = utils::get_filled_hash_map(nb_values); auto it = map.begin(); // Use second map to check for key after delete as we may not copy the key with move-only types. auto it2 = map2.begin(); while(it != map.end()) { it = map.erase(it); --nb_values; BOOST_CHECK_EQUAL(map.count(it2->first), 0); BOOST_CHECK_EQUAL(map.size(), nb_values); ++it2; } BOOST_CHECK(map.empty()); } BOOST_AUTO_TEST_CASE_TEMPLATE(test_erase_loop_range, HMap, test_types) { // insert x values, delete all five by five const std::size_t range = 5; std::size_t nb_values = 1000; BOOST_REQUIRE_EQUAL(1000 % 5, 0); HMap map = utils::get_filled_hash_map(nb_values); HMap map2 = utils::get_filled_hash_map(nb_values); auto it = map.begin(); // Use second map to check for key after delete as we may not copy the key with move-only types. auto it2 = map2.begin(); while(it != map.end()) { it = map.erase(it, std::next(it, range)); nb_values -= range; BOOST_CHECK_EQUAL(map.size(), nb_values); it2 = std::next(it2, range); } BOOST_CHECK(map.empty()); } BOOST_AUTO_TEST_CASE_TEMPLATE(test_insert_erase_insert, HMap, test_types) { // insert x/2 values, delete x/4 values, insert x/2 values, find each value using key_t = typename HMap::key_type; using value_t = typename HMap:: mapped_type; const std::size_t nb_values = 2000; HMap map; typename HMap::iterator it; bool inserted; // Insert nb_values/2 for(std::size_t i = 0; i < nb_values/2; i++) { std::tie(it, inserted) = map.insert({utils::get_key(i), utils::get_value(i)}); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); BOOST_CHECK(inserted); } BOOST_CHECK_EQUAL(map.size(), nb_values/2); // Delete nb_values/4 for(std::size_t i = 0; i < nb_values/2; i++) { if(i%2 == 0) { BOOST_CHECK_EQUAL(map.erase(utils::get_key(i)), 1); } } BOOST_CHECK_EQUAL(map.size(), nb_values/4); // Insert nb_values/2 for(std::size_t i = nb_values/2; i < nb_values; i++) { std::tie(it, inserted) = map.insert({utils::get_key(i), utils::get_value(i)}); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); BOOST_CHECK(inserted); } BOOST_CHECK_EQUAL(map.size(), nb_values-nb_values/4); // Find for(std::size_t i = 0; i < nb_values; i++) { if(i%2 == 0 && i < nb_values/2) { it = map.find(utils::get_key(i)); BOOST_CHECK(it == map.end()); } else { it = map.find(utils::get_key(i)); BOOST_REQUIRE(it != map.end()); BOOST_CHECK_EQUAL(it->first, utils::get_key(i)); BOOST_CHECK_EQUAL(it->second, utils::get_value(i)); } } } BOOST_AUTO_TEST_CASE(test_range_erase_same_iterators) { const std::size_t nb_values = 100; auto map = utils::get_filled_hash_map>(nb_values); tsl::robin_map::const_iterator it_const = map.cbegin(); std::advance(it_const, 10); tsl::robin_map::iterator it_mutable = map.erase(it_const, it_const); BOOST_CHECK(it_const == it_mutable); BOOST_CHECK(map.mutable_iterator(it_const) == it_mutable); BOOST_CHECK_EQUAL(map.size(), 100); it_mutable.value() = -100; BOOST_CHECK_EQUAL(it_const.value(), -100); } /** * operator== and operator!= */ BOOST_AUTO_TEST_CASE_TEMPLATE(test_compare, HMap, test_types) { // create 3 maps, 2 are the same, compare maps using key_t = typename HMap::key_type; using value_t = typename HMap:: mapped_type; const std::size_t nb_values = 1000; HMap map_1_1; HMap map_1_2; HMap map_2_1; for(std::size_t i = 0; i < nb_values; i++) { map_1_1.insert({utils::get_key(i), utils::get_value(i)}); if(i != 0) { map_2_1.insert({utils::get_key(i), utils::get_value(i)}); } } // Same as map_1_1 but insertion order inverted for(std::size_t i = nb_values; i != 0; i--) { map_1_2.insert({utils::get_key(i-1), utils::get_value(i-1)}); } BOOST_CHECK_EQUAL(map_1_1.size(), nb_values); BOOST_CHECK_EQUAL(map_1_2.size(), nb_values); BOOST_CHECK_EQUAL(map_2_1.size(), nb_values-1); BOOST_CHECK(map_1_1 == map_1_2); BOOST_CHECK(map_1_2 == map_1_1); BOOST_CHECK(map_1_1 != map_2_1); BOOST_CHECK(map_2_1 != map_1_1); BOOST_CHECK(map_1_2 != map_2_1); BOOST_CHECK(map_2_1 != map_1_2); } /** * clear */ BOOST_AUTO_TEST_CASE(test_clear) { // insert x values, clear map using HMap = tsl::robin_map; const std::size_t nb_values = 1000; auto map = utils::get_filled_hash_map(nb_values); map.clear(); BOOST_CHECK_EQUAL(map.size(), 0); BOOST_CHECK_EQUAL(std::distance(map.begin(), map.end()), 0); map.insert({5, -5}); map.insert({{1, -1}, {2, -1}, {4, -4}, {3, -3}}); BOOST_CHECK(map == (HMap({{5, -5}, {1, -1}, {2, -1}, {4, -4}, {3, -3}}))); } /** * iterator.value() */ BOOST_AUTO_TEST_CASE(test_modify_value) { // insert x values, modify value of even keys, check values const std::size_t nb_values = 100; auto map = utils::get_filled_hash_map>(nb_values); for(auto it = map.begin(); it != map.end(); it++) { if(it.key() % 2 == 0) { it.value() = -1; } } for(auto& val : map) { if(val.first % 2 == 0) { BOOST_CHECK_EQUAL(val.second, -1); } else { BOOST_CHECK_NE(val.second, -1); } } } /** * operator=(std::initializer_list) */ BOOST_AUTO_TEST_CASE(test_assign_operator) { tsl::robin_map map = {{0, 10}, {-2, 20}}; BOOST_CHECK_EQUAL(map.size(), 2); map = {{1, 3}, {2, 4}}; BOOST_CHECK_EQUAL(map.size(), 2); BOOST_CHECK_EQUAL(map.at(1), 3); BOOST_CHECK_EQUAL(map.at(2), 4); BOOST_CHECK(map.find(0) == map.end()); } /** * move/copy constructor/operator */ BOOST_AUTO_TEST_CASE(test_move_constructor) { // insert x values in map, move map into map_move, check map and map_move, // insert additional values in map_move, check map_move using HMap = tsl::robin_map; const std::size_t nb_values = 100; HMap map = utils::get_filled_hash_map(nb_values); HMap map_move(std::move(map)); BOOST_CHECK(map_move == utils::get_filled_hash_map(nb_values)); BOOST_CHECK(map == (HMap())); for(std::size_t i = nb_values; i < nb_values*2; i++) { map_move.insert({utils::get_key(i), utils::get_value(i)}); } BOOST_CHECK_EQUAL(map_move.size(), nb_values*2); BOOST_CHECK(map_move == utils::get_filled_hash_map(nb_values*2)); } BOOST_AUTO_TEST_CASE(test_move_operator) { // insert x values in map, move map into map_move, check map and map_move, // insert additional values in map_move, check map_move using HMap = tsl::robin_map; const std::size_t nb_values = 100; HMap map = utils::get_filled_hash_map(nb_values); HMap map_move = utils::get_filled_hash_map(1); map_move = std::move(map); BOOST_CHECK(map_move == utils::get_filled_hash_map(nb_values)); BOOST_CHECK(map == (HMap())); for(std::size_t i = nb_values; i < nb_values*2; i++) { map_move.insert({utils::get_key(i), utils::get_value(i)}); } BOOST_CHECK_EQUAL(map_move.size(), nb_values*2); BOOST_CHECK(map_move == utils::get_filled_hash_map(nb_values*2)); } BOOST_AUTO_TEST_CASE(test_reassign_moved_object_move_constructor) { using HMap = tsl::robin_map; HMap map = {{"Key1", "Value1"}, {"Key2", "Value2"}, {"Key3", "Value3"}}; HMap map_move(std::move(map)); BOOST_CHECK_EQUAL(map_move.size(), 3); BOOST_CHECK_EQUAL(map.size(), 0); map = {{"Key4", "Value4"}, {"Key5", "Value5"}}; BOOST_CHECK(map == (HMap({{"Key4", "Value4"}, {"Key5", "Value5"}}))); } BOOST_AUTO_TEST_CASE(test_reassign_moved_object_move_operator) { using HMap = tsl::robin_map; HMap map = {{"Key1", "Value1"}, {"Key2", "Value2"}, {"Key3", "Value3"}}; HMap map_move = std::move(map); BOOST_CHECK_EQUAL(map_move.size(), 3); BOOST_CHECK_EQUAL(map.size(), 0); map = {{"Key4", "Value4"}, {"Key5", "Value5"}}; BOOST_CHECK(map == (HMap({{"Key4", "Value4"}, {"Key5", "Value5"}}))); } BOOST_AUTO_TEST_CASE(test_copy_constructor_operator) { using HMap = tsl::robin_map>; const std::size_t nb_values = 100; HMap map = utils::get_filled_hash_map(nb_values); HMap map_copy = map; HMap map_copy2(map); HMap map_copy3; map_copy3 = map; BOOST_CHECK(map == map_copy); map.clear(); BOOST_CHECK(map_copy == map_copy2); BOOST_CHECK(map_copy == map_copy3); } /** * at */ BOOST_AUTO_TEST_CASE(test_at) { // insert x values, use at for known and unknown values. tsl::robin_map map = {{0, 10}, {-2, 20}}; BOOST_CHECK_EQUAL(map.at(0), 10); BOOST_CHECK_EQUAL(map.at(-2), 20); BOOST_CHECK_THROW(map.at(1), std::out_of_range); } /** * equal_range */ BOOST_AUTO_TEST_CASE(test_equal_range) { tsl::robin_map map = {{0, 10}, {-2, 20}}; auto it_pair = map.equal_range(0); BOOST_REQUIRE_EQUAL(std::distance(it_pair.first, it_pair.second), 1); BOOST_CHECK_EQUAL(it_pair.first->second, 10); it_pair = map.equal_range(1); BOOST_CHECK(it_pair.first == it_pair.second); BOOST_CHECK(it_pair.first == map.end()); } /** * operator[] */ BOOST_AUTO_TEST_CASE(test_access_operator) { // insert x values, use at for known and unknown values. tsl::robin_map map = {{0, 10}, {-2, 20}}; BOOST_CHECK_EQUAL(map[0], 10); BOOST_CHECK_EQUAL(map[-2], 20); BOOST_CHECK_EQUAL(map[2], std::int64_t()); BOOST_CHECK_EQUAL(map.size(), 3); } /** * swap */ BOOST_AUTO_TEST_CASE(test_swap) { tsl::robin_map map = {{1, 10}, {8, 80}, {3, 30}}; tsl::robin_map map2 = {{4, 40}, {5, 50}}; using std::swap; swap(map, map2); BOOST_CHECK(map == (tsl::robin_map{{4, 40}, {5, 50}})); BOOST_CHECK(map2 == (tsl::robin_map{{1, 10}, {8, 80}, {3, 30}})); } /** * other */ BOOST_AUTO_TEST_CASE(test_heterogeneous_lookups) { struct hash_ptr { std::size_t operator()(const std::unique_ptr& p) const { return std::hash()(reinterpret_cast(p.get())); } std::size_t operator()(std::uintptr_t p) const { return std::hash()(p); } std::size_t operator()(const int* const& p) const { return std::hash()(reinterpret_cast(p)); } }; struct equal_to_ptr { using is_transparent = std::true_type; bool operator()(const std::unique_ptr& p1, const std::unique_ptr& p2) const { return p1 == p2; } bool operator()(const std::unique_ptr& p1, std::uintptr_t p2) const { return reinterpret_cast(p1.get()) == p2; } bool operator()(std::uintptr_t p1, const std::unique_ptr& p2) const { return p1 == reinterpret_cast(p2.get()); } bool operator()(const std::unique_ptr& p1, const int* const& p2) const { return p1.get() == p2; } bool operator()(const int* const& p1, const std::unique_ptr& p2) const { return p1 == p2.get(); } }; std::unique_ptr ptr1(new int(1)); std::unique_ptr ptr2(new int(2)); std::unique_ptr ptr3(new int(3)); int other = -1; const std::uintptr_t addr1 = reinterpret_cast(ptr1.get()); const int* const addr2 = ptr2.get(); const int* const addr_unknown = &other; tsl::robin_map, int, hash_ptr, equal_to_ptr> map; map.insert({std::move(ptr1), 4}); map.insert({std::move(ptr2), 5}); map.insert({std::move(ptr3), 6}); BOOST_CHECK_EQUAL(map.size(), 3); BOOST_CHECK_EQUAL(map.at(addr1), 4); BOOST_CHECK_EQUAL(map.at(addr2), 5); BOOST_CHECK_THROW(map.at(addr_unknown), std::out_of_range); BOOST_REQUIRE(map.find(addr1) != map.end()); BOOST_CHECK_EQUAL(*map.find(addr1)->first, 1); BOOST_REQUIRE(map.find(addr2) != map.end()); BOOST_CHECK_EQUAL(*map.find(addr2)->first, 2); BOOST_CHECK(map.find(addr_unknown) == map.end()); BOOST_CHECK_EQUAL(map.count(addr1), 1); BOOST_CHECK_EQUAL(map.count(addr2), 1); BOOST_CHECK_EQUAL(map.count(addr_unknown), 0); BOOST_CHECK_EQUAL(map.erase(addr1), 1); BOOST_CHECK_EQUAL(map.erase(addr2), 1); BOOST_CHECK_EQUAL(map.erase(addr_unknown), 0); BOOST_CHECK_EQUAL(map.size(), 1); } /** * Various operations on empty map */ BOOST_AUTO_TEST_CASE(test_empty_map) { tsl::robin_map map(0); BOOST_CHECK_EQUAL(map.size(), 0); BOOST_CHECK(map.empty()); BOOST_CHECK(map.begin() == map.end()); BOOST_CHECK(map.begin() == map.cend()); BOOST_CHECK(map.cbegin() == map.cend()); BOOST_CHECK(map.find("") == map.end()); BOOST_CHECK(map.find("test") == map.end()); BOOST_CHECK_EQUAL(map.count(""), 0); BOOST_CHECK_EQUAL(map.count("test"), 0); BOOST_CHECK_THROW(map.at(""), std::out_of_range); BOOST_CHECK_THROW(map.at("test"), std::out_of_range); auto range = map.equal_range("test"); BOOST_CHECK(range.first == range.second); BOOST_CHECK_EQUAL(map.erase("test"), 0); BOOST_CHECK(map.erase(map.begin(), map.end()) == map.end()); BOOST_CHECK_EQUAL(map["new value"], int{}); } /** * Test precalculated hash */ BOOST_AUTO_TEST_CASE(test_precalculated_hash) { tsl::robin_map> map = {{1, -1}, {2, -2}, {3, -3}, {4, -4}, {5, -5}, {6, -6}}; const tsl::robin_map map_const = map; /** * find */ BOOST_REQUIRE(map.find(3, map.hash_function()(3)) != map.end()); BOOST_CHECK_EQUAL(map.find(3, map.hash_function()(3))->second, -3); BOOST_REQUIRE(map_const.find(3, map_const.hash_function()(3)) != map_const.end()); BOOST_CHECK_EQUAL(map_const.find(3, map_const.hash_function()(3))->second, -3); BOOST_REQUIRE_NE(map.hash_function()(2), map.hash_function()(3)); BOOST_CHECK(map.find(3, map.hash_function()(2)) == map.end()); /** * at */ BOOST_CHECK_EQUAL(map.at(3, map.hash_function()(3)), -3); BOOST_CHECK_EQUAL(map_const.at(3, map_const.hash_function()(3)), -3); BOOST_REQUIRE_NE(map.hash_function()(2), map.hash_function()(3)); BOOST_CHECK_THROW(map.at(3, map.hash_function()(2)), std::out_of_range); /** * count */ BOOST_CHECK_EQUAL(map.count(3, map.hash_function()(3)), 1); BOOST_CHECK_EQUAL(map_const.count(3, map_const.hash_function()(3)), 1); BOOST_REQUIRE_NE(map.hash_function()(2), map.hash_function()(3)); BOOST_CHECK_EQUAL(map.count(3, map.hash_function()(2)), 0); /** * equal_range */ auto it_range = map.equal_range(3, map.hash_function()(3)); BOOST_REQUIRE_EQUAL(std::distance(it_range.first, it_range.second), 1); BOOST_CHECK_EQUAL(it_range.first->second, -3); auto it_range_const = map_const.equal_range(3, map_const.hash_function()(3)); BOOST_REQUIRE_EQUAL(std::distance(it_range_const.first, it_range_const.second), 1); BOOST_CHECK_EQUAL(it_range_const.first->second, -3); it_range = map.equal_range(3, map.hash_function()(2)); BOOST_REQUIRE_NE(map.hash_function()(2), map.hash_function()(3)); BOOST_CHECK_EQUAL(std::distance(it_range.first, it_range.second), 0); /** * erase */ BOOST_CHECK_EQUAL(map.erase(3, map.hash_function()(3)), 1); BOOST_REQUIRE_NE(map.hash_function()(2), map.hash_function()(4)); BOOST_CHECK_EQUAL(map.erase(4, map.hash_function()(2)), 0); } BOOST_AUTO_TEST_SUITE_END()