• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // detail/hash_map.hpp
3 // ~~~~~~~~~~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2015 Christopher M. Kohlhoff (chris at kohlhoff dot com)
6 //
7 // Distributed under the Boost Software License, Version 1.0. (See accompanying
8 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9 //
10 
11 #ifndef ASIO_DETAIL_HASH_MAP_HPP
12 #define ASIO_DETAIL_HASH_MAP_HPP
13 
14 
15 #include "asio/detail/config.hpp"
16 #include <list>
17 #include <utility>
18 #include "asio/detail/assert.hpp"
19 #include "asio/detail/noncopyable.hpp"
20 
21 
22 #include "asio/detail/push_options.hpp"
23 
24 namespace asio {
25 namespace detail {
26 
calculate_hash_value(int i)27 inline std::size_t calculate_hash_value(int i)
28 {
29   return static_cast<std::size_t>(i);
30 }
31 
calculate_hash_value(void * p)32 inline std::size_t calculate_hash_value(void* p)
33 {
34   return reinterpret_cast<std::size_t>(p)
35     + (reinterpret_cast<std::size_t>(p) >> 3);
36 }
37 
38 
39 // Note: assumes K and V are POD types.
40 template <typename K, typename V>
41 class hash_map
42   : private noncopyable
43 {
44 public:
45   // The type of a value in the map.
46   typedef std::pair<K, V> value_type;
47 
48   // The type of a non-const iterator over the hash map.
49   typedef typename std::list<value_type>::iterator iterator;
50 
51   // The type of a const iterator over the hash map.
52   typedef typename std::list<value_type>::const_iterator const_iterator;
53 
54   // Constructor.
hash_map()55   hash_map()
56     : size_(0),
57       buckets_(0),
58       num_buckets_(0)
59   {
60   }
61 
62   // Destructor.
~hash_map()63   ~hash_map()
64   {
65     delete[] buckets_;
66   }
67 
68   // Get an iterator for the beginning of the map.
begin()69   iterator begin()
70   {
71     return values_.begin();
72   }
73 
74   // Get an iterator for the beginning of the map.
begin() const75   const_iterator begin() const
76   {
77     return values_.begin();
78   }
79 
80   // Get an iterator for the end of the map.
end()81   iterator end()
82   {
83     return values_.end();
84   }
85 
86   // Get an iterator for the end of the map.
end() const87   const_iterator end() const
88   {
89     return values_.end();
90   }
91 
92   // Check whether the map is empty.
empty() const93   bool empty() const
94   {
95     return values_.empty();
96   }
97 
98   // Find an entry in the map.
find(const K & k)99   iterator find(const K& k)
100   {
101     if (num_buckets_)
102     {
103       size_t bucket = calculate_hash_value(k) % num_buckets_;
104       iterator it = buckets_[bucket].first;
105       if (it == values_.end())
106         return values_.end();
107       iterator end_it = buckets_[bucket].last;
108       ++end_it;
109       while (it != end_it)
110       {
111         if (it->first == k)
112           return it;
113         ++it;
114       }
115     }
116     return values_.end();
117   }
118 
119   // Find an entry in the map.
find(const K & k) const120   const_iterator find(const K& k) const
121   {
122     if (num_buckets_)
123     {
124       size_t bucket = calculate_hash_value(k) % num_buckets_;
125       const_iterator it = buckets_[bucket].first;
126       if (it == values_.end())
127         return it;
128       const_iterator end_it = buckets_[bucket].last;
129       ++end_it;
130       while (it != end_it)
131       {
132         if (it->first == k)
133           return it;
134         ++it;
135       }
136     }
137     return values_.end();
138   }
139 
140   // Insert a new entry into the map.
insert(const value_type & v)141   std::pair<iterator, bool> insert(const value_type& v)
142   {
143     if (size_ + 1 >= num_buckets_)
144       rehash(hash_size(size_ + 1));
145     size_t bucket = calculate_hash_value(v.first) % num_buckets_;
146     iterator it = buckets_[bucket].first;
147     if (it == values_.end())
148     {
149       buckets_[bucket].first = buckets_[bucket].last =
150         values_insert(values_.end(), v);
151       ++size_;
152       return std::pair<iterator, bool>(buckets_[bucket].last, true);
153     }
154     iterator end_it = buckets_[bucket].last;
155     ++end_it;
156     while (it != end_it)
157     {
158       if (it->first == v.first)
159         return std::pair<iterator, bool>(it, false);
160       ++it;
161     }
162     buckets_[bucket].last = values_insert(end_it, v);
163     ++size_;
164     return std::pair<iterator, bool>(buckets_[bucket].last, true);
165   }
166 
167   // Erase an entry from the map.
erase(iterator it)168   void erase(iterator it)
169   {
170     ASIO_ASSERT(it != values_.end());
171     ASIO_ASSERT(num_buckets_ != 0);
172 
173     size_t bucket = calculate_hash_value(it->first) % num_buckets_;
174     bool is_first = (it == buckets_[bucket].first);
175     bool is_last = (it == buckets_[bucket].last);
176     if (is_first && is_last)
177       buckets_[bucket].first = buckets_[bucket].last = values_.end();
178     else if (is_first)
179       ++buckets_[bucket].first;
180     else if (is_last)
181       --buckets_[bucket].last;
182 
183     values_erase(it);
184     --size_;
185   }
186 
187   // Erase a key from the map.
erase(const K & k)188   void erase(const K& k)
189   {
190     iterator it = find(k);
191     if (it != values_.end())
192       erase(it);
193   }
194 
195   // Remove all entries from the map.
clear()196   void clear()
197   {
198     // Clear the values.
199     values_.clear();
200     size_ = 0;
201 
202     // Initialise all buckets to empty.
203     iterator end_it = values_.end();
204     for (size_t i = 0; i < num_buckets_; ++i)
205       buckets_[i].first = buckets_[i].last = end_it;
206   }
207 
208 private:
209   // Calculate the hash size for the specified number of elements.
hash_size(std::size_t num_elems)210   static std::size_t hash_size(std::size_t num_elems)
211   {
212     static std::size_t sizes[] =
213     {
214 #if defined(ASIO_HASH_MAP_BUCKETS)
215       ASIO_HASH_MAP_BUCKETS
216 #else // ASIO_HASH_MAP_BUCKETS
217       3, 13, 23, 53, 97, 193, 389, 769, 1543, 3079, 6151, 12289, 24593,
218       49157, 98317, 196613, 393241, 786433, 1572869, 3145739, 6291469,
219       12582917, 25165843
220 #endif // ASIO_HASH_MAP_BUCKETS
221     };
222     const std::size_t nth_size = sizeof(sizes) / sizeof(std::size_t) - 1;
223     for (std::size_t i = 0; i < nth_size; ++i)
224       if (num_elems < sizes[i])
225         return sizes[i];
226     return sizes[nth_size];
227   }
228 
229   // Re-initialise the hash from the values already contained in the list.
rehash(std::size_t num_buckets)230   void rehash(std::size_t num_buckets)
231   {
232     if (num_buckets == num_buckets_)
233       return;
234     num_buckets_ = num_buckets;
235     ASIO_ASSERT(num_buckets_ != 0);
236 
237     iterator end_iter = values_.end();
238 
239     // Update number of buckets and initialise all buckets to empty.
240     bucket_type* tmp = new bucket_type[num_buckets_];
241     delete[] buckets_;
242     buckets_ = tmp;
243     for (std::size_t i = 0; i < num_buckets_; ++i)
244       buckets_[i].first = buckets_[i].last = end_iter;
245 
246     // Put all values back into the hash.
247     iterator iter = values_.begin();
248     while (iter != end_iter)
249     {
250       std::size_t bucket = calculate_hash_value(iter->first) % num_buckets_;
251       if (buckets_[bucket].last == end_iter)
252       {
253         buckets_[bucket].first = buckets_[bucket].last = iter++;
254       }
255       else if (++buckets_[bucket].last == iter)
256       {
257         ++iter;
258       }
259       else
260       {
261         values_.splice(buckets_[bucket].last, values_, iter++);
262         --buckets_[bucket].last;
263       }
264     }
265   }
266 
267   // Insert an element into the values list by splicing from the spares list,
268   // if a spare is available, and otherwise by inserting a new element.
values_insert(iterator it,const value_type & v)269   iterator values_insert(iterator it, const value_type& v)
270   {
271     if (spares_.empty())
272     {
273       return values_.insert(it, v);
274     }
275     else
276     {
277       spares_.front() = v;
278       values_.splice(it, spares_, spares_.begin());
279       return --it;
280     }
281   }
282 
283   // Erase an element from the values list by splicing it to the spares list.
values_erase(iterator it)284   void values_erase(iterator it)
285   {
286     *it = value_type();
287     spares_.splice(spares_.begin(), values_, it);
288   }
289 
290   // The number of elements in the hash.
291   std::size_t size_;
292 
293   // The list of all values in the hash map.
294   std::list<value_type> values_;
295 
296   // The list of spare nodes waiting to be recycled. Assumes that POD types only
297   // are stored in the hash map.
298   std::list<value_type> spares_;
299 
300   // The type for a bucket in the hash table.
301   struct bucket_type
302   {
303     iterator first;
304     iterator last;
305   };
306 
307   // The buckets in the hash.
308   bucket_type* buckets_;
309 
310   // The number of buckets in the hash.
311   std::size_t num_buckets_;
312 };
313 
314 } // namespace detail
315 } // namespace asio
316 
317 #include "asio/detail/pop_options.hpp"
318 
319 #endif // ASIO_DETAIL_HASH_MAP_HPP
320