libstdc++
stl_tree.h
Go to the documentation of this file.
1// RB tree implementation -*- C++ -*-
2
3// Copyright (C) 2001-2026 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/*
26 *
27 * Copyright (c) 1996,1997
28 * Silicon Graphics Computer Systems, Inc.
29 *
30 * Permission to use, copy, modify, distribute and sell this software
31 * and its documentation for any purpose is hereby granted without fee,
32 * provided that the above copyright notice appear in all copies and
33 * that both that copyright notice and this permission notice appear
34 * in supporting documentation. Silicon Graphics makes no
35 * representations about the suitability of this software for any
36 * purpose. It is provided "as is" without express or implied warranty.
37 *
38 *
39 * Copyright (c) 1994
40 * Hewlett-Packard Company
41 *
42 * Permission to use, copy, modify, distribute and sell this software
43 * and its documentation for any purpose is hereby granted without fee,
44 * provided that the above copyright notice appear in all copies and
45 * that both that copyright notice and this permission notice appear
46 * in supporting documentation. Hewlett-Packard Company makes no
47 * representations about the suitability of this software for any
48 * purpose. It is provided "as is" without express or implied warranty.
49 *
50 *
51 */
52
53/** @file bits/stl_tree.h
54 * This is an internal header file, included by other library headers.
55 * Do not attempt to use it directly. @headername{map,set}
56 */
57
58#ifndef _STL_TREE_H
59#define _STL_TREE_H 1
60
61#ifdef _GLIBCXX_SYSHDR
62#pragma GCC system_header
63#endif
64
65#include <bits/stl_algobase.h>
66#include <bits/allocator.h>
67#include <bits/stl_function.h>
69#include <bits/ptr_traits.h>
70#include <ext/alloc_traits.h>
71#if __cplusplus >= 201103L
72# include <ext/aligned_buffer.h>
73#endif
74#ifdef __glibcxx_node_extract // >= C++17
75# include <bits/node_handle.h>
76#endif
77
78#if __cplusplus < 201103L
79# undef _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
80# define _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE 0
81#elif ! defined _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
82# define _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE 1
83#endif
84
85namespace std _GLIBCXX_VISIBILITY(default)
86{
87_GLIBCXX_BEGIN_NAMESPACE_VERSION
88
89 // Red-black tree class, designed for use in implementing STL
90 // associative containers (set, multiset, map, and multimap). The
91 // insertion and deletion algorithms are based on those in Cormen,
92 // Leiserson, and Rivest, Introduction to Algorithms (MIT Press,
93 // 1990), except that
94 //
95 // (1) the header cell is maintained with links not only to the root
96 // but also to the leftmost node of the tree, to enable constant
97 // time begin(), and to the rightmost node of the tree, to enable
98 // linear time performance when used with the generic set algorithms
99 // (set_union, etc.)
100 //
101 // (2) when a node being deleted has two children its successor node
102 // is relinked into its place, rather than copied, so that the only
103 // iterators invalidated are those referring to the deleted node.
104
105 enum _Rb_tree_color { _S_red = false, _S_black = true };
106
107 struct _Rb_tree_node_base
108 {
109 typedef _Rb_tree_node_base* _Base_ptr;
110
111 _Rb_tree_color _M_color;
112 _Base_ptr _M_parent;
113 _Base_ptr _M_left;
114 _Base_ptr _M_right;
115
116 static _Base_ptr
117 _S_minimum(_Base_ptr __x) _GLIBCXX_NOEXCEPT
118 {
119 while (__x->_M_left != 0) __x = __x->_M_left;
120 return __x;
121 }
122
123 static _Base_ptr
124 _S_maximum(_Base_ptr __x) _GLIBCXX_NOEXCEPT
125 {
126 while (__x->_M_right != 0) __x = __x->_M_right;
127 return __x;
128 }
129
130 // This is not const-correct, but it's only used in a const access path
131 // by std::_Rb_tree::_M_end() where the pointer is used to initialize a
132 // const_iterator and so constness is restored.
133 _Base_ptr
134 _M_base_ptr() const _GLIBCXX_NOEXCEPT
135 { return const_cast<_Rb_tree_node_base*>(this); }
136 };
137
138 // Helper type offering value initialization guarantee on the compare functor.
139 template<typename _Key_compare>
140 struct _Rb_tree_key_compare
141 {
142 _Key_compare _M_key_compare;
143
144 _Rb_tree_key_compare()
145 _GLIBCXX_NOEXCEPT_IF(
146 is_nothrow_default_constructible<_Key_compare>::value)
147 : _M_key_compare()
148 { }
149
150 _Rb_tree_key_compare(const _Key_compare& __comp)
151 : _M_key_compare(__comp)
152 { }
153
154#if __cplusplus >= 201103L
155 // Copy constructor added for consistency with C++98 mode.
156 _Rb_tree_key_compare(const _Rb_tree_key_compare&) = default;
157
158 _Rb_tree_key_compare(_Rb_tree_key_compare&& __x)
159 noexcept(is_nothrow_copy_constructible<_Key_compare>::value)
160 : _M_key_compare(__x._M_key_compare)
161 { }
162#endif
163 };
164
165 // Helper type to manage default initialization of node count and header.
166 struct _Rb_tree_header
167 {
168 _Rb_tree_node_base _M_header;
169 size_t _M_node_count; // Keeps track of size of tree.
170
171 _Rb_tree_header() _GLIBCXX_NOEXCEPT
172 {
173 _M_header._M_color = _S_red;
174 _M_reset();
175 }
176
177#if __cplusplus >= 201103L
178 _Rb_tree_header(_Rb_tree_header&& __x) noexcept
179 {
180 if (__x._M_header._M_parent != nullptr)
181 _M_move_data(__x);
182 else
183 {
184 _M_header._M_color = _S_red;
185 _M_reset();
186 }
187 }
188#endif
189
190 void
191 _M_move_data(_Rb_tree_header& __from)
192 {
193 _M_header._M_color = __from._M_header._M_color;
194 _M_header._M_parent = __from._M_header._M_parent;
195 _M_header._M_left = __from._M_header._M_left;
196 _M_header._M_right = __from._M_header._M_right;
197 _M_header._M_parent->_M_parent = &_M_header;
198 _M_node_count = __from._M_node_count;
199
200 __from._M_reset();
201 }
202
203 void
204 _M_reset()
205 {
206 _M_header._M_parent = 0;
207 _M_header._M_left = &_M_header;
208 _M_header._M_right = &_M_header;
209 _M_node_count = 0;
210 }
211 };
212
213 template<typename _Val>
214 struct _Rb_tree_node : public _Rb_tree_node_base
215 {
216#if __cplusplus < 201103L
217 _Val _M_value_field;
218
219 _Val*
220 _M_valptr()
221 { return std::__addressof(_M_value_field); }
222
223 const _Val*
224 _M_valptr() const
225 { return std::__addressof(_M_value_field); }
226#else
227 __gnu_cxx::__aligned_membuf<_Val> _M_storage;
228
229 _Val*
230 _M_valptr()
231 { return _M_storage._M_ptr(); }
232
233 const _Val*
234 _M_valptr() const
235 { return _M_storage._M_ptr(); }
236#endif
237
238 _Rb_tree_node*
239 _M_node_ptr() _GLIBCXX_NOEXCEPT
240 { return this; }
241 };
242
243#if _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
244namespace __rb_tree
245{
246 template<typename _VoidPtr>
247 struct _Node_base
248 {
249 using _Base_ptr = __ptr_rebind<_VoidPtr, _Node_base>;
250
251 _Rb_tree_color _M_color;
252 _Base_ptr _M_parent;
253 _Base_ptr _M_left;
254 _Base_ptr _M_right;
255
256 static _Base_ptr
257 _S_minimum(_Base_ptr __x) _GLIBCXX_NOEXCEPT
258 {
259 while (__x->_M_left) __x = __x->_M_left;
260 return __x;
261 }
262
263 static _Base_ptr
264 _S_maximum(_Base_ptr __x) _GLIBCXX_NOEXCEPT
265 {
266 while (__x->_M_right) __x = __x->_M_right;
267 return __x;
268 }
269
270 // This is not const-correct, but it's only used in a const access path
271 // by std::_Rb_tree::_M_end() where the pointer is used to initialize a
272 // const_iterator and so constness is restored.
273 _Base_ptr
274 _M_base_ptr() const noexcept
275 {
276 return pointer_traits<_Base_ptr>::pointer_to
277 (*const_cast<_Node_base*>(this));
278 }
279 };
280
281 // Helper type to manage default initialization of node count and header.
282 template<typename _NodeBase>
283 struct _Header
284 {
285 private:
286 using _Base_ptr = typename _NodeBase::_Base_ptr;
287
288 public:
289 _NodeBase _M_header;
290 size_t _M_node_count; // Keeps track of size of tree.
291
292 _Header() noexcept
293 {
294 _M_header._M_color = _S_red;
295 _M_reset();
296 }
297
298 _Header(_Header&& __x) noexcept
299 {
300 if (__x._M_header._M_parent)
301 _M_move_data(__x);
302 else
303 {
304 _M_header._M_color = _S_red;
305 _M_reset();
306 }
307 }
308
309 void
310 _M_move_data(_Header& __from)
311 {
312 _M_header._M_color = __from._M_header._M_color;
313 _M_header._M_parent = __from._M_header._M_parent;
314 _M_header._M_left = __from._M_header._M_left;
315 _M_header._M_right = __from._M_header._M_right;
316 _M_header._M_parent->_M_parent = _M_header._M_base_ptr();
317 _M_node_count = __from._M_node_count;
318
319 __from._M_reset();
320 }
321
322 void
323 _M_reset()
324 {
325 _M_header._M_parent = nullptr;
326 _M_header._M_left = _M_header._M_right = _M_header._M_base_ptr();
327 _M_node_count = 0;
328 }
329 };
330
331 template<typename _ValPtr>
332 struct _Node : public __rb_tree::_Node_base<__ptr_rebind<_ValPtr, void>>
333 {
334 using value_type = typename pointer_traits<_ValPtr>::element_type;
335 using _Node_ptr = __ptr_rebind<_ValPtr, _Node>;
336
337 _Node() noexcept { }
338 ~_Node() { }
339 _Node(_Node&&) = delete;
340
341 union _Uninit_storage
342 {
343 _Uninit_storage() noexcept { }
344 ~_Uninit_storage() { }
345
346 value_type _M_data;
347 };
348 _Uninit_storage _M_u;
349
350 value_type*
351 _M_valptr()
352 { return std::addressof(_M_u._M_data); }
353
354 value_type const*
355 _M_valptr() const
356 { return std::addressof(_M_u._M_data); }
357
358 _Node_ptr
359 _M_node_ptr() noexcept
360 { return pointer_traits<_Node_ptr>::pointer_to(*this); }
361 };
362} // namespace __rb_tree
363#endif // _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
364
365 _GLIBCXX_PURE _Rb_tree_node_base*
366 _Rb_tree_increment(_Rb_tree_node_base* __x) throw ();
367
368 _GLIBCXX_PURE _Rb_tree_node_base*
369 _Rb_tree_decrement(_Rb_tree_node_base* __x) throw ();
370
371 template<typename _Tp>
372 struct _Rb_tree_iterator
373 {
374 typedef _Tp value_type;
375 typedef _Tp& reference;
376 typedef _Tp* pointer;
377
378 typedef bidirectional_iterator_tag iterator_category;
379 typedef ptrdiff_t difference_type;
380
381 typedef _Rb_tree_node_base::_Base_ptr _Base_ptr;
382 typedef _Rb_tree_node<_Tp>* _Node_ptr;
383
384 _Rb_tree_iterator() _GLIBCXX_NOEXCEPT
385 : _M_node() { }
386
387 explicit
388 _Rb_tree_iterator(_Base_ptr __x) _GLIBCXX_NOEXCEPT
389 : _M_node(__x) { }
390
391 reference
392 operator*() const _GLIBCXX_NOEXCEPT
393 { return *static_cast<_Node_ptr>(_M_node)->_M_valptr(); }
394
395 pointer
396 operator->() const _GLIBCXX_NOEXCEPT
397 { return static_cast<_Node_ptr>(_M_node)->_M_valptr(); }
398
399 _Rb_tree_iterator&
400 operator++() _GLIBCXX_NOEXCEPT
401 {
402 _M_node = _Rb_tree_increment(_M_node);
403 return *this;
404 }
405
406 _Rb_tree_iterator
407 operator++(int) _GLIBCXX_NOEXCEPT
408 {
409 _Rb_tree_iterator __tmp = *this;
410 _M_node = _Rb_tree_increment(_M_node);
411 return __tmp;
412 }
413
414 _Rb_tree_iterator&
415 operator--() _GLIBCXX_NOEXCEPT
416 {
417 _M_node = _Rb_tree_decrement(_M_node);
418 return *this;
419 }
420
421 _Rb_tree_iterator
422 operator--(int) _GLIBCXX_NOEXCEPT
423 {
424 _Rb_tree_iterator __tmp = *this;
425 _M_node = _Rb_tree_decrement(_M_node);
426 return __tmp;
427 }
428
429 friend bool
430 operator==(const _Rb_tree_iterator& __x,
431 const _Rb_tree_iterator& __y) _GLIBCXX_NOEXCEPT
432 { return __x._M_node == __y._M_node; }
433
434#if ! __cpp_lib_three_way_comparison
435 friend bool
436 operator!=(const _Rb_tree_iterator& __x,
437 const _Rb_tree_iterator& __y) _GLIBCXX_NOEXCEPT
438 { return __x._M_node != __y._M_node; }
439#endif
440
441 _Base_ptr _M_node;
442 };
443
444 template<typename _Tp>
445 struct _Rb_tree_const_iterator
446 {
447 typedef _Tp value_type;
448 typedef const _Tp& reference;
449 typedef const _Tp* pointer;
450
451 typedef _Rb_tree_iterator<_Tp> iterator;
452
453 typedef bidirectional_iterator_tag iterator_category;
454 typedef ptrdiff_t difference_type;
455
456 typedef _Rb_tree_node_base::_Base_ptr _Base_ptr;
457 typedef const _Rb_tree_node<_Tp>* _Node_ptr;
458
459 _Rb_tree_const_iterator() _GLIBCXX_NOEXCEPT
460 : _M_node() { }
461
462 explicit
463 _Rb_tree_const_iterator(_Base_ptr __x) _GLIBCXX_NOEXCEPT
464 : _M_node(__x) { }
465
466 _Rb_tree_const_iterator(const iterator& __it) _GLIBCXX_NOEXCEPT
467 : _M_node(__it._M_node) { }
468
469 reference
470 operator*() const _GLIBCXX_NOEXCEPT
471 { return *static_cast<_Node_ptr>(_M_node)->_M_valptr(); }
472
473 pointer
474 operator->() const _GLIBCXX_NOEXCEPT
475 { return static_cast<_Node_ptr>(_M_node)->_M_valptr(); }
476
477 _Rb_tree_const_iterator&
478 operator++() _GLIBCXX_NOEXCEPT
479 {
480 _M_node = _Rb_tree_increment(_M_node);
481 return *this;
482 }
483
484 _Rb_tree_const_iterator
485 operator++(int) _GLIBCXX_NOEXCEPT
486 {
487 _Rb_tree_const_iterator __tmp = *this;
488 _M_node = _Rb_tree_increment(_M_node);
489 return __tmp;
490 }
491
492 _Rb_tree_const_iterator&
493 operator--() _GLIBCXX_NOEXCEPT
494 {
495 _M_node = _Rb_tree_decrement(_M_node);
496 return *this;
497 }
498
499 _Rb_tree_const_iterator
500 operator--(int) _GLIBCXX_NOEXCEPT
501 {
502 _Rb_tree_const_iterator __tmp = *this;
503 _M_node = _Rb_tree_decrement(_M_node);
504 return __tmp;
505 }
506
507 friend bool
508 operator==(const _Rb_tree_const_iterator& __x,
509 const _Rb_tree_const_iterator& __y) _GLIBCXX_NOEXCEPT
510 { return __x._M_node == __y._M_node; }
511
512#if ! __cpp_lib_three_way_comparison
513 friend bool
514 operator!=(const _Rb_tree_const_iterator& __x,
515 const _Rb_tree_const_iterator& __y) _GLIBCXX_NOEXCEPT
516 { return __x._M_node != __y._M_node; }
517#endif
518
519 _Base_ptr _M_node;
520 };
521
522 __attribute__((__nonnull__))
523 void
524 _Rb_tree_insert_and_rebalance(const bool __insert_left,
525 _Rb_tree_node_base* __x,
526 _Rb_tree_node_base* __p,
527 _Rb_tree_node_base& __header) throw ();
528
529 __attribute__((__nonnull__,__returns_nonnull__))
530 _Rb_tree_node_base*
531 _Rb_tree_rebalance_for_erase(_Rb_tree_node_base* const __z,
532 _Rb_tree_node_base& __header) throw ();
533
534namespace __rb_tree
535{
536#if _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
537 template<bool _Const, typename _ValPtr>
538 struct _Iterator
539 {
540 template<typename _Tp>
541 using __maybe_const = __conditional_t<_Const, const _Tp, _Tp>;
542
543 using __ptr_traits = pointer_traits<_ValPtr>;
544 using value_type = typename __ptr_traits::element_type;
545 using reference = __maybe_const<value_type>&;
546 using pointer = __maybe_const<value_type>*;
547
548 using iterator_category = bidirectional_iterator_tag;
549 using difference_type = ptrdiff_t;
550
551 using _Node = __rb_tree::_Node<_ValPtr>;
552 using _Node_base = __rb_tree::_Node_base<__ptr_rebind<_ValPtr, void>>;
553 using _Base_ptr = typename _Node_base::_Base_ptr;
554
555 _Iterator() noexcept
556 : _M_node() { }
557
558 constexpr explicit
559 _Iterator(_Base_ptr __x) noexcept
560 : _M_node(__x) { }
561
562 _Iterator(const _Iterator&) = default;
563 _Iterator& operator=(const _Iterator&) = default;
564
565#ifdef __glibcxx_concepts
566 constexpr
567 _Iterator(const _Iterator<false, _ValPtr>& __it) requires _Const
568#else
569 template<bool _OtherConst,
570 typename = __enable_if_t<_Const && !_OtherConst>>
571 constexpr
572 _Iterator(const _Iterator<_OtherConst, _ValPtr>& __it)
573#endif
574 : _M_node(__it._M_node) { }
575
576 [[__nodiscard__]]
577 reference
578 operator*() const noexcept
579 { return *static_cast<_Node&>(*_M_node)._M_valptr(); }
580
581 [[__nodiscard__]]
582 pointer
583 operator->() const noexcept
584 { return static_cast<_Node&>(*_M_node)._M_valptr(); }
585
586 _GLIBCXX14_CONSTEXPR _Iterator&
587 operator++() noexcept
588 {
589 if (_M_node->_M_right)
590 {
591 _M_node = _M_node->_M_right;
592 while (_M_node->_M_left)
593 _M_node = _M_node->_M_left;
594 }
595 else
596 {
597 _Base_ptr __y = _M_node->_M_parent;
598 while (_M_node == __y->_M_right)
599 {
600 _M_node = __y;
601 __y = __y->_M_parent;
602 }
603 if (_M_node->_M_right != __y)
604 _M_node = __y;
605 }
606
607 return *this;
608 }
609
610 _GLIBCXX14_CONSTEXPR _Iterator
611 operator++(int) noexcept
612 {
613 _Iterator __tmp(this->_M_node);
614 ++*this;
615 return __tmp;
616 }
617
618 _GLIBCXX14_CONSTEXPR _Iterator&
619 operator--() noexcept
620 {
621 if (_M_node->_M_color == _S_red
622 && _M_node->_M_parent->_M_parent == _M_node)
623 _M_node = _M_node->_M_right;
624 else if (_M_node->_M_left)
625 {
626 _Base_ptr __y = _M_node->_M_left;
627 while (__y->_M_right)
628 __y = __y->_M_right;
629 _M_node = __y;
630 }
631 else
632 {
633 _Base_ptr __y = _M_node->_M_parent;
634 while (_M_node == __y->_M_left)
635 {
636 _M_node = __y;
637 __y = __y->_M_parent;
638 }
639 _M_node = __y;
640 }
641 return *this;
642 }
643
644 _GLIBCXX14_CONSTEXPR _Iterator
645 operator--(int) noexcept
646 {
647 _Iterator __tmp(this->_M_node);
648 --*this;
649 return __tmp;
650 }
651
652 [[__nodiscard__]]
653 friend bool
654 operator==(const _Iterator& __x, const _Iterator& __y) _GLIBCXX_NOEXCEPT
655 { return __x._M_node == __y._M_node; }
656
657#if ! __cpp_lib_three_way_comparison
658 [[__nodiscard__]]
659 friend bool
660 operator!=(const _Iterator& __x, const _Iterator& __y) _GLIBCXX_NOEXCEPT
661 { return __x._M_node != __y._M_node; }
662#endif
663
664 _Base_ptr _M_node;
665 };
666#endif // USE_ALLOC_PTR_FOR_RB_TREE
667
668 // Determine the node and iterator types used by std::_Rb_tree.
669 template<typename _Val, typename _Ptr>
670 struct _Node_traits;
671
672#if _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE <= 9000
673 // Specialization for the simple case where the allocator's pointer type
674 // is the same type as value_type*.
675 // For ABI compatibility we can't change the types used for this case.
676 template<typename _Val>
677 struct _Node_traits<_Val, _Val*>
678 {
679 typedef _Rb_tree_node<_Val> _Node;
680 typedef _Node* _Node_ptr;
681 typedef _Rb_tree_node_base _Node_base;
682 typedef _Node_base* _Base_ptr;
683 typedef _Rb_tree_header _Header_t;
684 typedef _Rb_tree_iterator<_Val> _Iterator;
685 typedef _Rb_tree_const_iterator<_Val> _Const_iterator;
686
687 __attribute__((__nonnull__))
688 static void
689 _S_insert_and_rebalance(const bool __insert_left,
690 _Node_base* __x, _Node_base* __p,
691 _Node_base& __header) _GLIBCXX_USE_NOEXCEPT
692 {
693 return _Rb_tree_insert_and_rebalance(__insert_left, __x, __p, __header);
694 }
695
696 __attribute__((__nonnull__,__returns_nonnull__))
697 static _Node_base*
698 _S_rebalance_for_erase(_Node_base* const __z,
699 _Node_base& __header) _GLIBCXX_USE_NOEXCEPT
700 { return _Rb_tree_rebalance_for_erase(__z, __header); }
701 };
702#endif
703
704#if ! _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
705 // Always use the T* specialization.
706 template<typename _Val, typename _Ptr>
707 struct _Node_traits
708 : _Node_traits<_Val, _Val*>
709 { };
710#else
711 // Primary template used when the allocator uses fancy pointers.
712 template<typename _Val, typename _ValPtr>
713 struct _Node_traits
714 {
715 using _Node = __rb_tree::_Node<_ValPtr>;
716 using _Node_ptr = __ptr_rebind<_ValPtr, _Node>;
717 using _Node_base = __rb_tree::_Node_base<__ptr_rebind<_ValPtr, void>>;
718 using _Base_ptr = __ptr_rebind<_ValPtr, _Node_base>;
719 using _Header_t = __rb_tree::_Header<_Node_base>;
720 using _Iterator = __rb_tree::_Iterator<false, _ValPtr>;
721 using _Const_iterator = __rb_tree::_Iterator<true, _ValPtr>;
722
723 static void
724 _Rotate_left(_Base_ptr __x, _Base_ptr& __root)
725 {
726 const _Base_ptr __y = __x->_M_right;
727
728 __x->_M_right = __y->_M_left;
729 if (__y->_M_left)
730 __y->_M_left->_M_parent = __x;
731 __y->_M_parent = __x->_M_parent;
732
733 if (__x == __root)
734 __root = __y;
735 else if (__x == __x->_M_parent->_M_left)
736 __x->_M_parent->_M_left = __y;
737 else
738 __x->_M_parent->_M_right = __y;
739 __y->_M_left = __x;
740 __x->_M_parent = __y;
741 }
742
743 static void
744 _Rotate_right(_Base_ptr __x, _Base_ptr& __root)
745 {
746 const _Base_ptr __y = __x->_M_left;
747
748 __x->_M_left = __y->_M_right;
749 if (__y->_M_right)
750 __y->_M_right->_M_parent = __x;
751 __y->_M_parent = __x->_M_parent;
752
753 if (__x == __root)
754 __root = __y;
755 else if (__x == __x->_M_parent->_M_right)
756 __x->_M_parent->_M_right = __y;
757 else
758 __x->_M_parent->_M_left = __y;
759 __y->_M_right = __x;
760 __x->_M_parent = __y;
761 }
762
763 static void
764 _S_insert_and_rebalance(const bool __insert_left,
765 _Base_ptr __x, _Base_ptr __p,
766 _Node_base& __header)
767 {
768 _Base_ptr& __root = __header._M_parent;
769
770 // Initialize fields in new node to insert.
771 __x->_M_parent = __p;
772 __x->_M_left = __x->_M_right = nullptr;
773 __x->_M_color = _S_red;
774
775 // Insert.
776 // Make new node child of parent and maintain root, leftmost and
777 // rightmost nodes.
778 // N.B. First node is always inserted left.
779 if (__insert_left)
780 {
781 __p->_M_left = __x; // also makes leftmost = __x when __p == &__header
782
783 if (std::__to_address(__p) == std::addressof(__header))
784 {
785 __header._M_parent = __x;
786 __header._M_right = __x;
787 }
788 else if (__p == __header._M_left)
789 __header._M_left = __x; // maintain leftmost pointing to min node
790 }
791 else
792 {
793 __p->_M_right = __x;
794
795 if (__p == __header._M_right)
796 __header._M_right = __x; // maintain rightmost pointing to max node
797 }
798 // Rebalance.
799 while (__x != __root
800 && __x->_M_parent->_M_color == _S_red)
801 {
802 const _Base_ptr __xpp = __x->_M_parent->_M_parent;
803
804 if (__x->_M_parent == __xpp->_M_left)
805 {
806 const _Base_ptr __y = __xpp->_M_right;
807 if (__y && __y->_M_color == _S_red)
808 {
809 __x->_M_parent->_M_color = _S_black;
810 __y->_M_color = _S_black;
811 __xpp->_M_color = _S_red;
812 __x = __xpp;
813 }
814 else
815 {
816 if (__x == __x->_M_parent->_M_right)
817 {
818 __x = __x->_M_parent;
819 _Rotate_left(__x, __root);
820 }
821 __x->_M_parent->_M_color = _S_black;
822 __xpp->_M_color = _S_red;
823 _Rotate_right(__xpp, __root);
824 }
825 }
826 else
827 {
828 const _Base_ptr __y = __xpp->_M_left;
829 if (__y && __y->_M_color == _S_red)
830 {
831 __x->_M_parent->_M_color = _S_black;
832 __y->_M_color = _S_black;
833 __xpp->_M_color = _S_red;
834 __x = __xpp;
835 }
836 else
837 {
838 if (__x == __x->_M_parent->_M_left)
839 {
840 __x = __x->_M_parent;
841 _Rotate_right(__x, __root);
842 }
843 __x->_M_parent->_M_color = _S_black;
844 __xpp->_M_color = _S_red;
845 _Rotate_left(__xpp, __root);
846 }
847 }
848 }
849 __root->_M_color = _S_black;
850 }
851
852 static _Base_ptr
853 _S_rebalance_for_erase(_Base_ptr __z, _Node_base& __header)
854 {
855 _Base_ptr& __root = __header._M_parent;
856 _Base_ptr& __leftmost = __header._M_left;
857 _Base_ptr& __rightmost = __header._M_right;
858 _Base_ptr __y = __z;
859 _Base_ptr __x{};
860 _Base_ptr __x_parent{};
861
862 if (!__y->_M_left) // __z has at most one non-null child. y == z.
863 __x = __y->_M_right; // __x might be null.
864 else
865 if (!__y->_M_right) // __z has exactly one non-null child. y == z.
866 __x = __y->_M_left; // __x is not null.
867 else
868 {
869 // __z has two non-null children. Set __y to
870 __y = __y->_M_right; // __z's successor. __x might be null.
871 while (__y->_M_left)
872 __y = __y->_M_left;
873 __x = __y->_M_right;
874 }
875 if (__y != __z)
876 {
877 // relink y in place of z. y is z's successor
878 __z->_M_left->_M_parent = __y;
879 __y->_M_left = __z->_M_left;
880 if (__y != __z->_M_right)
881 {
882 __x_parent = __y->_M_parent;
883 if (__x)
884 __x->_M_parent = __y->_M_parent;
885 __y->_M_parent->_M_left = __x; // __y must be a child of _M_left
886 __y->_M_right = __z->_M_right;
887 __z->_M_right->_M_parent = __y;
888 }
889 else
890 __x_parent = __y;
891 if (__root == __z)
892 __root = __y;
893 else if (__z->_M_parent->_M_left == __z)
894 __z->_M_parent->_M_left = __y;
895 else
896 __z->_M_parent->_M_right = __y;
897 __y->_M_parent = __z->_M_parent;
898 std::swap(__y->_M_color, __z->_M_color);
899 __y = __z;
900 // __y now points to node to be actually deleted
901 }
902 else
903 { // __y == __z
904 __x_parent = __y->_M_parent;
905 if (__x)
906 __x->_M_parent = __y->_M_parent;
907 if (__root == __z)
908 __root = __x;
909 else
910 if (__z->_M_parent->_M_left == __z)
911 __z->_M_parent->_M_left = __x;
912 else
913 __z->_M_parent->_M_right = __x;
914 if (__leftmost == __z)
915 {
916 if (!__z->_M_right) // __z->_M_left must be null also
917 __leftmost = __z->_M_parent;
918 // makes __leftmost == _M_header if __z == __root
919 else
920 __leftmost = _Node_base::_S_minimum(__x);
921 }
922 if (__rightmost == __z)
923 {
924 if (__z->_M_left == 0) // __z->_M_right must be null also
925 __rightmost = __z->_M_parent;
926 // makes __rightmost == _M_header if __z == __root
927 else // __x == __z->_M_left
928 __rightmost = _Node_base::_S_maximum(__x);
929 }
930 }
931 if (__y->_M_color != _S_red)
932 {
933 while (__x != __root && (__x == 0 || __x->_M_color == _S_black))
934 if (__x == __x_parent->_M_left)
935 {
936 _Base_ptr __w = __x_parent->_M_right;
937 if (__w->_M_color == _S_red)
938 {
939 __w->_M_color = _S_black;
940 __x_parent->_M_color = _S_red;
941 _Rotate_left(__x_parent, __root);
942 __w = __x_parent->_M_right;
943 }
944 if ((!__w->_M_left || __w->_M_left->_M_color == _S_black) &&
945 (!__w->_M_right || __w->_M_right->_M_color == _S_black))
946 {
947 __w->_M_color = _S_red;
948 __x = __x_parent;
949 __x_parent = __x_parent->_M_parent;
950 }
951 else
952 {
953 if (!__w->_M_right || __w->_M_right->_M_color == _S_black)
954 {
955 __w->_M_left->_M_color = _S_black;
956 __w->_M_color = _S_red;
957 _Rotate_right(__w, __root);
958 __w = __x_parent->_M_right;
959 }
960 __w->_M_color = __x_parent->_M_color;
961 __x_parent->_M_color = _S_black;
962 if (__w->_M_right)
963 __w->_M_right->_M_color = _S_black;
964 _Rotate_left(__x_parent, __root);
965 break;
966 }
967 }
968 else
969 {
970 // same as above, with _M_right <-> _M_left.
971 _Base_ptr __w = __x_parent->_M_left;
972 if (__w->_M_color == _S_red)
973 {
974 __w->_M_color = _S_black;
975 __x_parent->_M_color = _S_red;
976 _Rotate_right(__x_parent, __root);
977 __w = __x_parent->_M_left;
978 }
979 if ((!__w->_M_right || __w->_M_right->_M_color == _S_black) &&
980 (!__w->_M_left || __w->_M_left->_M_color == _S_black))
981 {
982 __w->_M_color = _S_red;
983 __x = __x_parent;
984 __x_parent = __x_parent->_M_parent;
985 }
986 else
987 {
988 if (!__w->_M_left || __w->_M_left->_M_color == _S_black)
989 {
990 __w->_M_right->_M_color = _S_black;
991 __w->_M_color = _S_red;
992 _Rotate_left(__w, __root);
993 __w = __x_parent->_M_left;
994 }
995 __w->_M_color = __x_parent->_M_color;
996 __x_parent->_M_color = _S_black;
997 if (__w->_M_left)
998 __w->_M_left->_M_color = _S_black;
999 _Rotate_right(__x_parent, __root);
1000 break;
1001 }
1002 }
1003 if (__x)
1004 __x->_M_color = _S_black;
1005 }
1006
1007 return __y;
1008 }
1009 };
1010#endif
1011} // namespace __rb_tree
1012
1013#ifdef __glibcxx_node_extract // >= C++17
1014 template<typename _Tree1, typename _Cmp2>
1015 struct _Rb_tree_merge_helper { };
1016#endif
1017
1018 template<typename _Key, typename _Val, typename _KeyOfValue,
1019 typename _Compare, typename _Alloc = allocator<_Val> >
1020 class _Rb_tree
1021 {
1022 typedef typename __gnu_cxx::__alloc_traits<_Alloc>::template
1023 rebind<_Val>::other _Val_alloc_type;
1024
1025 typedef __gnu_cxx::__alloc_traits<_Val_alloc_type> _Val_alloc_traits;
1026 typedef typename _Val_alloc_traits::pointer _ValPtr;
1027 typedef __rb_tree::_Node_traits<_Val, _ValPtr> _Node_traits;
1028
1029 typedef typename _Node_traits::_Node_base _Node_base;
1030 typedef typename _Node_traits::_Node _Node;
1031
1032 typedef typename __gnu_cxx::__alloc_traits<_Alloc>::template
1033 rebind<_Node>::other _Node_allocator;
1034
1035 typedef __gnu_cxx::__alloc_traits<_Node_allocator> _Node_alloc_traits;
1036
1037 protected:
1038 typedef typename _Node_traits::_Base_ptr _Base_ptr;
1039 typedef typename _Node_traits::_Node_ptr _Node_ptr;
1040
1041 private:
1042 // Functor recycling a pool of nodes and using allocation once the pool
1043 // is empty.
1044 struct _Reuse_or_alloc_node
1045 {
1046 _Reuse_or_alloc_node(_Rb_tree& __t)
1047 : _M_root(__t._M_root()), _M_nodes(__t._M_rightmost()), _M_t(__t)
1048 {
1049 if (_M_root)
1050 {
1051 _M_root->_M_parent = _Base_ptr();
1052
1053 if (_M_nodes->_M_left)
1054 _M_nodes = _M_nodes->_M_left;
1055 }
1056 else
1057 _M_nodes = _Base_ptr();
1058 }
1059
1060#if __cplusplus >= 201103L
1061 _Reuse_or_alloc_node(const _Reuse_or_alloc_node&) = delete;
1062#endif
1063
1064 ~_Reuse_or_alloc_node()
1065 {
1066 if (_M_root)
1067 _M_t._M_erase(static_cast<_Node&>(*_M_root)._M_node_ptr());
1068 }
1069
1070 template<typename _Arg>
1071 _Node_ptr
1072 operator()(_GLIBCXX_FWDREF(_Arg) __arg)
1073 {
1074 _Base_ptr __base = _M_extract();
1075 if (__base)
1076 {
1077 _Node_ptr __node = static_cast<_Node&>(*__base)._M_node_ptr();
1078 _M_t._M_destroy_node(__node);
1079 _M_t._M_construct_node(__node, _GLIBCXX_FORWARD(_Arg, __arg));
1080 return __node;
1081 }
1082
1083 return _M_t._M_create_node(_GLIBCXX_FORWARD(_Arg, __arg));
1084 }
1085
1086 private:
1087 _Base_ptr
1088 _M_extract()
1089 {
1090 if (!_M_nodes)
1091 return _M_nodes;
1092
1093 _Base_ptr __node = _M_nodes;
1094 _M_nodes = _M_nodes->_M_parent;
1095 if (_M_nodes)
1096 {
1097 if (_M_nodes->_M_right == __node)
1098 {
1099 _M_nodes->_M_right = _Base_ptr();
1100
1101 if (_M_nodes->_M_left)
1102 {
1103 _M_nodes = _M_nodes->_M_left;
1104
1105 while (_M_nodes->_M_right)
1106 _M_nodes = _M_nodes->_M_right;
1107
1108 if (_M_nodes->_M_left)
1109 _M_nodes = _M_nodes->_M_left;
1110 }
1111 }
1112 else // __node is on the left.
1113 _M_nodes->_M_left = _Base_ptr();
1114 }
1115 else
1116 _M_root = _Base_ptr();
1117
1118 return __node;
1119 }
1120
1121 _Base_ptr _M_root;
1122 _Base_ptr _M_nodes;
1123 _Rb_tree& _M_t;
1124 };
1125
1126 // Functor similar to the previous one but without any pool of nodes to
1127 // recycle.
1128 struct _Alloc_node
1129 {
1130 _Alloc_node(_Rb_tree& __t)
1131 : _M_t(__t) { }
1132
1133 template<typename _Arg>
1134 _Node_ptr
1135 operator()(_GLIBCXX_FWDREF(_Arg) __arg) const
1136 { return _M_t._M_create_node(_GLIBCXX_FORWARD(_Arg, __arg)); }
1137
1138 private:
1139 _Rb_tree& _M_t;
1140 };
1141
1142 public:
1143 typedef _Key key_type;
1144 typedef _Val value_type;
1145 typedef value_type* pointer;
1146 typedef const value_type* const_pointer;
1147 typedef value_type& reference;
1148 typedef const value_type& const_reference;
1149 typedef size_t size_type;
1150 typedef ptrdiff_t difference_type;
1151 typedef _Alloc allocator_type;
1152
1153 _Node_allocator&
1154 _M_get_Node_allocator() _GLIBCXX_NOEXCEPT
1155 { return this->_M_impl; }
1156
1157 const _Node_allocator&
1158 _M_get_Node_allocator() const _GLIBCXX_NOEXCEPT
1159 { return this->_M_impl; }
1160
1161 allocator_type
1162 get_allocator() const _GLIBCXX_NOEXCEPT
1163 { return allocator_type(_M_get_Node_allocator()); }
1164
1165 protected:
1166 _Node_ptr
1167 _M_get_node()
1168 {
1169#if __cplusplus < 201102L || _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
1170 return _Node_alloc_traits::allocate(_M_get_Node_allocator(), 1);
1171#else
1172#pragma GCC diagnostic push
1173#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
1174 using __alloc_pointer = typename _Node_alloc_traits::pointer;
1175 if constexpr (is_same<_Node_ptr, __alloc_pointer>::value)
1176 return _Node_alloc_traits::allocate(_M_get_Node_allocator(), 1);
1177 else
1178 {
1179 auto __ptr =
1180 _Node_alloc_traits::allocate(_M_get_Node_allocator(), 1);
1181 return std::__to_address(__ptr);
1182 }
1183#pragma GCC diagnostic pop
1184#endif
1185 }
1186
1187 void
1188 _M_put_node(_Node_ptr __p) _GLIBCXX_NOEXCEPT
1189 {
1190#if __cplusplus < 201102L || _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
1191 _Node_alloc_traits::deallocate(_M_get_Node_allocator(), __p, 1);
1192#else
1193#pragma GCC diagnostic push
1194#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
1195 using __alloc_pointer = typename _Node_alloc_traits::pointer;
1196 if constexpr (is_same<_Node_ptr, __alloc_pointer>::value)
1197 _Node_alloc_traits::deallocate(_M_get_Node_allocator(), __p, 1);
1198 else
1199 {
1200 // When not using the allocator's pointer type internally we must
1201 // convert __p to __alloc_pointer so it can be deallocated.
1202 auto __ap = pointer_traits<__alloc_pointer>::pointer_to(*__p);
1203 _Node_alloc_traits::deallocate(_M_get_Node_allocator(), __ap, 1);
1204 }
1205#pragma GCC diagnostic pop
1206#endif
1207 }
1208
1209#if __cplusplus < 201103L
1210 void
1211 _M_construct_node(_Node_ptr __node, const value_type& __x)
1212 {
1213 __try
1214 { get_allocator().construct(__node->_M_valptr(), __x); }
1215 __catch(...)
1216 {
1217 _M_put_node(__node);
1218 __throw_exception_again;
1219 }
1220 }
1221
1222 _Node_ptr
1223 _M_create_node(const value_type& __x)
1224 {
1225 _Node_ptr __tmp = _M_get_node();
1226 _M_construct_node(__tmp, __x);
1227 return __tmp;
1228 }
1229#else
1230 template<typename... _Args>
1231 void
1232 _M_construct_node(_Node_ptr __node, _Args&&... __args)
1233 {
1234 __try
1235 {
1236 ::new(std::addressof(*__node)) _Node;
1237 _Node_alloc_traits::construct(_M_get_Node_allocator(),
1238 __node->_M_valptr(),
1239 std::forward<_Args>(__args)...);
1240 }
1241 __catch(...)
1242 {
1243 __node->~_Node();
1244 _M_put_node(__node);
1245 __throw_exception_again;
1246 }
1247 }
1248
1249 template<typename... _Args>
1250 _Node_ptr
1251 _M_create_node(_Args&&... __args)
1252 {
1253 _Node_ptr __tmp = _M_get_node();
1254 _M_construct_node(__tmp, std::forward<_Args>(__args)...);
1255 return __tmp;
1256 }
1257#endif
1258
1259 void
1260 _M_destroy_node(_Node_ptr __p) _GLIBCXX_NOEXCEPT
1261 {
1262#if __cplusplus < 201103L
1263 get_allocator().destroy(__p->_M_valptr());
1264#else
1265 _Node_alloc_traits::destroy(_M_get_Node_allocator(), __p->_M_valptr());
1266 __p->~_Node();
1267#endif
1268 }
1269
1270 void
1271 _M_drop_node(_Node_ptr __p) _GLIBCXX_NOEXCEPT
1272 {
1273 _M_destroy_node(__p);
1274 _M_put_node(__p);
1275 }
1276
1277 template<bool _MoveValue, typename _NodeGen>
1278 _Node_ptr
1279 _M_clone_node(_Node_ptr __x, _NodeGen& __node_gen)
1280 {
1281#if __cplusplus >= 201103L
1282 using _Vp = __conditional_t<_MoveValue,
1283 value_type&&,
1284 const value_type&>;
1285#endif
1286 _Node_ptr __tmp
1287 = __node_gen(_GLIBCXX_FORWARD(_Vp, *__x->_M_valptr()));
1288 __tmp->_M_color = __x->_M_color;
1289 __tmp->_M_left = __tmp->_M_right = _Base_ptr();
1290 return __tmp;
1291 }
1292
1293 protected:
1294 typedef typename _Node_traits::_Header_t _Header_t;
1295
1296#if _GLIBCXX_INLINE_VERSION
1297 template<typename _Key_compare>
1298#else
1299 // Unused _Is_pod_comparator is kept as it is part of mangled name.
1300 template<typename _Key_compare,
1301 bool /* _Is_pod_comparator */ = __is_pod(_Key_compare)>
1302#endif
1303 struct _Rb_tree_impl
1304 : public _Node_allocator
1305 , public _Rb_tree_key_compare<_Key_compare>
1306 , public _Header_t
1307 {
1308 typedef _Rb_tree_key_compare<_Key_compare> _Base_key_compare;
1309
1310 _Rb_tree_impl()
1311 _GLIBCXX_NOEXCEPT_IF(
1312 is_nothrow_default_constructible<_Node_allocator>::value
1313 && is_nothrow_default_constructible<_Base_key_compare>::value )
1314 : _Node_allocator()
1315 { }
1316
1317 _Rb_tree_impl(const _Rb_tree_impl& __x)
1318 : _Node_allocator(_Node_alloc_traits::_S_select_on_copy(__x))
1319 , _Base_key_compare(__x._M_key_compare)
1320 , _Header_t()
1321 { }
1322
1323#if __cplusplus < 201103L
1324 _Rb_tree_impl(const _Key_compare& __comp, const _Node_allocator& __a)
1325 : _Node_allocator(__a), _Base_key_compare(__comp)
1326 { }
1327#else
1328 _Rb_tree_impl(_Rb_tree_impl&&)
1329 noexcept( is_nothrow_move_constructible<_Base_key_compare>::value )
1330 = default;
1331
1332 explicit
1333 _Rb_tree_impl(_Node_allocator&& __a)
1334 : _Node_allocator(std::move(__a))
1335 { }
1336
1337 _Rb_tree_impl(_Rb_tree_impl&& __x, _Node_allocator&& __a)
1338 : _Node_allocator(std::move(__a)),
1339 _Base_key_compare(std::move(__x)),
1340 _Header_t(std::move(__x))
1341 { }
1342
1343 _Rb_tree_impl(const _Key_compare& __comp, _Node_allocator&& __a)
1344 : _Node_allocator(std::move(__a)), _Base_key_compare(__comp)
1345 { }
1346#endif
1347 };
1348
1349 _Rb_tree_impl<_Compare> _M_impl;
1350
1351 protected:
1352 _Base_ptr&
1353 _M_root() _GLIBCXX_NOEXCEPT
1354 { return this->_M_impl._M_header._M_parent; }
1355
1356 _Base_ptr
1357 _M_root() const _GLIBCXX_NOEXCEPT
1358 { return this->_M_impl._M_header._M_parent; }
1359
1360 _Base_ptr&
1361 _M_leftmost() _GLIBCXX_NOEXCEPT
1362 { return this->_M_impl._M_header._M_left; }
1363
1364 _Base_ptr
1365 _M_leftmost() const _GLIBCXX_NOEXCEPT
1366 { return this->_M_impl._M_header._M_left; }
1367
1368 _Base_ptr&
1369 _M_rightmost() _GLIBCXX_NOEXCEPT
1370 { return this->_M_impl._M_header._M_right; }
1371
1372 _Base_ptr
1373 _M_rightmost() const _GLIBCXX_NOEXCEPT
1374 { return this->_M_impl._M_header._M_right; }
1375
1376 _Base_ptr
1377 _M_begin() const _GLIBCXX_NOEXCEPT
1378 { return this->_M_impl._M_header._M_parent; }
1379
1380 _Node_ptr
1381 _M_begin_node() const _GLIBCXX_NOEXCEPT
1382 {
1383 _Base_ptr __begin = this->_M_impl._M_header._M_parent;
1384 return __begin
1385 ? static_cast<_Node&>(*__begin)._M_node_ptr()
1386 : _Node_ptr();
1387 }
1388
1389 _Base_ptr
1390 _M_end() const _GLIBCXX_NOEXCEPT
1391 { return this->_M_impl._M_header._M_base_ptr(); }
1392
1393 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1394 // 2542. Missing const requirements for associative containers
1395 template<typename _Key1, typename _Key2>
1396 bool
1397 _M_key_compare(const _Key1& __k1, const _Key2& __k2) const
1398 {
1399#if __cplusplus >= 201103L
1400 // Enforce this here with a user-friendly message.
1401 static_assert(
1402 __is_invocable<const _Compare&, const _Key&, const _Key&>::value,
1403 "comparison object must be invocable with arguments of key_type"
1404 );
1405#endif
1406 return _M_impl._M_key_compare(__k1, __k2);
1407 }
1408
1409 static const _Key&
1410 _S_key(const _Node& __node)
1411 { return _KeyOfValue()(*__node._M_valptr()); }
1412
1413 static const _Key&
1414 _S_key(_Base_ptr __x)
1415 { return _S_key(static_cast<const _Node&>(*__x)); }
1416
1417 static const _Key&
1418 _S_key(_Node_ptr __x)
1419 { return _S_key(*__x); }
1420
1421 static _Base_ptr
1422 _S_left(_Base_ptr __x) _GLIBCXX_NOEXCEPT
1423 { return __x->_M_left; }
1424
1425 static _Node_ptr
1426 _S_left(_Node_ptr __x)
1427 {
1428 return __x->_M_left
1429 ? static_cast<_Node&>(*__x->_M_left)._M_node_ptr()
1430 : _Node_ptr();
1431 }
1432
1433 static _Base_ptr
1434 _S_right(_Base_ptr __x) _GLIBCXX_NOEXCEPT
1435 { return __x->_M_right; }
1436
1437 static _Node_ptr
1438 _S_right(_Node_ptr __x) _GLIBCXX_NOEXCEPT
1439 {
1440 return __x->_M_right
1441 ? static_cast<_Node&>(*__x->_M_right)._M_node_ptr()
1442 : _Node_ptr();
1443 }
1444
1445 public:
1446 typedef typename _Node_traits::_Iterator iterator;
1447 typedef typename _Node_traits::_Const_iterator const_iterator;
1448
1449 typedef std::reverse_iterator<iterator> reverse_iterator;
1450 typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
1451
1452#ifdef __glibcxx_node_extract // >= C++17
1453 using node_type = _Node_handle<_Key, _Val, _Node_allocator>;
1454 using insert_return_type = _Node_insert_return<
1455 __conditional_t<is_same_v<_Key, _Val>, const_iterator, iterator>,
1456 node_type>;
1457#endif
1458
1460 _M_get_insert_unique_pos(const key_type& __k);
1461
1463 _M_get_insert_equal_pos(const key_type& __k);
1464
1466 _M_get_insert_hint_unique_pos(const_iterator __pos,
1467 const key_type& __k);
1468
1470 _M_get_insert_hint_equal_pos(const_iterator __pos,
1471 const key_type& __k);
1472
1473 private:
1474#if __cplusplus >= 201103L
1475 template<typename _Arg, typename _NodeGen>
1476 iterator
1477 _M_insert_(_Base_ptr __x, _Base_ptr __y, _Arg&& __v, _NodeGen&);
1478
1479 iterator
1480 _M_insert_node(_Base_ptr __x, _Base_ptr __y, _Node_ptr __z);
1481
1482 template<typename _Arg>
1483 iterator
1484 _M_insert_lower(_Base_ptr __y, _Arg&& __v);
1485
1486 template<typename _Arg>
1487 iterator
1488 _M_insert_equal_lower(_Arg&& __x);
1489
1490 iterator
1491 _M_insert_lower_node(_Base_ptr __p, _Node_ptr __z);
1492
1493 iterator
1494 _M_insert_equal_lower_node(_Node_ptr __z);
1495#else
1496 template<typename _NodeGen>
1497 iterator
1498 _M_insert_(_Base_ptr __x, _Base_ptr __y,
1499 const value_type& __v, _NodeGen&);
1500
1501 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1502 // 233. Insertion hints in associative containers.
1503 iterator
1504 _M_insert_lower(_Base_ptr __y, const value_type& __v);
1505
1506 iterator
1507 _M_insert_equal_lower(const value_type& __x);
1508#endif
1509
1510 enum { __as_lvalue, __as_rvalue };
1511
1512 template<bool _MoveValues, typename _NodeGen>
1513 _Base_ptr
1514 _M_copy(_Node_ptr, _Base_ptr, _NodeGen&);
1515
1516 template<bool _MoveValues, typename _NodeGen>
1517 _Base_ptr
1518 _M_copy(const _Rb_tree& __x, _NodeGen& __gen)
1519 {
1520 _Base_ptr __root =
1521 _M_copy<_MoveValues>(__x._M_begin_node(), _M_end(), __gen);
1522 _M_leftmost() = _Node_base::_S_minimum(__root);
1523 _M_rightmost() = _Node_base::_S_maximum(__root);
1524 _M_impl._M_node_count = __x._M_impl._M_node_count;
1525 return __root;
1526 }
1527
1528 _Base_ptr
1529 _M_copy(const _Rb_tree& __x)
1530 {
1531 _Alloc_node __an(*this);
1532 return _M_copy<__as_lvalue>(__x, __an);
1533 }
1534
1535 void
1536 _M_erase(_Node_ptr __x);
1537
1538 _Base_ptr
1539 _M_lower_bound(_Base_ptr __x, _Base_ptr __y,
1540 const _Key& __k) const;
1541
1542 template <typename _Kt>
1543 _Base_ptr
1544 _M_lower_bound_tr(_Base_ptr __x, _Base_ptr __y, const _Kt& __k) const;
1545
1546 _Base_ptr
1547 _M_upper_bound(_Base_ptr __x, _Base_ptr __y,
1548 const _Key& __k) const;
1549
1550 template <typename _Kt>
1551 _Base_ptr
1552 _M_upper_bound_tr(_Base_ptr __x, _Base_ptr __y, const _Kt& __k) const;
1553
1554 public:
1555 // allocation/deallocation
1556#if __cplusplus < 201103L
1557 _Rb_tree() { }
1558#else
1559 _Rb_tree() = default;
1560#endif
1561
1562 _Rb_tree(const _Compare& __comp,
1563 const allocator_type& __a = allocator_type())
1564 : _M_impl(__comp, _Node_allocator(__a)) { }
1565
1566 _Rb_tree(const _Rb_tree& __x)
1567 : _M_impl(__x._M_impl)
1568 {
1569 if (__x._M_root())
1570 _M_root() = _M_copy(__x);
1571 }
1572
1573#if __cplusplus >= 201103L
1574 _Rb_tree(const allocator_type& __a)
1575 : _M_impl(_Node_allocator(__a))
1576 { }
1577
1578 _Rb_tree(const _Rb_tree& __x, const allocator_type& __a)
1579 : _M_impl(__x._M_impl._M_key_compare, _Node_allocator(__a))
1580 {
1581 if (__x._M_root())
1582 _M_root() = _M_copy(__x);
1583 }
1584
1585 _Rb_tree(_Rb_tree&&) = default;
1586
1587 _Rb_tree(_Rb_tree&& __x, const allocator_type& __a)
1588 : _Rb_tree(std::move(__x), _Node_allocator(__a))
1589 { }
1590
1591 private:
1592 _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a, true_type)
1593 noexcept(is_nothrow_default_constructible<_Compare>::value)
1594 : _M_impl(std::move(__x._M_impl), std::move(__a))
1595 { }
1596
1597 _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a, false_type)
1598 : _M_impl(__x._M_impl._M_key_compare, std::move(__a))
1599 {
1600 if (__x._M_root())
1601 _M_move_data(__x, false_type{});
1602 }
1603
1604 public:
1605 _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a)
1606 noexcept( noexcept(
1609 : _Rb_tree(std::move(__x), std::move(__a),
1610 typename _Node_alloc_traits::is_always_equal{})
1611 { }
1612#endif
1613
1614 ~_Rb_tree() _GLIBCXX_NOEXCEPT
1615 { _M_erase(_M_begin_node()); }
1616
1617 _Rb_tree&
1618 operator=(const _Rb_tree& __x);
1619
1620 // Accessors.
1621 _Compare
1622 key_comp() const
1623 { return _M_impl._M_key_compare; }
1624
1625 iterator
1626 begin() _GLIBCXX_NOEXCEPT
1627 { return iterator(this->_M_impl._M_header._M_left); }
1628
1629 const_iterator
1630 begin() const _GLIBCXX_NOEXCEPT
1631 { return const_iterator(this->_M_impl._M_header._M_left); }
1632
1633 iterator
1634 end() _GLIBCXX_NOEXCEPT
1635 { return iterator(_M_end()); }
1636
1637 const_iterator
1638 end() const _GLIBCXX_NOEXCEPT
1639 { return const_iterator(_M_end()); }
1640
1641 reverse_iterator
1642 rbegin() _GLIBCXX_NOEXCEPT
1643 { return reverse_iterator(end()); }
1644
1645 const_reverse_iterator
1646 rbegin() const _GLIBCXX_NOEXCEPT
1647 { return const_reverse_iterator(end()); }
1648
1649 reverse_iterator
1650 rend() _GLIBCXX_NOEXCEPT
1651 { return reverse_iterator(begin()); }
1652
1653 const_reverse_iterator
1654 rend() const _GLIBCXX_NOEXCEPT
1655 { return const_reverse_iterator(begin()); }
1656
1657 _GLIBCXX_NODISCARD bool
1658 empty() const _GLIBCXX_NOEXCEPT
1659 { return _M_impl._M_node_count == 0; }
1660
1661 size_type
1662 size() const _GLIBCXX_NOEXCEPT
1663 { return _M_impl._M_node_count; }
1664
1665 size_type
1666 max_size() const _GLIBCXX_NOEXCEPT
1667 { return _Node_alloc_traits::max_size(_M_get_Node_allocator()); }
1668
1669 void
1670 swap(_Rb_tree& __t)
1671 _GLIBCXX_NOEXCEPT_IF(__is_nothrow_swappable<_Compare>::value);
1672
1673 // Insert/erase.
1674#if __cplusplus >= 201103L
1675 template<typename _Arg>
1677 _M_insert_unique(_Arg&& __x);
1678
1679 template<typename _Arg>
1680 iterator
1681 _M_insert_equal(_Arg&& __x);
1682
1683 template<typename _Arg, typename _NodeGen>
1684 iterator
1685 _M_insert_unique_(const_iterator __pos, _Arg&& __x, _NodeGen&);
1686
1687 template<typename _Arg>
1688 iterator
1689 _M_insert_unique_(const_iterator __pos, _Arg&& __x)
1690 {
1691 _Alloc_node __an(*this);
1692 return _M_insert_unique_(__pos, std::forward<_Arg>(__x), __an);
1693 }
1694
1695 template<typename _Arg, typename _NodeGen>
1696 iterator
1697 _M_insert_equal_(const_iterator __pos, _Arg&& __x, _NodeGen&);
1698
1699 template<typename _Arg>
1700 iterator
1701 _M_insert_equal_(const_iterator __pos, _Arg&& __x)
1702 {
1703 _Alloc_node __an(*this);
1704 return _M_insert_equal_(__pos, std::forward<_Arg>(__x), __an);
1705 }
1706
1707 template<typename... _Args>
1709 _M_emplace_unique(_Args&&... __args);
1710
1711 template<typename... _Args>
1712 iterator
1713 _M_emplace_equal(_Args&&... __args);
1714
1715 template<typename... _Args>
1716 iterator
1717 _M_emplace_hint_unique(const_iterator __pos, _Args&&... __args);
1718
1719 template<typename... _Args>
1720 iterator
1721 _M_emplace_hint_equal(const_iterator __pos, _Args&&... __args);
1722
1723 template<typename _Iter>
1724 using __same_value_type
1725 = is_same<value_type, typename iterator_traits<_Iter>::value_type>;
1726
1727 template<typename _InputIterator>
1728 __enable_if_t<__same_value_type<_InputIterator>::value>
1729 _M_insert_range_unique(_InputIterator __first, _InputIterator __last)
1730 {
1731 _Alloc_node __an(*this);
1732 for (; __first != __last; ++__first)
1733 _M_insert_unique_(end(), *__first, __an);
1734 }
1735
1736 template<typename _InputIterator>
1737 __enable_if_t<!__same_value_type<_InputIterator>::value>
1738 _M_insert_range_unique(_InputIterator __first, _InputIterator __last)
1739 {
1740 for (; __first != __last; ++__first)
1741 _M_emplace_unique(*__first);
1742 }
1743
1744 template<typename _InputIterator>
1745 __enable_if_t<__same_value_type<_InputIterator>::value>
1746 _M_insert_range_equal(_InputIterator __first, _InputIterator __last)
1747 {
1748 _Alloc_node __an(*this);
1749 for (; __first != __last; ++__first)
1750 _M_insert_equal_(end(), *__first, __an);
1751 }
1752
1753 template<typename _InputIterator>
1754 __enable_if_t<!__same_value_type<_InputIterator>::value>
1755 _M_insert_range_equal(_InputIterator __first, _InputIterator __last)
1756 {
1757 for (; __first != __last; ++__first)
1758 _M_emplace_equal(*__first);
1759 }
1760#else
1762 _M_insert_unique(const value_type& __x);
1763
1764 iterator
1765 _M_insert_equal(const value_type& __x);
1766
1767 template<typename _NodeGen>
1768 iterator
1769 _M_insert_unique_(const_iterator __pos, const value_type& __x,
1770 _NodeGen&);
1771
1772 iterator
1773 _M_insert_unique_(const_iterator __pos, const value_type& __x)
1774 {
1775 _Alloc_node __an(*this);
1776 return _M_insert_unique_(__pos, __x, __an);
1777 }
1778
1779 template<typename _NodeGen>
1780 iterator
1781 _M_insert_equal_(const_iterator __pos, const value_type& __x,
1782 _NodeGen&);
1783 iterator
1784 _M_insert_equal_(const_iterator __pos, const value_type& __x)
1785 {
1786 _Alloc_node __an(*this);
1787 return _M_insert_equal_(__pos, __x, __an);
1788 }
1789
1790 template<typename _InputIterator>
1791 void
1792 _M_insert_range_unique(_InputIterator __first, _InputIterator __last)
1793 {
1794 _Alloc_node __an(*this);
1795 for (; __first != __last; ++__first)
1796 _M_insert_unique_(end(), *__first, __an);
1797 }
1798
1799 template<typename _InputIterator>
1800 void
1801 _M_insert_range_equal(_InputIterator __first, _InputIterator __last)
1802 {
1803 _Alloc_node __an(*this);
1804 for (; __first != __last; ++__first)
1805 _M_insert_equal_(end(), *__first, __an);
1806 }
1807#endif
1808
1809 private:
1810 void
1811 _M_erase_aux(const_iterator __position);
1812
1813 void
1814 _M_erase_aux(const_iterator __first, const_iterator __last);
1815
1816 public:
1817#if __cplusplus >= 201103L
1818 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1819 // DR 130. Associative erase should return an iterator.
1820 _GLIBCXX_ABI_TAG_CXX11
1821 iterator
1822 erase(const_iterator __position)
1823 {
1824 __glibcxx_assert(__position != end());
1825 const_iterator __result = __position;
1826 ++__result;
1827 _M_erase_aux(__position);
1828 return iterator(__result._M_node);
1829 }
1830
1831 // LWG 2059.
1832 _GLIBCXX_ABI_TAG_CXX11
1833 iterator
1834 erase(iterator __position)
1835 {
1836 __glibcxx_assert(__position != end());
1837 iterator __result = __position;
1838 ++__result;
1839 _M_erase_aux(__position);
1840 return __result;
1841 }
1842#else
1843 void
1844 erase(iterator __position)
1845 {
1846 __glibcxx_assert(__position != end());
1847 _M_erase_aux(__position);
1848 }
1849
1850 void
1851 erase(const_iterator __position)
1852 {
1853 __glibcxx_assert(__position != end());
1854 _M_erase_aux(__position);
1855 }
1856#endif
1857
1858 size_type
1859 erase(const key_type& __x);
1860
1861 template <typename _Kt>
1862 size_type
1863 _M_erase_tr(const _Kt& __x);
1864
1865 size_type
1866 _M_erase_unique(const key_type& __x);
1867
1868#if __cplusplus >= 201103L
1869 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1870 // DR 130. Associative erase should return an iterator.
1871 _GLIBCXX_ABI_TAG_CXX11
1872 iterator
1873 erase(const_iterator __first, const_iterator __last)
1874 {
1875 _M_erase_aux(__first, __last);
1876 return iterator(__last._M_node);
1877 }
1878#else
1879 void
1880 erase(iterator __first, iterator __last)
1881 { _M_erase_aux(__first, __last); }
1882
1883 void
1884 erase(const_iterator __first, const_iterator __last)
1885 { _M_erase_aux(__first, __last); }
1886#endif
1887
1888 void
1889 clear() _GLIBCXX_NOEXCEPT
1890 {
1891 _M_erase(_M_begin_node());
1892 _M_impl._M_reset();
1893 }
1894
1895 // Set operations.
1896 iterator
1897 find(const key_type& __k);
1898
1899 const_iterator
1900 find(const key_type& __k) const;
1901
1902 size_type
1903 count(const key_type& __k) const;
1904
1905 iterator
1906 lower_bound(const key_type& __k)
1907 { return iterator(_M_lower_bound(_M_begin(), _M_end(), __k)); }
1908
1909 const_iterator
1910 lower_bound(const key_type& __k) const
1911 {
1912 return const_iterator
1913 (_M_lower_bound(_M_begin(), _M_end(), __k));
1914 }
1915
1916 iterator
1917 upper_bound(const key_type& __k)
1918 { return iterator(_M_upper_bound(_M_begin(), _M_end(), __k)); }
1919
1920 const_iterator
1921 upper_bound(const key_type& __k) const
1922 {
1923 return const_iterator
1924 (_M_upper_bound(_M_begin(), _M_end(), __k));
1925 }
1926
1928 equal_range(const key_type& __k);
1929
1931 equal_range(const key_type& __k) const;
1932
1933#ifdef __glibcxx_generic_associative_lookup // C++ >= 14
1934 template<typename _Kt,
1935 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
1936 iterator
1937 _M_find_tr(const _Kt& __k)
1938 {
1939 const _Rb_tree* __const_this = this;
1940 return iterator(__const_this->_M_find_tr(__k)._M_node);
1941 }
1942
1943 template<typename _Kt,
1944 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
1945 const_iterator
1946 _M_find_tr(const _Kt& __k) const
1947 {
1948 const_iterator __j(_M_lower_bound_tr(__k));
1949 if (__j != end() && _M_key_compare(__k, _S_key(__j._M_node)))
1950 __j = end();
1951 return __j;
1952 }
1953
1954 template<typename _Kt,
1955 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
1956 size_type
1957 _M_count_tr(const _Kt& __k) const
1958 {
1959 auto __p = _M_equal_range_tr(__k);
1960 return std::distance(__p.first, __p.second);
1961 }
1962
1963 template<typename _Kt,
1964 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
1965 _Base_ptr
1966 _M_lower_bound_tr(const _Kt& __k) const
1967 {
1968 auto __x = _M_begin();
1969 auto __y = _M_end();
1970 while (__x)
1971 if (!_M_key_compare(_S_key(__x), __k))
1972 {
1973 __y = __x;
1974 __x = _S_left(__x);
1975 }
1976 else
1977 __x = _S_right(__x);
1978 return __y;
1979 }
1980
1981 template<typename _Kt,
1982 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
1983 _Base_ptr
1984 _M_upper_bound_tr(const _Kt& __k) const
1985 {
1986 auto __x = _M_begin();
1987 auto __y = _M_end();
1988 while (__x)
1989 if (_M_key_compare(__k, _S_key(__x)))
1990 {
1991 __y = __x;
1992 __x = _S_left(__x);
1993 }
1994 else
1995 __x = _S_right(__x);
1996 return __y;
1997 }
1998
1999 template<typename _Kt,
2000 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
2002 _M_equal_range_tr(const _Kt& __k)
2003 {
2004 const _Rb_tree* __const_this = this;
2005 auto __ret = __const_this->_M_equal_range_tr(__k);
2006 return
2007 { iterator(__ret.first._M_node), iterator(__ret.second._M_node) };
2008 }
2009
2010 template<typename _Kt,
2011 typename _Req = __has_is_transparent_t<_Compare, _Kt>>
2013 _M_equal_range_tr(const _Kt& __k) const
2014 {
2015 const_iterator __low(_M_lower_bound_tr(__k));
2016 auto __high = __low;
2017 auto& __cmp = _M_impl._M_key_compare;
2018 while (__high != end() && !__cmp(__k, _S_key(__high._M_node)))
2019 ++__high;
2020 return { __low, __high };
2021 }
2022#endif // __glibcxx_generic_associative_lookup
2023
2024 // Debugging.
2025 bool
2026 __rb_verify() const;
2027
2028#if __cplusplus >= 201103L
2029 _Rb_tree&
2030 operator=(_Rb_tree&&)
2031 noexcept(_Node_alloc_traits::_S_nothrow_move()
2032 && is_nothrow_move_assignable<_Compare>::value);
2033
2034 template<typename _Iterator>
2035 void
2036 _M_assign_unique(_Iterator, _Iterator);
2037
2038 template<typename _Iterator>
2039 void
2040 _M_assign_equal(_Iterator, _Iterator);
2041
2042 private:
2043 // Move elements from container with equal allocator.
2044 void
2045 _M_move_data(_Rb_tree& __x, true_type)
2046 { _M_impl._M_move_data(__x._M_impl); }
2047
2048 // Move elements from container with possibly non-equal allocator,
2049 // which might result in a copy not a move.
2050 void
2051 _M_move_data(_Rb_tree&, false_type);
2052
2053 // Move assignment from container with equal allocator.
2054 void
2055 _M_move_assign(_Rb_tree&, true_type);
2056
2057 // Move assignment from container with possibly non-equal allocator,
2058 // which might result in a copy not a move.
2059 void
2060 _M_move_assign(_Rb_tree&, false_type);
2061#endif
2062
2063#if __glibcxx_node_extract // >= C++17
2064 static _Node_ptr
2065 _S_adapt(typename _Node_alloc_traits::pointer __ptr)
2066 {
2067#if _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
2068 return __ptr;
2069#else
2070#pragma GCC diagnostic push
2071#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
2072 using __alloc_ptr = typename _Node_alloc_traits::pointer;
2073 if constexpr (is_same<_Node_ptr, __alloc_ptr>::value)
2074 return __ptr;
2075 else
2076 return std::__to_address(__ptr);
2077#pragma GCC diagnostic pop
2078#endif
2079 }
2080
2081 public:
2082 /// Re-insert an extracted node.
2083 insert_return_type
2084 _M_reinsert_node_unique(node_type&& __nh)
2085 {
2086 insert_return_type __ret;
2087 if (__nh.empty())
2088 __ret.position = end();
2089 else
2090 {
2091 __glibcxx_assert(_M_get_Node_allocator() == *__nh._M_alloc);
2092
2093 auto __res = _M_get_insert_unique_pos(__nh._M_key());
2094 if (__res.second)
2095 {
2096 __ret.position
2097 = _M_insert_node(__res.first, __res.second,
2098 _S_adapt(__nh._M_ptr));
2099 __nh.release();
2100 __ret.inserted = true;
2101 }
2102 else
2103 {
2104 __ret.node = std::move(__nh);
2105 __ret.position = iterator(__res.first);
2106 __ret.inserted = false;
2107 }
2108 }
2109 return __ret;
2110 }
2111
2112 /// Re-insert an extracted node.
2113 iterator
2114 _M_reinsert_node_equal(node_type&& __nh)
2115 {
2116 iterator __ret;
2117 if (__nh.empty())
2118 __ret = end();
2119 else
2120 {
2121 __glibcxx_assert(_M_get_Node_allocator() == *__nh._M_alloc);
2122 auto __res = _M_get_insert_equal_pos(__nh._M_key());
2123 if (__res.second)
2124 __ret = _M_insert_node(__res.first, __res.second,
2125 _S_adapt(__nh._M_ptr));
2126 else
2127 __ret = _M_insert_equal_lower_node(_S_adapt(__nh._M_ptr));
2128 __nh.release();
2129 }
2130 return __ret;
2131 }
2132
2133 /// Re-insert an extracted node.
2134 iterator
2135 _M_reinsert_node_hint_unique(const_iterator __hint, node_type&& __nh)
2136 {
2137 iterator __ret;
2138 if (__nh.empty())
2139 __ret = end();
2140 else
2141 {
2142 __glibcxx_assert(_M_get_Node_allocator() == *__nh._M_alloc);
2143 auto __res = _M_get_insert_hint_unique_pos(__hint, __nh._M_key());
2144 if (__res.second)
2145 {
2146 __ret = _M_insert_node(__res.first, __res.second,
2147 _S_adapt(__nh._M_ptr));
2148 __nh.release();
2149 }
2150 else
2151 __ret = iterator(__res.first);
2152 }
2153 return __ret;
2154 }
2155
2156 /// Re-insert an extracted node.
2157 iterator
2158 _M_reinsert_node_hint_equal(const_iterator __hint, node_type&& __nh)
2159 {
2160 iterator __ret;
2161 if (__nh.empty())
2162 __ret = end();
2163 else
2164 {
2165 __glibcxx_assert(_M_get_Node_allocator() == *__nh._M_alloc);
2166 auto __res = _M_get_insert_hint_equal_pos(__hint, __nh._M_key());
2167 if (__res.second)
2168 __ret = _M_insert_node(__res.first, __res.second,
2169 _S_adapt(__nh._M_ptr));
2170 else
2171 __ret = _M_insert_equal_lower_node(_S_adapt(__nh._M_ptr));
2172 __nh.release();
2173 }
2174 return __ret;
2175 }
2176
2177 /// Extract a node.
2178 node_type
2179 extract(const_iterator __pos)
2180 {
2181 auto __ptr = _Node_traits::_S_rebalance_for_erase
2182 (__pos._M_node, _M_impl._M_header);
2183 --_M_impl._M_node_count;
2184 auto __node_ptr = static_cast<_Node&>(*__ptr)._M_node_ptr();
2185#if _GLIBCXX_USE_ALLOC_PTR_FOR_RB_TREE
2186 return { __node_ptr, _M_get_Node_allocator() };
2187#else
2188#pragma GCC diagnostic push
2189#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
2190 using __alloc_ptr = typename _Node_alloc_traits::pointer;
2191 if constexpr (is_same<_Node_ptr, __alloc_ptr>::value)
2192 return { __node_ptr, _M_get_Node_allocator() };
2193 else
2194 {
2195 auto __ap = pointer_traits<__alloc_ptr>::pointer_to(*__node_ptr);
2196 return { __ap, _M_get_Node_allocator() };
2197 }
2198#pragma GCC diagnostic pop
2199#endif
2200 }
2201
2202 /// Extract a node.
2203 node_type
2204 extract(const key_type& __k)
2205 {
2206 node_type __nh;
2207 auto __pos = find(__k);
2208 if (__pos != end())
2209 __nh = extract(const_iterator(__pos));
2210 return __nh;
2211 }
2212
2213 template <typename _Kt>
2214 node_type
2215 _M_extract_tr(const _Kt& __k)
2216 {
2217 node_type __nh;
2218 auto __pos = _M_find_tr(__k);
2219 if (__pos != end())
2220 __nh = extract(const_iterator(__pos));
2221 return __nh;
2222 }
2223
2224 template<typename _Compare2>
2225 using _Compatible_tree
2226 = _Rb_tree<_Key, _Val, _KeyOfValue, _Compare2, _Alloc>;
2227
2228 template<typename, typename>
2229 friend struct _Rb_tree_merge_helper;
2230
2231 /// Merge from a compatible container into one with unique keys.
2232 template<typename _Compare2>
2233 void
2234 _M_merge_unique(_Compatible_tree<_Compare2>& __src) noexcept
2235 {
2236 using _Merge_helper = _Rb_tree_merge_helper<_Rb_tree, _Compare2>;
2237 for (auto __i = __src.begin(), __end = __src.end(); __i != __end;)
2238 {
2239 auto __pos = __i++;
2240 auto __res = _M_get_insert_unique_pos(_KeyOfValue()(*__pos));
2241 if (__res.second)
2242 {
2243 auto& __src_impl = _Merge_helper::_S_get_impl(__src);
2244 auto __ptr = _Node_traits::_S_rebalance_for_erase
2245 (__pos._M_node, __src_impl._M_header);
2246 --__src_impl._M_node_count;
2247 auto __node_ptr = static_cast<_Node&>(*__ptr)._M_node_ptr();
2248 _M_insert_node(__res.first, __res.second, __node_ptr);
2249 }
2250 }
2251 }
2252
2253 /// Merge from a compatible container into one with equivalent keys.
2254 template<typename _Compare2>
2255 void
2256 _M_merge_equal(_Compatible_tree<_Compare2>& __src) noexcept
2257 {
2258 using _Merge_helper = _Rb_tree_merge_helper<_Rb_tree, _Compare2>;
2259 for (auto __i = __src.begin(), __end = __src.end(); __i != __end;)
2260 {
2261 auto __pos = __i++;
2262 auto __res = _M_get_insert_equal_pos(_KeyOfValue()(*__pos));
2263 if (__res.second)
2264 {
2265 auto& __src_impl = _Merge_helper::_S_get_impl(__src);
2266 auto __ptr = _Node_traits::_S_rebalance_for_erase
2267 (__pos._M_node, __src_impl._M_header);
2268 --__src_impl._M_node_count;
2269 auto __node_ptr = static_cast<_Node&>(*__ptr)._M_node_ptr();
2270 _M_insert_node(__res.first, __res.second, __node_ptr);
2271 }
2272 }
2273 }
2274#endif // C++17 node_extract
2275
2276 friend bool
2277 operator==(const _Rb_tree& __x, const _Rb_tree& __y)
2278 {
2279 return __x.size() == __y.size()
2280 && std::equal(__x.begin(), __x.end(), __y.begin());
2281 }
2282
2283#if __cpp_lib_three_way_comparison
2284 friend auto
2285 operator<=>(const _Rb_tree& __x, const _Rb_tree& __y)
2286 {
2287 if constexpr (requires { typename __detail::__synth3way_t<_Val>; })
2288 return std::lexicographical_compare_three_way(__x.begin(), __x.end(),
2289 __y.begin(), __y.end(),
2290 __detail::__synth3way);
2291 }
2292#else
2293 friend bool
2294 operator<(const _Rb_tree& __x, const _Rb_tree& __y)
2295 {
2296 return std::lexicographical_compare(__x.begin(), __x.end(),
2297 __y.begin(), __y.end());
2298 }
2299#endif
2300
2301 private:
2302#if __cplusplus >= 201103L
2303 // An RAII _Node handle
2304 struct _Auto_node
2305 {
2306 template<typename... _Args>
2307 _Auto_node(_Rb_tree& __t, _Args&&... __args)
2308 : _M_t(__t),
2309 _M_node(__t._M_create_node(std::forward<_Args>(__args)...))
2310 { }
2311
2312 ~_Auto_node()
2313 {
2314 if (_M_node)
2315 _M_t._M_drop_node(_M_node);
2316 }
2317
2318 _Auto_node(_Auto_node&& __n)
2319 : _M_t(__n._M_t), _M_node(__n._M_node)
2320 { __n._M_node = nullptr; }
2321
2322 const _Key&
2323 _M_key() const
2324 { return _S_key(_M_node); }
2325
2326 iterator
2327 _M_insert(pair<_Base_ptr, _Base_ptr> __p)
2328 {
2329 auto __it = _M_t._M_insert_node(__p.first, __p.second, _M_node);
2330 _M_node = nullptr;
2331 return __it;
2332 }
2333
2334 iterator
2335 _M_insert_equal_lower()
2336 {
2337 auto __it = _M_t._M_insert_equal_lower_node(_M_node);
2338 _M_node = nullptr;
2339 return __it;
2340 }
2341
2342 _Rb_tree& _M_t;
2343 _Node_ptr _M_node;
2344 };
2345#endif // C++11
2346 };
2347
2348 template<typename _Key, typename _Val, typename _KeyOfValue,
2349 typename _Compare, typename _Alloc>
2350 inline void
2351 swap(_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>& __x,
2352 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>& __y)
2353 { __x.swap(__y); }
2354
2355#if __cplusplus >= 201103L
2356 template<typename _Key, typename _Val, typename _KeyOfValue,
2357 typename _Compare, typename _Alloc>
2358 void
2359 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2360 _M_move_data(_Rb_tree& __x, false_type)
2361 {
2362 if (_M_get_Node_allocator() == __x._M_get_Node_allocator())
2363 _M_move_data(__x, true_type());
2364 else
2365 {
2366 constexpr bool __move = !__move_if_noexcept_cond<value_type>::value;
2367 _Alloc_node __an(*this);
2368 _M_root() = _M_copy<__move>(__x, __an);
2369#pragma GCC diagnostic push
2370#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
2371 if constexpr (__move)
2372 __x.clear();
2373#pragma GCC diagnostic pop
2374 }
2375 }
2376
2377 template<typename _Key, typename _Val, typename _KeyOfValue,
2378 typename _Compare, typename _Alloc>
2379 inline void
2380 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2381 _M_move_assign(_Rb_tree& __x, true_type)
2382 {
2383 clear();
2384 if (__x._M_root())
2385 _M_move_data(__x, true_type());
2386 std::__alloc_on_move(_M_get_Node_allocator(),
2387 __x._M_get_Node_allocator());
2388 }
2389
2390 template<typename _Key, typename _Val, typename _KeyOfValue,
2391 typename _Compare, typename _Alloc>
2392 void
2393 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2394 _M_move_assign(_Rb_tree& __x, false_type)
2395 {
2396 if (_M_get_Node_allocator() == __x._M_get_Node_allocator())
2397 return _M_move_assign(__x, true_type{});
2398
2399 // Try to move each node reusing existing nodes and copying __x nodes
2400 // structure.
2401 _Reuse_or_alloc_node __roan(*this);
2402 _M_impl._M_reset();
2403 if (__x._M_root())
2404 {
2405 _M_root() = _M_copy<__as_rvalue>(__x, __roan);
2406 __x.clear();
2407 }
2408 }
2409
2410 template<typename _Key, typename _Val, typename _KeyOfValue,
2411 typename _Compare, typename _Alloc>
2412 inline _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>&
2413 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2414 operator=(_Rb_tree&& __x)
2415 noexcept(_Node_alloc_traits::_S_nothrow_move()
2417 {
2418 _M_impl._M_key_compare = std::move(__x._M_impl._M_key_compare);
2419 _M_move_assign(__x,
2420 __bool_constant<_Node_alloc_traits::_S_nothrow_move()>());
2421 return *this;
2422 }
2423
2424 template<typename _Key, typename _Val, typename _KeyOfValue,
2425 typename _Compare, typename _Alloc>
2426 template<typename _Iterator>
2427 void
2428 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2429 _M_assign_unique(_Iterator __first, _Iterator __last)
2430 {
2431 _Reuse_or_alloc_node __roan(*this);
2432 _M_impl._M_reset();
2433 for (; __first != __last; ++__first)
2434 _M_insert_unique_(end(), *__first, __roan);
2435 }
2436
2437 template<typename _Key, typename _Val, typename _KeyOfValue,
2438 typename _Compare, typename _Alloc>
2439 template<typename _Iterator>
2440 void
2441 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2442 _M_assign_equal(_Iterator __first, _Iterator __last)
2443 {
2444 _Reuse_or_alloc_node __roan(*this);
2445 _M_impl._M_reset();
2446 for (; __first != __last; ++__first)
2447 _M_insert_equal_(end(), *__first, __roan);
2448 }
2449#endif
2450
2451 template<typename _Key, typename _Val, typename _KeyOfValue,
2452 typename _Compare, typename _Alloc>
2453 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>&
2454 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2455 operator=(const _Rb_tree& __x)
2456 {
2457 if (this != std::__addressof(__x))
2458 {
2459 // Note that _Key may be a constant type.
2460#if __cplusplus >= 201103L
2461 if (_Node_alloc_traits::_S_propagate_on_copy_assign())
2462 {
2463 auto& __this_alloc = this->_M_get_Node_allocator();
2464 auto& __that_alloc = __x._M_get_Node_allocator();
2465 if (!_Node_alloc_traits::_S_always_equal()
2466 && __this_alloc != __that_alloc)
2467 {
2468 // Replacement allocator cannot free existing storage, we need
2469 // to erase nodes first.
2470 clear();
2471 std::__alloc_on_copy(__this_alloc, __that_alloc);
2472 }
2473 }
2474#endif
2475
2476 _Reuse_or_alloc_node __roan(*this);
2477 _M_impl._M_reset();
2478 _M_impl._M_key_compare = __x._M_impl._M_key_compare;
2479 if (__x._M_root())
2480 _M_root() = _M_copy<__as_lvalue>(__x, __roan);
2481 }
2482
2483 return *this;
2484 }
2485
2486 template<typename _Key, typename _Val, typename _KeyOfValue,
2487 typename _Compare, typename _Alloc>
2488#if __cplusplus >= 201103L
2489 template<typename _Arg, typename _NodeGen>
2490#else
2491 template<typename _NodeGen>
2492#endif
2493 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
2494 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2495 _M_insert_(_Base_ptr __x, _Base_ptr __p,
2496#if __cplusplus >= 201103L
2497 _Arg&& __v,
2498#else
2499 const _Val& __v,
2500#endif
2501 _NodeGen& __node_gen)
2502 {
2503 bool __insert_left = (__x || __p == _M_end()
2504 || _M_key_compare(_KeyOfValue()(__v),
2505 _S_key(__p)));
2506
2507 _Base_ptr __z =
2508 __node_gen(_GLIBCXX_FORWARD(_Arg, __v))->_M_base_ptr();
2509
2510 _Node_traits::_S_insert_and_rebalance
2511 (__insert_left, __z, __p, this->_M_impl._M_header);
2512 ++_M_impl._M_node_count;
2513 return iterator(__z);
2514 }
2515
2516 template<typename _Key, typename _Val, typename _KeyOfValue,
2517 typename _Compare, typename _Alloc>
2518#if __cplusplus >= 201103L
2519 template<typename _Arg>
2520#endif
2521 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
2522 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2523#if __cplusplus >= 201103L
2524 _M_insert_lower(_Base_ptr __p, _Arg&& __v)
2525#else
2526 _M_insert_lower(_Base_ptr __p, const _Val& __v)
2527#endif
2528 {
2529 bool __insert_left = (__p == _M_end()
2530 || !_M_key_compare(_S_key(__p),
2531 _KeyOfValue()(__v)));
2532
2533 _Base_ptr __z =
2534 _M_create_node(_GLIBCXX_FORWARD(_Arg, __v))->_M_base_ptr();
2535 _Node_traits::_S_insert_and_rebalance
2536 (__insert_left, __z, __p, this->_M_impl._M_header);
2537 ++_M_impl._M_node_count;
2538 return iterator(__z);
2539 }
2540
2541 template<typename _Key, typename _Val, typename _KeyOfValue,
2542 typename _Compare, typename _Alloc>
2543#if __cplusplus >= 201103L
2544 template<typename _Arg>
2545#endif
2546 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
2547 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2548#if __cplusplus >= 201103L
2549 _M_insert_equal_lower(_Arg&& __v)
2550#else
2551 _M_insert_equal_lower(const _Val& __v)
2552#endif
2553 {
2554 _Base_ptr __x = _M_begin();
2555 _Base_ptr __y = _M_end();
2556 while (__x)
2557 {
2558 __y = __x;
2559 __x = !_M_key_compare(_S_key(__x), _KeyOfValue()(__v)) ?
2560 _S_left(__x) : _S_right(__x);
2561 }
2562 return _M_insert_lower(__y, _GLIBCXX_FORWARD(_Arg, __v));
2563 }
2564
2565 template<typename _Key, typename _Val, typename _KoV,
2566 typename _Compare, typename _Alloc>
2567 template<bool _MoveValues, typename _NodeGen>
2568 typename _Rb_tree<_Key, _Val, _KoV, _Compare, _Alloc>::_Base_ptr
2569 _Rb_tree<_Key, _Val, _KoV, _Compare, _Alloc>::
2570 _M_copy(_Node_ptr __x, _Base_ptr __p, _NodeGen& __node_gen)
2571 {
2572 // Structural copy. __x and __p must be non-null.
2573 _Node_ptr __top = _M_clone_node<_MoveValues>(__x, __node_gen);
2574 _Base_ptr __top_base = __top->_M_base_ptr();
2575 __top->_M_parent = __p;
2576
2577 __try
2578 {
2579 if (__x->_M_right)
2580 __top->_M_right =
2581 _M_copy<_MoveValues>(_S_right(__x), __top_base, __node_gen);
2582 __p = __top_base;
2583 __x = _S_left(__x);
2584
2585 while (__x)
2586 {
2587 _Base_ptr __y =
2588 _M_clone_node<_MoveValues>(__x, __node_gen)->_M_base_ptr();
2589 __p->_M_left = __y;
2590 __y->_M_parent = __p;
2591 if (__x->_M_right)
2592 __y->_M_right = _M_copy<_MoveValues>(_S_right(__x),
2593 __y, __node_gen);
2594 __p = __y;
2595 __x = _S_left(__x);
2596 }
2597 }
2598 __catch(...)
2599 {
2600 _M_erase(__top);
2601 __throw_exception_again;
2602 }
2603 return __top_base;
2604 }
2605
2606 template<typename _Key, typename _Val, typename _KeyOfValue,
2607 typename _Compare, typename _Alloc>
2608 void
2609 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2610 _M_erase(_Node_ptr __x)
2611 {
2612 // Erase without rebalancing.
2613 while (__x)
2614 {
2615 _M_erase(_S_right(__x));
2616 _Node_ptr __y = _S_left(__x);
2617 _M_drop_node(__x);
2618 __x = __y;
2619 }
2620 }
2621
2622 template<typename _Key, typename _Val, typename _KeyOfValue,
2623 typename _Compare, typename _Alloc>
2624 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2625 _Compare, _Alloc>::_Base_ptr
2626 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2627 _M_lower_bound(_Base_ptr __x, _Base_ptr __y,
2628 const _Key& __k) const
2629 {
2630 while (__x)
2631 if (!_M_key_compare(_S_key(__x), __k))
2632 __y = __x, __x = _S_left(__x);
2633 else
2634 __x = _S_right(__x);
2635 return __y;
2636 }
2637
2638 template<typename _Key, typename _Val, typename _KeyOfValue,
2639 typename _Compare, typename _Alloc>
2640 template <typename _Kt>
2641 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::_Base_ptr
2642 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2643 _M_lower_bound_tr(_Base_ptr __x, _Base_ptr __y, const _Kt& __k) const
2644 {
2645 while (__x)
2646 if (!_M_key_compare(_S_key(__x), __k))
2647 __y = __x, __x = _S_left(__x);
2648 else
2649 __x = _S_right(__x);
2650 return __y;
2651 }
2652
2653 template<typename _Key, typename _Val, typename _KeyOfValue,
2654 typename _Compare, typename _Alloc>
2655 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2656 _Compare, _Alloc>::_Base_ptr
2657 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2658 _M_upper_bound(_Base_ptr __x, _Base_ptr __y,
2659 const _Key& __k) const
2660 {
2661 while (__x)
2662 if (_M_key_compare(__k, _S_key(__x)))
2663 __y = __x, __x = _S_left(__x);
2664 else
2665 __x = _S_right(__x);
2666 return __y;
2667 }
2668
2669 template<typename _Key, typename _Val, typename _KeyOfValue,
2670 typename _Compare, typename _Alloc>
2671 template <typename _Kt>
2672 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::_Base_ptr
2673 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2674 _M_upper_bound_tr(_Base_ptr __x, _Base_ptr __y, const _Kt& __k) const
2675 {
2676 while (__x)
2677 if (_M_key_compare(__k, _S_key(__x)))
2678 __y = __x, __x = _S_left(__x);
2679 else
2680 __x = _S_right(__x);
2681 return __y;
2682 }
2683
2684 template<typename _Key, typename _Val, typename _KeyOfValue,
2685 typename _Compare, typename _Alloc>
2686 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2687 _Compare, _Alloc>::iterator,
2688 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2689 _Compare, _Alloc>::iterator>
2690 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2691 equal_range(const _Key& __k)
2692 {
2693 typedef pair<iterator, iterator> _Ret;
2694
2695 _Base_ptr __x = _M_begin();
2696 _Base_ptr __y = _M_end();
2697 while (__x)
2698 {
2699 if (_M_key_compare(_S_key(__x), __k))
2700 __x = _S_right(__x);
2701 else if (_M_key_compare(__k, _S_key(__x)))
2702 __y = __x, __x = _S_left(__x);
2703 else
2704 {
2705 _Base_ptr __xu(__x);
2706 _Base_ptr __yu(__y);
2707 __y = __x, __x = _S_left(__x);
2708 __xu = _S_right(__xu);
2709 return _Ret(iterator(_M_lower_bound(__x, __y, __k)),
2710 iterator(_M_upper_bound(__xu, __yu, __k)));
2711 }
2712 }
2713 return _Ret(iterator(__y), iterator(__y));
2714 }
2715
2716 template<typename _Key, typename _Val, typename _KeyOfValue,
2717 typename _Compare, typename _Alloc>
2718 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2719 _Compare, _Alloc>::const_iterator,
2720 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2721 _Compare, _Alloc>::const_iterator>
2722 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2723 equal_range(const _Key& __k) const
2724 {
2726
2727 _Base_ptr __x = _M_begin();
2728 _Base_ptr __y = _M_end();
2729 while (__x)
2730 {
2731 if (_M_key_compare(_S_key(__x), __k))
2732 __x = _S_right(__x);
2733 else if (_M_key_compare(__k, _S_key(__x)))
2734 __y = __x, __x = _S_left(__x);
2735 else
2736 {
2737 _Base_ptr __xu(__x);
2738 _Base_ptr __yu(__y);
2739 __y = __x, __x = _S_left(__x);
2740 __xu = _S_right(__xu);
2741 return _Ret(const_iterator(_M_lower_bound(__x, __y, __k)),
2742 const_iterator(_M_upper_bound(__xu, __yu, __k)));
2743 }
2744 }
2745 return _Ret(const_iterator(__y), const_iterator(__y));
2746 }
2747
2748 template<typename _Key, typename _Val, typename _KeyOfValue,
2749 typename _Compare, typename _Alloc>
2750 void
2751 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2752 swap(_Rb_tree& __t)
2753 _GLIBCXX_NOEXCEPT_IF(__is_nothrow_swappable<_Compare>::value)
2754 {
2755 if (!_M_root())
2756 {
2757 if (__t._M_root())
2758 _M_impl._M_move_data(__t._M_impl);
2759 }
2760 else if (!__t._M_root())
2761 __t._M_impl._M_move_data(_M_impl);
2762 else
2763 {
2764 std::swap(_M_root(),__t._M_root());
2765 std::swap(_M_leftmost(),__t._M_leftmost());
2766 std::swap(_M_rightmost(),__t._M_rightmost());
2767
2768 _M_root()->_M_parent = _M_end();
2769 __t._M_root()->_M_parent = __t._M_end();
2770 std::swap(this->_M_impl._M_node_count, __t._M_impl._M_node_count);
2771 }
2772 // No need to swap header's color as it does not change.
2773
2774 using std::swap;
2775 swap(this->_M_impl._M_key_compare, __t._M_impl._M_key_compare);
2776
2777 _Node_alloc_traits::_S_on_swap(_M_get_Node_allocator(),
2778 __t._M_get_Node_allocator());
2779 }
2780
2781 template<typename _Key, typename _Val, typename _KeyOfValue,
2782 typename _Compare, typename _Alloc>
2783 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2784 _Compare, _Alloc>::_Base_ptr,
2785 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2786 _Compare, _Alloc>::_Base_ptr>
2787 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2788 _M_get_insert_unique_pos(const key_type& __k)
2789 {
2790 typedef pair<_Base_ptr, _Base_ptr> _Res;
2791 _Base_ptr __x = _M_begin();
2792 _Base_ptr __y = _M_end();
2793 bool __comp = true;
2794 while (__x)
2795 {
2796 __y = __x;
2797 __comp = _M_key_compare(__k, _S_key(__x));
2798 __x = __comp ? _S_left(__x) : _S_right(__x);
2799 }
2800 iterator __j = iterator(__y);
2801 if (__comp)
2802 {
2803 if (__j == begin())
2804 return _Res(__x, __y);
2805 else
2806 --__j;
2807 }
2808 if (_M_key_compare(_S_key(__j._M_node), __k))
2809 return _Res(__x, __y);
2810 return _Res(__j._M_node, _Base_ptr());
2811 }
2812
2813 template<typename _Key, typename _Val, typename _KeyOfValue,
2814 typename _Compare, typename _Alloc>
2815 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2816 _Compare, _Alloc>::_Base_ptr,
2817 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2818 _Compare, _Alloc>::_Base_ptr>
2819 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2820 _M_get_insert_equal_pos(const key_type& __k)
2821 {
2822 typedef pair<_Base_ptr, _Base_ptr> _Res;
2823 _Base_ptr __x = _M_begin();
2824 _Base_ptr __y = _M_end();
2825 while (__x)
2826 {
2827 __y = __x;
2828 __x = _M_key_compare(__k, _S_key(__x)) ? _S_left(__x) : _S_right(__x);
2829 }
2830 return _Res(__x, __y);
2831 }
2832
2833 template<typename _Key, typename _Val, typename _KeyOfValue,
2834 typename _Compare, typename _Alloc>
2835#if __cplusplus >= 201103L
2836 template<typename _Arg>
2837#endif
2838 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2839 _Compare, _Alloc>::iterator, bool>
2840 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2841#if __cplusplus >= 201103L
2842 _M_insert_unique(_Arg&& __v)
2843#else
2844 _M_insert_unique(const _Val& __v)
2845#endif
2846 {
2847 typedef pair<iterator, bool> _Res;
2849 = _M_get_insert_unique_pos(_KeyOfValue()(__v));
2850
2851 if (__res.second)
2852 {
2853 _Alloc_node __an(*this);
2854 return _Res(_M_insert_(__res.first, __res.second,
2855 _GLIBCXX_FORWARD(_Arg, __v), __an),
2856 true);
2857 }
2858
2859 return _Res(iterator(__res.first), false);
2860 }
2861
2862 template<typename _Key, typename _Val, typename _KeyOfValue,
2863 typename _Compare, typename _Alloc>
2864#if __cplusplus >= 201103L
2865 template<typename _Arg>
2866#endif
2867 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
2868 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2869#if __cplusplus >= 201103L
2870 _M_insert_equal(_Arg&& __v)
2871#else
2872 _M_insert_equal(const _Val& __v)
2873#endif
2874 {
2876 = _M_get_insert_equal_pos(_KeyOfValue()(__v));
2877 _Alloc_node __an(*this);
2878 return _M_insert_(__res.first, __res.second,
2879 _GLIBCXX_FORWARD(_Arg, __v), __an);
2880 }
2881
2882 template<typename _Key, typename _Val, typename _KeyOfValue,
2883 typename _Compare, typename _Alloc>
2884 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2885 _Compare, _Alloc>::_Base_ptr,
2886 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2887 _Compare, _Alloc>::_Base_ptr>
2888 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2889 _M_get_insert_hint_unique_pos(const_iterator __position,
2890 const key_type& __k)
2891 {
2892 typedef pair<_Base_ptr, _Base_ptr> _Res;
2893
2894 // end()
2895 if (__position._M_node == _M_end())
2896 {
2897 if (size() > 0 && _M_key_compare(_S_key(_M_rightmost()), __k))
2898 return _Res(_Base_ptr(), _M_rightmost());
2899 else
2900 return _M_get_insert_unique_pos(__k);
2901 }
2902 else if (_M_key_compare(__k, _S_key(__position._M_node)))
2903 {
2904 // First, try before...
2905 iterator __before(__position._M_node);
2906 if (__position._M_node == _M_leftmost()) // begin()
2907 return _Res(_M_leftmost(), _M_leftmost());
2908 else if (_M_key_compare(_S_key((--__before)._M_node), __k))
2909 {
2910 if (!_S_right(__before._M_node))
2911 return _Res(_Base_ptr(), __before._M_node);
2912 else
2913 return _Res(__position._M_node, __position._M_node);
2914 }
2915 else
2916 return _M_get_insert_unique_pos(__k);
2917 }
2918 else if (_M_key_compare(_S_key(__position._M_node), __k))
2919 {
2920 // ... then try after.
2921 iterator __after(__position._M_node);
2922 if (__position._M_node == _M_rightmost())
2923 return _Res(_Base_ptr(), _M_rightmost());
2924 else if (_M_key_compare(__k, _S_key((++__after)._M_node)))
2925 {
2926 if (!_S_right(__position._M_node))
2927 return _Res(_Base_ptr(), __position._M_node);
2928 else
2929 return _Res(__after._M_node, __after._M_node);
2930 }
2931 else
2932 return _M_get_insert_unique_pos(__k);
2933 }
2934 else
2935 // Equivalent keys.
2936 return _Res(__position._M_node, _Base_ptr());
2937 }
2938
2939 template<typename _Key, typename _Val, typename _KeyOfValue,
2940 typename _Compare, typename _Alloc>
2941#if __cplusplus >= 201103L
2942 template<typename _Arg, typename _NodeGen>
2943#else
2944 template<typename _NodeGen>
2945#endif
2946 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
2947 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2948 _M_insert_unique_(const_iterator __position,
2949#if __cplusplus >= 201103L
2950 _Arg&& __v,
2951#else
2952 const _Val& __v,
2953#endif
2954 _NodeGen& __node_gen)
2955 {
2957 = _M_get_insert_hint_unique_pos(__position, _KeyOfValue()(__v));
2958
2959 if (__res.second)
2960 return _M_insert_(__res.first, __res.second,
2961 _GLIBCXX_FORWARD(_Arg, __v),
2962 __node_gen);
2963 return iterator(__res.first);
2964 }
2965
2966 template<typename _Key, typename _Val, typename _KeyOfValue,
2967 typename _Compare, typename _Alloc>
2968 pair<typename _Rb_tree<_Key, _Val, _KeyOfValue,
2969 _Compare, _Alloc>::_Base_ptr,
2970 typename _Rb_tree<_Key, _Val, _KeyOfValue,
2971 _Compare, _Alloc>::_Base_ptr>
2972 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
2973 _M_get_insert_hint_equal_pos(const_iterator __position, const key_type& __k)
2974 {
2975 typedef pair<_Base_ptr, _Base_ptr> _Res;
2976
2977 // end()
2978 if (__position._M_node == _M_end())
2979 {
2980 if (size() > 0
2981 && !_M_key_compare(__k, _S_key(_M_rightmost())))
2982 return _Res(_Base_ptr(), _M_rightmost());
2983 else
2984 return _M_get_insert_equal_pos(__k);
2985 }
2986 else if (!_M_key_compare(_S_key(__position._M_node), __k))
2987 {
2988 // First, try before...
2989 iterator __before(__position._M_node);
2990 if (__position._M_node == _M_leftmost()) // begin()
2991 return _Res(_M_leftmost(), _M_leftmost());
2992 else if (!_M_key_compare(__k, _S_key((--__before)._M_node)))
2993 {
2994 if (!_S_right(__before._M_node))
2995 return _Res(_Base_ptr(), __before._M_node);
2996 else
2997 return _Res(__position._M_node, __position._M_node);
2998 }
2999 else
3000 return _M_get_insert_equal_pos(__k);
3001 }
3002 else
3003 {
3004 // ... then try after.
3005 iterator __after(__position._M_node);
3006 if (__position._M_node == _M_rightmost())
3007 return _Res(_Base_ptr(), _M_rightmost());
3008 else if (!_M_key_compare(_S_key((++__after)._M_node), __k))
3009 {
3010 if (!_S_right(__position._M_node))
3011 return _Res(_Base_ptr(), __position._M_node);
3012 else
3013 return _Res(__after._M_node, __after._M_node);
3014 }
3015 else
3016 return _Res(_Base_ptr(), _Base_ptr());
3017 }
3018 }
3019
3020 template<typename _Key, typename _Val, typename _KeyOfValue,
3021 typename _Compare, typename _Alloc>
3022#if __cplusplus >= 201103L
3023 template<typename _Arg, typename _NodeGen>
3024#else
3025 template<typename _NodeGen>
3026#endif
3027 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator
3028 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3029 _M_insert_equal_(const_iterator __position,
3030#if __cplusplus >= 201103L
3031 _Arg&& __v,
3032#else
3033 const _Val& __v,
3034#endif
3035 _NodeGen& __node_gen)
3036 {
3038 = _M_get_insert_hint_equal_pos(__position, _KeyOfValue()(__v));
3039
3040 if (__res.second)
3041 return _M_insert_(__res.first, __res.second,
3042 _GLIBCXX_FORWARD(_Arg, __v),
3043 __node_gen);
3044
3045 return _M_insert_equal_lower(_GLIBCXX_FORWARD(_Arg, __v));
3046 }
3047
3048#if __cplusplus >= 201103L
3049 template<typename _Key, typename _Val, typename _KeyOfValue,
3050 typename _Compare, typename _Alloc>
3051 auto
3052 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3053 _M_insert_node(_Base_ptr __x, _Base_ptr __p, _Node_ptr __z)
3054 -> iterator
3055 {
3056 bool __insert_left = (__x || __p == _M_end()
3057 || _M_key_compare(_S_key(__z), _S_key(__p)));
3058
3059 _Base_ptr __base_z = __z->_M_base_ptr();
3060 _Node_traits::_S_insert_and_rebalance
3061 (__insert_left, __base_z, __p, this->_M_impl._M_header);
3062 ++_M_impl._M_node_count;
3063 return iterator(__base_z);
3064 }
3065
3066 template<typename _Key, typename _Val, typename _KeyOfValue,
3067 typename _Compare, typename _Alloc>
3068 auto
3069 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3070 _M_insert_lower_node(_Base_ptr __p, _Node_ptr __z)
3071 -> iterator
3072 {
3073 bool __insert_left = (__p == _M_end()
3074 || !_M_key_compare(_S_key(__p), _S_key(__z)));
3075
3076 _Base_ptr __base_z = __z->_M_base_ptr();
3077 _Node_traits::_S_insert_and_rebalance
3078 (__insert_left, __base_z, __p, this->_M_impl._M_header);
3079 ++_M_impl._M_node_count;
3080 return iterator(__base_z);
3081 }
3082
3083 template<typename _Key, typename _Val, typename _KeyOfValue,
3084 typename _Compare, typename _Alloc>
3085 auto
3086 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3087 _M_insert_equal_lower_node(_Node_ptr __z)
3088 -> iterator
3089 {
3090 _Base_ptr __x = _M_begin();
3091 _Base_ptr __y = _M_end();
3092 while (__x)
3093 {
3094 __y = __x;
3095 __x = !_M_key_compare(_S_key(__x), _S_key(__z)) ?
3096 _S_left(__x) : _S_right(__x);
3097 }
3098 return _M_insert_lower_node(__y, __z);
3099 }
3100
3101 template<typename _Key, typename _Val, typename _KeyOfValue,
3102 typename _Compare, typename _Alloc>
3103 template<typename... _Args>
3104 auto
3105 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3106 _M_emplace_unique(_Args&&... __args)
3108 {
3109 _Auto_node __z(*this, std::forward<_Args>(__args)...);
3110 auto __res = _M_get_insert_unique_pos(__z._M_key());
3111 if (__res.second)
3112 return {__z._M_insert(__res), true};
3113 return {iterator(__res.first), false};
3114 }
3115
3116 template<typename _Key, typename _Val, typename _KeyOfValue,
3117 typename _Compare, typename _Alloc>
3118 template<typename... _Args>
3119 auto
3120 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3121 _M_emplace_equal(_Args&&... __args)
3122 -> iterator
3123 {
3124 _Auto_node __z(*this, std::forward<_Args>(__args)...);
3125 auto __res = _M_get_insert_equal_pos(__z._M_key());
3126 return __z._M_insert(__res);
3127 }
3128
3129 template<typename _Key, typename _Val, typename _KeyOfValue,
3130 typename _Compare, typename _Alloc>
3131 template<typename... _Args>
3132 auto
3133 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3134 _M_emplace_hint_unique(const_iterator __pos, _Args&&... __args)
3135 -> iterator
3136 {
3137 _Auto_node __z(*this, std::forward<_Args>(__args)...);
3138 auto __res = _M_get_insert_hint_unique_pos(__pos, __z._M_key());
3139 if (__res.second)
3140 return __z._M_insert(__res);
3141 return iterator(__res.first);
3142 }
3143
3144 template<typename _Key, typename _Val, typename _KeyOfValue,
3145 typename _Compare, typename _Alloc>
3146 template<typename... _Args>
3147 auto
3148 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3149 _M_emplace_hint_equal(const_iterator __pos, _Args&&... __args)
3150 -> iterator
3151 {
3152 _Auto_node __z(*this, std::forward<_Args>(__args)...);
3153 auto __res = _M_get_insert_hint_equal_pos(__pos, __z._M_key());
3154 if (__res.second)
3155 return __z._M_insert(__res);
3156 return __z._M_insert_equal_lower();
3157 }
3158#endif
3159
3160
3161 template<typename _Key, typename _Val, typename _KeyOfValue,
3162 typename _Compare, typename _Alloc>
3163 void
3164 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3165 _M_erase_aux(const_iterator __position)
3166 {
3167 _Base_ptr __y = _Node_traits::_S_rebalance_for_erase
3168 (__position._M_node, this->_M_impl._M_header);
3169 _M_drop_node(static_cast<_Node&>(*__y)._M_node_ptr());
3170 --_M_impl._M_node_count;
3171 }
3172
3173 template<typename _Key, typename _Val, typename _KeyOfValue,
3174 typename _Compare, typename _Alloc>
3175 void
3176 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3177 _M_erase_aux(const_iterator __first, const_iterator __last)
3178 {
3179 if (__first == begin() && __last == end())
3180 clear();
3181 else
3182 while (__first != __last)
3183 _M_erase_aux(__first++);
3184 }
3185
3186 template<typename _Key, typename _Val, typename _KeyOfValue,
3187 typename _Compare, typename _Alloc>
3188 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::size_type
3189 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3190 erase(const _Key& __x)
3191 {
3192 pair<iterator, iterator> __p = equal_range(__x);
3193 const size_type __old_size = size();
3194 _M_erase_aux(__p.first, __p.second);
3195 return __old_size - size();
3196 }
3197
3198 template<typename _Key, typename _Val, typename _KeyOfValue,
3199 typename _Compare, typename _Alloc>
3200 template <typename _Kt>
3201 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::size_type
3202 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3203 _M_erase_tr(const _Kt& __x)
3204 {
3205 pair<iterator, iterator> __p = _M_equal_range_tr(__x);
3206 const size_type __old_size = size();
3207 _M_erase_aux(__p.first, __p.second);
3208 return __old_size - size();
3209 }
3210
3211 template<typename _Key, typename _Val, typename _KeyOfValue,
3212 typename _Compare, typename _Alloc>
3213 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::size_type
3214 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3215 _M_erase_unique(const _Key& __x)
3216 {
3217 iterator __it = find(__x);
3218 if (__it == end())
3219 return 0;
3220
3221 _M_erase_aux(__it);
3222 return 1;
3223 }
3224
3225 template<typename _Key, typename _Val, typename _KeyOfValue,
3226 typename _Compare, typename _Alloc>
3227 typename _Rb_tree<_Key, _Val, _KeyOfValue,
3228 _Compare, _Alloc>::iterator
3229 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3230 find(const _Key& __k)
3231 {
3232 iterator __j(_M_lower_bound(_M_begin(), _M_end(), __k));
3233 return (__j == end()
3234 || _M_key_compare(__k, _S_key(__j._M_node))) ? end() : __j;
3235 }
3236
3237 template<typename _Key, typename _Val, typename _KeyOfValue,
3238 typename _Compare, typename _Alloc>
3239 typename _Rb_tree<_Key, _Val, _KeyOfValue,
3240 _Compare, _Alloc>::const_iterator
3241 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3242 find(const _Key& __k) const
3243 {
3244 const_iterator __j(_M_lower_bound(_M_begin(), _M_end(), __k));
3245 return (__j == end()
3246 || _M_key_compare(__k, _S_key(__j._M_node))) ? end() : __j;
3247 }
3248
3249 template<typename _Key, typename _Val, typename _KeyOfValue,
3250 typename _Compare, typename _Alloc>
3251 typename _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::size_type
3252 _Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
3253 count(const _Key& __k) const
3254 {
3255 pair<const_iterator, const_iterator> __p = equal_range(__k);
3256 const size_type __n = std::distance(__p.first, __p.second);
3257 return __n;
3258 }
3259
3260 _GLIBCXX_PURE unsigned int
3261 _Rb_tree_black_count(const _Rb_tree_node_base* __node,
3262 const _Rb_tree_node_base* __root) throw ();
3263
3264 template<typename _Key, typename _Val, typename _KeyOfValue,
3265 typename _Compare, typename _Alloc>
3266 bool
3267 _Rb_tree<_Key,_Val,_KeyOfValue,_Compare,_Alloc>::__rb_verify() const
3268 {
3269 if (_M_impl._M_node_count == 0 || begin() == end())
3270 return _M_impl._M_node_count == 0 && begin() == end()
3271 && this->_M_impl._M_header._M_left == _M_end()
3272 && this->_M_impl._M_header._M_right == _M_end();
3273
3274 unsigned int __len = _Rb_tree_black_count(_M_leftmost(), _M_root());
3275 for (const_iterator __it = begin(); __it != end(); ++__it)
3276 {
3277 _Base_ptr __x = __it._M_node;
3278 _Base_ptr __L = _S_left(__x);
3279 _Base_ptr __R = _S_right(__x);
3280
3281 if (__x->_M_color == _S_red)
3282 if ((__L && __L->_M_color == _S_red)
3283 || (__R && __R->_M_color == _S_red))
3284 return false;
3285
3286 if (__L && _M_key_compare(_S_key(__x), _S_key(__L)))
3287 return false;
3288 if (__R && _M_key_compare(_S_key(__R), _S_key(__x)))
3289 return false;
3290
3291 if (!__L && !__R && _Rb_tree_black_count(__x, _M_root()) != __len)
3292 return false;
3293 }
3294
3295 if (_M_leftmost() != _Node_base::_S_minimum(_M_root()))
3296 return false;
3297 if (_M_rightmost() != _Node_base::_S_maximum(_M_root()))
3298 return false;
3299 return true;
3300 }
3301
3302#ifdef __glibcxx_node_extract // >= C++17
3303 // Allow access to internals of compatible _Rb_tree specializations.
3304 template<typename _Key, typename _Val, typename _Sel, typename _Cmp1,
3305 typename _Alloc, typename _Cmp2>
3306 struct _Rb_tree_merge_helper<_Rb_tree<_Key, _Val, _Sel, _Cmp1, _Alloc>,
3307 _Cmp2>
3308 {
3309 private:
3310 friend class _Rb_tree<_Key, _Val, _Sel, _Cmp1, _Alloc>;
3311
3312 static auto&
3313 _S_get_impl(_Rb_tree<_Key, _Val, _Sel, _Cmp2, _Alloc>& __tree)
3314 { return __tree._M_impl; }
3315 };
3316#endif // C++17
3317
3318#ifdef __glibcxx_associative_heterogeneous_erasure // C++ >= 23
3319template <typename _Kt, typename _Container>
3320 concept __heterogeneous_tree_key =
3321 __transparent_comparator<typename _Container::key_compare> &&
3322 __heterogeneous_key<_Kt, _Container>;
3323#endif
3324
3325_GLIBCXX_END_NAMESPACE_VERSION
3326} // namespace
3327
3328#endif
constexpr bool operator<(const duration< _Rep1, _Period1 > &__lhs, const duration< _Rep2, _Period2 > &__rhs)
Definition chrono.h:826
__bool_constant< true > true_type
The type used as a compile-time boolean with true value.
Definition type_traits:119
__bool_constant< false > false_type
The type used as a compile-time boolean with false value.
Definition type_traits:122
pair(_T1, _T2) -> pair< _T1, _T2 >
Two pairs are equal iff their members are equal.
auto declval() noexcept -> decltype(__declval< _Tp >(0))
Definition type_traits:2714
constexpr _Tp * addressof(_Tp &__r) noexcept
Returns the actual address of the object or function referenced by r, even in the presence of an over...
Definition move.h:176
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
Definition move.h:138
constexpr _Tp * __addressof(_Tp &__r) noexcept
Same as C++11 std::addressof.
Definition move.h:52
constexpr _Tp && forward(typename std::remove_reference< _Tp >::type &__t) noexcept
Forward an lvalue.
Definition move.h:72
_Tp * end(valarray< _Tp > &__va) noexcept
Return an iterator pointing to one past the last element of the valarray.
Definition valarray:1251
_Tp * begin(valarray< _Tp > &__va) noexcept
Return an iterator pointing to the first element of the valarray.
Definition valarray:1229
constexpr auto lexicographical_compare_three_way(_InputIter1 __first1, _InputIter1 __last1, _InputIter2 __first2, _InputIter2 __last2, _Comp __comp) -> decltype(__comp(*__first1, *__first2))
Performs dictionary comparison on ranges.
ISO C++ entities toplevel namespace is std.
constexpr iterator_traits< _InputIterator >::difference_type distance(_InputIterator __first, _InputIterator __last)
A generalization of pointer arithmetic.
typename pointer_traits< _Ptr >::template rebind< _Tp > __ptr_rebind
Convenience alias for rebinding pointers.
Definition ptr_traits.h:201
constexpr auto size(const _Container &__cont) noexcept(noexcept(__cont.size())) -> decltype(__cont.size())
Return the size of a container.
constexpr _Iterator __base(_Iterator __it)
is_nothrow_move_assignable
Definition type_traits:1411
The standard allocator, as per C++03 [20.4.1].
Definition allocator.h:134
Struct holding two objects of arbitrary type.
Definition stl_pair.h:304
Common iterator class.
static constexpr pointer allocate(_Node_allocator &__a, size_type __n)
static constexpr void deallocate(_Node_allocator &__a, pointer __p, size_type __n)
static constexpr size_type max_size(const _Node_allocator &__a) noexcept