blob: 880ef967e17e3b9eac34b49769bd9d3a27d18f5a [file] [log] [blame]
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001// -*- C++ -*-
2//===----------------------------------------------------------------------===//
3//
Howard Hinnantf5256e12010-05-11 21:36:01 +00004// The LLVM Compiler Infrastructure
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00005//
Howard Hinnantb64f8b02010-11-16 22:09:02 +00006// This file is dual licensed under the MIT and the University of Illinois Open
7// Source Licenses. See LICENSE.TXT for details.
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00008//
9//===----------------------------------------------------------------------===//
10
11#ifndef _LIBCPP__HASH_TABLE
12#define _LIBCPP__HASH_TABLE
13
14#include <__config>
15#include <initializer_list>
16#include <memory>
17#include <iterator>
18#include <algorithm>
19#include <cmath>
20
Howard Hinnant66c6f972011-11-29 16:45:27 +000021#include <__undef_min_max>
22
Howard Hinnant8b00e6c2013-08-02 00:26:35 +000023#ifdef _LIBCPP_DEBUG2
24# include <__debug>
25#else
26# define _LIBCPP_ASSERT(x, m) ((void)0)
27#endif
28
Howard Hinnant08e17472011-10-17 20:05:10 +000029#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000030#pragma GCC system_header
Howard Hinnant08e17472011-10-17 20:05:10 +000031#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000032
33_LIBCPP_BEGIN_NAMESPACE_STD
34
Howard Hinnant83eade62013-03-06 23:30:19 +000035_LIBCPP_FUNC_VIS
Howard Hinnant2b1b2d42011-06-14 19:58:17 +000036size_t __next_prime(size_t __n);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000037
38template <class _NodePtr>
39struct __hash_node_base
40{
41 typedef __hash_node_base __first_node;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000042
Howard Hinnantdf85e572011-02-27 18:02:02 +000043 _NodePtr __next_;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000044
Howard Hinnant5f2f14c2011-06-04 18:54:24 +000045 _LIBCPP_INLINE_VISIBILITY __hash_node_base() _NOEXCEPT : __next_(nullptr) {}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000046};
47
48template <class _Tp, class _VoidPtr>
49struct __hash_node
50 : public __hash_node_base
51 <
52 typename pointer_traits<_VoidPtr>::template
53#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
54 rebind<__hash_node<_Tp, _VoidPtr> >
55#else
56 rebind<__hash_node<_Tp, _VoidPtr> >::other
57#endif
58 >
59{
60 typedef _Tp value_type;
61
62 size_t __hash_;
63 value_type __value_;
64};
65
Howard Hinnant7a445152012-07-06 17:31:14 +000066inline _LIBCPP_INLINE_VISIBILITY
67bool
68__is_power2(size_t __bc)
69{
70 return __bc > 2 && !(__bc & (__bc - 1));
71}
72
73inline _LIBCPP_INLINE_VISIBILITY
74size_t
75__constrain_hash(size_t __h, size_t __bc)
76{
77 return !(__bc & (__bc - 1)) ? __h & (__bc - 1) : __h % __bc;
78}
79
80inline _LIBCPP_INLINE_VISIBILITY
81size_t
82__next_pow2(size_t __n)
83{
84 return size_t(1) << (std::numeric_limits<size_t>::digits - __clz(__n-1));
85}
86
Howard Hinnant2b1b2d42011-06-14 19:58:17 +000087template <class _Tp, class _Hash, class _Equal, class _Alloc> class __hash_table;
Howard Hinnant0f678bd2013-08-12 18:38:34 +000088template <class _ConstNodePtr> class _LIBCPP_TYPE_VIS_ONLY __hash_const_iterator;
89template <class _HashIterator> class _LIBCPP_TYPE_VIS_ONLY __hash_map_iterator;
90template <class _HashIterator> class _LIBCPP_TYPE_VIS_ONLY __hash_map_const_iterator;
Howard Hinnant2b1b2d42011-06-14 19:58:17 +000091template <class _Key, class _Tp, class _Hash, class _Pred, class _Alloc>
Howard Hinnant0f678bd2013-08-12 18:38:34 +000092 class _LIBCPP_TYPE_VIS_ONLY unordered_map;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000093
94template <class _NodePtr>
Howard Hinnant0f678bd2013-08-12 18:38:34 +000095class _LIBCPP_TYPE_VIS_ONLY __hash_iterator
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000096{
97 typedef _NodePtr __node_pointer;
98
99 __node_pointer __node_;
100
101public:
102 typedef forward_iterator_tag iterator_category;
103 typedef typename pointer_traits<__node_pointer>::element_type::value_type value_type;
104 typedef typename pointer_traits<__node_pointer>::difference_type difference_type;
105 typedef value_type& reference;
106 typedef typename pointer_traits<__node_pointer>::template
107#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
108 rebind<value_type>
109#else
110 rebind<value_type>::other
111#endif
112 pointer;
113
Howard Hinnant39213642013-07-23 22:01:58 +0000114 _LIBCPP_INLINE_VISIBILITY __hash_iterator() _NOEXCEPT
Marshall Clow193ef032013-08-07 21:30:44 +0000115#if _LIBCPP_STD_VER > 11
116 : __node_(nullptr)
117#endif
Howard Hinnant39213642013-07-23 22:01:58 +0000118 {
119#if _LIBCPP_DEBUG_LEVEL >= 2
120 __get_db()->__insert_i(this);
121#endif
122 }
123
124#if _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000125
Howard Hinnant99acc502010-09-21 17:32:39 +0000126 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000127 __hash_iterator(const __hash_iterator& __i)
128 : __node_(__i.__node_)
129 {
130 __get_db()->__iterator_copy(this, &__i);
131 }
132
Howard Hinnant99acc502010-09-21 17:32:39 +0000133 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000134 ~__hash_iterator()
135 {
136 __get_db()->__erase_i(this);
137 }
138
139 _LIBCPP_INLINE_VISIBILITY
140 __hash_iterator& operator=(const __hash_iterator& __i)
141 {
142 if (this != &__i)
143 {
144 __get_db()->__iterator_copy(this, &__i);
145 __node_ = __i.__node_;
146 }
147 return *this;
148 }
149
150#endif // _LIBCPP_DEBUG_LEVEL >= 2
151
152 _LIBCPP_INLINE_VISIBILITY
153 reference operator*() const
154 {
155#if _LIBCPP_DEBUG_LEVEL >= 2
156 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
157 "Attempted to dereference a non-dereferenceable unordered container iterator");
158#endif
159 return __node_->__value_;
160 }
161 _LIBCPP_INLINE_VISIBILITY
162 pointer operator->() const
163 {
164#if _LIBCPP_DEBUG_LEVEL >= 2
165 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
166 "Attempted to dereference a non-dereferenceable unordered container iterator");
167#endif
168 return pointer_traits<pointer>::pointer_to(__node_->__value_);
169 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000170
Howard Hinnant99acc502010-09-21 17:32:39 +0000171 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000172 __hash_iterator& operator++()
173 {
Howard Hinnant39213642013-07-23 22:01:58 +0000174#if _LIBCPP_DEBUG_LEVEL >= 2
175 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
176 "Attempted to increment non-incrementable unordered container iterator");
177#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000178 __node_ = __node_->__next_;
179 return *this;
180 }
181
Howard Hinnant99acc502010-09-21 17:32:39 +0000182 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000183 __hash_iterator operator++(int)
184 {
185 __hash_iterator __t(*this);
186 ++(*this);
187 return __t;
188 }
189
Howard Hinnant99acc502010-09-21 17:32:39 +0000190 friend _LIBCPP_INLINE_VISIBILITY
191 bool operator==(const __hash_iterator& __x, const __hash_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000192 {
Howard Hinnant39213642013-07-23 22:01:58 +0000193 return __x.__node_ == __y.__node_;
194 }
Howard Hinnant99acc502010-09-21 17:32:39 +0000195 friend _LIBCPP_INLINE_VISIBILITY
196 bool operator!=(const __hash_iterator& __x, const __hash_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000197 {return !(__x == __y);}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000198
199private:
Howard Hinnant39213642013-07-23 22:01:58 +0000200#if _LIBCPP_DEBUG_LEVEL >= 2
201 _LIBCPP_INLINE_VISIBILITY
202 __hash_iterator(__node_pointer __node, const void* __c) _NOEXCEPT
203 : __node_(__node)
204 {
205 __get_db()->__insert_ic(this, __c);
206 }
207#else
Howard Hinnant99acc502010-09-21 17:32:39 +0000208 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000209 __hash_iterator(__node_pointer __node) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000210 : __node_(__node)
211 {}
Howard Hinnant39213642013-07-23 22:01:58 +0000212#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000213
214 template <class, class, class, class> friend class __hash_table;
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000215 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_const_iterator;
216 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_map_iterator;
217 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_map;
218 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_multimap;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000219};
220
221template <class _ConstNodePtr>
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000222class _LIBCPP_TYPE_VIS_ONLY __hash_const_iterator
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000223{
224 typedef _ConstNodePtr __node_pointer;
225
226 __node_pointer __node_;
227
228 typedef typename remove_const<
229 typename pointer_traits<__node_pointer>::element_type
230 >::type __node;
231
232public:
233 typedef forward_iterator_tag iterator_category;
234 typedef typename __node::value_type value_type;
235 typedef typename pointer_traits<__node_pointer>::difference_type difference_type;
236 typedef const value_type& reference;
237 typedef typename pointer_traits<__node_pointer>::template
238#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
239 rebind<const value_type>
240#else
241 rebind<const value_type>::other
242#endif
243 pointer;
244 typedef typename pointer_traits<__node_pointer>::template
245#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
246 rebind<__node>
247#else
248 rebind<__node>::other
249#endif
250 __non_const_node_pointer;
251 typedef __hash_iterator<__non_const_node_pointer> __non_const_iterator;
252
Howard Hinnant39213642013-07-23 22:01:58 +0000253 _LIBCPP_INLINE_VISIBILITY __hash_const_iterator() _NOEXCEPT
Marshall Clow193ef032013-08-07 21:30:44 +0000254#if _LIBCPP_STD_VER > 11
255 : __node_(nullptr)
256#endif
Howard Hinnant39213642013-07-23 22:01:58 +0000257 {
258#if _LIBCPP_DEBUG_LEVEL >= 2
259 __get_db()->__insert_i(this);
260#endif
261 }
Howard Hinnant99acc502010-09-21 17:32:39 +0000262 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000263 __hash_const_iterator(const __non_const_iterator& __x) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000264 : __node_(__x.__node_)
Howard Hinnant39213642013-07-23 22:01:58 +0000265 {
266#if _LIBCPP_DEBUG_LEVEL >= 2
267 __get_db()->__iterator_copy(this, &__x);
268#endif
269 }
270
271#if _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000272
Howard Hinnant99acc502010-09-21 17:32:39 +0000273 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000274 __hash_const_iterator(const __hash_const_iterator& __i)
275 : __node_(__i.__node_)
276 {
277 __get_db()->__iterator_copy(this, &__i);
278 }
279
Howard Hinnant99acc502010-09-21 17:32:39 +0000280 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000281 ~__hash_const_iterator()
282 {
283 __get_db()->__erase_i(this);
284 }
285
286 _LIBCPP_INLINE_VISIBILITY
287 __hash_const_iterator& operator=(const __hash_const_iterator& __i)
288 {
289 if (this != &__i)
290 {
291 __get_db()->__iterator_copy(this, &__i);
292 __node_ = __i.__node_;
293 }
294 return *this;
295 }
296
297#endif // _LIBCPP_DEBUG_LEVEL >= 2
298
299 _LIBCPP_INLINE_VISIBILITY
300 reference operator*() const
301 {
302#if _LIBCPP_DEBUG_LEVEL >= 2
303 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
304 "Attempted to dereference a non-dereferenceable unordered container const_iterator");
305#endif
306 return __node_->__value_;
307 }
308 _LIBCPP_INLINE_VISIBILITY
309 pointer operator->() const
310 {
311#if _LIBCPP_DEBUG_LEVEL >= 2
312 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
313 "Attempted to dereference a non-dereferenceable unordered container const_iterator");
314#endif
315 return pointer_traits<pointer>::pointer_to(__node_->__value_);
316 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000317
Howard Hinnant99acc502010-09-21 17:32:39 +0000318 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000319 __hash_const_iterator& operator++()
320 {
Howard Hinnant39213642013-07-23 22:01:58 +0000321#if _LIBCPP_DEBUG_LEVEL >= 2
322 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
323 "Attempted to increment non-incrementable unordered container const_iterator");
324#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000325 __node_ = __node_->__next_;
326 return *this;
327 }
328
Howard Hinnant99acc502010-09-21 17:32:39 +0000329 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000330 __hash_const_iterator operator++(int)
331 {
332 __hash_const_iterator __t(*this);
333 ++(*this);
334 return __t;
335 }
336
Howard Hinnant99acc502010-09-21 17:32:39 +0000337 friend _LIBCPP_INLINE_VISIBILITY
338 bool operator==(const __hash_const_iterator& __x, const __hash_const_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000339 {
Howard Hinnant39213642013-07-23 22:01:58 +0000340 return __x.__node_ == __y.__node_;
341 }
Howard Hinnant99acc502010-09-21 17:32:39 +0000342 friend _LIBCPP_INLINE_VISIBILITY
343 bool operator!=(const __hash_const_iterator& __x, const __hash_const_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000344 {return !(__x == __y);}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000345
346private:
Howard Hinnant39213642013-07-23 22:01:58 +0000347#if _LIBCPP_DEBUG_LEVEL >= 2
348 _LIBCPP_INLINE_VISIBILITY
349 __hash_const_iterator(__node_pointer __node, const void* __c) _NOEXCEPT
350 : __node_(__node)
351 {
352 __get_db()->__insert_ic(this, __c);
353 }
354#else
Howard Hinnant99acc502010-09-21 17:32:39 +0000355 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000356 __hash_const_iterator(__node_pointer __node) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000357 : __node_(__node)
358 {}
Howard Hinnant39213642013-07-23 22:01:58 +0000359#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000360
361 template <class, class, class, class> friend class __hash_table;
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000362 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_map_const_iterator;
363 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_map;
364 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_multimap;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000365};
366
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000367template <class _ConstNodePtr> class _LIBCPP_TYPE_VIS_ONLY __hash_const_local_iterator;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000368
369template <class _NodePtr>
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000370class _LIBCPP_TYPE_VIS_ONLY __hash_local_iterator
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000371{
372 typedef _NodePtr __node_pointer;
373
374 __node_pointer __node_;
375 size_t __bucket_;
376 size_t __bucket_count_;
377
378 typedef pointer_traits<__node_pointer> __pointer_traits;
379public:
380 typedef forward_iterator_tag iterator_category;
381 typedef typename __pointer_traits::element_type::value_type value_type;
382 typedef typename __pointer_traits::difference_type difference_type;
383 typedef value_type& reference;
384 typedef typename __pointer_traits::template
385#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
386 rebind<value_type>
387#else
388 rebind<value_type>::other
389#endif
390 pointer;
391
Howard Hinnant39213642013-07-23 22:01:58 +0000392 _LIBCPP_INLINE_VISIBILITY __hash_local_iterator() _NOEXCEPT
393 {
394#if _LIBCPP_DEBUG_LEVEL >= 2
395 __get_db()->__insert_i(this);
396#endif
397 }
398
399#if _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000400
Howard Hinnant99acc502010-09-21 17:32:39 +0000401 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000402 __hash_local_iterator(const __hash_local_iterator& __i)
403 : __node_(__i.__node_),
404 __bucket_(__i.__bucket_),
405 __bucket_count_(__i.__bucket_count_)
406 {
407 __get_db()->__iterator_copy(this, &__i);
408 }
409
Howard Hinnant99acc502010-09-21 17:32:39 +0000410 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000411 ~__hash_local_iterator()
412 {
413 __get_db()->__erase_i(this);
414 }
415
416 _LIBCPP_INLINE_VISIBILITY
417 __hash_local_iterator& operator=(const __hash_local_iterator& __i)
418 {
419 if (this != &__i)
420 {
421 __get_db()->__iterator_copy(this, &__i);
422 __node_ = __i.__node_;
423 __bucket_ = __i.__bucket_;
424 __bucket_count_ = __i.__bucket_count_;
425 }
426 return *this;
427 }
428
429#endif // _LIBCPP_DEBUG_LEVEL >= 2
430
431 _LIBCPP_INLINE_VISIBILITY
432 reference operator*() const
433 {
434#if _LIBCPP_DEBUG_LEVEL >= 2
435 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
436 "Attempted to dereference a non-dereferenceable unordered container local_iterator");
437#endif
438 return __node_->__value_;
439 }
440 _LIBCPP_INLINE_VISIBILITY
441 pointer operator->() const
442 {
443#if _LIBCPP_DEBUG_LEVEL >= 2
444 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
445 "Attempted to dereference a non-dereferenceable unordered container local_iterator");
446#endif
447 return pointer_traits<pointer>::pointer_to(__node_->__value_);
448 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000449
Howard Hinnant99acc502010-09-21 17:32:39 +0000450 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000451 __hash_local_iterator& operator++()
452 {
Howard Hinnant39213642013-07-23 22:01:58 +0000453#if _LIBCPP_DEBUG_LEVEL >= 2
454 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
455 "Attempted to increment non-incrementable unordered container local_iterator");
456#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000457 __node_ = __node_->__next_;
Howard Hinnant7a445152012-07-06 17:31:14 +0000458 if (__node_ != nullptr && __constrain_hash(__node_->__hash_, __bucket_count_) != __bucket_)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000459 __node_ = nullptr;
460 return *this;
461 }
462
Howard Hinnant99acc502010-09-21 17:32:39 +0000463 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000464 __hash_local_iterator operator++(int)
465 {
466 __hash_local_iterator __t(*this);
467 ++(*this);
468 return __t;
469 }
470
Howard Hinnant99acc502010-09-21 17:32:39 +0000471 friend _LIBCPP_INLINE_VISIBILITY
472 bool operator==(const __hash_local_iterator& __x, const __hash_local_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000473 {
Howard Hinnant39213642013-07-23 22:01:58 +0000474 return __x.__node_ == __y.__node_;
475 }
Howard Hinnant99acc502010-09-21 17:32:39 +0000476 friend _LIBCPP_INLINE_VISIBILITY
477 bool operator!=(const __hash_local_iterator& __x, const __hash_local_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000478 {return !(__x == __y);}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000479
480private:
Howard Hinnant39213642013-07-23 22:01:58 +0000481#if _LIBCPP_DEBUG_LEVEL >= 2
482 _LIBCPP_INLINE_VISIBILITY
483 __hash_local_iterator(__node_pointer __node, size_t __bucket,
484 size_t __bucket_count, const void* __c) _NOEXCEPT
485 : __node_(__node),
486 __bucket_(__bucket),
487 __bucket_count_(__bucket_count)
488 {
489 __get_db()->__insert_ic(this, __c);
490 if (__node_ != nullptr)
491 __node_ = __node_->__next_;
492 }
493#else
Howard Hinnant99acc502010-09-21 17:32:39 +0000494 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000495 __hash_local_iterator(__node_pointer __node, size_t __bucket,
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000496 size_t __bucket_count) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000497 : __node_(__node),
498 __bucket_(__bucket),
499 __bucket_count_(__bucket_count)
500 {
501 if (__node_ != nullptr)
502 __node_ = __node_->__next_;
503 }
Howard Hinnant39213642013-07-23 22:01:58 +0000504#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000505 template <class, class, class, class> friend class __hash_table;
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000506 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_const_local_iterator;
507 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_map_iterator;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000508};
509
510template <class _ConstNodePtr>
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000511class _LIBCPP_TYPE_VIS_ONLY __hash_const_local_iterator
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000512{
513 typedef _ConstNodePtr __node_pointer;
514
515 __node_pointer __node_;
516 size_t __bucket_;
517 size_t __bucket_count_;
518
519 typedef pointer_traits<__node_pointer> __pointer_traits;
520 typedef typename __pointer_traits::element_type __node;
521 typedef typename remove_const<__node>::type __non_const_node;
522 typedef typename pointer_traits<__node_pointer>::template
523#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
524 rebind<__non_const_node>
525#else
526 rebind<__non_const_node>::other
527#endif
528 __non_const_node_pointer;
529 typedef __hash_local_iterator<__non_const_node_pointer>
530 __non_const_iterator;
531public:
532 typedef forward_iterator_tag iterator_category;
533 typedef typename remove_const<
534 typename __pointer_traits::element_type::value_type
535 >::type value_type;
536 typedef typename __pointer_traits::difference_type difference_type;
537 typedef const value_type& reference;
538 typedef typename __pointer_traits::template
539#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
540 rebind<const value_type>
541#else
542 rebind<const value_type>::other
543#endif
544 pointer;
545
Howard Hinnant39213642013-07-23 22:01:58 +0000546 _LIBCPP_INLINE_VISIBILITY __hash_const_local_iterator() _NOEXCEPT
547 {
548#if _LIBCPP_DEBUG_LEVEL >= 2
549 __get_db()->__insert_i(this);
550#endif
551 }
552
Howard Hinnant99acc502010-09-21 17:32:39 +0000553 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000554 __hash_const_local_iterator(const __non_const_iterator& __x) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000555 : __node_(__x.__node_),
556 __bucket_(__x.__bucket_),
557 __bucket_count_(__x.__bucket_count_)
Howard Hinnant39213642013-07-23 22:01:58 +0000558 {
559#if _LIBCPP_DEBUG_LEVEL >= 2
560 __get_db()->__iterator_copy(this, &__x);
561#endif
562 }
563
564#if _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000565
Howard Hinnant99acc502010-09-21 17:32:39 +0000566 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000567 __hash_const_local_iterator(const __hash_const_local_iterator& __i)
568 : __node_(__i.__node_),
569 __bucket_(__i.__bucket_),
570 __bucket_count_(__i.__bucket_count_)
571 {
572 __get_db()->__iterator_copy(this, &__i);
573 }
574
Howard Hinnant99acc502010-09-21 17:32:39 +0000575 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant39213642013-07-23 22:01:58 +0000576 ~__hash_const_local_iterator()
577 {
578 __get_db()->__erase_i(this);
579 }
580
581 _LIBCPP_INLINE_VISIBILITY
582 __hash_const_local_iterator& operator=(const __hash_const_local_iterator& __i)
583 {
584 if (this != &__i)
585 {
586 __get_db()->__iterator_copy(this, &__i);
587 __node_ = __i.__node_;
588 __bucket_ = __i.__bucket_;
589 __bucket_count_ = __i.__bucket_count_;
590 }
591 return *this;
592 }
593
594#endif // _LIBCPP_DEBUG_LEVEL >= 2
595
596 _LIBCPP_INLINE_VISIBILITY
597 reference operator*() const
598 {
599#if _LIBCPP_DEBUG_LEVEL >= 2
600 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
601 "Attempted to dereference a non-dereferenceable unordered container const_local_iterator");
602#endif
603 return __node_->__value_;
604 }
605 _LIBCPP_INLINE_VISIBILITY
606 pointer operator->() const
607 {
608#if _LIBCPP_DEBUG_LEVEL >= 2
609 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
610 "Attempted to dereference a non-dereferenceable unordered container const_local_iterator");
611#endif
612 return pointer_traits<pointer>::pointer_to(__node_->__value_);
613 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000614
Howard Hinnant99acc502010-09-21 17:32:39 +0000615 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000616 __hash_const_local_iterator& operator++()
617 {
Howard Hinnant39213642013-07-23 22:01:58 +0000618#if _LIBCPP_DEBUG_LEVEL >= 2
619 _LIBCPP_ASSERT(__get_const_db()->__dereferenceable(this),
620 "Attempted to increment non-incrementable unordered container const_local_iterator");
621#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000622 __node_ = __node_->__next_;
Howard Hinnant7a445152012-07-06 17:31:14 +0000623 if (__node_ != nullptr && __constrain_hash(__node_->__hash_, __bucket_count_) != __bucket_)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000624 __node_ = nullptr;
625 return *this;
626 }
627
Howard Hinnant99acc502010-09-21 17:32:39 +0000628 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000629 __hash_const_local_iterator operator++(int)
630 {
631 __hash_const_local_iterator __t(*this);
632 ++(*this);
633 return __t;
634 }
635
Howard Hinnant99acc502010-09-21 17:32:39 +0000636 friend _LIBCPP_INLINE_VISIBILITY
637 bool operator==(const __hash_const_local_iterator& __x, const __hash_const_local_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000638 {
Howard Hinnant39213642013-07-23 22:01:58 +0000639 return __x.__node_ == __y.__node_;
640 }
Howard Hinnant99acc502010-09-21 17:32:39 +0000641 friend _LIBCPP_INLINE_VISIBILITY
642 bool operator!=(const __hash_const_local_iterator& __x, const __hash_const_local_iterator& __y)
Howard Hinnant39213642013-07-23 22:01:58 +0000643 {return !(__x == __y);}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000644
645private:
Howard Hinnant39213642013-07-23 22:01:58 +0000646#if _LIBCPP_DEBUG_LEVEL >= 2
647 _LIBCPP_INLINE_VISIBILITY
648 __hash_const_local_iterator(__node_pointer __node, size_t __bucket,
649 size_t __bucket_count, const void* __c) _NOEXCEPT
650 : __node_(__node),
651 __bucket_(__bucket),
652 __bucket_count_(__bucket_count)
653 {
654 __get_db()->__insert_ic(this, __c);
655 if (__node_ != nullptr)
656 __node_ = __node_->__next_;
657 }
658#else
Howard Hinnant99acc502010-09-21 17:32:39 +0000659 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000660 __hash_const_local_iterator(__node_pointer __node, size_t __bucket,
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000661 size_t __bucket_count) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000662 : __node_(__node),
663 __bucket_(__bucket),
664 __bucket_count_(__bucket_count)
665 {
666 if (__node_ != nullptr)
667 __node_ = __node_->__next_;
668 }
Howard Hinnant39213642013-07-23 22:01:58 +0000669#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000670 template <class, class, class, class> friend class __hash_table;
Howard Hinnant0f678bd2013-08-12 18:38:34 +0000671 template <class> friend class _LIBCPP_TYPE_VIS_ONLY __hash_map_const_iterator;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000672};
673
674template <class _Alloc>
675class __bucket_list_deallocator
676{
677 typedef _Alloc allocator_type;
678 typedef allocator_traits<allocator_type> __alloc_traits;
679 typedef typename __alloc_traits::size_type size_type;
680
681 __compressed_pair<size_type, allocator_type> __data_;
682public:
683 typedef typename __alloc_traits::pointer pointer;
684
Howard Hinnant99acc502010-09-21 17:32:39 +0000685 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000686 __bucket_list_deallocator()
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000687 _NOEXCEPT_(is_nothrow_default_constructible<allocator_type>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000688 : __data_(0) {}
Howard Hinnant99acc502010-09-21 17:32:39 +0000689
690 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000691 __bucket_list_deallocator(const allocator_type& __a, size_type __size)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000692 _NOEXCEPT_(is_nothrow_copy_constructible<allocator_type>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000693 : __data_(__size, __a) {}
694
Howard Hinnant73d21a42010-09-04 23:28:19 +0000695#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000696
Howard Hinnant99acc502010-09-21 17:32:39 +0000697 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000698 __bucket_list_deallocator(__bucket_list_deallocator&& __x)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000699 _NOEXCEPT_(is_nothrow_move_constructible<allocator_type>::value)
Howard Hinnant0949eed2011-06-30 21:18:19 +0000700 : __data_(_VSTD::move(__x.__data_))
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000701 {
702 __x.size() = 0;
703 }
704
Howard Hinnant73d21a42010-09-04 23:28:19 +0000705#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000706
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000707 _LIBCPP_INLINE_VISIBILITY
708 size_type& size() _NOEXCEPT {return __data_.first();}
709 _LIBCPP_INLINE_VISIBILITY
710 size_type size() const _NOEXCEPT {return __data_.first();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000711
Howard Hinnant99acc502010-09-21 17:32:39 +0000712 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000713 allocator_type& __alloc() _NOEXCEPT {return __data_.second();}
714 _LIBCPP_INLINE_VISIBILITY
715 const allocator_type& __alloc() const _NOEXCEPT {return __data_.second();}
716
717 _LIBCPP_INLINE_VISIBILITY
718 void operator()(pointer __p) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000719 {
720 __alloc_traits::deallocate(__alloc(), __p, size());
721 }
722};
723
Howard Hinnant2b1b2d42011-06-14 19:58:17 +0000724template <class _Alloc> class __hash_map_node_destructor;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000725
726template <class _Alloc>
727class __hash_node_destructor
728{
729 typedef _Alloc allocator_type;
730 typedef allocator_traits<allocator_type> __alloc_traits;
731 typedef typename __alloc_traits::value_type::value_type value_type;
732public:
733 typedef typename __alloc_traits::pointer pointer;
734private:
735
736 allocator_type& __na_;
737
738 __hash_node_destructor& operator=(const __hash_node_destructor&);
739
740public:
741 bool __value_constructed;
742
Howard Hinnant99acc502010-09-21 17:32:39 +0000743 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant199d0ae2011-07-31 17:04:30 +0000744 explicit __hash_node_destructor(allocator_type& __na,
745 bool __constructed = false) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000746 : __na_(__na),
Howard Hinnant199d0ae2011-07-31 17:04:30 +0000747 __value_constructed(__constructed)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000748 {}
749
Howard Hinnant99acc502010-09-21 17:32:39 +0000750 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000751 void operator()(pointer __p) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000752 {
753 if (__value_constructed)
Howard Hinnant0949eed2011-06-30 21:18:19 +0000754 __alloc_traits::destroy(__na_, _VSTD::addressof(__p->__value_));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000755 if (__p)
756 __alloc_traits::deallocate(__na_, __p, 1);
757 }
758
759 template <class> friend class __hash_map_node_destructor;
760};
761
762template <class _Tp, class _Hash, class _Equal, class _Alloc>
763class __hash_table
764{
765public:
766 typedef _Tp value_type;
767 typedef _Hash hasher;
768 typedef _Equal key_equal;
769 typedef _Alloc allocator_type;
770
771private:
772 typedef allocator_traits<allocator_type> __alloc_traits;
773public:
774 typedef value_type& reference;
775 typedef const value_type& const_reference;
776 typedef typename __alloc_traits::pointer pointer;
777 typedef typename __alloc_traits::const_pointer const_pointer;
778 typedef typename __alloc_traits::size_type size_type;
779 typedef typename __alloc_traits::difference_type difference_type;
780public:
781 // Create __node
782 typedef __hash_node<value_type, typename __alloc_traits::void_pointer> __node;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000783 typedef typename __alloc_traits::template
784#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
785 rebind_alloc<__node>
786#else
787 rebind_alloc<__node>::other
788#endif
789 __node_allocator;
790 typedef allocator_traits<__node_allocator> __node_traits;
791 typedef typename __node_traits::pointer __node_pointer;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +0000792 typedef typename __node_traits::pointer __node_const_pointer;
Howard Hinnantf8880d02011-12-12 17:26:24 +0000793 typedef __hash_node_base<__node_pointer> __first_node;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +0000794 typedef typename pointer_traits<__node_pointer>::template
795#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
796 rebind<__first_node>
797#else
798 rebind<__first_node>::other
799#endif
800 __node_base_pointer;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000801
802private:
803
804 typedef typename __node_traits::template
805#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
806 rebind_alloc<__node_pointer>
807#else
808 rebind_alloc<__node_pointer>::other
809#endif
810 __pointer_allocator;
811 typedef __bucket_list_deallocator<__pointer_allocator> __bucket_list_deleter;
812 typedef unique_ptr<__node_pointer[], __bucket_list_deleter> __bucket_list;
813 typedef allocator_traits<__pointer_allocator> __pointer_alloc_traits;
814 typedef typename __bucket_list_deleter::pointer __node_pointer_pointer;
815
816 // --- Member data begin ---
817 __bucket_list __bucket_list_;
818 __compressed_pair<__first_node, __node_allocator> __p1_;
819 __compressed_pair<size_type, hasher> __p2_;
820 __compressed_pair<float, key_equal> __p3_;
821 // --- Member data end ---
822
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000823 _LIBCPP_INLINE_VISIBILITY
824 size_type& size() _NOEXCEPT {return __p2_.first();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000825public:
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000826 _LIBCPP_INLINE_VISIBILITY
827 size_type size() const _NOEXCEPT {return __p2_.first();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000828
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000829 _LIBCPP_INLINE_VISIBILITY
830 hasher& hash_function() _NOEXCEPT {return __p2_.second();}
831 _LIBCPP_INLINE_VISIBILITY
832 const hasher& hash_function() const _NOEXCEPT {return __p2_.second();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000833
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000834 _LIBCPP_INLINE_VISIBILITY
835 float& max_load_factor() _NOEXCEPT {return __p3_.first();}
836 _LIBCPP_INLINE_VISIBILITY
837 float max_load_factor() const _NOEXCEPT {return __p3_.first();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000838
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000839 _LIBCPP_INLINE_VISIBILITY
840 key_equal& key_eq() _NOEXCEPT {return __p3_.second();}
841 _LIBCPP_INLINE_VISIBILITY
842 const key_equal& key_eq() const _NOEXCEPT {return __p3_.second();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000843
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000844 _LIBCPP_INLINE_VISIBILITY
845 __node_allocator& __node_alloc() _NOEXCEPT {return __p1_.second();}
846 _LIBCPP_INLINE_VISIBILITY
847 const __node_allocator& __node_alloc() const _NOEXCEPT
848 {return __p1_.second();}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000849
850public:
851 typedef __hash_iterator<__node_pointer> iterator;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +0000852 typedef __hash_const_iterator<__node_pointer> const_iterator;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000853 typedef __hash_local_iterator<__node_pointer> local_iterator;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +0000854 typedef __hash_const_local_iterator<__node_pointer> const_local_iterator;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000855
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000856 __hash_table()
857 _NOEXCEPT_(
858 is_nothrow_default_constructible<__bucket_list>::value &&
859 is_nothrow_default_constructible<__first_node>::value &&
860 is_nothrow_default_constructible<__node_allocator>::value &&
861 is_nothrow_default_constructible<hasher>::value &&
862 is_nothrow_default_constructible<key_equal>::value);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000863 __hash_table(const hasher& __hf, const key_equal& __eql);
864 __hash_table(const hasher& __hf, const key_equal& __eql,
865 const allocator_type& __a);
866 explicit __hash_table(const allocator_type& __a);
867 __hash_table(const __hash_table& __u);
868 __hash_table(const __hash_table& __u, const allocator_type& __a);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000869#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000870 __hash_table(__hash_table&& __u)
871 _NOEXCEPT_(
872 is_nothrow_move_constructible<__bucket_list>::value &&
873 is_nothrow_move_constructible<__first_node>::value &&
874 is_nothrow_move_constructible<__node_allocator>::value &&
875 is_nothrow_move_constructible<hasher>::value &&
876 is_nothrow_move_constructible<key_equal>::value);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000877 __hash_table(__hash_table&& __u, const allocator_type& __a);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000878#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000879 ~__hash_table();
880
881 __hash_table& operator=(const __hash_table& __u);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000882#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000883 __hash_table& operator=(__hash_table&& __u)
884 _NOEXCEPT_(
885 __node_traits::propagate_on_container_move_assignment::value &&
886 is_nothrow_move_assignable<__node_allocator>::value &&
887 is_nothrow_move_assignable<hasher>::value &&
888 is_nothrow_move_assignable<key_equal>::value);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000889#endif
890 template <class _InputIterator>
891 void __assign_unique(_InputIterator __first, _InputIterator __last);
892 template <class _InputIterator>
893 void __assign_multi(_InputIterator __first, _InputIterator __last);
894
Howard Hinnant99acc502010-09-21 17:32:39 +0000895 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000896 size_type max_size() const _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000897 {
898 return allocator_traits<__pointer_allocator>::max_size(
899 __bucket_list_.get_deleter().__alloc());
900 }
901
902 pair<iterator, bool> __node_insert_unique(__node_pointer __nd);
903 iterator __node_insert_multi(__node_pointer __nd);
904 iterator __node_insert_multi(const_iterator __p,
905 __node_pointer __nd);
906
Howard Hinnant73d21a42010-09-04 23:28:19 +0000907#if !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000908 template <class... _Args>
909 pair<iterator, bool> __emplace_unique(_Args&&... __args);
910 template <class... _Args>
911 iterator __emplace_multi(_Args&&... __args);
912 template <class... _Args>
913 iterator __emplace_hint_multi(const_iterator __p, _Args&&... __args);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000914#endif // !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000915
916 pair<iterator, bool> __insert_unique(const value_type& __x);
917
Howard Hinnant73d21a42010-09-04 23:28:19 +0000918#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnant99968442011-11-29 18:15:50 +0000919 template <class _Pp>
920 pair<iterator, bool> __insert_unique(_Pp&& __x);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000921#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000922
Howard Hinnant73d21a42010-09-04 23:28:19 +0000923#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnant99968442011-11-29 18:15:50 +0000924 template <class _Pp>
925 iterator __insert_multi(_Pp&& __x);
926 template <class _Pp>
927 iterator __insert_multi(const_iterator __p, _Pp&& __x);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000928#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000929 iterator __insert_multi(const value_type& __x);
930 iterator __insert_multi(const_iterator __p, const value_type& __x);
Howard Hinnant73d21a42010-09-04 23:28:19 +0000931#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000932
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000933 void clear() _NOEXCEPT;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000934 void rehash(size_type __n);
Howard Hinnant99acc502010-09-21 17:32:39 +0000935 _LIBCPP_INLINE_VISIBILITY void reserve(size_type __n)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000936 {rehash(static_cast<size_type>(ceil(__n / max_load_factor())));}
Howard Hinnant99acc502010-09-21 17:32:39 +0000937
938 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000939 size_type bucket_count() const _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000940 {
941 return __bucket_list_.get_deleter().size();
942 }
943
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000944 iterator begin() _NOEXCEPT;
945 iterator end() _NOEXCEPT;
946 const_iterator begin() const _NOEXCEPT;
947 const_iterator end() const _NOEXCEPT;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000948
949 template <class _Key>
Howard Hinnant99acc502010-09-21 17:32:39 +0000950 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000951 size_type bucket(const _Key& __k) const
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +0000952 {
953 _LIBCPP_ASSERT(bucket_count() > 0,
954 "unordered container::bucket(key) called when bucket_count() == 0");
955 return __constrain_hash(hash_function()(__k), bucket_count());
956 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000957
958 template <class _Key>
959 iterator find(const _Key& __x);
960 template <class _Key>
961 const_iterator find(const _Key& __x) const;
962
Howard Hinnant99968442011-11-29 18:15:50 +0000963 typedef __hash_node_destructor<__node_allocator> _Dp;
964 typedef unique_ptr<__node, _Dp> __node_holder;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000965
966 iterator erase(const_iterator __p);
967 iterator erase(const_iterator __first, const_iterator __last);
968 template <class _Key>
969 size_type __erase_unique(const _Key& __k);
970 template <class _Key>
971 size_type __erase_multi(const _Key& __k);
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000972 __node_holder remove(const_iterator __p) _NOEXCEPT;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000973
974 template <class _Key>
975 size_type __count_unique(const _Key& __k) const;
976 template <class _Key>
977 size_type __count_multi(const _Key& __k) const;
978
979 template <class _Key>
980 pair<iterator, iterator>
981 __equal_range_unique(const _Key& __k);
982 template <class _Key>
983 pair<const_iterator, const_iterator>
984 __equal_range_unique(const _Key& __k) const;
985
986 template <class _Key>
987 pair<iterator, iterator>
988 __equal_range_multi(const _Key& __k);
989 template <class _Key>
990 pair<const_iterator, const_iterator>
991 __equal_range_multi(const _Key& __k) const;
992
Howard Hinnant5f2f14c2011-06-04 18:54:24 +0000993 void swap(__hash_table& __u)
994 _NOEXCEPT_(
995 (!allocator_traits<__pointer_allocator>::propagate_on_container_swap::value ||
996 __is_nothrow_swappable<__pointer_allocator>::value) &&
997 (!__node_traits::propagate_on_container_swap::value ||
998 __is_nothrow_swappable<__node_allocator>::value) &&
999 __is_nothrow_swappable<hasher>::value &&
1000 __is_nothrow_swappable<key_equal>::value);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001001
Howard Hinnant99acc502010-09-21 17:32:39 +00001002 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001003 size_type max_bucket_count() const _NOEXCEPT
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001004 {return __pointer_alloc_traits::max_size(__bucket_list_.get_deleter().__alloc());}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001005 size_type bucket_size(size_type __n) const;
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001006 _LIBCPP_INLINE_VISIBILITY float load_factor() const _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001007 {
1008 size_type __bc = bucket_count();
1009 return __bc != 0 ? (float)size() / __bc : 0.f;
1010 }
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001011 _LIBCPP_INLINE_VISIBILITY void max_load_factor(float __mlf) _NOEXCEPT
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001012 {
1013 _LIBCPP_ASSERT(__mlf > 0,
1014 "unordered container::max_load_factor(lf) called with lf <= 0");
1015 max_load_factor() = _VSTD::max(__mlf, load_factor());
1016 }
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001017
Howard Hinnant39213642013-07-23 22:01:58 +00001018 _LIBCPP_INLINE_VISIBILITY
1019 local_iterator
1020 begin(size_type __n)
1021 {
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001022 _LIBCPP_ASSERT(__n < bucket_count(),
1023 "unordered container::begin(n) called with n >= bucket_count()");
Howard Hinnant39213642013-07-23 22:01:58 +00001024#if _LIBCPP_DEBUG_LEVEL >= 2
1025 return local_iterator(__bucket_list_[__n], __n, bucket_count(), this);
1026#else
1027 return local_iterator(__bucket_list_[__n], __n, bucket_count());
1028#endif
1029 }
1030
1031 _LIBCPP_INLINE_VISIBILITY
1032 local_iterator
1033 end(size_type __n)
1034 {
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001035 _LIBCPP_ASSERT(__n < bucket_count(),
1036 "unordered container::end(n) called with n >= bucket_count()");
Howard Hinnant39213642013-07-23 22:01:58 +00001037#if _LIBCPP_DEBUG_LEVEL >= 2
1038 return local_iterator(nullptr, __n, bucket_count(), this);
1039#else
1040 return local_iterator(nullptr, __n, bucket_count());
1041#endif
1042 }
1043
1044 _LIBCPP_INLINE_VISIBILITY
1045 const_local_iterator
1046 cbegin(size_type __n) const
1047 {
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001048 _LIBCPP_ASSERT(__n < bucket_count(),
1049 "unordered container::cbegin(n) called with n >= bucket_count()");
Howard Hinnant39213642013-07-23 22:01:58 +00001050#if _LIBCPP_DEBUG_LEVEL >= 2
1051 return const_local_iterator(__bucket_list_[__n], __n, bucket_count(), this);
1052#else
1053 return const_local_iterator(__bucket_list_[__n], __n, bucket_count());
1054#endif
1055 }
1056
1057 _LIBCPP_INLINE_VISIBILITY
1058 const_local_iterator
1059 cend(size_type __n) const
1060 {
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001061 _LIBCPP_ASSERT(__n < bucket_count(),
1062 "unordered container::cend(n) called with n >= bucket_count()");
Howard Hinnant39213642013-07-23 22:01:58 +00001063#if _LIBCPP_DEBUG_LEVEL >= 2
1064 return const_local_iterator(nullptr, __n, bucket_count(), this);
1065#else
1066 return const_local_iterator(nullptr, __n, bucket_count());
1067#endif
1068 }
1069
1070#if _LIBCPP_DEBUG_LEVEL >= 2
1071
1072 bool __dereferenceable(const const_iterator* __i) const;
1073 bool __decrementable(const const_iterator* __i) const;
1074 bool __addable(const const_iterator* __i, ptrdiff_t __n) const;
1075 bool __subscriptable(const const_iterator* __i, ptrdiff_t __n) const;
1076
1077#endif // _LIBCPP_DEBUG_LEVEL >= 2
1078
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001079private:
1080 void __rehash(size_type __n);
1081
Howard Hinnant73d21a42010-09-04 23:28:19 +00001082#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
1083#ifndef _LIBCPP_HAS_NO_VARIADICS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001084 template <class ..._Args>
1085 __node_holder __construct_node(_Args&& ...__args);
Howard Hinnantbfd55302010-09-04 23:46:48 +00001086#endif // _LIBCPP_HAS_NO_VARIADICS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001087 __node_holder __construct_node(value_type&& __v, size_t __hash);
Howard Hinnant73d21a42010-09-04 23:28:19 +00001088#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001089 __node_holder __construct_node(const value_type& __v);
1090#endif
1091 __node_holder __construct_node(const value_type& __v, size_t __hash);
1092
Howard Hinnant99acc502010-09-21 17:32:39 +00001093 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001094 void __copy_assign_alloc(const __hash_table& __u)
1095 {__copy_assign_alloc(__u, integral_constant<bool,
1096 __node_traits::propagate_on_container_copy_assignment::value>());}
1097 void __copy_assign_alloc(const __hash_table& __u, true_type);
Howard Hinnant99acc502010-09-21 17:32:39 +00001098 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantec3773c2011-12-01 20:21:04 +00001099 void __copy_assign_alloc(const __hash_table&, false_type) {}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001100
1101 void __move_assign(__hash_table& __u, false_type);
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001102 void __move_assign(__hash_table& __u, true_type)
1103 _NOEXCEPT_(
1104 is_nothrow_move_assignable<__node_allocator>::value &&
1105 is_nothrow_move_assignable<hasher>::value &&
1106 is_nothrow_move_assignable<key_equal>::value);
1107 _LIBCPP_INLINE_VISIBILITY
1108 void __move_assign_alloc(__hash_table& __u)
1109 _NOEXCEPT_(
1110 !__node_traits::propagate_on_container_move_assignment::value ||
1111 (is_nothrow_move_assignable<__pointer_allocator>::value &&
1112 is_nothrow_move_assignable<__node_allocator>::value))
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001113 {__move_assign_alloc(__u, integral_constant<bool,
1114 __node_traits::propagate_on_container_move_assignment::value>());}
Howard Hinnant99acc502010-09-21 17:32:39 +00001115 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001116 void __move_assign_alloc(__hash_table& __u, true_type)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001117 _NOEXCEPT_(
1118 is_nothrow_move_assignable<__pointer_allocator>::value &&
1119 is_nothrow_move_assignable<__node_allocator>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001120 {
1121 __bucket_list_.get_deleter().__alloc() =
Howard Hinnant0949eed2011-06-30 21:18:19 +00001122 _VSTD::move(__u.__bucket_list_.get_deleter().__alloc());
1123 __node_alloc() = _VSTD::move(__u.__node_alloc());
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001124 }
Howard Hinnant99acc502010-09-21 17:32:39 +00001125 _LIBCPP_INLINE_VISIBILITY
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001126 void __move_assign_alloc(__hash_table&, false_type) _NOEXCEPT {}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001127
Howard Hinnant99968442011-11-29 18:15:50 +00001128 template <class _Ap>
Howard Hinnant99acc502010-09-21 17:32:39 +00001129 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001130 static
1131 void
Howard Hinnant99968442011-11-29 18:15:50 +00001132 __swap_alloc(_Ap& __x, _Ap& __y)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001133 _NOEXCEPT_(
Howard Hinnant99968442011-11-29 18:15:50 +00001134 !allocator_traits<_Ap>::propagate_on_container_swap::value ||
1135 __is_nothrow_swappable<_Ap>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001136 {
1137 __swap_alloc(__x, __y,
1138 integral_constant<bool,
Howard Hinnant99968442011-11-29 18:15:50 +00001139 allocator_traits<_Ap>::propagate_on_container_swap::value
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001140 >());
1141 }
1142
Howard Hinnant99968442011-11-29 18:15:50 +00001143 template <class _Ap>
Howard Hinnant99acc502010-09-21 17:32:39 +00001144 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001145 static
1146 void
Howard Hinnant99968442011-11-29 18:15:50 +00001147 __swap_alloc(_Ap& __x, _Ap& __y, true_type)
1148 _NOEXCEPT_(__is_nothrow_swappable<_Ap>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001149 {
Howard Hinnant0949eed2011-06-30 21:18:19 +00001150 using _VSTD::swap;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001151 swap(__x, __y);
1152 }
1153
Howard Hinnant99968442011-11-29 18:15:50 +00001154 template <class _Ap>
Howard Hinnant99acc502010-09-21 17:32:39 +00001155 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001156 static
1157 void
Howard Hinnantec3773c2011-12-01 20:21:04 +00001158 __swap_alloc(_Ap&, _Ap&, false_type) _NOEXCEPT {}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001159
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001160 void __deallocate(__node_pointer __np) _NOEXCEPT;
1161 __node_pointer __detach() _NOEXCEPT;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001162
Howard Hinnant0f678bd2013-08-12 18:38:34 +00001163 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_map;
1164 template <class, class, class, class, class> friend class _LIBCPP_TYPE_VIS_ONLY unordered_multimap;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001165};
1166
1167template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001168inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001169__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table()
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001170 _NOEXCEPT_(
1171 is_nothrow_default_constructible<__bucket_list>::value &&
1172 is_nothrow_default_constructible<__first_node>::value &&
1173 is_nothrow_default_constructible<hasher>::value &&
1174 is_nothrow_default_constructible<key_equal>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001175 : __p2_(0),
1176 __p3_(1.0f)
1177{
1178}
1179
1180template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001181inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001182__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(const hasher& __hf,
1183 const key_equal& __eql)
1184 : __bucket_list_(nullptr, __bucket_list_deleter()),
1185 __p1_(),
1186 __p2_(0, __hf),
1187 __p3_(1.0f, __eql)
1188{
1189}
1190
1191template <class _Tp, class _Hash, class _Equal, class _Alloc>
1192__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(const hasher& __hf,
1193 const key_equal& __eql,
1194 const allocator_type& __a)
1195 : __bucket_list_(nullptr, __bucket_list_deleter(__pointer_allocator(__a), 0)),
1196 __p1_(__node_allocator(__a)),
1197 __p2_(0, __hf),
1198 __p3_(1.0f, __eql)
1199{
1200}
1201
1202template <class _Tp, class _Hash, class _Equal, class _Alloc>
1203__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(const allocator_type& __a)
1204 : __bucket_list_(nullptr, __bucket_list_deleter(__pointer_allocator(__a), 0)),
1205 __p1_(__node_allocator(__a)),
1206 __p2_(0),
1207 __p3_(1.0f)
1208{
1209}
1210
1211template <class _Tp, class _Hash, class _Equal, class _Alloc>
1212__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(const __hash_table& __u)
1213 : __bucket_list_(nullptr,
1214 __bucket_list_deleter(allocator_traits<__pointer_allocator>::
1215 select_on_container_copy_construction(
1216 __u.__bucket_list_.get_deleter().__alloc()), 0)),
1217 __p1_(allocator_traits<__node_allocator>::
1218 select_on_container_copy_construction(__u.__node_alloc())),
1219 __p2_(0, __u.hash_function()),
1220 __p3_(__u.__p3_)
1221{
1222}
1223
1224template <class _Tp, class _Hash, class _Equal, class _Alloc>
1225__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(const __hash_table& __u,
1226 const allocator_type& __a)
1227 : __bucket_list_(nullptr, __bucket_list_deleter(__pointer_allocator(__a), 0)),
1228 __p1_(__node_allocator(__a)),
1229 __p2_(0, __u.hash_function()),
1230 __p3_(__u.__p3_)
1231{
1232}
1233
Howard Hinnant73d21a42010-09-04 23:28:19 +00001234#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001235
1236template <class _Tp, class _Hash, class _Equal, class _Alloc>
1237__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(__hash_table&& __u)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001238 _NOEXCEPT_(
1239 is_nothrow_move_constructible<__bucket_list>::value &&
1240 is_nothrow_move_constructible<__first_node>::value &&
1241 is_nothrow_move_constructible<hasher>::value &&
1242 is_nothrow_move_constructible<key_equal>::value)
Howard Hinnant0949eed2011-06-30 21:18:19 +00001243 : __bucket_list_(_VSTD::move(__u.__bucket_list_)),
1244 __p1_(_VSTD::move(__u.__p1_)),
1245 __p2_(_VSTD::move(__u.__p2_)),
1246 __p3_(_VSTD::move(__u.__p3_))
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001247{
1248 if (size() > 0)
1249 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001250 __bucket_list_[__constrain_hash(__p1_.first().__next_->__hash_, bucket_count())] =
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001251 static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001252 __u.__p1_.first().__next_ = nullptr;
1253 __u.size() = 0;
1254 }
1255}
1256
1257template <class _Tp, class _Hash, class _Equal, class _Alloc>
1258__hash_table<_Tp, _Hash, _Equal, _Alloc>::__hash_table(__hash_table&& __u,
1259 const allocator_type& __a)
1260 : __bucket_list_(nullptr, __bucket_list_deleter(__pointer_allocator(__a), 0)),
1261 __p1_(__node_allocator(__a)),
Howard Hinnant0949eed2011-06-30 21:18:19 +00001262 __p2_(0, _VSTD::move(__u.hash_function())),
1263 __p3_(_VSTD::move(__u.__p3_))
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001264{
1265 if (__a == allocator_type(__u.__node_alloc()))
1266 {
1267 __bucket_list_.reset(__u.__bucket_list_.release());
1268 __bucket_list_.get_deleter().size() = __u.__bucket_list_.get_deleter().size();
1269 __u.__bucket_list_.get_deleter().size() = 0;
1270 if (__u.size() > 0)
1271 {
1272 __p1_.first().__next_ = __u.__p1_.first().__next_;
1273 __u.__p1_.first().__next_ = nullptr;
Howard Hinnant7a445152012-07-06 17:31:14 +00001274 __bucket_list_[__constrain_hash(__p1_.first().__next_->__hash_, bucket_count())] =
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001275 static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001276 size() = __u.size();
1277 __u.size() = 0;
1278 }
1279 }
1280}
1281
Howard Hinnant73d21a42010-09-04 23:28:19 +00001282#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001283
1284template <class _Tp, class _Hash, class _Equal, class _Alloc>
1285__hash_table<_Tp, _Hash, _Equal, _Alloc>::~__hash_table()
1286{
1287 __deallocate(__p1_.first().__next_);
Howard Hinnant39213642013-07-23 22:01:58 +00001288#if _LIBCPP_DEBUG_LEVEL >= 2
1289 __get_db()->__erase_c(this);
1290#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001291}
1292
1293template <class _Tp, class _Hash, class _Equal, class _Alloc>
1294void
1295__hash_table<_Tp, _Hash, _Equal, _Alloc>::__copy_assign_alloc(
1296 const __hash_table& __u, true_type)
1297{
1298 if (__node_alloc() != __u.__node_alloc())
1299 {
1300 clear();
1301 __bucket_list_.reset();
1302 __bucket_list_.get_deleter().size() = 0;
1303 }
1304 __bucket_list_.get_deleter().__alloc() = __u.__bucket_list_.get_deleter().__alloc();
1305 __node_alloc() = __u.__node_alloc();
1306}
1307
1308template <class _Tp, class _Hash, class _Equal, class _Alloc>
1309__hash_table<_Tp, _Hash, _Equal, _Alloc>&
1310__hash_table<_Tp, _Hash, _Equal, _Alloc>::operator=(const __hash_table& __u)
1311{
1312 if (this != &__u)
1313 {
1314 __copy_assign_alloc(__u);
1315 hash_function() = __u.hash_function();
1316 key_eq() = __u.key_eq();
1317 max_load_factor() = __u.max_load_factor();
1318 __assign_multi(__u.begin(), __u.end());
1319 }
1320 return *this;
1321}
1322
1323template <class _Tp, class _Hash, class _Equal, class _Alloc>
1324void
1325__hash_table<_Tp, _Hash, _Equal, _Alloc>::__deallocate(__node_pointer __np)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001326 _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001327{
1328 __node_allocator& __na = __node_alloc();
1329 while (__np != nullptr)
1330 {
1331 __node_pointer __next = __np->__next_;
Howard Hinnant39213642013-07-23 22:01:58 +00001332#if _LIBCPP_DEBUG_LEVEL >= 2
1333 __c_node* __c = __get_db()->__find_c_and_lock(this);
1334 for (__i_node** __p = __c->end_; __p != __c->beg_; )
1335 {
1336 --__p;
1337 iterator* __i = static_cast<iterator*>((*__p)->__i_);
1338 if (__i->__node_ == __np)
1339 {
1340 (*__p)->__c_ = nullptr;
1341 if (--__c->end_ != __p)
1342 memmove(__p, __p+1, (__c->end_ - __p)*sizeof(__i_node*));
1343 }
1344 }
1345 __get_db()->unlock();
1346#endif
Howard Hinnant0949eed2011-06-30 21:18:19 +00001347 __node_traits::destroy(__na, _VSTD::addressof(__np->__value_));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001348 __node_traits::deallocate(__na, __np, 1);
1349 __np = __next;
1350 }
1351}
1352
1353template <class _Tp, class _Hash, class _Equal, class _Alloc>
1354typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_pointer
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001355__hash_table<_Tp, _Hash, _Equal, _Alloc>::__detach() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001356{
1357 size_type __bc = bucket_count();
1358 for (size_type __i = 0; __i < __bc; ++__i)
1359 __bucket_list_[__i] = nullptr;
1360 size() = 0;
1361 __node_pointer __cache = __p1_.first().__next_;
1362 __p1_.first().__next_ = nullptr;
1363 return __cache;
1364}
1365
Howard Hinnant73d21a42010-09-04 23:28:19 +00001366#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001367
1368template <class _Tp, class _Hash, class _Equal, class _Alloc>
1369void
1370__hash_table<_Tp, _Hash, _Equal, _Alloc>::__move_assign(
1371 __hash_table& __u, true_type)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001372 _NOEXCEPT_(
1373 is_nothrow_move_assignable<__node_allocator>::value &&
1374 is_nothrow_move_assignable<hasher>::value &&
1375 is_nothrow_move_assignable<key_equal>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001376{
1377 clear();
1378 __bucket_list_.reset(__u.__bucket_list_.release());
1379 __bucket_list_.get_deleter().size() = __u.__bucket_list_.get_deleter().size();
1380 __u.__bucket_list_.get_deleter().size() = 0;
1381 __move_assign_alloc(__u);
1382 size() = __u.size();
Howard Hinnant0949eed2011-06-30 21:18:19 +00001383 hash_function() = _VSTD::move(__u.hash_function());
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001384 max_load_factor() = __u.max_load_factor();
Howard Hinnant0949eed2011-06-30 21:18:19 +00001385 key_eq() = _VSTD::move(__u.key_eq());
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001386 __p1_.first().__next_ = __u.__p1_.first().__next_;
1387 if (size() > 0)
1388 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001389 __bucket_list_[__constrain_hash(__p1_.first().__next_->__hash_, bucket_count())] =
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001390 static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001391 __u.__p1_.first().__next_ = nullptr;
1392 __u.size() = 0;
1393 }
Howard Hinnant39213642013-07-23 22:01:58 +00001394#if _LIBCPP_DEBUG_LEVEL >= 2
1395 __get_db()->swap(this, &__u);
1396#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001397}
1398
1399template <class _Tp, class _Hash, class _Equal, class _Alloc>
1400void
1401__hash_table<_Tp, _Hash, _Equal, _Alloc>::__move_assign(
1402 __hash_table& __u, false_type)
1403{
1404 if (__node_alloc() == __u.__node_alloc())
1405 __move_assign(__u, true_type());
1406 else
1407 {
Howard Hinnant0949eed2011-06-30 21:18:19 +00001408 hash_function() = _VSTD::move(__u.hash_function());
1409 key_eq() = _VSTD::move(__u.key_eq());
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001410 max_load_factor() = __u.max_load_factor();
1411 if (bucket_count() != 0)
1412 {
1413 __node_pointer __cache = __detach();
1414#ifndef _LIBCPP_NO_EXCEPTIONS
1415 try
1416 {
Howard Hinnant324bb032010-08-22 00:02:43 +00001417#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001418 const_iterator __i = __u.begin();
1419 while (__cache != nullptr && __u.size() != 0)
1420 {
Howard Hinnant0949eed2011-06-30 21:18:19 +00001421 __cache->__value_ = _VSTD::move(__u.remove(__i++)->__value_);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001422 __node_pointer __next = __cache->__next_;
1423 __node_insert_multi(__cache);
1424 __cache = __next;
1425 }
1426#ifndef _LIBCPP_NO_EXCEPTIONS
1427 }
1428 catch (...)
1429 {
1430 __deallocate(__cache);
1431 throw;
1432 }
Howard Hinnant324bb032010-08-22 00:02:43 +00001433#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001434 __deallocate(__cache);
1435 }
1436 const_iterator __i = __u.begin();
1437 while (__u.size() != 0)
1438 {
1439 __node_holder __h =
Howard Hinnant0949eed2011-06-30 21:18:19 +00001440 __construct_node(_VSTD::move(__u.remove(__i++)->__value_));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001441 __node_insert_multi(__h.get());
1442 __h.release();
1443 }
1444 }
1445}
1446
1447template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001448inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001449__hash_table<_Tp, _Hash, _Equal, _Alloc>&
1450__hash_table<_Tp, _Hash, _Equal, _Alloc>::operator=(__hash_table&& __u)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001451 _NOEXCEPT_(
1452 __node_traits::propagate_on_container_move_assignment::value &&
1453 is_nothrow_move_assignable<__node_allocator>::value &&
1454 is_nothrow_move_assignable<hasher>::value &&
1455 is_nothrow_move_assignable<key_equal>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001456{
1457 __move_assign(__u, integral_constant<bool,
1458 __node_traits::propagate_on_container_move_assignment::value>());
1459 return *this;
1460}
1461
Howard Hinnant73d21a42010-09-04 23:28:19 +00001462#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001463
1464template <class _Tp, class _Hash, class _Equal, class _Alloc>
1465template <class _InputIterator>
1466void
1467__hash_table<_Tp, _Hash, _Equal, _Alloc>::__assign_unique(_InputIterator __first,
1468 _InputIterator __last)
1469{
1470 if (bucket_count() != 0)
1471 {
1472 __node_pointer __cache = __detach();
1473#ifndef _LIBCPP_NO_EXCEPTIONS
1474 try
1475 {
Howard Hinnant324bb032010-08-22 00:02:43 +00001476#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001477 for (; __cache != nullptr && __first != __last; ++__first)
1478 {
1479 __cache->__value_ = *__first;
1480 __node_pointer __next = __cache->__next_;
1481 __node_insert_unique(__cache);
1482 __cache = __next;
1483 }
1484#ifndef _LIBCPP_NO_EXCEPTIONS
1485 }
1486 catch (...)
1487 {
1488 __deallocate(__cache);
1489 throw;
1490 }
Howard Hinnant324bb032010-08-22 00:02:43 +00001491#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001492 __deallocate(__cache);
1493 }
1494 for (; __first != __last; ++__first)
1495 __insert_unique(*__first);
1496}
1497
1498template <class _Tp, class _Hash, class _Equal, class _Alloc>
1499template <class _InputIterator>
1500void
1501__hash_table<_Tp, _Hash, _Equal, _Alloc>::__assign_multi(_InputIterator __first,
1502 _InputIterator __last)
1503{
1504 if (bucket_count() != 0)
1505 {
1506 __node_pointer __cache = __detach();
1507#ifndef _LIBCPP_NO_EXCEPTIONS
1508 try
1509 {
Howard Hinnant324bb032010-08-22 00:02:43 +00001510#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001511 for (; __cache != nullptr && __first != __last; ++__first)
1512 {
1513 __cache->__value_ = *__first;
1514 __node_pointer __next = __cache->__next_;
1515 __node_insert_multi(__cache);
1516 __cache = __next;
1517 }
1518#ifndef _LIBCPP_NO_EXCEPTIONS
1519 }
1520 catch (...)
1521 {
1522 __deallocate(__cache);
1523 throw;
1524 }
Howard Hinnant324bb032010-08-22 00:02:43 +00001525#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001526 __deallocate(__cache);
1527 }
1528 for (; __first != __last; ++__first)
1529 __insert_multi(*__first);
1530}
1531
1532template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001533inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001534typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001535__hash_table<_Tp, _Hash, _Equal, _Alloc>::begin() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001536{
Howard Hinnant39213642013-07-23 22:01:58 +00001537#if _LIBCPP_DEBUG_LEVEL >= 2
1538 return iterator(__p1_.first().__next_, this);
1539#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001540 return iterator(__p1_.first().__next_);
Howard Hinnant39213642013-07-23 22:01:58 +00001541#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001542}
1543
1544template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001545inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001546typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001547__hash_table<_Tp, _Hash, _Equal, _Alloc>::end() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001548{
Howard Hinnant39213642013-07-23 22:01:58 +00001549#if _LIBCPP_DEBUG_LEVEL >= 2
1550 return iterator(nullptr, this);
1551#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001552 return iterator(nullptr);
Howard Hinnant39213642013-07-23 22:01:58 +00001553#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001554}
1555
1556template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001557inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001558typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001559__hash_table<_Tp, _Hash, _Equal, _Alloc>::begin() const _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001560{
Howard Hinnant39213642013-07-23 22:01:58 +00001561#if _LIBCPP_DEBUG_LEVEL >= 2
1562 return const_iterator(__p1_.first().__next_, this);
1563#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001564 return const_iterator(__p1_.first().__next_);
Howard Hinnant39213642013-07-23 22:01:58 +00001565#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001566}
1567
1568template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99acc502010-09-21 17:32:39 +00001569inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001570typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001571__hash_table<_Tp, _Hash, _Equal, _Alloc>::end() const _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001572{
Howard Hinnant39213642013-07-23 22:01:58 +00001573#if _LIBCPP_DEBUG_LEVEL >= 2
1574 return const_iterator(nullptr, this);
1575#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001576 return const_iterator(nullptr);
Howard Hinnant39213642013-07-23 22:01:58 +00001577#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001578}
1579
1580template <class _Tp, class _Hash, class _Equal, class _Alloc>
1581void
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00001582__hash_table<_Tp, _Hash, _Equal, _Alloc>::clear() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001583{
1584 if (size() > 0)
1585 {
1586 __deallocate(__p1_.first().__next_);
1587 __p1_.first().__next_ = nullptr;
1588 size_type __bc = bucket_count();
Howard Hinnant9f66bff2011-07-05 14:14:17 +00001589 for (size_type __i = 0; __i < __bc; ++__i)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001590 __bucket_list_[__i] = nullptr;
1591 size() = 0;
1592 }
1593}
1594
1595template <class _Tp, class _Hash, class _Equal, class _Alloc>
1596pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator, bool>
1597__hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_insert_unique(__node_pointer __nd)
1598{
1599 __nd->__hash_ = hash_function()(__nd->__value_);
1600 size_type __bc = bucket_count();
1601 bool __inserted = false;
1602 __node_pointer __ndptr;
1603 size_t __chash;
1604 if (__bc != 0)
1605 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001606 __chash = __constrain_hash(__nd->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001607 __ndptr = __bucket_list_[__chash];
1608 if (__ndptr != nullptr)
1609 {
1610 for (__ndptr = __ndptr->__next_; __ndptr != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00001611 __constrain_hash(__ndptr->__hash_, __bc) == __chash;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001612 __ndptr = __ndptr->__next_)
1613 {
1614 if (key_eq()(__ndptr->__value_, __nd->__value_))
1615 goto __done;
1616 }
1617 }
1618 }
1619 {
1620 if (size()+1 > __bc * max_load_factor() || __bc == 0)
1621 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001622 rehash(_VSTD::max<size_type>(2 * __bc + !__is_power2(__bc),
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001623 size_type(ceil(float(size() + 1) / max_load_factor()))));
1624 __bc = bucket_count();
Howard Hinnant7a445152012-07-06 17:31:14 +00001625 __chash = __constrain_hash(__nd->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001626 }
1627 // insert_after __bucket_list_[__chash], or __first_node if bucket is null
1628 __node_pointer __pn = __bucket_list_[__chash];
1629 if (__pn == nullptr)
1630 {
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001631 __pn = static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001632 __nd->__next_ = __pn->__next_;
1633 __pn->__next_ = __nd;
1634 // fix up __bucket_list_
1635 __bucket_list_[__chash] = __pn;
1636 if (__nd->__next_ != nullptr)
Howard Hinnant7a445152012-07-06 17:31:14 +00001637 __bucket_list_[__constrain_hash(__nd->__next_->__hash_, __bc)] = __nd;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001638 }
1639 else
1640 {
1641 __nd->__next_ = __pn->__next_;
1642 __pn->__next_ = __nd;
1643 }
1644 __ndptr = __nd;
1645 // increment size
1646 ++size();
1647 __inserted = true;
1648 }
1649__done:
Howard Hinnant39213642013-07-23 22:01:58 +00001650#if _LIBCPP_DEBUG_LEVEL >= 2
1651 return pair<iterator, bool>(iterator(__ndptr, this), __inserted);
1652#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001653 return pair<iterator, bool>(iterator(__ndptr), __inserted);
Howard Hinnant39213642013-07-23 22:01:58 +00001654#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001655}
1656
1657template <class _Tp, class _Hash, class _Equal, class _Alloc>
1658typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1659__hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_insert_multi(__node_pointer __cp)
1660{
1661 __cp->__hash_ = hash_function()(__cp->__value_);
1662 size_type __bc = bucket_count();
1663 if (size()+1 > __bc * max_load_factor() || __bc == 0)
1664 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001665 rehash(_VSTD::max<size_type>(2 * __bc + !__is_power2(__bc),
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001666 size_type(ceil(float(size() + 1) / max_load_factor()))));
1667 __bc = bucket_count();
1668 }
Howard Hinnant7a445152012-07-06 17:31:14 +00001669 size_t __chash = __constrain_hash(__cp->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001670 __node_pointer __pn = __bucket_list_[__chash];
1671 if (__pn == nullptr)
1672 {
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001673 __pn = static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001674 __cp->__next_ = __pn->__next_;
1675 __pn->__next_ = __cp;
1676 // fix up __bucket_list_
1677 __bucket_list_[__chash] = __pn;
1678 if (__cp->__next_ != nullptr)
Howard Hinnant7a445152012-07-06 17:31:14 +00001679 __bucket_list_[__constrain_hash(__cp->__next_->__hash_, __bc)] = __cp;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001680 }
1681 else
1682 {
1683 for (bool __found = false; __pn->__next_ != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00001684 __constrain_hash(__pn->__next_->__hash_, __bc) == __chash;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001685 __pn = __pn->__next_)
Howard Hinnant324bb032010-08-22 00:02:43 +00001686 {
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001687 // __found key_eq() action
1688 // false false loop
1689 // true true loop
1690 // false true set __found to true
1691 // true false break
1692 if (__found != (__pn->__next_->__hash_ == __cp->__hash_ &&
1693 key_eq()(__pn->__next_->__value_, __cp->__value_)))
1694 {
1695 if (!__found)
1696 __found = true;
1697 else
1698 break;
1699 }
1700 }
1701 __cp->__next_ = __pn->__next_;
1702 __pn->__next_ = __cp;
1703 if (__cp->__next_ != nullptr)
1704 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001705 size_t __nhash = __constrain_hash(__cp->__next_->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001706 if (__nhash != __chash)
1707 __bucket_list_[__nhash] = __cp;
1708 }
1709 }
1710 ++size();
Howard Hinnant39213642013-07-23 22:01:58 +00001711#if _LIBCPP_DEBUG_LEVEL >= 2
1712 return iterator(__cp, this);
1713#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001714 return iterator(__cp);
Howard Hinnant39213642013-07-23 22:01:58 +00001715#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001716}
1717
1718template <class _Tp, class _Hash, class _Equal, class _Alloc>
1719typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1720__hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_insert_multi(
1721 const_iterator __p, __node_pointer __cp)
1722{
Howard Hinnant824c1992013-08-02 17:50:49 +00001723#if _LIBCPP_DEBUG_LEVEL >= 2
1724 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__p) == this,
1725 "unordered container::emplace_hint(const_iterator, args...) called with an iterator not"
1726 " referring to this unordered container");
1727#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001728 if (__p != end() && key_eq()(*__p, __cp->__value_))
1729 {
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001730 __node_pointer __np = __p.__node_;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001731 __cp->__hash_ = __np->__hash_;
1732 size_type __bc = bucket_count();
1733 if (size()+1 > __bc * max_load_factor() || __bc == 0)
1734 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001735 rehash(_VSTD::max<size_type>(2 * __bc + !__is_power2(__bc),
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001736 size_type(ceil(float(size() + 1) / max_load_factor()))));
1737 __bc = bucket_count();
1738 }
Howard Hinnant7a445152012-07-06 17:31:14 +00001739 size_t __chash = __constrain_hash(__cp->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001740 __node_pointer __pp = __bucket_list_[__chash];
1741 while (__pp->__next_ != __np)
1742 __pp = __pp->__next_;
1743 __cp->__next_ = __np;
1744 __pp->__next_ = __cp;
1745 ++size();
Howard Hinnant39213642013-07-23 22:01:58 +00001746#if _LIBCPP_DEBUG_LEVEL >= 2
1747 return iterator(__cp, this);
1748#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001749 return iterator(__cp);
Howard Hinnant39213642013-07-23 22:01:58 +00001750#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001751 }
1752 return __node_insert_multi(__cp);
1753}
1754
1755template <class _Tp, class _Hash, class _Equal, class _Alloc>
1756pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator, bool>
1757__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_unique(const value_type& __x)
1758{
1759 size_t __hash = hash_function()(__x);
1760 size_type __bc = bucket_count();
1761 bool __inserted = false;
1762 __node_pointer __nd;
1763 size_t __chash;
1764 if (__bc != 0)
1765 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001766 __chash = __constrain_hash(__hash, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001767 __nd = __bucket_list_[__chash];
1768 if (__nd != nullptr)
1769 {
1770 for (__nd = __nd->__next_; __nd != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00001771 __constrain_hash(__nd->__hash_, __bc) == __chash;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001772 __nd = __nd->__next_)
1773 {
1774 if (key_eq()(__nd->__value_, __x))
1775 goto __done;
1776 }
1777 }
1778 }
1779 {
1780 __node_holder __h = __construct_node(__x, __hash);
1781 if (size()+1 > __bc * max_load_factor() || __bc == 0)
1782 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001783 rehash(_VSTD::max<size_type>(2 * __bc + !__is_power2(__bc),
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001784 size_type(ceil(float(size() + 1) / max_load_factor()))));
1785 __bc = bucket_count();
Howard Hinnant7a445152012-07-06 17:31:14 +00001786 __chash = __constrain_hash(__hash, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001787 }
1788 // insert_after __bucket_list_[__chash], or __first_node if bucket is null
1789 __node_pointer __pn = __bucket_list_[__chash];
1790 if (__pn == nullptr)
1791 {
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001792 __pn = static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001793 __h->__next_ = __pn->__next_;
1794 __pn->__next_ = __h.get();
1795 // fix up __bucket_list_
1796 __bucket_list_[__chash] = __pn;
1797 if (__h->__next_ != nullptr)
Howard Hinnant7a445152012-07-06 17:31:14 +00001798 __bucket_list_[__constrain_hash(__h->__next_->__hash_, __bc)] = __h.get();
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001799 }
1800 else
1801 {
1802 __h->__next_ = __pn->__next_;
1803 __pn->__next_ = __h.get();
1804 }
1805 __nd = __h.release();
1806 // increment size
1807 ++size();
1808 __inserted = true;
1809 }
1810__done:
Howard Hinnant39213642013-07-23 22:01:58 +00001811#if _LIBCPP_DEBUG_LEVEL >= 2
1812 return pair<iterator, bool>(iterator(__nd, this), __inserted);
1813#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001814 return pair<iterator, bool>(iterator(__nd), __inserted);
Howard Hinnant39213642013-07-23 22:01:58 +00001815#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001816}
1817
Howard Hinnant73d21a42010-09-04 23:28:19 +00001818#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
1819#ifndef _LIBCPP_HAS_NO_VARIADICS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001820
1821template <class _Tp, class _Hash, class _Equal, class _Alloc>
1822template <class... _Args>
1823pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator, bool>
1824__hash_table<_Tp, _Hash, _Equal, _Alloc>::__emplace_unique(_Args&&... __args)
1825{
Howard Hinnant0949eed2011-06-30 21:18:19 +00001826 __node_holder __h = __construct_node(_VSTD::forward<_Args>(__args)...);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001827 pair<iterator, bool> __r = __node_insert_unique(__h.get());
1828 if (__r.second)
1829 __h.release();
1830 return __r;
1831}
1832
1833template <class _Tp, class _Hash, class _Equal, class _Alloc>
1834template <class... _Args>
1835typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1836__hash_table<_Tp, _Hash, _Equal, _Alloc>::__emplace_multi(_Args&&... __args)
1837{
Howard Hinnant0949eed2011-06-30 21:18:19 +00001838 __node_holder __h = __construct_node(_VSTD::forward<_Args>(__args)...);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001839 iterator __r = __node_insert_multi(__h.get());
1840 __h.release();
1841 return __r;
1842}
1843
1844template <class _Tp, class _Hash, class _Equal, class _Alloc>
1845template <class... _Args>
1846typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1847__hash_table<_Tp, _Hash, _Equal, _Alloc>::__emplace_hint_multi(
1848 const_iterator __p, _Args&&... __args)
1849{
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001850#if _LIBCPP_DEBUG_LEVEL >= 2
1851 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__p) == this,
1852 "unordered container::emplace_hint(const_iterator, args...) called with an iterator not"
1853 " referring to this unordered container");
1854#endif
Howard Hinnant0949eed2011-06-30 21:18:19 +00001855 __node_holder __h = __construct_node(_VSTD::forward<_Args>(__args)...);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001856 iterator __r = __node_insert_multi(__p, __h.get());
1857 __h.release();
1858 return __r;
1859}
1860
Howard Hinnant73d21a42010-09-04 23:28:19 +00001861#endif // _LIBCPP_HAS_NO_VARIADICS
1862
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001863template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99968442011-11-29 18:15:50 +00001864template <class _Pp>
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001865pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator, bool>
Howard Hinnant99968442011-11-29 18:15:50 +00001866__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_unique(_Pp&& __x)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001867{
Howard Hinnant99968442011-11-29 18:15:50 +00001868 __node_holder __h = __construct_node(_VSTD::forward<_Pp>(__x));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001869 pair<iterator, bool> __r = __node_insert_unique(__h.get());
1870 if (__r.second)
1871 __h.release();
1872 return __r;
1873}
1874
Howard Hinnant73d21a42010-09-04 23:28:19 +00001875#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001876
Howard Hinnant73d21a42010-09-04 23:28:19 +00001877#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001878
1879template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99968442011-11-29 18:15:50 +00001880template <class _Pp>
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001881typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
Howard Hinnant99968442011-11-29 18:15:50 +00001882__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_multi(_Pp&& __x)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001883{
Howard Hinnant99968442011-11-29 18:15:50 +00001884 __node_holder __h = __construct_node(_VSTD::forward<_Pp>(__x));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001885 iterator __r = __node_insert_multi(__h.get());
1886 __h.release();
1887 return __r;
1888}
1889
1890template <class _Tp, class _Hash, class _Equal, class _Alloc>
Howard Hinnant99968442011-11-29 18:15:50 +00001891template <class _Pp>
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001892typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1893__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_multi(const_iterator __p,
Howard Hinnant99968442011-11-29 18:15:50 +00001894 _Pp&& __x)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001895{
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001896#if _LIBCPP_DEBUG_LEVEL >= 2
1897 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__p) == this,
1898 "unordered container::insert(const_iterator, rvalue) called with an iterator not"
1899 " referring to this unordered container");
1900#endif
Howard Hinnant99968442011-11-29 18:15:50 +00001901 __node_holder __h = __construct_node(_VSTD::forward<_Pp>(__x));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001902 iterator __r = __node_insert_multi(__p, __h.get());
1903 __h.release();
1904 return __r;
1905}
1906
Howard Hinnant73d21a42010-09-04 23:28:19 +00001907#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001908
1909template <class _Tp, class _Hash, class _Equal, class _Alloc>
1910typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1911__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_multi(const value_type& __x)
1912{
1913 __node_holder __h = __construct_node(__x);
1914 iterator __r = __node_insert_multi(__h.get());
1915 __h.release();
1916 return __r;
1917}
1918
1919template <class _Tp, class _Hash, class _Equal, class _Alloc>
1920typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
1921__hash_table<_Tp, _Hash, _Equal, _Alloc>::__insert_multi(const_iterator __p,
1922 const value_type& __x)
1923{
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00001924#if _LIBCPP_DEBUG_LEVEL >= 2
1925 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__p) == this,
1926 "unordered container::insert(const_iterator, lvalue) called with an iterator not"
1927 " referring to this unordered container");
1928#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001929 __node_holder __h = __construct_node(__x);
1930 iterator __r = __node_insert_multi(__p, __h.get());
1931 __h.release();
1932 return __r;
1933}
1934
Howard Hinnant73d21a42010-09-04 23:28:19 +00001935#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001936
1937template <class _Tp, class _Hash, class _Equal, class _Alloc>
1938void
1939__hash_table<_Tp, _Hash, _Equal, _Alloc>::rehash(size_type __n)
1940{
Howard Hinnant7a445152012-07-06 17:31:14 +00001941 if (__n == 1)
1942 __n = 2;
1943 else if (__n & (__n - 1))
1944 __n = __next_prime(__n);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001945 size_type __bc = bucket_count();
1946 if (__n > __bc)
1947 __rehash(__n);
Howard Hinnant7a445152012-07-06 17:31:14 +00001948 else if (__n < __bc)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001949 {
Howard Hinnant0949eed2011-06-30 21:18:19 +00001950 __n = _VSTD::max<size_type>
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001951 (
1952 __n,
Howard Hinnant7a445152012-07-06 17:31:14 +00001953 __is_power2(__bc) ? __next_pow2(size_t(ceil(float(size()) / max_load_factor()))) :
1954 __next_prime(size_t(ceil(float(size()) / max_load_factor())))
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001955 );
1956 if (__n < __bc)
1957 __rehash(__n);
1958 }
1959}
1960
1961template <class _Tp, class _Hash, class _Equal, class _Alloc>
1962void
1963__hash_table<_Tp, _Hash, _Equal, _Alloc>::__rehash(size_type __nbc)
1964{
Howard Hinnant39213642013-07-23 22:01:58 +00001965#if _LIBCPP_DEBUG_LEVEL >= 2
1966 __get_db()->__invalidate_all(this);
1967#endif // _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001968 __pointer_allocator& __npa = __bucket_list_.get_deleter().__alloc();
1969 __bucket_list_.reset(__nbc > 0 ?
1970 __pointer_alloc_traits::allocate(__npa, __nbc) : nullptr);
1971 __bucket_list_.get_deleter().size() = __nbc;
1972 if (__nbc > 0)
1973 {
1974 for (size_type __i = 0; __i < __nbc; ++__i)
1975 __bucket_list_[__i] = nullptr;
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00001976 __node_pointer __pp(static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first())));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001977 __node_pointer __cp = __pp->__next_;
1978 if (__cp != nullptr)
1979 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001980 size_type __chash = __constrain_hash(__cp->__hash_, __nbc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001981 __bucket_list_[__chash] = __pp;
1982 size_type __phash = __chash;
1983 for (__pp = __cp, __cp = __cp->__next_; __cp != nullptr;
1984 __cp = __pp->__next_)
1985 {
Howard Hinnant7a445152012-07-06 17:31:14 +00001986 __chash = __constrain_hash(__cp->__hash_, __nbc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001987 if (__chash == __phash)
1988 __pp = __cp;
1989 else
1990 {
1991 if (__bucket_list_[__chash] == nullptr)
1992 {
1993 __bucket_list_[__chash] = __pp;
1994 __pp = __cp;
1995 __phash = __chash;
1996 }
1997 else
1998 {
1999 __node_pointer __np = __cp;
2000 for (; __np->__next_ != nullptr &&
2001 key_eq()(__cp->__value_, __np->__next_->__value_);
2002 __np = __np->__next_)
2003 ;
2004 __pp->__next_ = __np->__next_;
2005 __np->__next_ = __bucket_list_[__chash]->__next_;
2006 __bucket_list_[__chash]->__next_ = __cp;
Howard Hinnant324bb032010-08-22 00:02:43 +00002007
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002008 }
2009 }
2010 }
2011 }
2012 }
2013}
2014
2015template <class _Tp, class _Hash, class _Equal, class _Alloc>
2016template <class _Key>
2017typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
2018__hash_table<_Tp, _Hash, _Equal, _Alloc>::find(const _Key& __k)
2019{
2020 size_t __hash = hash_function()(__k);
2021 size_type __bc = bucket_count();
2022 if (__bc != 0)
2023 {
Howard Hinnant7a445152012-07-06 17:31:14 +00002024 size_t __chash = __constrain_hash(__hash, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002025 __node_pointer __nd = __bucket_list_[__chash];
2026 if (__nd != nullptr)
2027 {
2028 for (__nd = __nd->__next_; __nd != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00002029 __constrain_hash(__nd->__hash_, __bc) == __chash;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002030 __nd = __nd->__next_)
2031 {
2032 if (key_eq()(__nd->__value_, __k))
Howard Hinnant39213642013-07-23 22:01:58 +00002033#if _LIBCPP_DEBUG_LEVEL >= 2
2034 return iterator(__nd, this);
2035#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002036 return iterator(__nd);
Howard Hinnant39213642013-07-23 22:01:58 +00002037#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002038 }
2039 }
2040 }
2041 return end();
2042}
2043
2044template <class _Tp, class _Hash, class _Equal, class _Alloc>
2045template <class _Key>
2046typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator
2047__hash_table<_Tp, _Hash, _Equal, _Alloc>::find(const _Key& __k) const
2048{
2049 size_t __hash = hash_function()(__k);
2050 size_type __bc = bucket_count();
2051 if (__bc != 0)
2052 {
Howard Hinnant7a445152012-07-06 17:31:14 +00002053 size_t __chash = __constrain_hash(__hash, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002054 __node_const_pointer __nd = __bucket_list_[__chash];
2055 if (__nd != nullptr)
2056 {
2057 for (__nd = __nd->__next_; __nd != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00002058 __constrain_hash(__nd->__hash_, __bc) == __chash;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002059 __nd = __nd->__next_)
2060 {
2061 if (key_eq()(__nd->__value_, __k))
Howard Hinnant39213642013-07-23 22:01:58 +00002062#if _LIBCPP_DEBUG_LEVEL >= 2
2063 return const_iterator(__nd, this);
2064#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002065 return const_iterator(__nd);
Howard Hinnant39213642013-07-23 22:01:58 +00002066#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002067 }
2068 }
Howard Hinnant324bb032010-08-22 00:02:43 +00002069
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002070 }
2071 return end();
2072}
2073
Howard Hinnant73d21a42010-09-04 23:28:19 +00002074#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
2075#ifndef _LIBCPP_HAS_NO_VARIADICS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002076
2077template <class _Tp, class _Hash, class _Equal, class _Alloc>
2078template <class ..._Args>
2079typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_holder
2080__hash_table<_Tp, _Hash, _Equal, _Alloc>::__construct_node(_Args&& ...__args)
2081{
2082 __node_allocator& __na = __node_alloc();
Howard Hinnant99968442011-11-29 18:15:50 +00002083 __node_holder __h(__node_traits::allocate(__na, 1), _Dp(__na));
Howard Hinnant0949eed2011-06-30 21:18:19 +00002084 __node_traits::construct(__na, _VSTD::addressof(__h->__value_), _VSTD::forward<_Args>(__args)...);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002085 __h.get_deleter().__value_constructed = true;
2086 __h->__hash_ = hash_function()(__h->__value_);
2087 __h->__next_ = nullptr;
2088 return __h;
2089}
2090
Howard Hinnant73d21a42010-09-04 23:28:19 +00002091#endif // _LIBCPP_HAS_NO_VARIADICS
2092
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002093template <class _Tp, class _Hash, class _Equal, class _Alloc>
2094typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_holder
2095__hash_table<_Tp, _Hash, _Equal, _Alloc>::__construct_node(value_type&& __v,
2096 size_t __hash)
2097{
2098 __node_allocator& __na = __node_alloc();
Howard Hinnant99968442011-11-29 18:15:50 +00002099 __node_holder __h(__node_traits::allocate(__na, 1), _Dp(__na));
Howard Hinnant0949eed2011-06-30 21:18:19 +00002100 __node_traits::construct(__na, _VSTD::addressof(__h->__value_), _VSTD::move(__v));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002101 __h.get_deleter().__value_constructed = true;
2102 __h->__hash_ = __hash;
2103 __h->__next_ = nullptr;
Howard Hinnant0949eed2011-06-30 21:18:19 +00002104 return _VSTD::move(__h);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002105}
2106
Howard Hinnant73d21a42010-09-04 23:28:19 +00002107#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002108
2109template <class _Tp, class _Hash, class _Equal, class _Alloc>
2110typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_holder
2111__hash_table<_Tp, _Hash, _Equal, _Alloc>::__construct_node(const value_type& __v)
2112{
2113 __node_allocator& __na = __node_alloc();
Howard Hinnant99968442011-11-29 18:15:50 +00002114 __node_holder __h(__node_traits::allocate(__na, 1), _Dp(__na));
Howard Hinnant0949eed2011-06-30 21:18:19 +00002115 __node_traits::construct(__na, _VSTD::addressof(__h->__value_), __v);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002116 __h.get_deleter().__value_constructed = true;
2117 __h->__hash_ = hash_function()(__h->__value_);
2118 __h->__next_ = nullptr;
Howard Hinnant0949eed2011-06-30 21:18:19 +00002119 return _VSTD::move(__h);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002120}
2121
Howard Hinnant73d21a42010-09-04 23:28:19 +00002122#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002123
2124template <class _Tp, class _Hash, class _Equal, class _Alloc>
2125typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_holder
2126__hash_table<_Tp, _Hash, _Equal, _Alloc>::__construct_node(const value_type& __v,
2127 size_t __hash)
2128{
2129 __node_allocator& __na = __node_alloc();
Howard Hinnant99968442011-11-29 18:15:50 +00002130 __node_holder __h(__node_traits::allocate(__na, 1), _Dp(__na));
Howard Hinnant0949eed2011-06-30 21:18:19 +00002131 __node_traits::construct(__na, _VSTD::addressof(__h->__value_), __v);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002132 __h.get_deleter().__value_constructed = true;
2133 __h->__hash_ = __hash;
2134 __h->__next_ = nullptr;
Howard Hinnant0949eed2011-06-30 21:18:19 +00002135 return _VSTD::move(__h);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002136}
2137
2138template <class _Tp, class _Hash, class _Equal, class _Alloc>
2139typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
2140__hash_table<_Tp, _Hash, _Equal, _Alloc>::erase(const_iterator __p)
2141{
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002142 __node_pointer __np = __p.__node_;
Howard Hinnant39213642013-07-23 22:01:58 +00002143#if _LIBCPP_DEBUG_LEVEL >= 2
2144 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__p) == this,
2145 "unordered container erase(iterator) called with an iterator not"
2146 " referring to this container");
2147 _LIBCPP_ASSERT(__p != end(),
2148 "unordered container erase(iterator) called with a non-dereferenceable iterator");
2149 iterator __r(__np, this);
2150#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002151 iterator __r(__np);
Howard Hinnant39213642013-07-23 22:01:58 +00002152#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002153 ++__r;
2154 remove(__p);
2155 return __r;
2156}
2157
2158template <class _Tp, class _Hash, class _Equal, class _Alloc>
2159typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator
2160__hash_table<_Tp, _Hash, _Equal, _Alloc>::erase(const_iterator __first,
2161 const_iterator __last)
2162{
Howard Hinnant39213642013-07-23 22:01:58 +00002163#if _LIBCPP_DEBUG_LEVEL >= 2
2164 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__first) == this,
2165 "unodered container::erase(iterator, iterator) called with an iterator not"
2166 " referring to this unodered container");
2167 _LIBCPP_ASSERT(__get_const_db()->__find_c_from_i(&__last) == this,
2168 "unodered container::erase(iterator, iterator) called with an iterator not"
2169 " referring to this unodered container");
2170#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002171 for (const_iterator __p = __first; __first != __last; __p = __first)
2172 {
2173 ++__first;
2174 erase(__p);
2175 }
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002176 __node_pointer __np = __last.__node_;
Howard Hinnant39213642013-07-23 22:01:58 +00002177#if _LIBCPP_DEBUG_LEVEL >= 2
2178 return iterator (__np, this);
2179#else
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002180 return iterator (__np);
Howard Hinnant39213642013-07-23 22:01:58 +00002181#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002182}
2183
2184template <class _Tp, class _Hash, class _Equal, class _Alloc>
2185template <class _Key>
2186typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::size_type
2187__hash_table<_Tp, _Hash, _Equal, _Alloc>::__erase_unique(const _Key& __k)
2188{
2189 iterator __i = find(__k);
2190 if (__i == end())
2191 return 0;
2192 erase(__i);
2193 return 1;
2194}
2195
2196template <class _Tp, class _Hash, class _Equal, class _Alloc>
2197template <class _Key>
2198typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::size_type
2199__hash_table<_Tp, _Hash, _Equal, _Alloc>::__erase_multi(const _Key& __k)
2200{
2201 size_type __r = 0;
2202 iterator __i = find(__k);
2203 if (__i != end())
2204 {
2205 iterator __e = end();
2206 do
2207 {
2208 erase(__i++);
2209 ++__r;
2210 } while (__i != __e && key_eq()(*__i, __k));
2211 }
2212 return __r;
2213}
2214
2215template <class _Tp, class _Hash, class _Equal, class _Alloc>
2216typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::__node_holder
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00002217__hash_table<_Tp, _Hash, _Equal, _Alloc>::remove(const_iterator __p) _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002218{
2219 // current node
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002220 __node_pointer __cn = __p.__node_;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002221 size_type __bc = bucket_count();
Howard Hinnant7a445152012-07-06 17:31:14 +00002222 size_t __chash = __constrain_hash(__cn->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002223 // find previous node
2224 __node_pointer __pn = __bucket_list_[__chash];
2225 for (; __pn->__next_ != __cn; __pn = __pn->__next_)
2226 ;
2227 // Fix up __bucket_list_
2228 // if __pn is not in same bucket (before begin is not in same bucket) &&
2229 // if __cn->__next_ is not in same bucket (nullptr is not in same bucket)
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002230 if (__pn == static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()))
2231 || __constrain_hash(__pn->__hash_, __bc) != __chash)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002232 {
Howard Hinnant7a445152012-07-06 17:31:14 +00002233 if (__cn->__next_ == nullptr || __constrain_hash(__cn->__next_->__hash_, __bc) != __chash)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002234 __bucket_list_[__chash] = nullptr;
2235 }
2236 // if __cn->__next_ is not in same bucket (nullptr is in same bucket)
2237 if (__cn->__next_ != nullptr)
2238 {
Howard Hinnant7a445152012-07-06 17:31:14 +00002239 size_t __nhash = __constrain_hash(__cn->__next_->__hash_, __bc);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002240 if (__nhash != __chash)
2241 __bucket_list_[__nhash] = __pn;
2242 }
2243 // remove __cn
2244 __pn->__next_ = __cn->__next_;
2245 __cn->__next_ = nullptr;
2246 --size();
Howard Hinnant39213642013-07-23 22:01:58 +00002247#if _LIBCPP_DEBUG_LEVEL >= 2
2248 __c_node* __c = __get_db()->__find_c_and_lock(this);
2249 for (__i_node** __p = __c->end_; __p != __c->beg_; )
2250 {
2251 --__p;
2252 iterator* __i = static_cast<iterator*>((*__p)->__i_);
2253 if (__i->__node_ == __cn)
2254 {
2255 (*__p)->__c_ = nullptr;
2256 if (--__c->end_ != __p)
2257 memmove(__p, __p+1, (__c->end_ - __p)*sizeof(__i_node*));
2258 }
2259 }
2260 __get_db()->unlock();
2261#endif
Howard Hinnant99968442011-11-29 18:15:50 +00002262 return __node_holder(__cn, _Dp(__node_alloc(), true));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002263}
2264
2265template <class _Tp, class _Hash, class _Equal, class _Alloc>
2266template <class _Key>
Howard Hinnant99acc502010-09-21 17:32:39 +00002267inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002268typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::size_type
2269__hash_table<_Tp, _Hash, _Equal, _Alloc>::__count_unique(const _Key& __k) const
2270{
2271 return static_cast<size_type>(find(__k) != end());
2272}
2273
2274template <class _Tp, class _Hash, class _Equal, class _Alloc>
2275template <class _Key>
2276typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::size_type
2277__hash_table<_Tp, _Hash, _Equal, _Alloc>::__count_multi(const _Key& __k) const
2278{
2279 size_type __r = 0;
2280 const_iterator __i = find(__k);
2281 if (__i != end())
2282 {
2283 const_iterator __e = end();
2284 do
2285 {
2286 ++__i;
2287 ++__r;
2288 } while (__i != __e && key_eq()(*__i, __k));
2289 }
2290 return __r;
2291}
2292
2293template <class _Tp, class _Hash, class _Equal, class _Alloc>
2294template <class _Key>
2295pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator,
2296 typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator>
2297__hash_table<_Tp, _Hash, _Equal, _Alloc>::__equal_range_unique(
2298 const _Key& __k)
2299{
2300 iterator __i = find(__k);
2301 iterator __j = __i;
2302 if (__i != end())
2303 ++__j;
2304 return pair<iterator, iterator>(__i, __j);
2305}
2306
2307template <class _Tp, class _Hash, class _Equal, class _Alloc>
2308template <class _Key>
2309pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator,
2310 typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator>
2311__hash_table<_Tp, _Hash, _Equal, _Alloc>::__equal_range_unique(
2312 const _Key& __k) const
2313{
2314 const_iterator __i = find(__k);
2315 const_iterator __j = __i;
2316 if (__i != end())
2317 ++__j;
2318 return pair<const_iterator, const_iterator>(__i, __j);
2319}
2320
2321template <class _Tp, class _Hash, class _Equal, class _Alloc>
2322template <class _Key>
2323pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator,
2324 typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::iterator>
2325__hash_table<_Tp, _Hash, _Equal, _Alloc>::__equal_range_multi(
2326 const _Key& __k)
2327{
2328 iterator __i = find(__k);
2329 iterator __j = __i;
2330 if (__i != end())
2331 {
2332 iterator __e = end();
2333 do
2334 {
2335 ++__j;
2336 } while (__j != __e && key_eq()(*__j, __k));
2337 }
2338 return pair<iterator, iterator>(__i, __j);
2339}
2340
2341template <class _Tp, class _Hash, class _Equal, class _Alloc>
2342template <class _Key>
2343pair<typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator,
2344 typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::const_iterator>
2345__hash_table<_Tp, _Hash, _Equal, _Alloc>::__equal_range_multi(
2346 const _Key& __k) const
2347{
2348 const_iterator __i = find(__k);
2349 const_iterator __j = __i;
2350 if (__i != end())
2351 {
2352 const_iterator __e = end();
2353 do
2354 {
2355 ++__j;
2356 } while (__j != __e && key_eq()(*__j, __k));
2357 }
2358 return pair<const_iterator, const_iterator>(__i, __j);
2359}
2360
2361template <class _Tp, class _Hash, class _Equal, class _Alloc>
2362void
2363__hash_table<_Tp, _Hash, _Equal, _Alloc>::swap(__hash_table& __u)
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00002364 _NOEXCEPT_(
2365 (!allocator_traits<__pointer_allocator>::propagate_on_container_swap::value ||
2366 __is_nothrow_swappable<__pointer_allocator>::value) &&
2367 (!__node_traits::propagate_on_container_swap::value ||
2368 __is_nothrow_swappable<__node_allocator>::value) &&
2369 __is_nothrow_swappable<hasher>::value &&
2370 __is_nothrow_swappable<key_equal>::value)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002371{
2372 {
2373 __node_pointer_pointer __npp = __bucket_list_.release();
2374 __bucket_list_.reset(__u.__bucket_list_.release());
2375 __u.__bucket_list_.reset(__npp);
2376 }
Howard Hinnant0949eed2011-06-30 21:18:19 +00002377 _VSTD::swap(__bucket_list_.get_deleter().size(), __u.__bucket_list_.get_deleter().size());
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002378 __swap_alloc(__bucket_list_.get_deleter().__alloc(),
2379 __u.__bucket_list_.get_deleter().__alloc());
2380 __swap_alloc(__node_alloc(), __u.__node_alloc());
Howard Hinnant0949eed2011-06-30 21:18:19 +00002381 _VSTD::swap(__p1_.first().__next_, __u.__p1_.first().__next_);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002382 __p2_.swap(__u.__p2_);
2383 __p3_.swap(__u.__p3_);
2384 if (size() > 0)
Howard Hinnant7a445152012-07-06 17:31:14 +00002385 __bucket_list_[__constrain_hash(__p1_.first().__next_->__hash_, bucket_count())] =
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002386 static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__p1_.first()));
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002387 if (__u.size() > 0)
Howard Hinnant7a445152012-07-06 17:31:14 +00002388 __u.__bucket_list_[__constrain_hash(__u.__p1_.first().__next_->__hash_, __u.bucket_count())] =
Howard Hinnant7a6b7ce2013-06-22 15:21:29 +00002389 static_cast<__node_pointer>(pointer_traits<__node_base_pointer>::pointer_to(__u.__p1_.first()));
Howard Hinnant39213642013-07-23 22:01:58 +00002390#if _LIBCPP_DEBUG_LEVEL >= 2
2391 __get_db()->swap(this, &__u);
2392#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002393}
2394
2395template <class _Tp, class _Hash, class _Equal, class _Alloc>
2396typename __hash_table<_Tp, _Hash, _Equal, _Alloc>::size_type
2397__hash_table<_Tp, _Hash, _Equal, _Alloc>::bucket_size(size_type __n) const
2398{
Howard Hinnant0bb0a7c2013-07-29 19:05:47 +00002399 _LIBCPP_ASSERT(__n < bucket_count(),
2400 "unordered container::bucket_size(n) called with n >= bucket_count()");
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002401 __node_const_pointer __np = __bucket_list_[__n];
2402 size_type __bc = bucket_count();
2403 size_type __r = 0;
2404 if (__np != nullptr)
2405 {
2406 for (__np = __np->__next_; __np != nullptr &&
Howard Hinnant7a445152012-07-06 17:31:14 +00002407 __constrain_hash(__np->__hash_, __bc) == __n;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002408 __np = __np->__next_, ++__r)
2409 ;
2410 }
2411 return __r;
2412}
2413
Howard Hinnant5f2f14c2011-06-04 18:54:24 +00002414template <class _Tp, class _Hash, class _Equal, class _Alloc>
2415inline _LIBCPP_INLINE_VISIBILITY
2416void
2417swap(__hash_table<_Tp, _Hash, _Equal, _Alloc>& __x,
2418 __hash_table<_Tp, _Hash, _Equal, _Alloc>& __y)
2419 _NOEXCEPT_(_NOEXCEPT_(__x.swap(__y)))
2420{
2421 __x.swap(__y);
2422}
2423
Howard Hinnant39213642013-07-23 22:01:58 +00002424#if _LIBCPP_DEBUG_LEVEL >= 2
2425
2426template <class _Tp, class _Hash, class _Equal, class _Alloc>
2427bool
2428__hash_table<_Tp, _Hash, _Equal, _Alloc>::__dereferenceable(const const_iterator* __i) const
2429{
2430 return __i->__node_ != nullptr;
2431}
2432
2433template <class _Tp, class _Hash, class _Equal, class _Alloc>
2434bool
2435__hash_table<_Tp, _Hash, _Equal, _Alloc>::__decrementable(const const_iterator*) const
2436{
2437 return false;
2438}
2439
2440template <class _Tp, class _Hash, class _Equal, class _Alloc>
2441bool
2442__hash_table<_Tp, _Hash, _Equal, _Alloc>::__addable(const const_iterator*, ptrdiff_t) const
2443{
2444 return false;
2445}
2446
2447template <class _Tp, class _Hash, class _Equal, class _Alloc>
2448bool
2449__hash_table<_Tp, _Hash, _Equal, _Alloc>::__subscriptable(const const_iterator*, ptrdiff_t) const
2450{
2451 return false;
2452}
2453
2454#endif // _LIBCPP_DEBUG_LEVEL >= 2
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00002455_LIBCPP_END_NAMESPACE_STD
2456
2457#endif // _LIBCPP__HASH_TABLE