| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/containers/small_map.h" |
| |
| #include <stddef.h> |
| |
| #include <algorithm> |
| #include <functional> |
| #include <map> |
| #include <unordered_map> |
| |
| #include "base/logging.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace base { |
| |
| TEST(SmallMap, General) { |
| small_map<std::unordered_map<int, int>> m; |
| |
| EXPECT_TRUE(m.empty()); |
| |
| m[0] = 5; |
| |
| EXPECT_FALSE(m.empty()); |
| EXPECT_EQ(m.size(), 1u); |
| |
| m[9] = 2; |
| |
| EXPECT_FALSE(m.empty()); |
| EXPECT_EQ(m.size(), 2u); |
| |
| EXPECT_EQ(m[9], 2); |
| EXPECT_EQ(m[0], 5); |
| EXPECT_FALSE(m.UsingFullMap()); |
| |
| small_map<std::unordered_map<int, int>>::iterator iter(m.begin()); |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 0); |
| EXPECT_EQ(iter->second, 5); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ((*iter).first, 9); |
| EXPECT_EQ((*iter).second, 2); |
| ++iter; |
| EXPECT_TRUE(iter == m.end()); |
| |
| m[8] = 23; |
| m[1234] = 90; |
| m[-5] = 6; |
| |
| EXPECT_EQ(m[ 9], 2); |
| EXPECT_EQ(m[ 0], 5); |
| EXPECT_EQ(m[1234], 90); |
| EXPECT_EQ(m[ 8], 23); |
| EXPECT_EQ(m[ -5], 6); |
| EXPECT_EQ(m.size(), 5u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_TRUE(m.UsingFullMap()); |
| |
| iter = m.begin(); |
| for (int i = 0; i < 5; i++) { |
| EXPECT_TRUE(iter != m.end()); |
| ++iter; |
| } |
| EXPECT_TRUE(iter == m.end()); |
| |
| const small_map<std::unordered_map<int, int>>& ref = m; |
| EXPECT_TRUE(ref.find(1234) != m.end()); |
| EXPECT_TRUE(ref.find(5678) == m.end()); |
| } |
| |
| TEST(SmallMap, PostFixIteratorIncrement) { |
| small_map<std::unordered_map<int, int>> m; |
| m[0] = 5; |
| m[2] = 3; |
| |
| { |
| small_map<std::unordered_map<int, int>>::iterator iter(m.begin()); |
| small_map<std::unordered_map<int, int>>::iterator last(iter++); |
| ++last; |
| EXPECT_TRUE(last == iter); |
| } |
| |
| { |
| small_map<std::unordered_map<int, int>>::const_iterator iter(m.begin()); |
| small_map<std::unordered_map<int, int>>::const_iterator last(iter++); |
| ++last; |
| EXPECT_TRUE(last == iter); |
| } |
| } |
| |
| // Based on the General testcase. |
| TEST(SmallMap, CopyConstructor) { |
| small_map<std::unordered_map<int, int>> src; |
| |
| { |
| small_map<std::unordered_map<int, int>> m(src); |
| EXPECT_TRUE(m.empty()); |
| } |
| |
| src[0] = 5; |
| |
| { |
| small_map<std::unordered_map<int, int>> m(src); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_EQ(m.size(), 1u); |
| } |
| |
| src[9] = 2; |
| |
| { |
| small_map<std::unordered_map<int, int>> m(src); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_EQ(m.size(), 2u); |
| |
| EXPECT_EQ(m[9], 2); |
| EXPECT_EQ(m[0], 5); |
| EXPECT_FALSE(m.UsingFullMap()); |
| } |
| |
| src[8] = 23; |
| src[1234] = 90; |
| src[-5] = 6; |
| |
| { |
| small_map<std::unordered_map<int, int>> m(src); |
| EXPECT_EQ(m[ 9], 2); |
| EXPECT_EQ(m[ 0], 5); |
| EXPECT_EQ(m[1234], 90); |
| EXPECT_EQ(m[ 8], 23); |
| EXPECT_EQ(m[ -5], 6); |
| EXPECT_EQ(m.size(), 5u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_TRUE(m.UsingFullMap()); |
| } |
| } |
| |
| template <class inner> |
| static bool SmallMapIsSubset(small_map<inner> const& a, |
| small_map<inner> const& b) { |
| typename small_map<inner>::const_iterator it; |
| for (it = a.begin(); it != a.end(); ++it) { |
| typename small_map<inner>::const_iterator it_in_b = b.find(it->first); |
| if (it_in_b == b.end() || it_in_b->second != it->second) |
| return false; |
| } |
| return true; |
| } |
| |
| template <class inner> |
| static bool SmallMapEqual(small_map<inner> const& a, |
| small_map<inner> const& b) { |
| return SmallMapIsSubset(a, b) && SmallMapIsSubset(b, a); |
| } |
| |
| TEST(SmallMap, AssignmentOperator) { |
| small_map<std::unordered_map<int, int>> src_small; |
| small_map<std::unordered_map<int, int>> src_large; |
| |
| src_small[1] = 20; |
| src_small[2] = 21; |
| src_small[3] = 22; |
| EXPECT_FALSE(src_small.UsingFullMap()); |
| |
| src_large[1] = 20; |
| src_large[2] = 21; |
| src_large[3] = 22; |
| src_large[5] = 23; |
| src_large[6] = 24; |
| src_large[7] = 25; |
| EXPECT_TRUE(src_large.UsingFullMap()); |
| |
| // Assignments to empty. |
| small_map<std::unordered_map<int, int>> dest_small; |
| dest_small = src_small; |
| EXPECT_TRUE(SmallMapEqual(dest_small, src_small)); |
| EXPECT_EQ(dest_small.UsingFullMap(), |
| src_small.UsingFullMap()); |
| |
| small_map<std::unordered_map<int, int>> dest_large; |
| dest_large = src_large; |
| EXPECT_TRUE(SmallMapEqual(dest_large, src_large)); |
| EXPECT_EQ(dest_large.UsingFullMap(), |
| src_large.UsingFullMap()); |
| |
| // Assignments which assign from full to small, and vice versa. |
| dest_small = src_large; |
| EXPECT_TRUE(SmallMapEqual(dest_small, src_large)); |
| EXPECT_EQ(dest_small.UsingFullMap(), |
| src_large.UsingFullMap()); |
| |
| dest_large = src_small; |
| EXPECT_TRUE(SmallMapEqual(dest_large, src_small)); |
| EXPECT_EQ(dest_large.UsingFullMap(), |
| src_small.UsingFullMap()); |
| |
| // Double check that SmallMapEqual works: |
| dest_large[42] = 666; |
| EXPECT_FALSE(SmallMapEqual(dest_large, src_small)); |
| } |
| |
| TEST(SmallMap, Insert) { |
| small_map<std::unordered_map<int, int>> sm; |
| |
| // loop through the transition from small map to map. |
| for (int i = 1; i <= 10; ++i) { |
| VLOG(1) << "Iteration " << i; |
| // insert an element |
| std::pair<small_map<std::unordered_map<int, int>>::iterator, bool> ret; |
| ret = sm.insert(std::make_pair(i, 100*i)); |
| EXPECT_TRUE(ret.second); |
| EXPECT_TRUE(ret.first == sm.find(i)); |
| EXPECT_EQ(ret.first->first, i); |
| EXPECT_EQ(ret.first->second, 100*i); |
| |
| // try to insert it again with different value, fails, but we still get an |
| // iterator back with the original value. |
| ret = sm.insert(std::make_pair(i, -i)); |
| EXPECT_FALSE(ret.second); |
| EXPECT_TRUE(ret.first == sm.find(i)); |
| EXPECT_EQ(ret.first->first, i); |
| EXPECT_EQ(ret.first->second, 100*i); |
| |
| // check the state of the map. |
| for (int j = 1; j <= i; ++j) { |
| small_map<std::unordered_map<int, int>>::iterator it = sm.find(j); |
| EXPECT_TRUE(it != sm.end()); |
| EXPECT_EQ(it->first, j); |
| EXPECT_EQ(it->second, j * 100); |
| } |
| EXPECT_EQ(sm.size(), static_cast<size_t>(i)); |
| EXPECT_FALSE(sm.empty()); |
| } |
| } |
| |
| TEST(SmallMap, InsertRange) { |
| // loop through the transition from small map to map. |
| for (int elements = 0; elements <= 10; ++elements) { |
| VLOG(1) << "Elements " << elements; |
| std::unordered_map<int, int> normal_map; |
| for (int i = 1; i <= elements; ++i) { |
| normal_map.insert(std::make_pair(i, 100*i)); |
| } |
| |
| small_map<std::unordered_map<int, int>> sm; |
| sm.insert(normal_map.begin(), normal_map.end()); |
| EXPECT_EQ(normal_map.size(), sm.size()); |
| for (int i = 1; i <= elements; ++i) { |
| VLOG(1) << "Iteration " << i; |
| EXPECT_TRUE(sm.find(i) != sm.end()); |
| EXPECT_EQ(sm.find(i)->first, i); |
| EXPECT_EQ(sm.find(i)->second, 100*i); |
| } |
| } |
| } |
| |
| TEST(SmallMap, Erase) { |
| small_map<std::unordered_map<std::string, int>> m; |
| small_map<std::unordered_map<std::string, int>>::iterator iter; |
| |
| m["monday"] = 1; |
| m["tuesday"] = 2; |
| m["wednesday"] = 3; |
| |
| EXPECT_EQ(m["monday" ], 1); |
| EXPECT_EQ(m["tuesday" ], 2); |
| EXPECT_EQ(m["wednesday"], 3); |
| EXPECT_EQ(m.count("tuesday"), 1u); |
| EXPECT_FALSE(m.UsingFullMap()); |
| |
| iter = m.begin(); |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, "monday"); |
| EXPECT_EQ(iter->second, 1); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, "tuesday"); |
| EXPECT_EQ(iter->second, 2); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, "wednesday"); |
| EXPECT_EQ(iter->second, 3); |
| ++iter; |
| EXPECT_TRUE(iter == m.end()); |
| |
| EXPECT_EQ(m.erase("tuesday"), 1u); |
| |
| EXPECT_EQ(m["monday" ], 1); |
| EXPECT_EQ(m["wednesday"], 3); |
| EXPECT_EQ(m.count("tuesday"), 0u); |
| EXPECT_EQ(m.erase("tuesday"), 0u); |
| |
| iter = m.begin(); |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, "monday"); |
| EXPECT_EQ(iter->second, 1); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, "wednesday"); |
| EXPECT_EQ(iter->second, 3); |
| ++iter; |
| EXPECT_TRUE(iter == m.end()); |
| |
| m["thursday"] = 4; |
| m["friday"] = 5; |
| EXPECT_EQ(m.size(), 4u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_FALSE(m.UsingFullMap()); |
| |
| m["saturday"] = 6; |
| EXPECT_TRUE(m.UsingFullMap()); |
| |
| EXPECT_EQ(m.count("friday"), 1u); |
| EXPECT_EQ(m.erase("friday"), 1u); |
| EXPECT_TRUE(m.UsingFullMap()); |
| EXPECT_EQ(m.count("friday"), 0u); |
| EXPECT_EQ(m.erase("friday"), 0u); |
| |
| EXPECT_EQ(m.size(), 4u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_EQ(m.erase("monday"), 1u); |
| EXPECT_EQ(m.size(), 3u); |
| EXPECT_FALSE(m.empty()); |
| |
| m.clear(); |
| EXPECT_FALSE(m.UsingFullMap()); |
| EXPECT_EQ(m.size(), 0u); |
| EXPECT_TRUE(m.empty()); |
| } |
| |
| TEST(SmallMap, EraseReturnsIteratorFollowingRemovedElement) { |
| small_map<std::unordered_map<std::string, int>> m; |
| small_map<std::unordered_map<std::string, int>>::iterator iter; |
| |
| m["a"] = 0; |
| m["b"] = 1; |
| m["c"] = 2; |
| |
| // Erase first item. |
| auto following_iter = m.erase(m.begin()); |
| EXPECT_EQ(m.begin(), following_iter); |
| EXPECT_EQ(2u, m.size()); |
| EXPECT_EQ(m.count("a"), 0u); |
| EXPECT_EQ(m.count("b"), 1u); |
| EXPECT_EQ(m.count("c"), 1u); |
| |
| // Iterate to last item and erase it. |
| ++following_iter; |
| following_iter = m.erase(following_iter); |
| ASSERT_EQ(1u, m.size()); |
| EXPECT_EQ(m.end(), following_iter); |
| EXPECT_EQ(m.count("b"), 0u); |
| EXPECT_EQ(m.count("c"), 1u); |
| |
| // Erase remaining item. |
| following_iter = m.erase(m.begin()); |
| EXPECT_TRUE(m.empty()); |
| EXPECT_EQ(m.end(), following_iter); |
| } |
| |
| TEST(SmallMap, NonHashMap) { |
| small_map<std::map<int, int>, 4, std::equal_to<int>> m; |
| EXPECT_TRUE(m.empty()); |
| |
| m[9] = 2; |
| m[0] = 5; |
| |
| EXPECT_EQ(m[9], 2); |
| EXPECT_EQ(m[0], 5); |
| EXPECT_EQ(m.size(), 2u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_FALSE(m.UsingFullMap()); |
| |
| small_map<std::map<int, int>, 4, std::equal_to<int>>::iterator iter( |
| m.begin()); |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 9); |
| EXPECT_EQ(iter->second, 2); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 0); |
| EXPECT_EQ(iter->second, 5); |
| ++iter; |
| EXPECT_TRUE(iter == m.end()); |
| --iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 0); |
| EXPECT_EQ(iter->second, 5); |
| |
| m[8] = 23; |
| m[1234] = 90; |
| m[-5] = 6; |
| |
| EXPECT_EQ(m[ 9], 2); |
| EXPECT_EQ(m[ 0], 5); |
| EXPECT_EQ(m[1234], 90); |
| EXPECT_EQ(m[ 8], 23); |
| EXPECT_EQ(m[ -5], 6); |
| EXPECT_EQ(m.size(), 5u); |
| EXPECT_FALSE(m.empty()); |
| EXPECT_TRUE(m.UsingFullMap()); |
| |
| iter = m.begin(); |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, -5); |
| EXPECT_EQ(iter->second, 6); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 0); |
| EXPECT_EQ(iter->second, 5); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 8); |
| EXPECT_EQ(iter->second, 23); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 9); |
| EXPECT_EQ(iter->second, 2); |
| ++iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 1234); |
| EXPECT_EQ(iter->second, 90); |
| ++iter; |
| EXPECT_TRUE(iter == m.end()); |
| --iter; |
| ASSERT_TRUE(iter != m.end()); |
| EXPECT_EQ(iter->first, 1234); |
| EXPECT_EQ(iter->second, 90); |
| } |
| |
| TEST(SmallMap, DefaultEqualKeyWorks) { |
| // If these tests compile, they pass. The EXPECT calls are only there to avoid |
| // unused variable warnings. |
| small_map<std::unordered_map<int, int>> hm; |
| EXPECT_EQ(0u, hm.size()); |
| small_map<std::map<int, int>> m; |
| EXPECT_EQ(0u, m.size()); |
| } |
| |
| namespace { |
| |
| class unordered_map_add_item : public std::unordered_map<int, int> { |
| public: |
| unordered_map_add_item() = default; |
| explicit unordered_map_add_item(const std::pair<int, int>& item) { |
| insert(item); |
| } |
| }; |
| |
| void InitMap(unordered_map_add_item* map_ctor) { |
| new (map_ctor) unordered_map_add_item(std::make_pair(0, 0)); |
| } |
| |
| class unordered_map_add_item_initializer { |
| public: |
| explicit unordered_map_add_item_initializer(int item_to_add) |
| : item_(item_to_add) {} |
| unordered_map_add_item_initializer() : item_(0) {} |
| void operator()(unordered_map_add_item* map_ctor) const { |
| new (map_ctor) unordered_map_add_item(std::make_pair(item_, item_)); |
| } |
| |
| int item_; |
| }; |
| |
| } // anonymous namespace |
| |
| TEST(SmallMap, SubclassInitializationWithFunctionPointer) { |
| small_map<unordered_map_add_item, 4, std::equal_to<int>, |
| void (&)(unordered_map_add_item*)> |
| m(InitMap); |
| |
| EXPECT_TRUE(m.empty()); |
| |
| m[1] = 1; |
| m[2] = 2; |
| m[3] = 3; |
| m[4] = 4; |
| |
| EXPECT_EQ(4u, m.size()); |
| EXPECT_EQ(0u, m.count(0)); |
| |
| m[5] = 5; |
| EXPECT_EQ(6u, m.size()); |
| // Our function adds an extra item when we convert to a map. |
| EXPECT_EQ(1u, m.count(0)); |
| } |
| |
| TEST(SmallMap, SubclassInitializationWithFunctionObject) { |
| small_map<unordered_map_add_item, 4, std::equal_to<int>, |
| unordered_map_add_item_initializer> |
| m(unordered_map_add_item_initializer(-1)); |
| |
| EXPECT_TRUE(m.empty()); |
| |
| m[1] = 1; |
| m[2] = 2; |
| m[3] = 3; |
| m[4] = 4; |
| |
| EXPECT_EQ(4u, m.size()); |
| EXPECT_EQ(0u, m.count(-1)); |
| |
| m[5] = 5; |
| EXPECT_EQ(6u, m.size()); |
| // Our functor adds an extra item when we convert to a map. |
| EXPECT_EQ(1u, m.count(-1)); |
| } |
| |
| namespace { |
| |
| // This class acts as a basic implementation of a move-only type. The canonical |
| // example of such a type is scoped_ptr/unique_ptr. |
| template <typename V> |
| class MoveOnlyType { |
| public: |
| MoveOnlyType() : value_(0) {} |
| explicit MoveOnlyType(V value) : value_(value) {} |
| |
| MoveOnlyType(MoveOnlyType&& other) { |
| *this = std::move(other); |
| } |
| |
| MoveOnlyType& operator=(MoveOnlyType&& other) { |
| value_ = other.value_; |
| other.value_ = 0; |
| return *this; |
| } |
| |
| MoveOnlyType(const MoveOnlyType&) = delete; |
| MoveOnlyType& operator=(const MoveOnlyType&) = delete; |
| |
| V value() const { return value_; } |
| |
| private: |
| V value_; |
| }; |
| |
| } // namespace |
| |
| TEST(SmallMap, MoveOnlyValueType) { |
| small_map<std::map<int, MoveOnlyType<int>>, 2> m; |
| |
| m[0] = MoveOnlyType<int>(1); |
| m[1] = MoveOnlyType<int>(2); |
| m.erase(m.begin()); |
| |
| // small_map will move m[1] to an earlier index in the internal array. |
| EXPECT_EQ(m.size(), 1u); |
| EXPECT_EQ(m[1].value(), 2); |
| |
| m[0] = MoveOnlyType<int>(1); |
| // small_map must move the values from the array into the internal std::map. |
| m[2] = MoveOnlyType<int>(3); |
| |
| EXPECT_EQ(m.size(), 3u); |
| EXPECT_EQ(m[0].value(), 1); |
| EXPECT_EQ(m[1].value(), 2); |
| EXPECT_EQ(m[2].value(), 3); |
| |
| m.erase(m.begin()); |
| |
| // small_map should also let internal std::map erase with a move-only type. |
| EXPECT_EQ(m.size(), 2u); |
| EXPECT_EQ(m[1].value(), 2); |
| EXPECT_EQ(m[2].value(), 3); |
| } |
| |
| TEST(SmallMap, Emplace) { |
| small_map<std::map<size_t, MoveOnlyType<size_t>>> sm; |
| |
| // loop through the transition from small map to map. |
| for (size_t i = 1; i <= 10; ++i) { |
| // insert an element |
| auto ret = sm.emplace(i, MoveOnlyType<size_t>(100 * i)); |
| EXPECT_TRUE(ret.second); |
| EXPECT_TRUE(ret.first == sm.find(i)); |
| EXPECT_EQ(ret.first->first, i); |
| EXPECT_EQ(ret.first->second.value(), 100 * i); |
| |
| // try to insert it again with different value, fails, but we still get an |
| // iterator back with the original value. |
| ret = sm.emplace(i, MoveOnlyType<size_t>(i)); |
| EXPECT_FALSE(ret.second); |
| EXPECT_TRUE(ret.first == sm.find(i)); |
| EXPECT_EQ(ret.first->first, i); |
| EXPECT_EQ(ret.first->second.value(), 100 * i); |
| |
| // check the state of the map. |
| for (size_t j = 1; j <= i; ++j) { |
| const auto it = sm.find(j); |
| EXPECT_TRUE(it != sm.end()); |
| EXPECT_EQ(it->first, j); |
| EXPECT_EQ(it->second.value(), j * 100); |
| } |
| EXPECT_EQ(sm.size(), i); |
| EXPECT_FALSE(sm.empty()); |
| } |
| } |
| |
| } // namespace base |