OpenCores
URL https://opencores.org/ocsvn/altor32/altor32/trunk

Subversion Repositories altor32

[/] [altor32/] [trunk/] [gcc-x64/] [or1knd-elf/] [or1knd-elf/] [include/] [c++/] [4.8.0/] [tr1/] [hashtable.h] - Blame information for rev 35

Details | Compare with Previous | View Log

Line No. Rev Author Line
1 35 ultra_embe
// TR1 hashtable.h header -*- C++ -*-
2
 
3
// Copyright (C) 2007, 2009, 2010, 2011 Free Software Foundation, Inc.
4
//
5
// This file is part of the GNU ISO C++ Library.  This library is free
6
// software; you can redistribute it and/or modify it under the
7
// terms of the GNU General Public License as published by the
8
// Free Software Foundation; either version 3, or (at your option)
9
// any later version.
10
 
11
// This library is distributed in the hope that it will be useful,
12
// but WITHOUT ANY WARRANTY; without even the implied warranty of
13
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
// GNU General Public License for more details.
15
 
16
// Under Section 7 of GPL version 3, you are granted additional
17
// permissions described in the GCC Runtime Library Exception, version
18
// 3.1, as published by the Free Software Foundation.
19
 
20
// You should have received a copy of the GNU General Public License and
21
// a copy of the GCC Runtime Library Exception along with this program;
22
// see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
23
// <http://www.gnu.org/licenses/>.
24
 
25
/** @file tr1/hashtable.h
26
 *  This is an internal header file, included by other library headers.
27
 *  Do not attempt to use it directly.
28
 *  @headername{tr1/unordered_set, tr1/unordered_map}
29
 */
30
 
31
#ifndef _GLIBCXX_TR1_HASHTABLE_H
32
#define _GLIBCXX_TR1_HASHTABLE_H 1
33
 
34
#pragma GCC system_header
35
 
36
#include <tr1/hashtable_policy.h>
37
 
38
namespace std _GLIBCXX_VISIBILITY(default)
39
{
40
namespace tr1
41
{
42
_GLIBCXX_BEGIN_NAMESPACE_VERSION
43
 
44
  // Class template _Hashtable, class definition.
45
 
46
  // Meaning of class template _Hashtable's template parameters
47
 
48
  // _Key and _Value: arbitrary CopyConstructible types.
49
 
50
  // _Allocator: an allocator type ([lib.allocator.requirements]) whose
51
  // value type is Value.  As a conforming extension, we allow for
52
  // value type != Value.
53
 
54
  // _ExtractKey: function object that takes a object of type Value
55
  // and returns a value of type _Key.
56
 
57
  // _Equal: function object that takes two objects of type k and returns
58
  // a bool-like value that is true if the two objects are considered equal.
59
 
60
  // _H1: the hash function.  A unary function object with argument type
61
  // Key and result type size_t.  Return values should be distributed
62
  // over the entire range [0, numeric_limits<size_t>:::max()].
63
 
64
  // _H2: the range-hashing function (in the terminology of Tavori and
65
  // Dreizin).  A binary function object whose argument types and result
66
  // type are all size_t.  Given arguments r and N, the return value is
67
  // in the range [0, N).
68
 
69
  // _Hash: the ranged hash function (Tavori and Dreizin). A binary function
70
  // whose argument types are _Key and size_t and whose result type is
71
  // size_t.  Given arguments k and N, the return value is in the range
72
  // [0, N).  Default: hash(k, N) = h2(h1(k), N).  If _Hash is anything other
73
  // than the default, _H1 and _H2 are ignored.
74
 
75
  // _RehashPolicy: Policy class with three members, all of which govern
76
  // the bucket count. _M_next_bkt(n) returns a bucket count no smaller
77
  // than n.  _M_bkt_for_elements(n) returns a bucket count appropriate
78
  // for an element count of n.  _M_need_rehash(n_bkt, n_elt, n_ins)
79
  // determines whether, if the current bucket count is n_bkt and the
80
  // current element count is n_elt, we need to increase the bucket
81
  // count.  If so, returns make_pair(true, n), where n is the new
82
  // bucket count.  If not, returns make_pair(false, <anything>).
83
 
84
  // ??? Right now it is hard-wired that the number of buckets never
85
  // shrinks.  Should we allow _RehashPolicy to change that?
86
 
87
  // __cache_hash_code: bool.  true if we store the value of the hash
88
  // function along with the value.  This is a time-space tradeoff.
89
  // Storing it may improve lookup speed by reducing the number of times
90
  // we need to call the Equal function.
91
 
92
  // __constant_iterators: bool.  true if iterator and const_iterator are
93
  // both constant iterator types.  This is true for unordered_set and
94
  // unordered_multiset, false for unordered_map and unordered_multimap.
95
 
96
  // __unique_keys: bool.  true if the return value of _Hashtable::count(k)
97
  // is always at most one, false if it may be an arbitrary number.  This
98
  // true for unordered_set and unordered_map, false for unordered_multiset
99
  // and unordered_multimap.
100
 
101
  template<typename _Key, typename _Value, typename _Allocator,
102
           typename _ExtractKey, typename _Equal,
103
           typename _H1, typename _H2, typename _Hash,
104
           typename _RehashPolicy,
105
           bool __cache_hash_code,
106
           bool __constant_iterators,
107
           bool __unique_keys>
108
    class _Hashtable
109
    : public __detail::_Rehash_base<_RehashPolicy,
110
                                    _Hashtable<_Key, _Value, _Allocator,
111
                                               _ExtractKey,
112
                                               _Equal, _H1, _H2, _Hash,
113
                                               _RehashPolicy,
114
                                               __cache_hash_code,
115
                                               __constant_iterators,
116
                                               __unique_keys> >,
117
      public __detail::_Hash_code_base<_Key, _Value, _ExtractKey, _Equal,
118
                                       _H1, _H2, _Hash, __cache_hash_code>,
119
      public __detail::_Map_base<_Key, _Value, _ExtractKey, __unique_keys,
120
                                 _Hashtable<_Key, _Value, _Allocator,
121
                                            _ExtractKey,
122
                                            _Equal, _H1, _H2, _Hash,
123
                                            _RehashPolicy,
124
                                            __cache_hash_code,
125
                                            __constant_iterators,
126
                                            __unique_keys> >
127
    {
128
    public:
129
      typedef _Allocator                                  allocator_type;
130
      typedef _Value                                      value_type;
131
      typedef _Key                                        key_type;
132
      typedef _Equal                                      key_equal;
133
      // mapped_type, if present, comes from _Map_base.
134
      // hasher, if present, comes from _Hash_code_base.
135
      typedef typename _Allocator::difference_type        difference_type;
136
      typedef typename _Allocator::size_type              size_type;
137
      typedef typename _Allocator::pointer                pointer;
138
      typedef typename _Allocator::const_pointer          const_pointer;
139
      typedef typename _Allocator::reference              reference;
140
      typedef typename _Allocator::const_reference        const_reference;
141
 
142
      typedef __detail::_Node_iterator<value_type, __constant_iterators,
143
                                       __cache_hash_code>
144
                                                          local_iterator;
145
      typedef __detail::_Node_const_iterator<value_type,
146
                                             __constant_iterators,
147
                                             __cache_hash_code>
148
                                                          const_local_iterator;
149
 
150
      typedef __detail::_Hashtable_iterator<value_type, __constant_iterators,
151
                                            __cache_hash_code>
152
                                                          iterator;
153
      typedef __detail::_Hashtable_const_iterator<value_type,
154
                                                  __constant_iterators,
155
                                                  __cache_hash_code>
156
                                                          const_iterator;
157
 
158
      template<typename _Key2, typename _Value2, typename _Ex2, bool __unique2,
159
               typename _Hashtable2>
160
        friend struct __detail::_Map_base;
161
 
162
    private:
163
      typedef __detail::_Hash_node<_Value, __cache_hash_code> _Node;
164
      typedef typename _Allocator::template rebind<_Node>::other
165
                                                        _Node_allocator_type;
166
      typedef typename _Allocator::template rebind<_Node*>::other
167
                                                        _Bucket_allocator_type;
168
 
169
      typedef typename _Allocator::template rebind<_Value>::other
170
                                                        _Value_allocator_type;
171
 
172
      _Node_allocator_type   _M_node_allocator;
173
      _Node**                _M_buckets;
174
      size_type              _M_bucket_count;
175
      size_type              _M_element_count;
176
      _RehashPolicy          _M_rehash_policy;
177
 
178
      _Node*
179
      _M_allocate_node(const value_type& __v);
180
 
181
      void
182
      _M_deallocate_node(_Node* __n);
183
 
184
      void
185
      _M_deallocate_nodes(_Node**, size_type);
186
 
187
      _Node**
188
      _M_allocate_buckets(size_type __n);
189
 
190
      void
191
      _M_deallocate_buckets(_Node**, size_type __n);
192
 
193
    public:
194
      // Constructor, destructor, assignment, swap
195
      _Hashtable(size_type __bucket_hint,
196
                 const _H1&, const _H2&, const _Hash&,
197
                 const _Equal&, const _ExtractKey&,
198
                 const allocator_type&);
199
 
200
      template<typename _InputIterator>
201
        _Hashtable(_InputIterator __first, _InputIterator __last,
202
                   size_type __bucket_hint,
203
                   const _H1&, const _H2&, const _Hash&,
204
                   const _Equal&, const _ExtractKey&,
205
                   const allocator_type&);
206
 
207
      _Hashtable(const _Hashtable&);
208
 
209
      _Hashtable&
210
      operator=(const _Hashtable&);
211
 
212
      ~_Hashtable();
213
 
214
      void swap(_Hashtable&);
215
 
216
      // Basic container operations
217
      iterator
218
      begin()
219
      {
220
        iterator __i(_M_buckets);
221
        if (!__i._M_cur_node)
222
          __i._M_incr_bucket();
223
        return __i;
224
      }
225
 
226
      const_iterator
227
      begin() const
228
      {
229
        const_iterator __i(_M_buckets);
230
        if (!__i._M_cur_node)
231
          __i._M_incr_bucket();
232
        return __i;
233
      }
234
 
235
      iterator
236
      end()
237
      { return iterator(_M_buckets + _M_bucket_count); }
238
 
239
      const_iterator
240
      end() const
241
      { return const_iterator(_M_buckets + _M_bucket_count); }
242
 
243
      size_type
244
      size() const
245
      { return _M_element_count; }
246
 
247
      bool
248
      empty() const
249
      { return size() == 0; }
250
 
251
      allocator_type
252
      get_allocator() const
253
      { return allocator_type(_M_node_allocator); }
254
 
255
      _Value_allocator_type
256
      _M_get_Value_allocator() const
257
      { return _Value_allocator_type(_M_node_allocator); }
258
 
259
      size_type
260
      max_size() const
261
      { return _M_node_allocator.max_size(); }
262
 
263
      // Observers
264
      key_equal
265
      key_eq() const
266
      { return this->_M_eq; }
267
 
268
      // hash_function, if present, comes from _Hash_code_base.
269
 
270
      // Bucket operations
271
      size_type
272
      bucket_count() const
273
      { return _M_bucket_count; }
274
 
275
      size_type
276
      max_bucket_count() const
277
      { return max_size(); }
278
 
279
      size_type
280
      bucket_size(size_type __n) const
281
      { return std::distance(begin(__n), end(__n)); }
282
 
283
      size_type
284
      bucket(const key_type& __k) const
285
      {
286
        return this->_M_bucket_index(__k, this->_M_hash_code(__k),
287
                                     bucket_count());
288
      }
289
 
290
      local_iterator
291
      begin(size_type __n)
292
      { return local_iterator(_M_buckets[__n]); }
293
 
294
      local_iterator
295
      end(size_type)
296
      { return local_iterator(0); }
297
 
298
      const_local_iterator
299
      begin(size_type __n) const
300
      { return const_local_iterator(_M_buckets[__n]); }
301
 
302
      const_local_iterator
303
      end(size_type) const
304
      { return const_local_iterator(0); }
305
 
306
      float
307
      load_factor() const
308
      {
309
        return static_cast<float>(size()) / static_cast<float>(bucket_count());
310
      }
311
 
312
      // max_load_factor, if present, comes from _Rehash_base.
313
 
314
      // Generalization of max_load_factor.  Extension, not found in TR1.  Only
315
      // useful if _RehashPolicy is something other than the default.
316
      const _RehashPolicy&
317
      __rehash_policy() const
318
      { return _M_rehash_policy; }
319
 
320
      void
321
      __rehash_policy(const _RehashPolicy&);
322
 
323
      // Lookup.
324
      iterator
325
      find(const key_type& __k);
326
 
327
      const_iterator
328
      find(const key_type& __k) const;
329
 
330
      size_type
331
      count(const key_type& __k) const;
332
 
333
      std::pair<iterator, iterator>
334
      equal_range(const key_type& __k);
335
 
336
      std::pair<const_iterator, const_iterator>
337
      equal_range(const key_type& __k) const;
338
 
339
    private:                    // Find, insert and erase helper functions
340
      // ??? This dispatching is a workaround for the fact that we don't
341
      // have partial specialization of member templates; it would be
342
      // better to just specialize insert on __unique_keys.  There may be a
343
      // cleaner workaround.
344
      typedef typename __gnu_cxx::__conditional_type<__unique_keys,
345
                            std::pair<iterator, bool>, iterator>::__type
346
        _Insert_Return_Type;
347
 
348
      typedef typename __gnu_cxx::__conditional_type<__unique_keys,
349
                                          std::_Select1st<_Insert_Return_Type>,
350
                                          std::_Identity<_Insert_Return_Type>
351
                                   >::__type
352
        _Insert_Conv_Type;
353
 
354
      _Node*
355
      _M_find_node(_Node*, const key_type&,
356
                   typename _Hashtable::_Hash_code_type) const;
357
 
358
      iterator
359
      _M_insert_bucket(const value_type&, size_type,
360
                       typename _Hashtable::_Hash_code_type);
361
 
362
      std::pair<iterator, bool>
363
      _M_insert(const value_type&, std::tr1::true_type);
364
 
365
      iterator
366
      _M_insert(const value_type&, std::tr1::false_type);
367
 
368
      void
369
      _M_erase_node(_Node*, _Node**);
370
 
371
    public:
372
      // Insert and erase
373
      _Insert_Return_Type
374
      insert(const value_type& __v)
375
      { return _M_insert(__v, std::tr1::integral_constant<bool,
376
                         __unique_keys>()); }
377
 
378
      iterator
379
      insert(iterator, const value_type& __v)
380
      { return iterator(_Insert_Conv_Type()(this->insert(__v))); }
381
 
382
      const_iterator
383
      insert(const_iterator, const value_type& __v)
384
      { return const_iterator(_Insert_Conv_Type()(this->insert(__v))); }
385
 
386
      template<typename _InputIterator>
387
        void
388
        insert(_InputIterator __first, _InputIterator __last);
389
 
390
      iterator
391
      erase(iterator);
392
 
393
      const_iterator
394
      erase(const_iterator);
395
 
396
      size_type
397
      erase(const key_type&);
398
 
399
      iterator
400
      erase(iterator, iterator);
401
 
402
      const_iterator
403
      erase(const_iterator, const_iterator);
404
 
405
      void
406
      clear();
407
 
408
      // Set number of buckets to be appropriate for container of n element.
409
      void rehash(size_type __n);
410
 
411
    private:
412
      // Unconditionally change size of bucket array to n.
413
      void _M_rehash(size_type __n);
414
    };
415
 
416
 
417
  // Definitions of class template _Hashtable's out-of-line member functions.
418
  template<typename _Key, typename _Value,
419
           typename _Allocator, typename _ExtractKey, typename _Equal,
420
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
421
           bool __chc, bool __cit, bool __uk>
422
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
423
                        _H1, _H2, _Hash, _RehashPolicy,
424
                        __chc, __cit, __uk>::_Node*
425
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
426
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
427
    _M_allocate_node(const value_type& __v)
428
    {
429
      _Node* __n = _M_node_allocator.allocate(1);
430
      __try
431
        {
432
          _M_get_Value_allocator().construct(&__n->_M_v, __v);
433
          __n->_M_next = 0;
434
          return __n;
435
        }
436
      __catch(...)
437
        {
438
          _M_node_allocator.deallocate(__n, 1);
439
          __throw_exception_again;
440
        }
441
    }
442
 
443
  template<typename _Key, typename _Value,
444
           typename _Allocator, typename _ExtractKey, typename _Equal,
445
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
446
           bool __chc, bool __cit, bool __uk>
447
    void
448
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
449
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
450
    _M_deallocate_node(_Node* __n)
451
    {
452
      _M_get_Value_allocator().destroy(&__n->_M_v);
453
      _M_node_allocator.deallocate(__n, 1);
454
    }
455
 
456
  template<typename _Key, typename _Value,
457
           typename _Allocator, typename _ExtractKey, typename _Equal,
458
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
459
           bool __chc, bool __cit, bool __uk>
460
    void
461
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
462
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
463
    _M_deallocate_nodes(_Node** __array, size_type __n)
464
    {
465
      for (size_type __i = 0; __i < __n; ++__i)
466
        {
467
          _Node* __p = __array[__i];
468
          while (__p)
469
            {
470
              _Node* __tmp = __p;
471
              __p = __p->_M_next;
472
              _M_deallocate_node(__tmp);
473
            }
474
          __array[__i] = 0;
475
        }
476
    }
477
 
478
  template<typename _Key, typename _Value,
479
           typename _Allocator, typename _ExtractKey, typename _Equal,
480
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
481
           bool __chc, bool __cit, bool __uk>
482
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
483
                        _H1, _H2, _Hash, _RehashPolicy,
484
                        __chc, __cit, __uk>::_Node**
485
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
486
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
487
    _M_allocate_buckets(size_type __n)
488
    {
489
      _Bucket_allocator_type __alloc(_M_node_allocator);
490
 
491
      // We allocate one extra bucket to hold a sentinel, an arbitrary
492
      // non-null pointer.  Iterator increment relies on this.
493
      _Node** __p = __alloc.allocate(__n + 1);
494
      std::fill(__p, __p + __n, (_Node*) 0);
495
      __p[__n] = reinterpret_cast<_Node*>(0x1000);
496
      return __p;
497
    }
498
 
499
  template<typename _Key, typename _Value,
500
           typename _Allocator, typename _ExtractKey, typename _Equal,
501
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
502
           bool __chc, bool __cit, bool __uk>
503
    void
504
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
505
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
506
    _M_deallocate_buckets(_Node** __p, size_type __n)
507
    {
508
      _Bucket_allocator_type __alloc(_M_node_allocator);
509
      __alloc.deallocate(__p, __n + 1);
510
    }
511
 
512
  template<typename _Key, typename _Value,
513
           typename _Allocator, typename _ExtractKey, typename _Equal,
514
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
515
           bool __chc, bool __cit, bool __uk>
516
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
517
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
518
    _Hashtable(size_type __bucket_hint,
519
               const _H1& __h1, const _H2& __h2, const _Hash& __h,
520
               const _Equal& __eq, const _ExtractKey& __exk,
521
               const allocator_type& __a)
522
    : __detail::_Rehash_base<_RehashPolicy, _Hashtable>(),
523
      __detail::_Hash_code_base<_Key, _Value, _ExtractKey, _Equal,
524
                                _H1, _H2, _Hash, __chc>(__exk, __eq,
525
                                                        __h1, __h2, __h),
526
      __detail::_Map_base<_Key, _Value, _ExtractKey, __uk, _Hashtable>(),
527
      _M_node_allocator(__a),
528
      _M_bucket_count(0),
529
      _M_element_count(0),
530
      _M_rehash_policy()
531
    {
532
      _M_bucket_count = _M_rehash_policy._M_next_bkt(__bucket_hint);
533
      _M_buckets = _M_allocate_buckets(_M_bucket_count);
534
    }
535
 
536
  template<typename _Key, typename _Value,
537
           typename _Allocator, typename _ExtractKey, typename _Equal,
538
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
539
           bool __chc, bool __cit, bool __uk>
540
    template<typename _InputIterator>
541
      _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
542
                 _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
543
      _Hashtable(_InputIterator __f, _InputIterator __l,
544
                 size_type __bucket_hint,
545
                 const _H1& __h1, const _H2& __h2, const _Hash& __h,
546
                 const _Equal& __eq, const _ExtractKey& __exk,
547
                 const allocator_type& __a)
548
      : __detail::_Rehash_base<_RehashPolicy, _Hashtable>(),
549
        __detail::_Hash_code_base<_Key, _Value, _ExtractKey, _Equal,
550
                                  _H1, _H2, _Hash, __chc>(__exk, __eq,
551
                                                          __h1, __h2, __h),
552
        __detail::_Map_base<_Key, _Value, _ExtractKey, __uk, _Hashtable>(),
553
        _M_node_allocator(__a),
554
        _M_bucket_count(0),
555
        _M_element_count(0),
556
        _M_rehash_policy()
557
      {
558
        _M_bucket_count = std::max(_M_rehash_policy._M_next_bkt(__bucket_hint),
559
                                   _M_rehash_policy.
560
                                   _M_bkt_for_elements(__detail::
561
                                                       __distance_fw(__f,
562
                                                                     __l)));
563
        _M_buckets = _M_allocate_buckets(_M_bucket_count);
564
        __try
565
          {
566
            for (; __f != __l; ++__f)
567
              this->insert(*__f);
568
          }
569
        __catch(...)
570
          {
571
            clear();
572
            _M_deallocate_buckets(_M_buckets, _M_bucket_count);
573
            __throw_exception_again;
574
          }
575
      }
576
 
577
  template<typename _Key, typename _Value,
578
           typename _Allocator, typename _ExtractKey, typename _Equal,
579
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
580
           bool __chc, bool __cit, bool __uk>
581
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
582
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
583
    _Hashtable(const _Hashtable& __ht)
584
    : __detail::_Rehash_base<_RehashPolicy, _Hashtable>(__ht),
585
      __detail::_Hash_code_base<_Key, _Value, _ExtractKey, _Equal,
586
                                _H1, _H2, _Hash, __chc>(__ht),
587
      __detail::_Map_base<_Key, _Value, _ExtractKey, __uk, _Hashtable>(__ht),
588
      _M_node_allocator(__ht._M_node_allocator),
589
      _M_bucket_count(__ht._M_bucket_count),
590
      _M_element_count(__ht._M_element_count),
591
      _M_rehash_policy(__ht._M_rehash_policy)
592
    {
593
      _M_buckets = _M_allocate_buckets(_M_bucket_count);
594
      __try
595
        {
596
          for (size_type __i = 0; __i < __ht._M_bucket_count; ++__i)
597
            {
598
              _Node* __n = __ht._M_buckets[__i];
599
              _Node** __tail = _M_buckets + __i;
600
              while (__n)
601
                {
602
                  *__tail = _M_allocate_node(__n->_M_v);
603
                  this->_M_copy_code(*__tail, __n);
604
                  __tail = &((*__tail)->_M_next);
605
                  __n = __n->_M_next;
606
                }
607
            }
608
        }
609
      __catch(...)
610
        {
611
          clear();
612
          _M_deallocate_buckets(_M_buckets, _M_bucket_count);
613
          __throw_exception_again;
614
        }
615
    }
616
 
617
  template<typename _Key, typename _Value,
618
           typename _Allocator, typename _ExtractKey, typename _Equal,
619
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
620
           bool __chc, bool __cit, bool __uk>
621
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
622
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>&
623
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
624
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
625
    operator=(const _Hashtable& __ht)
626
    {
627
      _Hashtable __tmp(__ht);
628
      this->swap(__tmp);
629
      return *this;
630
    }
631
 
632
  template<typename _Key, typename _Value,
633
           typename _Allocator, typename _ExtractKey, typename _Equal,
634
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
635
           bool __chc, bool __cit, bool __uk>
636
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
637
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
638
    ~_Hashtable()
639
    {
640
      clear();
641
      _M_deallocate_buckets(_M_buckets, _M_bucket_count);
642
    }
643
 
644
  template<typename _Key, typename _Value,
645
           typename _Allocator, typename _ExtractKey, typename _Equal,
646
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
647
           bool __chc, bool __cit, bool __uk>
648
    void
649
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
650
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
651
    swap(_Hashtable& __x)
652
    {
653
      // The only base class with member variables is hash_code_base.  We
654
      // define _Hash_code_base::_M_swap because different specializations
655
      // have different members.
656
      __detail::_Hash_code_base<_Key, _Value, _ExtractKey, _Equal,
657
        _H1, _H2, _Hash, __chc>::_M_swap(__x);
658
 
659
      // _GLIBCXX_RESOLVE_LIB_DEFECTS
660
      // 431. Swapping containers with unequal allocators.
661
      std::__alloc_swap<_Node_allocator_type>::_S_do_it(_M_node_allocator,
662
                                                        __x._M_node_allocator);
663
 
664
      std::swap(_M_rehash_policy, __x._M_rehash_policy);
665
      std::swap(_M_buckets, __x._M_buckets);
666
      std::swap(_M_bucket_count, __x._M_bucket_count);
667
      std::swap(_M_element_count, __x._M_element_count);
668
    }
669
 
670
  template<typename _Key, typename _Value,
671
           typename _Allocator, typename _ExtractKey, typename _Equal,
672
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
673
           bool __chc, bool __cit, bool __uk>
674
    void
675
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
676
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
677
    __rehash_policy(const _RehashPolicy& __pol)
678
    {
679
      _M_rehash_policy = __pol;
680
      size_type __n_bkt = __pol._M_bkt_for_elements(_M_element_count);
681
      if (__n_bkt > _M_bucket_count)
682
        _M_rehash(__n_bkt);
683
    }
684
 
685
  template<typename _Key, typename _Value,
686
           typename _Allocator, typename _ExtractKey, typename _Equal,
687
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
688
           bool __chc, bool __cit, bool __uk>
689
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
690
                        _H1, _H2, _Hash, _RehashPolicy,
691
                        __chc, __cit, __uk>::iterator
692
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
693
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
694
    find(const key_type& __k)
695
    {
696
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
697
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
698
      _Node* __p = _M_find_node(_M_buckets[__n], __k, __code);
699
      return __p ? iterator(__p, _M_buckets + __n) : this->end();
700
    }
701
 
702
  template<typename _Key, typename _Value,
703
           typename _Allocator, typename _ExtractKey, typename _Equal,
704
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
705
           bool __chc, bool __cit, bool __uk>
706
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
707
                        _H1, _H2, _Hash, _RehashPolicy,
708
                        __chc, __cit, __uk>::const_iterator
709
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
710
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
711
    find(const key_type& __k) const
712
    {
713
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
714
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
715
      _Node* __p = _M_find_node(_M_buckets[__n], __k, __code);
716
      return __p ? const_iterator(__p, _M_buckets + __n) : this->end();
717
    }
718
 
719
  template<typename _Key, typename _Value,
720
           typename _Allocator, typename _ExtractKey, typename _Equal,
721
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
722
           bool __chc, bool __cit, bool __uk>
723
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
724
                        _H1, _H2, _Hash, _RehashPolicy,
725
                        __chc, __cit, __uk>::size_type
726
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
727
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
728
    count(const key_type& __k) const
729
    {
730
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
731
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
732
      std::size_t __result = 0;
733
      for (_Node* __p = _M_buckets[__n]; __p; __p = __p->_M_next)
734
        if (this->_M_compare(__k, __code, __p))
735
          ++__result;
736
      return __result;
737
    }
738
 
739
  template<typename _Key, typename _Value,
740
           typename _Allocator, typename _ExtractKey, typename _Equal,
741
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
742
           bool __chc, bool __cit, bool __uk>
743
    std::pair<typename _Hashtable<_Key, _Value, _Allocator,
744
                                  _ExtractKey, _Equal, _H1,
745
                                  _H2, _Hash, _RehashPolicy,
746
                                  __chc, __cit, __uk>::iterator,
747
              typename _Hashtable<_Key, _Value, _Allocator,
748
                                  _ExtractKey, _Equal, _H1,
749
                                  _H2, _Hash, _RehashPolicy,
750
                                  __chc, __cit, __uk>::iterator>
751
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
752
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
753
    equal_range(const key_type& __k)
754
    {
755
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
756
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
757
      _Node** __head = _M_buckets + __n;
758
      _Node* __p = _M_find_node(*__head, __k, __code);
759
 
760
      if (__p)
761
        {
762
          _Node* __p1 = __p->_M_next;
763
          for (; __p1; __p1 = __p1->_M_next)
764
            if (!this->_M_compare(__k, __code, __p1))
765
              break;
766
 
767
          iterator __first(__p, __head);
768
          iterator __last(__p1, __head);
769
          if (!__p1)
770
            __last._M_incr_bucket();
771
          return std::make_pair(__first, __last);
772
        }
773
      else
774
        return std::make_pair(this->end(), this->end());
775
    }
776
 
777
  template<typename _Key, typename _Value,
778
           typename _Allocator, typename _ExtractKey, typename _Equal,
779
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
780
           bool __chc, bool __cit, bool __uk>
781
    std::pair<typename _Hashtable<_Key, _Value, _Allocator,
782
                                  _ExtractKey, _Equal, _H1,
783
                                  _H2, _Hash, _RehashPolicy,
784
                                  __chc, __cit, __uk>::const_iterator,
785
              typename _Hashtable<_Key, _Value, _Allocator,
786
                                  _ExtractKey, _Equal, _H1,
787
                                  _H2, _Hash, _RehashPolicy,
788
                                  __chc, __cit, __uk>::const_iterator>
789
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
790
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
791
    equal_range(const key_type& __k) const
792
    {
793
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
794
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
795
      _Node** __head = _M_buckets + __n;
796
      _Node* __p = _M_find_node(*__head, __k, __code);
797
 
798
      if (__p)
799
        {
800
          _Node* __p1 = __p->_M_next;
801
          for (; __p1; __p1 = __p1->_M_next)
802
            if (!this->_M_compare(__k, __code, __p1))
803
              break;
804
 
805
          const_iterator __first(__p, __head);
806
          const_iterator __last(__p1, __head);
807
          if (!__p1)
808
            __last._M_incr_bucket();
809
          return std::make_pair(__first, __last);
810
        }
811
      else
812
        return std::make_pair(this->end(), this->end());
813
    }
814
 
815
  // Find the node whose key compares equal to k, beginning the search
816
  // at p (usually the head of a bucket).  Return zero if no node is found.
817
  template<typename _Key, typename _Value,
818
           typename _Allocator, typename _ExtractKey, typename _Equal,
819
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
820
           bool __chc, bool __cit, bool __uk>
821
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey,
822
                        _Equal, _H1, _H2, _Hash, _RehashPolicy,
823
                        __chc, __cit, __uk>::_Node*
824
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
825
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
826
    _M_find_node(_Node* __p, const key_type& __k,
827
                typename _Hashtable::_Hash_code_type __code) const
828
    {
829
      for (; __p; __p = __p->_M_next)
830
        if (this->_M_compare(__k, __code, __p))
831
          return __p;
832
      return 0;
833
    }
834
 
835
  // Insert v in bucket n (assumes no element with its key already present).
836
  template<typename _Key, typename _Value,
837
           typename _Allocator, typename _ExtractKey, typename _Equal,
838
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
839
           bool __chc, bool __cit, bool __uk>
840
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
841
                        _H1, _H2, _Hash, _RehashPolicy,
842
                        __chc, __cit, __uk>::iterator
843
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
844
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
845
    _M_insert_bucket(const value_type& __v, size_type __n,
846
                    typename _Hashtable::_Hash_code_type __code)
847
    {
848
      std::pair<bool, std::size_t> __do_rehash
849
        = _M_rehash_policy._M_need_rehash(_M_bucket_count,
850
                                          _M_element_count, 1);
851
 
852
      // Allocate the new node before doing the rehash so that we don't
853
      // do a rehash if the allocation throws.
854
      _Node* __new_node = _M_allocate_node(__v);
855
 
856
      __try
857
        {
858
          if (__do_rehash.first)
859
            {
860
              const key_type& __k = this->_M_extract(__v);
861
              __n = this->_M_bucket_index(__k, __code, __do_rehash.second);
862
              _M_rehash(__do_rehash.second);
863
            }
864
 
865
          __new_node->_M_next = _M_buckets[__n];
866
          this->_M_store_code(__new_node, __code);
867
          _M_buckets[__n] = __new_node;
868
          ++_M_element_count;
869
          return iterator(__new_node, _M_buckets + __n);
870
        }
871
      __catch(...)
872
        {
873
          _M_deallocate_node(__new_node);
874
          __throw_exception_again;
875
        }
876
    }
877
 
878
  // Insert v if no element with its key is already present.
879
  template<typename _Key, typename _Value,
880
           typename _Allocator, typename _ExtractKey, typename _Equal,
881
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
882
           bool __chc, bool __cit, bool __uk>
883
    std::pair<typename _Hashtable<_Key, _Value, _Allocator,
884
                                  _ExtractKey, _Equal, _H1,
885
                                  _H2, _Hash, _RehashPolicy,
886
                                  __chc, __cit, __uk>::iterator, bool>
887
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
888
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
889
  _M_insert(const value_type& __v, std::tr1::true_type)
890
    {
891
      const key_type& __k = this->_M_extract(__v);
892
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
893
      size_type __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
894
 
895
      if (_Node* __p = _M_find_node(_M_buckets[__n], __k, __code))
896
        return std::make_pair(iterator(__p, _M_buckets + __n), false);
897
      return std::make_pair(_M_insert_bucket(__v, __n, __code), true);
898
    }
899
 
900
  // Insert v unconditionally.
901
  template<typename _Key, typename _Value,
902
           typename _Allocator, typename _ExtractKey, typename _Equal,
903
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
904
           bool __chc, bool __cit, bool __uk>
905
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
906
                        _H1, _H2, _Hash, _RehashPolicy,
907
                        __chc, __cit, __uk>::iterator
908
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
909
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
910
    _M_insert(const value_type& __v, std::tr1::false_type)
911
    {
912
      std::pair<bool, std::size_t> __do_rehash
913
        = _M_rehash_policy._M_need_rehash(_M_bucket_count,
914
                                          _M_element_count, 1);
915
      if (__do_rehash.first)
916
        _M_rehash(__do_rehash.second);
917
 
918
      const key_type& __k = this->_M_extract(__v);
919
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
920
      size_type __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
921
 
922
      // First find the node, avoid leaking new_node if compare throws.
923
      _Node* __prev = _M_find_node(_M_buckets[__n], __k, __code);
924
      _Node* __new_node = _M_allocate_node(__v);
925
 
926
      if (__prev)
927
        {
928
          __new_node->_M_next = __prev->_M_next;
929
          __prev->_M_next = __new_node;
930
        }
931
      else
932
        {
933
          __new_node->_M_next = _M_buckets[__n];
934
          _M_buckets[__n] = __new_node;
935
        }
936
      this->_M_store_code(__new_node, __code);
937
 
938
      ++_M_element_count;
939
      return iterator(__new_node, _M_buckets + __n);
940
    }
941
 
942
  // For erase(iterator) and erase(const_iterator).
943
  template<typename _Key, typename _Value,
944
           typename _Allocator, typename _ExtractKey, typename _Equal,
945
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
946
           bool __chc, bool __cit, bool __uk>
947
    void
948
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
949
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
950
    _M_erase_node(_Node* __p, _Node** __b)
951
    {
952
      _Node* __cur = *__b;
953
      if (__cur == __p)
954
        *__b = __cur->_M_next;
955
      else
956
        {
957
          _Node* __next = __cur->_M_next;
958
          while (__next != __p)
959
            {
960
              __cur = __next;
961
              __next = __cur->_M_next;
962
            }
963
          __cur->_M_next = __next->_M_next;
964
        }
965
 
966
      _M_deallocate_node(__p);
967
      --_M_element_count;
968
    }
969
 
970
  template<typename _Key, typename _Value,
971
           typename _Allocator, typename _ExtractKey, typename _Equal,
972
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
973
           bool __chc, bool __cit, bool __uk>
974
    template<typename _InputIterator>
975
      void
976
      _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
977
                 _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
978
      insert(_InputIterator __first, _InputIterator __last)
979
      {
980
        size_type __n_elt = __detail::__distance_fw(__first, __last);
981
        std::pair<bool, std::size_t> __do_rehash
982
          = _M_rehash_policy._M_need_rehash(_M_bucket_count,
983
                                            _M_element_count, __n_elt);
984
        if (__do_rehash.first)
985
          _M_rehash(__do_rehash.second);
986
 
987
        for (; __first != __last; ++__first)
988
          this->insert(*__first);
989
      }
990
 
991
  template<typename _Key, typename _Value,
992
           typename _Allocator, typename _ExtractKey, typename _Equal,
993
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
994
           bool __chc, bool __cit, bool __uk>
995
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
996
                        _H1, _H2, _Hash, _RehashPolicy,
997
                        __chc, __cit, __uk>::iterator
998
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
999
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1000
    erase(iterator __it)
1001
    {
1002
      iterator __result = __it;
1003
      ++__result;
1004
      _M_erase_node(__it._M_cur_node, __it._M_cur_bucket);
1005
      return __result;
1006
    }
1007
 
1008
  template<typename _Key, typename _Value,
1009
           typename _Allocator, typename _ExtractKey, typename _Equal,
1010
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1011
           bool __chc, bool __cit, bool __uk>
1012
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1013
                        _H1, _H2, _Hash, _RehashPolicy,
1014
                        __chc, __cit, __uk>::const_iterator
1015
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1016
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1017
    erase(const_iterator __it)
1018
    {
1019
      const_iterator __result = __it;
1020
      ++__result;
1021
      _M_erase_node(__it._M_cur_node, __it._M_cur_bucket);
1022
      return __result;
1023
    }
1024
 
1025
  template<typename _Key, typename _Value,
1026
           typename _Allocator, typename _ExtractKey, typename _Equal,
1027
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1028
           bool __chc, bool __cit, bool __uk>
1029
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1030
                        _H1, _H2, _Hash, _RehashPolicy,
1031
                        __chc, __cit, __uk>::size_type
1032
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1033
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1034
    erase(const key_type& __k)
1035
    {
1036
      typename _Hashtable::_Hash_code_type __code = this->_M_hash_code(__k);
1037
      std::size_t __n = this->_M_bucket_index(__k, __code, _M_bucket_count);
1038
      size_type __result = 0;
1039
 
1040
      _Node** __slot = _M_buckets + __n;
1041
      while (*__slot && !this->_M_compare(__k, __code, *__slot))
1042
        __slot = &((*__slot)->_M_next);
1043
 
1044
      _Node** __saved_slot = 0;
1045
      while (*__slot && this->_M_compare(__k, __code, *__slot))
1046
        {
1047
          // _GLIBCXX_RESOLVE_LIB_DEFECTS
1048
          // 526. Is it undefined if a function in the standard changes
1049
          // in parameters?
1050
          if (&this->_M_extract((*__slot)->_M_v) != &__k)
1051
            {
1052
              _Node* __p = *__slot;
1053
              *__slot = __p->_M_next;
1054
              _M_deallocate_node(__p);
1055
              --_M_element_count;
1056
              ++__result;
1057
            }
1058
          else
1059
            {
1060
              __saved_slot = __slot;
1061
              __slot = &((*__slot)->_M_next);
1062
            }
1063
        }
1064
 
1065
      if (__saved_slot)
1066
        {
1067
          _Node* __p = *__saved_slot;
1068
          *__saved_slot = __p->_M_next;
1069
          _M_deallocate_node(__p);
1070
          --_M_element_count;
1071
          ++__result;
1072
        }
1073
 
1074
      return __result;
1075
    }
1076
 
1077
  // ??? This could be optimized by taking advantage of the bucket
1078
  // structure, but it's not clear that it's worth doing.  It probably
1079
  // wouldn't even be an optimization unless the load factor is large.
1080
  template<typename _Key, typename _Value,
1081
           typename _Allocator, typename _ExtractKey, typename _Equal,
1082
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1083
           bool __chc, bool __cit, bool __uk>
1084
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1085
                        _H1, _H2, _Hash, _RehashPolicy,
1086
                        __chc, __cit, __uk>::iterator
1087
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1088
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1089
    erase(iterator __first, iterator __last)
1090
    {
1091
      while (__first != __last)
1092
        __first = this->erase(__first);
1093
      return __last;
1094
    }
1095
 
1096
  template<typename _Key, typename _Value,
1097
           typename _Allocator, typename _ExtractKey, typename _Equal,
1098
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1099
           bool __chc, bool __cit, bool __uk>
1100
    typename _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1101
                        _H1, _H2, _Hash, _RehashPolicy,
1102
                        __chc, __cit, __uk>::const_iterator
1103
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1104
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1105
    erase(const_iterator __first, const_iterator __last)
1106
    {
1107
      while (__first != __last)
1108
        __first = this->erase(__first);
1109
      return __last;
1110
    }
1111
 
1112
  template<typename _Key, typename _Value,
1113
           typename _Allocator, typename _ExtractKey, typename _Equal,
1114
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1115
           bool __chc, bool __cit, bool __uk>
1116
    void
1117
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1118
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1119
    clear()
1120
    {
1121
      _M_deallocate_nodes(_M_buckets, _M_bucket_count);
1122
      _M_element_count = 0;
1123
    }
1124
 
1125
  template<typename _Key, typename _Value,
1126
           typename _Allocator, typename _ExtractKey, typename _Equal,
1127
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1128
           bool __chc, bool __cit, bool __uk>
1129
    void
1130
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1131
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1132
    rehash(size_type __n)
1133
    {
1134
      _M_rehash(std::max(_M_rehash_policy._M_next_bkt(__n),
1135
                         _M_rehash_policy._M_bkt_for_elements(_M_element_count
1136
                                                              + 1)));
1137
    }
1138
 
1139
  template<typename _Key, typename _Value,
1140
           typename _Allocator, typename _ExtractKey, typename _Equal,
1141
           typename _H1, typename _H2, typename _Hash, typename _RehashPolicy,
1142
           bool __chc, bool __cit, bool __uk>
1143
    void
1144
    _Hashtable<_Key, _Value, _Allocator, _ExtractKey, _Equal,
1145
               _H1, _H2, _Hash, _RehashPolicy, __chc, __cit, __uk>::
1146
    _M_rehash(size_type __n)
1147
    {
1148
      _Node** __new_array = _M_allocate_buckets(__n);
1149
      __try
1150
        {
1151
          for (size_type __i = 0; __i < _M_bucket_count; ++__i)
1152
            while (_Node* __p = _M_buckets[__i])
1153
              {
1154
                std::size_t __new_index = this->_M_bucket_index(__p, __n);
1155
                _M_buckets[__i] = __p->_M_next;
1156
                __p->_M_next = __new_array[__new_index];
1157
                __new_array[__new_index] = __p;
1158
              }
1159
          _M_deallocate_buckets(_M_buckets, _M_bucket_count);
1160
          _M_bucket_count = __n;
1161
          _M_buckets = __new_array;
1162
        }
1163
      __catch(...)
1164
        {
1165
          // A failure here means that a hash function threw an exception.
1166
          // We can't restore the previous state without calling the hash
1167
          // function again, so the only sensible recovery is to delete
1168
          // everything.
1169
          _M_deallocate_nodes(__new_array, __n);
1170
          _M_deallocate_buckets(__new_array, __n);
1171
          _M_deallocate_nodes(_M_buckets, _M_bucket_count);
1172
          _M_element_count = 0;
1173
          __throw_exception_again;
1174
        }
1175
    }
1176
 
1177
_GLIBCXX_END_NAMESPACE_VERSION
1178
} // namespace tr1
1179
} // namespace std
1180
 
1181
#endif // _GLIBCXX_TR1_HASHTABLE_H

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.