| // Copyright 2017 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. | 
 |  | 
 | #ifndef BASE_CONTAINERS_FLAT_MAP_H_ | 
 | #define BASE_CONTAINERS_FLAT_MAP_H_ | 
 |  | 
 | #include <functional> | 
 | #include <tuple> | 
 | #include <utility> | 
 |  | 
 | #include "base/containers/flat_tree.h" | 
 | #include "base/logging.h" | 
 | #include "base/template_util.h" | 
 |  | 
 | namespace base { | 
 |  | 
 | namespace internal { | 
 |  | 
 | // An implementation of the flat_tree GetKeyFromValue template parameter that | 
 | // extracts the key as the first element of a pair. | 
 | template <class Key, class Mapped> | 
 | struct GetKeyFromValuePairFirst { | 
 |   const Key& operator()(const std::pair<Key, Mapped>& p) const { | 
 |     return p.first; | 
 |   } | 
 | }; | 
 |  | 
 | }  // namespace internal | 
 |  | 
 | // flat_map is a container with a std::map-like interface that stores its | 
 | // contents in a sorted vector. | 
 | // | 
 | // Please see //base/containers/README.md for an overview of which container | 
 | // to select. | 
 | // | 
 | // PROS | 
 | // | 
 | //  - Good memory locality. | 
 | //  - Low overhead, especially for smaller maps. | 
 | //  - Performance is good for more workloads than you might expect (see | 
 | //    overview link above). | 
 | //  - Supports C++14 map interface. | 
 | // | 
 | // CONS | 
 | // | 
 | //  - Inserts and removals are O(n). | 
 | // | 
 | // IMPORTANT NOTES | 
 | // | 
 | //  - Iterators are invalidated across mutations. | 
 | //  - If possible, construct a flat_map in one operation by inserting into | 
 | //    a std::vector and moving that vector into the flat_map constructor. | 
 | // | 
 | // QUICK REFERENCE | 
 | // | 
 | // Most of the core functionality is inherited from flat_tree. Please see | 
 | // flat_tree.h for more details for most of these functions. As a quick | 
 | // reference, the functions available are: | 
 | // | 
 | // Constructors (inputs need not be sorted): | 
 | //   flat_map(InputIterator first, InputIterator last, | 
 | //            FlatContainerDupes = KEEP_FIRST_OF_DUPES, | 
 | //            const Compare& compare = Compare()); | 
 | //   flat_map(const flat_map&); | 
 | //   flat_map(flat_map&&); | 
 | //   flat_map(std::vector<value_type>, | 
 | //            FlatContainerDupes = KEEP_FIRST_OF_DUPES, | 
 | //            const Compare& compare = Compare()); // Re-use storage. | 
 | //   flat_map(std::initializer_list<value_type> ilist, | 
 | //            FlatContainerDupes = KEEP_FIRST_OF_DUPES, | 
 | //            const Compare& comp = Compare()); | 
 | // | 
 | // Assignment functions: | 
 | //   flat_map& operator=(const flat_map&); | 
 | //   flat_map& operator=(flat_map&&); | 
 | //   flat_map& operator=(initializer_list<value_type>); | 
 | // | 
 | // Memory management functions: | 
 | //   void   reserve(size_t); | 
 | //   size_t capacity() const; | 
 | //   void   shrink_to_fit(); | 
 | // | 
 | // Size management functions: | 
 | //   void   clear(); | 
 | //   size_t size() const; | 
 | //   size_t max_size() const; | 
 | //   bool   empty() const; | 
 | // | 
 | // Iterator functions: | 
 | //   iterator               begin(); | 
 | //   const_iterator         begin() const; | 
 | //   const_iterator         cbegin() const; | 
 | //   iterator               end(); | 
 | //   const_iterator         end() const; | 
 | //   const_iterator         cend() const; | 
 | //   reverse_iterator       rbegin(); | 
 | //   const reverse_iterator rbegin() const; | 
 | //   const_reverse_iterator crbegin() const; | 
 | //   reverse_iterator       rend(); | 
 | //   const_reverse_iterator rend() const; | 
 | //   const_reverse_iterator crend() const; | 
 | // | 
 | // Insert and accessor functions: | 
 | //   mapped_type&         operator[](const key_type&); | 
 | //   mapped_type&         operator[](key_type&&); | 
 | //   pair<iterator, bool> insert(const value_type&); | 
 | //   pair<iterator, bool> insert(value_type&&); | 
 | //   iterator             insert(const_iterator hint, const value_type&); | 
 | //   iterator             insert(const_iterator hint, value_type&&); | 
 | //   void                 insert(InputIterator first, InputIterator last, | 
 | //                               FlatContainerDupes = KEEP_FIRST_OF_DUPES); | 
 | //   pair<iterator, bool> insert_or_assign(K&&, M&&); | 
 | //   iterator             insert_or_assign(const_iterator hint, K&&, M&&); | 
 | //   pair<iterator, bool> emplace(Args&&...); | 
 | //   iterator             emplace_hint(const_iterator, Args&&...); | 
 | //   pair<iterator, bool> try_emplace(K&&, Args&&...); | 
 | //   iterator             try_emplace(const_iterator hint, K&&, Args&&...); | 
 | // | 
 | // Erase functions: | 
 | //   iterator erase(iterator); | 
 | //   iterator erase(const_iterator); | 
 | //   iterator erase(const_iterator first, const_iterator& last); | 
 | //   template <class K> size_t erase(const K& key); | 
 | // | 
 | // Comparators (see std::map documentation). | 
 | //   key_compare   key_comp() const; | 
 | //   value_compare value_comp() const; | 
 | // | 
 | // Search functions: | 
 | //   template <typename K> size_t                   count(const K&) const; | 
 | //   template <typename K> iterator                 find(const K&); | 
 | //   template <typename K> const_iterator           find(const K&) const; | 
 | //   template <typename K> pair<iterator, iterator> equal_range(const K&); | 
 | //   template <typename K> iterator                 lower_bound(const K&); | 
 | //   template <typename K> const_iterator           lower_bound(const K&) const; | 
 | //   template <typename K> iterator                 upper_bound(const K&); | 
 | //   template <typename K> const_iterator           upper_bound(const K&) const; | 
 | // | 
 | // General functions: | 
 | //   void swap(flat_map&&); | 
 | // | 
 | // Non-member operators: | 
 | //   bool operator==(const flat_map&, const flat_map); | 
 | //   bool operator!=(const flat_map&, const flat_map); | 
 | //   bool operator<(const flat_map&, const flat_map); | 
 | //   bool operator>(const flat_map&, const flat_map); | 
 | //   bool operator>=(const flat_map&, const flat_map); | 
 | //   bool operator<=(const flat_map&, const flat_map); | 
 | // | 
 | template <class Key, class Mapped, class Compare = std::less<>> | 
 | class flat_map : public ::base::internal::flat_tree< | 
 |                      Key, | 
 |                      std::pair<Key, Mapped>, | 
 |                      ::base::internal::GetKeyFromValuePairFirst<Key, Mapped>, | 
 |                      Compare> { | 
 |  private: | 
 |   using tree = typename ::base::internal::flat_tree< | 
 |       Key, | 
 |       std::pair<Key, Mapped>, | 
 |       ::base::internal::GetKeyFromValuePairFirst<Key, Mapped>, | 
 |       Compare>; | 
 |  | 
 |  public: | 
 |   using key_type = typename tree::key_type; | 
 |   using mapped_type = Mapped; | 
 |   using value_type = typename tree::value_type; | 
 |   using iterator = typename tree::iterator; | 
 |   using const_iterator = typename tree::const_iterator; | 
 |  | 
 |   // -------------------------------------------------------------------------- | 
 |   // Lifetime and assignments. | 
 |   // | 
 |   // Note: we could do away with these constructors, destructor and assignment | 
 |   // operator overloads by inheriting |tree|'s, but this breaks the GCC build | 
 |   // due to https://gcc.gnu.org/bugzilla/show_bug.cgi?id=84782 (see | 
 |   // https://crbug.com/837221). | 
 |  | 
 |   flat_map() = default; | 
 |   explicit flat_map(const Compare& comp); | 
 |  | 
 |   template <class InputIterator> | 
 |   flat_map(InputIterator first, | 
 |            InputIterator last, | 
 |            FlatContainerDupes dupe_handling = KEEP_FIRST_OF_DUPES, | 
 |            const Compare& comp = Compare()); | 
 |  | 
 |   flat_map(const flat_map&) = default; | 
 |   flat_map(flat_map&&) noexcept = default; | 
 |  | 
 |   flat_map(std::vector<value_type> items, | 
 |            FlatContainerDupes dupe_handling = KEEP_FIRST_OF_DUPES, | 
 |            const Compare& comp = Compare()); | 
 |  | 
 |   flat_map(std::initializer_list<value_type> ilist, | 
 |            FlatContainerDupes dupe_handling = KEEP_FIRST_OF_DUPES, | 
 |            const Compare& comp = Compare()); | 
 |  | 
 |   ~flat_map() = default; | 
 |  | 
 |   flat_map& operator=(const flat_map&) = default; | 
 |   flat_map& operator=(flat_map&&) = default; | 
 |   // Takes the first if there are duplicates in the initializer list. | 
 |   flat_map& operator=(std::initializer_list<value_type> ilist); | 
 |  | 
 |   // -------------------------------------------------------------------------- | 
 |   // Map-specific insert operations. | 
 |   // | 
 |   // Normal insert() functions are inherited from flat_tree. | 
 |   // | 
 |   // Assume that every operation invalidates iterators and references. | 
 |   // Insertion of one element can take O(size). | 
 |  | 
 |   mapped_type& operator[](const key_type& key); | 
 |   mapped_type& operator[](key_type&& key); | 
 |  | 
 |   template <class K, class M> | 
 |   std::pair<iterator, bool> insert_or_assign(K&& key, M&& obj); | 
 |   template <class K, class M> | 
 |   iterator insert_or_assign(const_iterator hint, K&& key, M&& obj); | 
 |  | 
 |   template <class K, class... Args> | 
 |   std::enable_if_t<std::is_constructible<key_type, K&&>::value, | 
 |                    std::pair<iterator, bool>> | 
 |   try_emplace(K&& key, Args&&... args); | 
 |  | 
 |   template <class K, class... Args> | 
 |   std::enable_if_t<std::is_constructible<key_type, K&&>::value, iterator> | 
 |   try_emplace(const_iterator hint, K&& key, Args&&... args); | 
 |  | 
 |   // -------------------------------------------------------------------------- | 
 |   // General operations. | 
 |   // | 
 |   // Assume that swap invalidates iterators and references. | 
 |  | 
 |   void swap(flat_map& other) noexcept; | 
 |  | 
 |   friend void swap(flat_map& lhs, flat_map& rhs) noexcept { lhs.swap(rhs); } | 
 | }; | 
 |  | 
 | // ---------------------------------------------------------------------------- | 
 | // Lifetime. | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | flat_map<Key, Mapped, Compare>::flat_map(const Compare& comp) : tree(comp) {} | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | template <class InputIterator> | 
 | flat_map<Key, Mapped, Compare>::flat_map(InputIterator first, | 
 |                                          InputIterator last, | 
 |                                          FlatContainerDupes dupe_handling, | 
 |                                          const Compare& comp) | 
 |     : tree(first, last, dupe_handling, comp) {} | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | flat_map<Key, Mapped, Compare>::flat_map(std::vector<value_type> items, | 
 |                                          FlatContainerDupes dupe_handling, | 
 |                                          const Compare& comp) | 
 |     : tree(std::move(items), dupe_handling, comp) {} | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | flat_map<Key, Mapped, Compare>::flat_map( | 
 |     std::initializer_list<value_type> ilist, | 
 |     FlatContainerDupes dupe_handling, | 
 |     const Compare& comp) | 
 |     : flat_map(std::begin(ilist), std::end(ilist), dupe_handling, comp) {} | 
 |  | 
 | // ---------------------------------------------------------------------------- | 
 | // Assignments. | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | auto flat_map<Key, Mapped, Compare>::operator=( | 
 |     std::initializer_list<value_type> ilist) -> flat_map& { | 
 |   // When https://gcc.gnu.org/bugzilla/show_bug.cgi?id=84782 gets fixed, we | 
 |   // need to remember to inherit tree::operator= to prevent | 
 |   //   flat_map<...> x; | 
 |   //   x = {...}; | 
 |   // from first creating a flat_map and then move assigning it. This most | 
 |   // likely would be optimized away but still affects our debug builds. | 
 |   tree::operator=(ilist); | 
 |   return *this; | 
 | } | 
 |  | 
 | // ---------------------------------------------------------------------------- | 
 | // Insert operations. | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | auto flat_map<Key, Mapped, Compare>::operator[](const key_type& key) | 
 |     -> mapped_type& { | 
 |   iterator found = tree::lower_bound(key); | 
 |   if (found == tree::end() || tree::key_comp()(key, found->first)) | 
 |     found = tree::unsafe_emplace(found, key, mapped_type()); | 
 |   return found->second; | 
 | } | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | auto flat_map<Key, Mapped, Compare>::operator[](key_type&& key) | 
 |     -> mapped_type& { | 
 |   iterator found = tree::lower_bound(key); | 
 |   if (found == tree::end() || tree::key_comp()(key, found->first)) | 
 |     found = tree::unsafe_emplace(found, std::move(key), mapped_type()); | 
 |   return found->second; | 
 | } | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | template <class K, class M> | 
 | auto flat_map<Key, Mapped, Compare>::insert_or_assign(K&& key, M&& obj) | 
 |     -> std::pair<iterator, bool> { | 
 |   auto result = | 
 |       tree::emplace_key_args(key, std::forward<K>(key), std::forward<M>(obj)); | 
 |   if (!result.second) | 
 |     result.first->second = std::forward<M>(obj); | 
 |   return result; | 
 | } | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | template <class K, class M> | 
 | auto flat_map<Key, Mapped, Compare>::insert_or_assign(const_iterator hint, | 
 |                                                       K&& key, | 
 |                                                       M&& obj) -> iterator { | 
 |   auto result = tree::emplace_hint_key_args(hint, key, std::forward<K>(key), | 
 |                                             std::forward<M>(obj)); | 
 |   if (!result.second) | 
 |     result.first->second = std::forward<M>(obj); | 
 |   return result.first; | 
 | } | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | template <class K, class... Args> | 
 | auto flat_map<Key, Mapped, Compare>::try_emplace(K&& key, Args&&... args) | 
 |     -> std::enable_if_t<std::is_constructible<key_type, K&&>::value, | 
 |                         std::pair<iterator, bool>> { | 
 |   return tree::emplace_key_args( | 
 |       key, std::piecewise_construct, | 
 |       std::forward_as_tuple(std::forward<K>(key)), | 
 |       std::forward_as_tuple(std::forward<Args>(args)...)); | 
 | } | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | template <class K, class... Args> | 
 | auto flat_map<Key, Mapped, Compare>::try_emplace(const_iterator hint, | 
 |                                                  K&& key, | 
 |                                                  Args&&... args) | 
 |     -> std::enable_if_t<std::is_constructible<key_type, K&&>::value, iterator> { | 
 |   return tree::emplace_hint_key_args( | 
 |              hint, key, std::piecewise_construct, | 
 |              std::forward_as_tuple(std::forward<K>(key)), | 
 |              std::forward_as_tuple(std::forward<Args>(args)...)) | 
 |       .first; | 
 | } | 
 |  | 
 | // ---------------------------------------------------------------------------- | 
 | // General operations. | 
 |  | 
 | template <class Key, class Mapped, class Compare> | 
 | void flat_map<Key, Mapped, Compare>::swap(flat_map& other) noexcept { | 
 |   tree::swap(other); | 
 | } | 
 |  | 
 | }  // namespace base | 
 |  | 
 | #endif  // BASE_CONTAINERS_FLAT_MAP_H_ |