xref: /freebsd/contrib/llvm-project/libcxx/include/list (revision 035dd78d30ba28a3dc15c05ec85ad10127165677)
1// -*- C++ -*-
2//===----------------------------------------------------------------------===//
3//
4// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5// See https://llvm.org/LICENSE.txt for license information.
6// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7//
8//===----------------------------------------------------------------------===//
9
10#ifndef _LIBCPP_LIST
11#define _LIBCPP_LIST
12
13/*
14    list synopsis
15
16namespace std
17{
18
19template <class T, class Alloc = allocator<T> >
20class list
21{
22public:
23
24    // types:
25    typedef T value_type;
26    typedef Alloc allocator_type;
27    typedef typename allocator_type::reference reference;
28    typedef typename allocator_type::const_reference const_reference;
29    typedef typename allocator_type::pointer pointer;
30    typedef typename allocator_type::const_pointer const_pointer;
31    typedef implementation-defined iterator;
32    typedef implementation-defined const_iterator;
33    typedef implementation-defined size_type;
34    typedef implementation-defined difference_type;
35    typedef reverse_iterator<iterator> reverse_iterator;
36    typedef reverse_iterator<const_iterator> const_reverse_iterator;
37
38    list()
39        noexcept(is_nothrow_default_constructible<allocator_type>::value);
40    explicit list(const allocator_type& a);
41    explicit list(size_type n);
42    explicit list(size_type n, const allocator_type& a); // C++14
43    list(size_type n, const value_type& value);
44    list(size_type n, const value_type& value, const allocator_type& a);
45    template <class Iter>
46        list(Iter first, Iter last);
47    template <class Iter>
48        list(Iter first, Iter last, const allocator_type& a);
49    list(const list& x);
50    list(const list&, const allocator_type& a);
51    list(list&& x)
52        noexcept(is_nothrow_move_constructible<allocator_type>::value);
53    list(list&&, const allocator_type& a);
54    list(initializer_list<value_type>);
55    list(initializer_list<value_type>, const allocator_type& a);
56
57    ~list();
58
59    list& operator=(const list& x);
60    list& operator=(list&& x)
61        noexcept(
62             allocator_type::propagate_on_container_move_assignment::value &&
63             is_nothrow_move_assignable<allocator_type>::value);
64    list& operator=(initializer_list<value_type>);
65    template <class Iter>
66        void assign(Iter first, Iter last);
67    void assign(size_type n, const value_type& t);
68    void assign(initializer_list<value_type>);
69
70    allocator_type get_allocator() const noexcept;
71
72    iterator begin() noexcept;
73    const_iterator begin() const noexcept;
74    iterator end() noexcept;
75    const_iterator end() const noexcept;
76    reverse_iterator rbegin() noexcept;
77    const_reverse_iterator rbegin() const noexcept;
78    reverse_iterator rend() noexcept;
79    const_reverse_iterator rend() const noexcept;
80    const_iterator cbegin() const noexcept;
81    const_iterator cend() const noexcept;
82    const_reverse_iterator crbegin() const noexcept;
83    const_reverse_iterator crend() const noexcept;
84
85    reference front();
86    const_reference front() const;
87    reference back();
88    const_reference back() const;
89
90    bool empty() const noexcept;
91    size_type size() const noexcept;
92    size_type max_size() const noexcept;
93
94    template <class... Args>
95        reference emplace_front(Args&&... args); // reference in C++17
96    void pop_front();
97    template <class... Args>
98        reference emplace_back(Args&&... args);  // reference in C++17
99    void pop_back();
100    void push_front(const value_type& x);
101    void push_front(value_type&& x);
102    void push_back(const value_type& x);
103    void push_back(value_type&& x);
104    template <class... Args>
105        iterator emplace(const_iterator position, Args&&... args);
106    iterator insert(const_iterator position, const value_type& x);
107    iterator insert(const_iterator position, value_type&& x);
108    iterator insert(const_iterator position, size_type n, const value_type& x);
109    template <class Iter>
110        iterator insert(const_iterator position, Iter first, Iter last);
111    iterator insert(const_iterator position, initializer_list<value_type> il);
112
113    iterator erase(const_iterator position);
114    iterator erase(const_iterator position, const_iterator last);
115
116    void resize(size_type sz);
117    void resize(size_type sz, const value_type& c);
118
119    void swap(list&)
120        noexcept(allocator_traits<allocator_type>::is_always_equal::value);  // C++17
121    void clear() noexcept;
122
123    void splice(const_iterator position, list& x);
124    void splice(const_iterator position, list&& x);
125    void splice(const_iterator position, list& x, const_iterator i);
126    void splice(const_iterator position, list&& x, const_iterator i);
127    void splice(const_iterator position, list& x, const_iterator first,
128                                                  const_iterator last);
129    void splice(const_iterator position, list&& x, const_iterator first,
130                                                  const_iterator last);
131
132    size_type remove(const value_type& value);       // void before C++20
133    template <class Pred>
134      size_type remove_if(Pred pred);                // void before C++20
135    size_type unique();                              // void before C++20
136    template <class BinaryPredicate>
137      size_type unique(BinaryPredicate binary_pred); // void before C++20
138    void merge(list& x);
139    void merge(list&& x);
140    template <class Compare>
141        void merge(list& x, Compare comp);
142    template <class Compare>
143        void merge(list&& x, Compare comp);
144    void sort();
145    template <class Compare>
146        void sort(Compare comp);
147    void reverse() noexcept;
148};
149
150
151template <class InputIterator, class Allocator = allocator<typename iterator_traits<InputIterator>::value_type>>
152    list(InputIterator, InputIterator, Allocator = Allocator())
153    -> list<typename iterator_traits<InputIterator>::value_type, Allocator>;  // C++17
154
155template <class T, class Alloc>
156    bool operator==(const list<T,Alloc>& x, const list<T,Alloc>& y);
157template <class T, class Alloc>
158    bool operator< (const list<T,Alloc>& x, const list<T,Alloc>& y);
159template <class T, class Alloc>
160    bool operator!=(const list<T,Alloc>& x, const list<T,Alloc>& y);
161template <class T, class Alloc>
162    bool operator> (const list<T,Alloc>& x, const list<T,Alloc>& y);
163template <class T, class Alloc>
164    bool operator>=(const list<T,Alloc>& x, const list<T,Alloc>& y);
165template <class T, class Alloc>
166    bool operator<=(const list<T,Alloc>& x, const list<T,Alloc>& y);
167
168template <class T, class Alloc>
169    void swap(list<T,Alloc>& x, list<T,Alloc>& y)
170         noexcept(noexcept(x.swap(y)));
171
172template <class T, class Allocator, class U>
173    typename list<T, Allocator>::size_type
174    erase(list<T, Allocator>& c, const U& value);       // C++20
175template <class T, class Allocator, class Predicate>
176    typename list<T, Allocator>::size_type
177    erase_if(list<T, Allocator>& c, Predicate pred);    // C++20
178
179}  // std
180
181*/
182
183#include <__algorithm/comp.h>
184#include <__algorithm/equal.h>
185#include <__algorithm/lexicographical_compare.h>
186#include <__algorithm/min.h>
187#include <__assert> // all public C++ headers provide the assertion handler
188#include <__config>
189#include <__debug>
190#include <__format/enable_insertable.h>
191#include <__iterator/distance.h>
192#include <__iterator/iterator_traits.h>
193#include <__iterator/move_iterator.h>
194#include <__iterator/next.h>
195#include <__iterator/prev.h>
196#include <__iterator/reverse_iterator.h>
197#include <__memory/swap_allocator.h>
198#include <__utility/forward.h>
199#include <__utility/move.h>
200#include <__utility/swap.h>
201#include <limits>
202#include <memory>
203#include <type_traits>
204#include <version>
205
206#ifndef _LIBCPP_REMOVE_TRANSITIVE_INCLUDES
207#  include <algorithm>
208#  include <functional>
209#  include <iterator>
210#endif
211
212// standard-mandated includes
213
214// [iterator.range]
215#include <__iterator/access.h>
216#include <__iterator/data.h>
217#include <__iterator/empty.h>
218#include <__iterator/reverse_access.h>
219#include <__iterator/size.h>
220
221// [list.syn]
222#include <compare>
223#include <initializer_list>
224
225#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
226#  pragma GCC system_header
227#endif
228
229_LIBCPP_PUSH_MACROS
230#include <__undef_macros>
231
232
233_LIBCPP_BEGIN_NAMESPACE_STD
234
235template <class _Tp, class _VoidPtr> struct __list_node;
236template <class _Tp, class _VoidPtr> struct __list_node_base;
237
238template <class _Tp, class _VoidPtr>
239struct __list_node_pointer_traits {
240  typedef typename __rebind_pointer<_VoidPtr, __list_node<_Tp, _VoidPtr> >::type
241        __node_pointer;
242  typedef typename __rebind_pointer<_VoidPtr, __list_node_base<_Tp, _VoidPtr> >::type
243        __base_pointer;
244
245#if defined(_LIBCPP_ABI_LIST_REMOVE_NODE_POINTER_UB)
246  typedef __base_pointer __link_pointer;
247#else
248  typedef typename conditional<
249          is_pointer<_VoidPtr>::value,
250          __base_pointer,
251          __node_pointer
252  >::type __link_pointer;
253#endif
254
255  typedef typename conditional<
256          is_same<__link_pointer, __node_pointer>::value,
257          __base_pointer,
258          __node_pointer
259  >::type __non_link_pointer;
260
261  static _LIBCPP_INLINE_VISIBILITY
262  __link_pointer __unsafe_link_pointer_cast(__link_pointer __p) {
263      return __p;
264  }
265
266  static _LIBCPP_INLINE_VISIBILITY
267  __link_pointer __unsafe_link_pointer_cast(__non_link_pointer __p) {
268      return static_cast<__link_pointer>(static_cast<_VoidPtr>(__p));
269  }
270
271};
272
273template <class _Tp, class _VoidPtr>
274struct __list_node_base
275{
276    typedef __list_node_pointer_traits<_Tp, _VoidPtr> _NodeTraits;
277    typedef typename _NodeTraits::__node_pointer __node_pointer;
278    typedef typename _NodeTraits::__base_pointer __base_pointer;
279    typedef typename _NodeTraits::__link_pointer __link_pointer;
280
281    __link_pointer __prev_;
282    __link_pointer __next_;
283
284    _LIBCPP_INLINE_VISIBILITY
285    __list_node_base() : __prev_(_NodeTraits::__unsafe_link_pointer_cast(__self())),
286                         __next_(_NodeTraits::__unsafe_link_pointer_cast(__self())) {}
287
288    _LIBCPP_INLINE_VISIBILITY
289    __base_pointer __self() {
290        return pointer_traits<__base_pointer>::pointer_to(*this);
291    }
292
293    _LIBCPP_INLINE_VISIBILITY
294    __node_pointer __as_node() {
295        return static_cast<__node_pointer>(__self());
296    }
297};
298
299template <class _Tp, class _VoidPtr>
300struct _LIBCPP_STANDALONE_DEBUG __list_node
301    : public __list_node_base<_Tp, _VoidPtr>
302{
303    _Tp __value_;
304
305    typedef __list_node_base<_Tp, _VoidPtr> __base;
306    typedef typename __base::__link_pointer __link_pointer;
307
308    _LIBCPP_INLINE_VISIBILITY
309    __link_pointer __as_link() {
310        return static_cast<__link_pointer>(__base::__self());
311    }
312};
313
314template <class _Tp, class _Alloc = allocator<_Tp> > class _LIBCPP_TEMPLATE_VIS list;
315template <class _Tp, class _Alloc> class __list_imp;
316template <class _Tp, class _VoidPtr> class _LIBCPP_TEMPLATE_VIS __list_const_iterator;
317
318template <class _Tp, class _VoidPtr>
319class _LIBCPP_TEMPLATE_VIS __list_iterator
320{
321    typedef __list_node_pointer_traits<_Tp, _VoidPtr> _NodeTraits;
322    typedef typename _NodeTraits::__link_pointer __link_pointer;
323
324    __link_pointer __ptr_;
325
326    _LIBCPP_INLINE_VISIBILITY
327    explicit __list_iterator(__link_pointer __p, const void* __c) _NOEXCEPT
328        : __ptr_(__p)
329    {
330        (void)__c;
331#ifdef _LIBCPP_ENABLE_DEBUG_MODE
332        __get_db()->__insert_ic(this, __c);
333#endif
334    }
335
336    template<class, class> friend class list;
337    template<class, class> friend class __list_imp;
338    template<class, class> friend class __list_const_iterator;
339public:
340    typedef bidirectional_iterator_tag       iterator_category;
341    typedef _Tp                              value_type;
342    typedef value_type&                      reference;
343    typedef typename __rebind_pointer<_VoidPtr, value_type>::type pointer;
344    typedef typename pointer_traits<pointer>::difference_type difference_type;
345
346    _LIBCPP_INLINE_VISIBILITY
347    __list_iterator() _NOEXCEPT : __ptr_(nullptr)
348    {
349        _VSTD::__debug_db_insert_i(this);
350    }
351
352#ifdef _LIBCPP_ENABLE_DEBUG_MODE
353
354    _LIBCPP_INLINE_VISIBILITY
355    __list_iterator(const __list_iterator& __p)
356        : __ptr_(__p.__ptr_)
357    {
358        __get_db()->__iterator_copy(this, _VSTD::addressof(__p));
359    }
360
361    _LIBCPP_INLINE_VISIBILITY
362    ~__list_iterator()
363    {
364        __get_db()->__erase_i(this);
365    }
366
367    _LIBCPP_INLINE_VISIBILITY
368    __list_iterator& operator=(const __list_iterator& __p)
369    {
370        if (this != _VSTD::addressof(__p))
371        {
372            __get_db()->__iterator_copy(this, _VSTD::addressof(__p));
373            __ptr_ = __p.__ptr_;
374        }
375        return *this;
376    }
377
378#endif // _LIBCPP_ENABLE_DEBUG_MODE
379
380    _LIBCPP_INLINE_VISIBILITY
381    reference operator*() const
382    {
383        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
384                             "Attempted to dereference a non-dereferenceable list::iterator");
385        return __ptr_->__as_node()->__value_;
386    }
387    _LIBCPP_INLINE_VISIBILITY
388    pointer operator->() const
389    {
390        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
391                             "Attempted to dereference a non-dereferenceable list::iterator");
392        return pointer_traits<pointer>::pointer_to(__ptr_->__as_node()->__value_);
393    }
394
395    _LIBCPP_INLINE_VISIBILITY
396    __list_iterator& operator++()
397    {
398        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
399                             "Attempted to increment a non-incrementable list::iterator");
400        __ptr_ = __ptr_->__next_;
401        return *this;
402    }
403    _LIBCPP_INLINE_VISIBILITY
404    __list_iterator operator++(int) {__list_iterator __t(*this); ++(*this); return __t;}
405
406    _LIBCPP_INLINE_VISIBILITY
407    __list_iterator& operator--()
408    {
409        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__decrementable(this),
410                             "Attempted to decrement a non-decrementable list::iterator");
411        __ptr_ = __ptr_->__prev_;
412        return *this;
413    }
414    _LIBCPP_INLINE_VISIBILITY
415    __list_iterator operator--(int) {__list_iterator __t(*this); --(*this); return __t;}
416
417    friend _LIBCPP_INLINE_VISIBILITY
418    bool operator==(const __list_iterator& __x, const __list_iterator& __y)
419    {
420        return __x.__ptr_ == __y.__ptr_;
421    }
422    friend _LIBCPP_INLINE_VISIBILITY
423     bool operator!=(const __list_iterator& __x, const __list_iterator& __y)
424        {return !(__x == __y);}
425};
426
427template <class _Tp, class _VoidPtr>
428class _LIBCPP_TEMPLATE_VIS __list_const_iterator
429{
430    typedef __list_node_pointer_traits<_Tp, _VoidPtr> _NodeTraits;
431    typedef typename _NodeTraits::__link_pointer __link_pointer;
432
433    __link_pointer __ptr_;
434
435    _LIBCPP_INLINE_VISIBILITY
436    explicit __list_const_iterator(__link_pointer __p, const void* __c) _NOEXCEPT
437        : __ptr_(__p)
438    {
439        (void)__c;
440#ifdef _LIBCPP_ENABLE_DEBUG_MODE
441        __get_db()->__insert_ic(this, __c);
442#endif
443    }
444
445    template<class, class> friend class list;
446    template<class, class> friend class __list_imp;
447public:
448    typedef bidirectional_iterator_tag       iterator_category;
449    typedef _Tp                              value_type;
450    typedef const value_type&                reference;
451    typedef typename __rebind_pointer<_VoidPtr, const value_type>::type pointer;
452    typedef typename pointer_traits<pointer>::difference_type difference_type;
453
454    _LIBCPP_INLINE_VISIBILITY
455    __list_const_iterator() _NOEXCEPT : __ptr_(nullptr)
456    {
457        _VSTD::__debug_db_insert_i(this);
458    }
459    _LIBCPP_INLINE_VISIBILITY
460    __list_const_iterator(const __list_iterator<_Tp, _VoidPtr>& __p) _NOEXCEPT
461        : __ptr_(__p.__ptr_)
462    {
463#ifdef _LIBCPP_ENABLE_DEBUG_MODE
464        __get_db()->__iterator_copy(this, _VSTD::addressof(__p));
465#endif
466    }
467
468#ifdef _LIBCPP_ENABLE_DEBUG_MODE
469
470    _LIBCPP_INLINE_VISIBILITY
471    __list_const_iterator(const __list_const_iterator& __p)
472        : __ptr_(__p.__ptr_)
473    {
474        __get_db()->__iterator_copy(this, _VSTD::addressof(__p));
475    }
476
477    _LIBCPP_INLINE_VISIBILITY
478    ~__list_const_iterator()
479    {
480        __get_db()->__erase_i(this);
481    }
482
483    _LIBCPP_INLINE_VISIBILITY
484    __list_const_iterator& operator=(const __list_const_iterator& __p)
485    {
486        if (this != _VSTD::addressof(__p))
487        {
488            __get_db()->__iterator_copy(this, _VSTD::addressof(__p));
489            __ptr_ = __p.__ptr_;
490        }
491        return *this;
492    }
493
494#endif // _LIBCPP_ENABLE_DEBUG_MODE
495    _LIBCPP_INLINE_VISIBILITY
496    reference operator*() const
497    {
498        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
499                             "Attempted to dereference a non-dereferenceable list::const_iterator");
500        return __ptr_->__as_node()->__value_;
501    }
502    _LIBCPP_INLINE_VISIBILITY
503    pointer operator->() const
504    {
505        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
506                             "Attempted to dereference a non-dereferenceable list::const_iterator");
507        return pointer_traits<pointer>::pointer_to(__ptr_->__as_node()->__value_);
508    }
509
510    _LIBCPP_INLINE_VISIBILITY
511    __list_const_iterator& operator++()
512    {
513        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(this),
514                             "Attempted to increment a non-incrementable list::const_iterator");
515        __ptr_ = __ptr_->__next_;
516        return *this;
517    }
518    _LIBCPP_INLINE_VISIBILITY
519    __list_const_iterator operator++(int) {__list_const_iterator __t(*this); ++(*this); return __t;}
520
521    _LIBCPP_INLINE_VISIBILITY
522    __list_const_iterator& operator--()
523    {
524        _LIBCPP_DEBUG_ASSERT(__get_const_db()->__decrementable(this),
525                             "Attempted to decrement a non-decrementable list::const_iterator");
526        __ptr_ = __ptr_->__prev_;
527        return *this;
528    }
529    _LIBCPP_INLINE_VISIBILITY
530    __list_const_iterator operator--(int) {__list_const_iterator __t(*this); --(*this); return __t;}
531
532    friend _LIBCPP_INLINE_VISIBILITY
533    bool operator==(const __list_const_iterator& __x, const __list_const_iterator& __y)
534    {
535        return __x.__ptr_ == __y.__ptr_;
536    }
537    friend _LIBCPP_INLINE_VISIBILITY
538    bool operator!=(const __list_const_iterator& __x, const __list_const_iterator& __y)
539        {return !(__x == __y);}
540};
541
542template <class _Tp, class _Alloc>
543class __list_imp
544{
545    __list_imp(const __list_imp&);
546    __list_imp& operator=(const __list_imp&);
547public:
548    typedef _Alloc                                                  allocator_type;
549    typedef allocator_traits<allocator_type>                        __alloc_traits;
550    typedef typename __alloc_traits::size_type                      size_type;
551protected:
552    typedef _Tp                                                     value_type;
553    typedef typename __alloc_traits::void_pointer                   __void_pointer;
554    typedef __list_iterator<value_type, __void_pointer>             iterator;
555    typedef __list_const_iterator<value_type, __void_pointer>       const_iterator;
556    typedef __list_node_base<value_type, __void_pointer>            __node_base;
557    typedef __list_node<value_type, __void_pointer>                 __node;
558    typedef typename __rebind_alloc_helper<__alloc_traits, __node>::type __node_allocator;
559    typedef allocator_traits<__node_allocator>                       __node_alloc_traits;
560    typedef typename __node_alloc_traits::pointer                    __node_pointer;
561    typedef typename __node_alloc_traits::pointer                    __node_const_pointer;
562    typedef __list_node_pointer_traits<value_type, __void_pointer> __node_pointer_traits;
563    typedef typename __node_pointer_traits::__link_pointer __link_pointer;
564    typedef __link_pointer __link_const_pointer;
565    typedef typename __alloc_traits::pointer                         pointer;
566    typedef typename __alloc_traits::const_pointer                   const_pointer;
567    typedef typename __alloc_traits::difference_type                 difference_type;
568
569    typedef typename __rebind_alloc_helper<__alloc_traits, __node_base>::type __node_base_allocator;
570    typedef typename allocator_traits<__node_base_allocator>::pointer __node_base_pointer;
571    static_assert((!is_same<allocator_type, __node_allocator>::value),
572                  "internal allocator type must differ from user-specified "
573                  "type; otherwise overload resolution breaks");
574
575    __node_base __end_;
576    __compressed_pair<size_type, __node_allocator> __size_alloc_;
577
578    _LIBCPP_INLINE_VISIBILITY
579    __link_pointer __end_as_link() const _NOEXCEPT {
580        return __node_pointer_traits::__unsafe_link_pointer_cast(
581                const_cast<__node_base&>(__end_).__self());
582    }
583
584    _LIBCPP_INLINE_VISIBILITY
585          size_type& __sz() _NOEXCEPT {return __size_alloc_.first();}
586    _LIBCPP_INLINE_VISIBILITY
587    const size_type& __sz() const _NOEXCEPT
588        {return __size_alloc_.first();}
589    _LIBCPP_INLINE_VISIBILITY
590          __node_allocator& __node_alloc() _NOEXCEPT
591          {return __size_alloc_.second();}
592    _LIBCPP_INLINE_VISIBILITY
593    const __node_allocator& __node_alloc() const _NOEXCEPT
594        {return __size_alloc_.second();}
595
596    _LIBCPP_INLINE_VISIBILITY
597    size_type __node_alloc_max_size() const _NOEXCEPT {
598        return __node_alloc_traits::max_size(__node_alloc());
599    }
600    _LIBCPP_INLINE_VISIBILITY
601    static void __unlink_nodes(__link_pointer __f, __link_pointer __l) _NOEXCEPT;
602
603    _LIBCPP_INLINE_VISIBILITY
604    __list_imp()
605        _NOEXCEPT_(is_nothrow_default_constructible<__node_allocator>::value);
606    _LIBCPP_INLINE_VISIBILITY
607    __list_imp(const allocator_type& __a);
608    _LIBCPP_INLINE_VISIBILITY
609    __list_imp(const __node_allocator& __a);
610#ifndef _LIBCPP_CXX03_LANG
611    __list_imp(__node_allocator&& __a) _NOEXCEPT;
612#endif
613    ~__list_imp();
614    void clear() _NOEXCEPT;
615    _LIBCPP_INLINE_VISIBILITY
616    bool empty() const _NOEXCEPT {return __sz() == 0;}
617
618    _LIBCPP_INLINE_VISIBILITY
619    iterator begin() _NOEXCEPT
620    {
621        return iterator(__end_.__next_, this);
622    }
623    _LIBCPP_INLINE_VISIBILITY
624    const_iterator begin() const  _NOEXCEPT
625    {
626        return const_iterator(__end_.__next_, this);
627    }
628    _LIBCPP_INLINE_VISIBILITY
629    iterator end() _NOEXCEPT
630    {
631        return iterator(__end_as_link(), this);
632    }
633    _LIBCPP_INLINE_VISIBILITY
634    const_iterator end() const _NOEXCEPT
635    {
636        return const_iterator(__end_as_link(), this);
637    }
638
639    void swap(__list_imp& __c)
640#if _LIBCPP_STD_VER >= 14
641        _NOEXCEPT;
642#else
643        _NOEXCEPT_(!__alloc_traits::propagate_on_container_swap::value ||
644                    __is_nothrow_swappable<allocator_type>::value);
645#endif
646
647    _LIBCPP_INLINE_VISIBILITY
648    void __copy_assign_alloc(const __list_imp& __c)
649        {__copy_assign_alloc(__c, integral_constant<bool,
650                      __node_alloc_traits::propagate_on_container_copy_assignment::value>());}
651
652    _LIBCPP_INLINE_VISIBILITY
653    void __move_assign_alloc(__list_imp& __c)
654        _NOEXCEPT_(
655            !__node_alloc_traits::propagate_on_container_move_assignment::value ||
656            is_nothrow_move_assignable<__node_allocator>::value)
657        {__move_assign_alloc(__c, integral_constant<bool,
658                      __node_alloc_traits::propagate_on_container_move_assignment::value>());}
659
660private:
661    _LIBCPP_INLINE_VISIBILITY
662    void __copy_assign_alloc(const __list_imp& __c, true_type)
663        {
664            if (__node_alloc() != __c.__node_alloc())
665                clear();
666            __node_alloc() = __c.__node_alloc();
667        }
668
669    _LIBCPP_INLINE_VISIBILITY
670    void __copy_assign_alloc(const __list_imp&, false_type)
671        {}
672
673    _LIBCPP_INLINE_VISIBILITY
674    void __move_assign_alloc(__list_imp& __c, true_type)
675        _NOEXCEPT_(is_nothrow_move_assignable<__node_allocator>::value)
676        {
677            __node_alloc() = _VSTD::move(__c.__node_alloc());
678        }
679
680    _LIBCPP_INLINE_VISIBILITY
681    void __move_assign_alloc(__list_imp&, false_type)
682        _NOEXCEPT
683        {}
684};
685
686// Unlink nodes [__f, __l]
687template <class _Tp, class _Alloc>
688inline
689void
690__list_imp<_Tp, _Alloc>::__unlink_nodes(__link_pointer __f, __link_pointer __l)
691    _NOEXCEPT
692{
693    __f->__prev_->__next_ = __l->__next_;
694    __l->__next_->__prev_ = __f->__prev_;
695}
696
697template <class _Tp, class _Alloc>
698inline
699__list_imp<_Tp, _Alloc>::__list_imp()
700        _NOEXCEPT_(is_nothrow_default_constructible<__node_allocator>::value)
701    : __size_alloc_(0, __default_init_tag())
702{
703}
704
705template <class _Tp, class _Alloc>
706inline
707__list_imp<_Tp, _Alloc>::__list_imp(const allocator_type& __a)
708    : __size_alloc_(0, __node_allocator(__a))
709{
710}
711
712template <class _Tp, class _Alloc>
713inline __list_imp<_Tp, _Alloc>::__list_imp(const __node_allocator& __a)
714    : __size_alloc_(0, __a) {}
715
716#ifndef _LIBCPP_CXX03_LANG
717template <class _Tp, class _Alloc>
718inline __list_imp<_Tp, _Alloc>::__list_imp(__node_allocator&& __a) _NOEXCEPT
719    : __size_alloc_(0, _VSTD::move(__a)) {}
720#endif
721
722template <class _Tp, class _Alloc>
723__list_imp<_Tp, _Alloc>::~__list_imp() {
724  clear();
725  std::__debug_db_erase_c(this);
726}
727
728template <class _Tp, class _Alloc>
729void
730__list_imp<_Tp, _Alloc>::clear() _NOEXCEPT
731{
732    if (!empty())
733    {
734        __node_allocator& __na = __node_alloc();
735        __link_pointer __f = __end_.__next_;
736        __link_pointer __l = __end_as_link();
737        __unlink_nodes(__f, __l->__prev_);
738        __sz() = 0;
739        while (__f != __l)
740        {
741            __node_pointer __np = __f->__as_node();
742            __f = __f->__next_;
743            __node_alloc_traits::destroy(__na, _VSTD::addressof(__np->__value_));
744            __node_alloc_traits::deallocate(__na, __np, 1);
745        }
746        std::__debug_db_invalidate_all(this);
747    }
748}
749
750template <class _Tp, class _Alloc>
751void
752__list_imp<_Tp, _Alloc>::swap(__list_imp& __c)
753#if _LIBCPP_STD_VER >= 14
754        _NOEXCEPT
755#else
756        _NOEXCEPT_(!__alloc_traits::propagate_on_container_swap::value ||
757                    __is_nothrow_swappable<allocator_type>::value)
758#endif
759{
760    _LIBCPP_ASSERT(__alloc_traits::propagate_on_container_swap::value ||
761                   this->__node_alloc() == __c.__node_alloc(),
762                   "list::swap: Either propagate_on_container_swap must be true"
763                   " or the allocators must compare equal");
764    using _VSTD::swap;
765    _VSTD::__swap_allocator(__node_alloc(), __c.__node_alloc());
766    swap(__sz(), __c.__sz());
767    swap(__end_, __c.__end_);
768    if (__sz() == 0)
769        __end_.__next_ = __end_.__prev_ = __end_as_link();
770    else
771        __end_.__prev_->__next_ = __end_.__next_->__prev_ = __end_as_link();
772    if (__c.__sz() == 0)
773        __c.__end_.__next_ = __c.__end_.__prev_ = __c.__end_as_link();
774    else
775        __c.__end_.__prev_->__next_ = __c.__end_.__next_->__prev_ = __c.__end_as_link();
776
777#ifdef _LIBCPP_ENABLE_DEBUG_MODE
778    __libcpp_db* __db = __get_db();
779    __c_node* __cn1 = __db->__find_c_and_lock(this);
780    __c_node* __cn2 = __db->__find_c(_VSTD::addressof(__c));
781    _VSTD::swap(__cn1->beg_, __cn2->beg_);
782    _VSTD::swap(__cn1->end_, __cn2->end_);
783    _VSTD::swap(__cn1->cap_, __cn2->cap_);
784    for (__i_node** __p = __cn1->end_; __p != __cn1->beg_;)
785    {
786        --__p;
787        const_iterator* __i = static_cast<const_iterator*>((*__p)->__i_);
788        if (__i->__ptr_ == __c.__end_as_link())
789        {
790            __cn2->__add(*__p);
791            if (--__cn1->end_ != __p)
792                _VSTD::memmove(__p, __p+1, (__cn1->end_ - __p)*sizeof(__i_node*));
793        }
794        else
795            (*__p)->__c_ = __cn1;
796    }
797    for (__i_node** __p = __cn2->end_; __p != __cn2->beg_;)
798    {
799        --__p;
800        const_iterator* __i = static_cast<const_iterator*>((*__p)->__i_);
801        if (__i->__ptr_ == __end_as_link())
802        {
803            __cn1->__add(*__p);
804            if (--__cn2->end_ != __p)
805                _VSTD::memmove(__p, __p+1, (__cn2->end_ - __p)*sizeof(__i_node*));
806        }
807        else
808            (*__p)->__c_ = __cn2;
809    }
810    __db->unlock();
811#endif
812}
813
814template <class _Tp, class _Alloc /*= allocator<_Tp>*/>
815class _LIBCPP_TEMPLATE_VIS list
816    : private __list_imp<_Tp, _Alloc>
817{
818    typedef __list_imp<_Tp, _Alloc> base;
819    typedef typename base::__node              __node;
820    typedef typename base::__node_allocator    __node_allocator;
821    typedef typename base::__node_pointer      __node_pointer;
822    typedef typename base::__node_alloc_traits __node_alloc_traits;
823    typedef typename base::__node_base         __node_base;
824    typedef typename base::__node_base_pointer __node_base_pointer;
825    typedef typename base::__link_pointer __link_pointer;
826
827public:
828    typedef _Tp                                            value_type;
829    typedef _Alloc                                         allocator_type;
830    static_assert((is_same<value_type, typename allocator_type::value_type>::value),
831                  "Invalid allocator::value_type");
832    typedef value_type&                                    reference;
833    typedef const value_type&                              const_reference;
834    typedef typename base::pointer                         pointer;
835    typedef typename base::const_pointer                   const_pointer;
836    typedef typename base::size_type                       size_type;
837    typedef typename base::difference_type                 difference_type;
838    typedef typename base::iterator                        iterator;
839    typedef typename base::const_iterator                  const_iterator;
840    typedef _VSTD::reverse_iterator<iterator>              reverse_iterator;
841    typedef _VSTD::reverse_iterator<const_iterator>        const_reverse_iterator;
842#if _LIBCPP_STD_VER > 17
843    typedef size_type                                      __remove_return_type;
844#else
845    typedef void                                           __remove_return_type;
846#endif
847
848    _LIBCPP_INLINE_VISIBILITY
849    list()
850        _NOEXCEPT_(is_nothrow_default_constructible<__node_allocator>::value)
851    {
852        _VSTD::__debug_db_insert_c(this);
853    }
854    _LIBCPP_INLINE_VISIBILITY
855    explicit list(const allocator_type& __a) : base(__a)
856    {
857        _VSTD::__debug_db_insert_c(this);
858    }
859    explicit list(size_type __n);
860#if _LIBCPP_STD_VER > 11
861    explicit list(size_type __n, const allocator_type& __a);
862#endif
863    list(size_type __n, const value_type& __x);
864    template <class = __enable_if_t<__is_allocator<_Alloc>::value> >
865    list(size_type __n, const value_type& __x, const allocator_type& __a) : base(__a)
866    {
867        _VSTD::__debug_db_insert_c(this);
868        for (; __n > 0; --__n)
869            push_back(__x);
870    }
871
872    template <class _InpIter>
873        list(_InpIter __f, _InpIter __l,
874             __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>* = 0);
875    template <class _InpIter>
876        list(_InpIter __f, _InpIter __l, const allocator_type& __a,
877             __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>* = 0);
878
879    list(const list& __c);
880    list(const list& __c, const __type_identity_t<allocator_type>& __a);
881    _LIBCPP_INLINE_VISIBILITY
882    list& operator=(const list& __c);
883#ifndef _LIBCPP_CXX03_LANG
884    list(initializer_list<value_type> __il);
885    list(initializer_list<value_type> __il, const allocator_type& __a);
886
887    _LIBCPP_INLINE_VISIBILITY
888    list(list&& __c)
889        _NOEXCEPT_(is_nothrow_move_constructible<__node_allocator>::value);
890    _LIBCPP_INLINE_VISIBILITY
891    list(list&& __c, const __type_identity_t<allocator_type>& __a);
892    _LIBCPP_INLINE_VISIBILITY
893    list& operator=(list&& __c)
894        _NOEXCEPT_(
895            __node_alloc_traits::propagate_on_container_move_assignment::value &&
896            is_nothrow_move_assignable<__node_allocator>::value);
897
898    _LIBCPP_INLINE_VISIBILITY
899    list& operator=(initializer_list<value_type> __il)
900        {assign(__il.begin(), __il.end()); return *this;}
901
902    _LIBCPP_INLINE_VISIBILITY
903    void assign(initializer_list<value_type> __il)
904        {assign(__il.begin(), __il.end());}
905#endif // _LIBCPP_CXX03_LANG
906
907    template <class _InpIter>
908        void assign(_InpIter __f, _InpIter __l,
909                    __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>* = 0);
910    void assign(size_type __n, const value_type& __x);
911
912    _LIBCPP_INLINE_VISIBILITY
913    allocator_type get_allocator() const _NOEXCEPT;
914
915    _LIBCPP_INLINE_VISIBILITY
916    size_type size() const _NOEXCEPT     {return base::__sz();}
917    _LIBCPP_NODISCARD_AFTER_CXX17 _LIBCPP_INLINE_VISIBILITY
918    bool empty() const _NOEXCEPT         {return base::empty();}
919    _LIBCPP_INLINE_VISIBILITY
920    size_type max_size() const _NOEXCEPT
921        {
922            return _VSTD::min<size_type>(
923                base::__node_alloc_max_size(),
924                numeric_limits<difference_type >::max());
925        }
926
927    _LIBCPP_INLINE_VISIBILITY
928          iterator begin() _NOEXCEPT        {return base::begin();}
929    _LIBCPP_INLINE_VISIBILITY
930    const_iterator begin()  const _NOEXCEPT {return base::begin();}
931    _LIBCPP_INLINE_VISIBILITY
932          iterator end() _NOEXCEPT          {return base::end();}
933    _LIBCPP_INLINE_VISIBILITY
934    const_iterator end()    const _NOEXCEPT {return base::end();}
935    _LIBCPP_INLINE_VISIBILITY
936    const_iterator cbegin() const _NOEXCEPT {return base::begin();}
937    _LIBCPP_INLINE_VISIBILITY
938    const_iterator cend()   const _NOEXCEPT {return base::end();}
939
940    _LIBCPP_INLINE_VISIBILITY
941          reverse_iterator rbegin() _NOEXCEPT
942            {return       reverse_iterator(end());}
943    _LIBCPP_INLINE_VISIBILITY
944    const_reverse_iterator rbegin()  const _NOEXCEPT
945        {return const_reverse_iterator(end());}
946    _LIBCPP_INLINE_VISIBILITY
947          reverse_iterator rend() _NOEXCEPT
948            {return       reverse_iterator(begin());}
949    _LIBCPP_INLINE_VISIBILITY
950    const_reverse_iterator rend()    const _NOEXCEPT
951        {return const_reverse_iterator(begin());}
952    _LIBCPP_INLINE_VISIBILITY
953    const_reverse_iterator crbegin() const _NOEXCEPT
954        {return const_reverse_iterator(end());}
955    _LIBCPP_INLINE_VISIBILITY
956    const_reverse_iterator crend()   const _NOEXCEPT
957        {return const_reverse_iterator(begin());}
958
959    _LIBCPP_INLINE_VISIBILITY
960    reference front()
961    {
962        _LIBCPP_ASSERT(!empty(), "list::front called on empty list");
963        return base::__end_.__next_->__as_node()->__value_;
964    }
965    _LIBCPP_INLINE_VISIBILITY
966    const_reference front() const
967    {
968        _LIBCPP_ASSERT(!empty(), "list::front called on empty list");
969        return base::__end_.__next_->__as_node()->__value_;
970    }
971    _LIBCPP_INLINE_VISIBILITY
972    reference back()
973    {
974        _LIBCPP_ASSERT(!empty(), "list::back called on empty list");
975        return base::__end_.__prev_->__as_node()->__value_;
976    }
977    _LIBCPP_INLINE_VISIBILITY
978    const_reference back() const
979    {
980        _LIBCPP_ASSERT(!empty(), "list::back called on empty list");
981        return base::__end_.__prev_->__as_node()->__value_;
982    }
983
984#ifndef _LIBCPP_CXX03_LANG
985    void push_front(value_type&& __x);
986    void push_back(value_type&& __x);
987
988    template <class... _Args>
989#if _LIBCPP_STD_VER > 14
990       reference emplace_front(_Args&&... __args);
991#else
992       void      emplace_front(_Args&&... __args);
993#endif
994    template <class... _Args>
995#if _LIBCPP_STD_VER > 14
996        reference emplace_back(_Args&&... __args);
997#else
998       void       emplace_back(_Args&&... __args);
999#endif
1000    template <class... _Args>
1001        iterator emplace(const_iterator __p, _Args&&... __args);
1002
1003    iterator insert(const_iterator __p, value_type&& __x);
1004
1005    _LIBCPP_INLINE_VISIBILITY
1006    iterator insert(const_iterator __p, initializer_list<value_type> __il)
1007        {return insert(__p, __il.begin(), __il.end());}
1008#endif // _LIBCPP_CXX03_LANG
1009
1010    void push_front(const value_type& __x);
1011    void push_back(const value_type& __x);
1012
1013#ifndef _LIBCPP_CXX03_LANG
1014    template <class _Arg>
1015    _LIBCPP_INLINE_VISIBILITY
1016    void __emplace_back(_Arg&& __arg) { emplace_back(_VSTD::forward<_Arg>(__arg)); }
1017#else
1018    _LIBCPP_INLINE_VISIBILITY
1019    void __emplace_back(value_type const& __arg) { push_back(__arg); }
1020#endif
1021
1022    iterator insert(const_iterator __p, const value_type& __x);
1023    iterator insert(const_iterator __p, size_type __n, const value_type& __x);
1024    template <class _InpIter>
1025        iterator insert(const_iterator __p, _InpIter __f, _InpIter __l,
1026                        __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>* = 0);
1027
1028    _LIBCPP_INLINE_VISIBILITY
1029    void swap(list& __c)
1030#if _LIBCPP_STD_VER >= 14
1031        _NOEXCEPT
1032#else
1033        _NOEXCEPT_(!__node_alloc_traits::propagate_on_container_swap::value ||
1034                   __is_nothrow_swappable<__node_allocator>::value)
1035#endif
1036        {base::swap(__c);}
1037    _LIBCPP_INLINE_VISIBILITY
1038    void clear() _NOEXCEPT {base::clear();}
1039
1040    void pop_front();
1041    void pop_back();
1042
1043    iterator erase(const_iterator __p);
1044    iterator erase(const_iterator __f, const_iterator __l);
1045
1046    void resize(size_type __n);
1047    void resize(size_type __n, const value_type& __x);
1048
1049    void splice(const_iterator __p, list& __c);
1050#ifndef _LIBCPP_CXX03_LANG
1051    _LIBCPP_INLINE_VISIBILITY
1052    void splice(const_iterator __p, list&& __c) {splice(__p, __c);}
1053    _LIBCPP_INLINE_VISIBILITY
1054    void splice(const_iterator __p, list&& __c, const_iterator __i)
1055        {splice(__p, __c, __i);}
1056    _LIBCPP_INLINE_VISIBILITY
1057    void splice(const_iterator __p, list&& __c, const_iterator __f, const_iterator __l)
1058        {splice(__p, __c, __f, __l);}
1059#endif
1060    void splice(const_iterator __p, list& __c, const_iterator __i);
1061    void splice(const_iterator __p, list& __c, const_iterator __f, const_iterator __l);
1062
1063    __remove_return_type remove(const value_type& __x);
1064    template <class _Pred> __remove_return_type remove_if(_Pred __pred);
1065    _LIBCPP_INLINE_VISIBILITY
1066    __remove_return_type unique() { return unique(__equal_to<value_type>()); }
1067    template <class _BinaryPred>
1068        __remove_return_type unique(_BinaryPred __binary_pred);
1069    _LIBCPP_INLINE_VISIBILITY
1070    void merge(list& __c);
1071#ifndef _LIBCPP_CXX03_LANG
1072    _LIBCPP_INLINE_VISIBILITY
1073    void merge(list&& __c) {merge(__c);}
1074
1075    template <class _Comp>
1076    _LIBCPP_INLINE_VISIBILITY
1077        void merge(list&& __c, _Comp __comp) {merge(__c, __comp);}
1078#endif
1079    template <class _Comp>
1080        void merge(list& __c, _Comp __comp);
1081
1082    _LIBCPP_INLINE_VISIBILITY
1083    void sort();
1084    template <class _Comp>
1085        _LIBCPP_INLINE_VISIBILITY
1086        void sort(_Comp __comp);
1087
1088    void reverse() _NOEXCEPT;
1089
1090    bool __invariants() const;
1091
1092    typedef __allocator_destructor<__node_allocator> __node_destructor;
1093    typedef unique_ptr<__node, __node_destructor> __hold_pointer;
1094
1095    _LIBCPP_INLINE_VISIBILITY
1096    __hold_pointer __allocate_node(__node_allocator& __na) {
1097      __node_pointer __p = __node_alloc_traits::allocate(__na, 1);
1098      __p->__prev_ = nullptr;
1099      return __hold_pointer(__p, __node_destructor(__na, 1));
1100    }
1101
1102#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1103
1104    bool __dereferenceable(const const_iterator* __i) const;
1105    bool __decrementable(const const_iterator* __i) const;
1106    bool __addable(const const_iterator* __i, ptrdiff_t __n) const;
1107    bool __subscriptable(const const_iterator* __i, ptrdiff_t __n) const;
1108
1109#endif // _LIBCPP_ENABLE_DEBUG_MODE
1110
1111private:
1112    _LIBCPP_INLINE_VISIBILITY
1113    static void __link_nodes  (__link_pointer __p, __link_pointer __f, __link_pointer __l);
1114    _LIBCPP_INLINE_VISIBILITY
1115    void __link_nodes_at_front(__link_pointer __f, __link_pointer __l);
1116    _LIBCPP_INLINE_VISIBILITY
1117    void __link_nodes_at_back (__link_pointer __f, __link_pointer __l);
1118    iterator __iterator(size_type __n);
1119    template <class _Comp>
1120        static iterator __sort(iterator __f1, iterator __e2, size_type __n, _Comp& __comp);
1121
1122    void __move_assign(list& __c, true_type)
1123        _NOEXCEPT_(is_nothrow_move_assignable<__node_allocator>::value);
1124    void __move_assign(list& __c, false_type);
1125};
1126
1127#if _LIBCPP_STD_VER >= 17
1128template<class _InputIterator,
1129         class _Alloc = allocator<__iter_value_type<_InputIterator>>,
1130         class = enable_if_t<__is_cpp17_input_iterator<_InputIterator>::value>,
1131         class = enable_if_t<__is_allocator<_Alloc>::value>
1132         >
1133list(_InputIterator, _InputIterator)
1134  -> list<__iter_value_type<_InputIterator>, _Alloc>;
1135
1136template<class _InputIterator,
1137         class _Alloc,
1138         class = enable_if_t<__is_cpp17_input_iterator<_InputIterator>::value>,
1139         class = enable_if_t<__is_allocator<_Alloc>::value>
1140         >
1141list(_InputIterator, _InputIterator, _Alloc)
1142  -> list<__iter_value_type<_InputIterator>, _Alloc>;
1143#endif
1144
1145// Link in nodes [__f, __l] just prior to __p
1146template <class _Tp, class _Alloc>
1147inline
1148void
1149list<_Tp, _Alloc>::__link_nodes(__link_pointer __p, __link_pointer __f, __link_pointer __l)
1150{
1151    __p->__prev_->__next_ = __f;
1152    __f->__prev_ = __p->__prev_;
1153    __p->__prev_ = __l;
1154    __l->__next_ = __p;
1155}
1156
1157// Link in nodes [__f, __l] at the front of the list
1158template <class _Tp, class _Alloc>
1159inline
1160void
1161list<_Tp, _Alloc>::__link_nodes_at_front(__link_pointer __f, __link_pointer __l)
1162{
1163    __f->__prev_ = base::__end_as_link();
1164    __l->__next_ = base::__end_.__next_;
1165    __l->__next_->__prev_ = __l;
1166    base::__end_.__next_ = __f;
1167}
1168
1169// Link in nodes [__f, __l] at the back of the list
1170template <class _Tp, class _Alloc>
1171inline
1172void
1173list<_Tp, _Alloc>::__link_nodes_at_back(__link_pointer __f, __link_pointer __l)
1174{
1175    __l->__next_ = base::__end_as_link();
1176    __f->__prev_ = base::__end_.__prev_;
1177    __f->__prev_->__next_ = __f;
1178    base::__end_.__prev_ = __l;
1179}
1180
1181
1182template <class _Tp, class _Alloc>
1183inline
1184typename list<_Tp, _Alloc>::iterator
1185list<_Tp, _Alloc>::__iterator(size_type __n)
1186{
1187    return __n <= base::__sz() / 2 ? _VSTD::next(begin(), __n)
1188                                   : _VSTD::prev(end(), base::__sz() - __n);
1189}
1190
1191template <class _Tp, class _Alloc>
1192list<_Tp, _Alloc>::list(size_type __n)
1193{
1194    _VSTD::__debug_db_insert_c(this);
1195    for (; __n > 0; --__n)
1196#ifndef _LIBCPP_CXX03_LANG
1197        emplace_back();
1198#else
1199        push_back(value_type());
1200#endif
1201}
1202
1203#if _LIBCPP_STD_VER > 11
1204template <class _Tp, class _Alloc>
1205list<_Tp, _Alloc>::list(size_type __n, const allocator_type& __a) : base(__a)
1206{
1207    _VSTD::__debug_db_insert_c(this);
1208    for (; __n > 0; --__n)
1209        emplace_back();
1210}
1211#endif
1212
1213template <class _Tp, class _Alloc>
1214list<_Tp, _Alloc>::list(size_type __n, const value_type& __x)
1215{
1216    _VSTD::__debug_db_insert_c(this);
1217    for (; __n > 0; --__n)
1218        push_back(__x);
1219}
1220
1221template <class _Tp, class _Alloc>
1222template <class _InpIter>
1223list<_Tp, _Alloc>::list(_InpIter __f, _InpIter __l,
1224                        __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>*)
1225{
1226    _VSTD::__debug_db_insert_c(this);
1227    for (; __f != __l; ++__f)
1228        __emplace_back(*__f);
1229}
1230
1231template <class _Tp, class _Alloc>
1232template <class _InpIter>
1233list<_Tp, _Alloc>::list(_InpIter __f, _InpIter __l, const allocator_type& __a,
1234                        __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>*)
1235    : base(__a)
1236{
1237    _VSTD::__debug_db_insert_c(this);
1238    for (; __f != __l; ++__f)
1239        __emplace_back(*__f);
1240}
1241
1242template <class _Tp, class _Alloc>
1243list<_Tp, _Alloc>::list(const list& __c)
1244    : base(__node_alloc_traits::select_on_container_copy_construction(
1245          __c.__node_alloc())) {
1246    _VSTD::__debug_db_insert_c(this);
1247    for (const_iterator __i = __c.begin(), __e = __c.end(); __i != __e; ++__i)
1248        push_back(*__i);
1249}
1250
1251template <class _Tp, class _Alloc>
1252list<_Tp, _Alloc>::list(const list& __c, const __type_identity_t<allocator_type>& __a)
1253    : base(__a)
1254{
1255    _VSTD::__debug_db_insert_c(this);
1256    for (const_iterator __i = __c.begin(), __e = __c.end(); __i != __e; ++__i)
1257        push_back(*__i);
1258}
1259
1260#ifndef _LIBCPP_CXX03_LANG
1261
1262template <class _Tp, class _Alloc>
1263list<_Tp, _Alloc>::list(initializer_list<value_type> __il, const allocator_type& __a)
1264    : base(__a)
1265{
1266    _VSTD::__debug_db_insert_c(this);
1267    for (typename initializer_list<value_type>::const_iterator __i = __il.begin(),
1268            __e = __il.end(); __i != __e; ++__i)
1269        push_back(*__i);
1270}
1271
1272template <class _Tp, class _Alloc>
1273list<_Tp, _Alloc>::list(initializer_list<value_type> __il)
1274{
1275    _VSTD::__debug_db_insert_c(this);
1276    for (typename initializer_list<value_type>::const_iterator __i = __il.begin(),
1277            __e = __il.end(); __i != __e; ++__i)
1278        push_back(*__i);
1279}
1280
1281template <class _Tp, class _Alloc>
1282inline list<_Tp, _Alloc>::list(list&& __c)
1283        _NOEXCEPT_(is_nothrow_move_constructible<__node_allocator>::value)
1284        : base(_VSTD::move(__c.__node_alloc())) {
1285    _VSTD::__debug_db_insert_c(this);
1286    splice(end(), __c);
1287}
1288
1289template <class _Tp, class _Alloc>
1290inline
1291list<_Tp, _Alloc>::list(list&& __c, const __type_identity_t<allocator_type>& __a)
1292    : base(__a)
1293{
1294    _VSTD::__debug_db_insert_c(this);
1295    if (__a == __c.get_allocator())
1296        splice(end(), __c);
1297    else
1298    {
1299        typedef move_iterator<iterator> _Ip;
1300        assign(_Ip(__c.begin()), _Ip(__c.end()));
1301    }
1302}
1303
1304template <class _Tp, class _Alloc>
1305inline
1306list<_Tp, _Alloc>&
1307list<_Tp, _Alloc>::operator=(list&& __c)
1308        _NOEXCEPT_(
1309            __node_alloc_traits::propagate_on_container_move_assignment::value &&
1310            is_nothrow_move_assignable<__node_allocator>::value)
1311{
1312    __move_assign(__c, integral_constant<bool,
1313          __node_alloc_traits::propagate_on_container_move_assignment::value>());
1314    return *this;
1315}
1316
1317template <class _Tp, class _Alloc>
1318void
1319list<_Tp, _Alloc>::__move_assign(list& __c, false_type)
1320{
1321    if (base::__node_alloc() != __c.__node_alloc())
1322    {
1323        typedef move_iterator<iterator> _Ip;
1324        assign(_Ip(__c.begin()), _Ip(__c.end()));
1325    }
1326    else
1327        __move_assign(__c, true_type());
1328}
1329
1330template <class _Tp, class _Alloc>
1331void
1332list<_Tp, _Alloc>::__move_assign(list& __c, true_type)
1333        _NOEXCEPT_(is_nothrow_move_assignable<__node_allocator>::value)
1334{
1335    clear();
1336    base::__move_assign_alloc(__c);
1337    splice(end(), __c);
1338}
1339
1340#endif // _LIBCPP_CXX03_LANG
1341
1342template <class _Tp, class _Alloc>
1343inline
1344list<_Tp, _Alloc>&
1345list<_Tp, _Alloc>::operator=(const list& __c)
1346{
1347    if (this != _VSTD::addressof(__c))
1348    {
1349        base::__copy_assign_alloc(__c);
1350        assign(__c.begin(), __c.end());
1351    }
1352    return *this;
1353}
1354
1355template <class _Tp, class _Alloc>
1356template <class _InpIter>
1357void
1358list<_Tp, _Alloc>::assign(_InpIter __f, _InpIter __l,
1359                          __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>*)
1360{
1361    iterator __i = begin();
1362    iterator __e = end();
1363    for (; __f != __l && __i != __e; ++__f, (void) ++__i)
1364        *__i = *__f;
1365    if (__i == __e)
1366        insert(__e, __f, __l);
1367    else
1368        erase(__i, __e);
1369    std::__debug_db_invalidate_all(this);
1370}
1371
1372template <class _Tp, class _Alloc>
1373void
1374list<_Tp, _Alloc>::assign(size_type __n, const value_type& __x)
1375{
1376    iterator __i = begin();
1377    iterator __e = end();
1378    for (; __n > 0 && __i != __e; --__n, (void) ++__i)
1379        *__i = __x;
1380    if (__i == __e)
1381        insert(__e, __n, __x);
1382    else
1383        erase(__i, __e);
1384    std::__debug_db_invalidate_all(this);
1385}
1386
1387template <class _Tp, class _Alloc>
1388inline
1389_Alloc
1390list<_Tp, _Alloc>::get_allocator() const _NOEXCEPT
1391{
1392    return allocator_type(base::__node_alloc());
1393}
1394
1395template <class _Tp, class _Alloc>
1396typename list<_Tp, _Alloc>::iterator
1397list<_Tp, _Alloc>::insert(const_iterator __p, const value_type& __x)
1398{
1399    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1400                         "list::insert(iterator, x) called with an iterator not referring to this list");
1401    __node_allocator& __na = base::__node_alloc();
1402    __hold_pointer __hold = __allocate_node(__na);
1403    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1404    __link_nodes(__p.__ptr_, __hold->__as_link(), __hold->__as_link());
1405    ++base::__sz();
1406    return iterator(__hold.release()->__as_link(), this);
1407}
1408
1409template <class _Tp, class _Alloc>
1410typename list<_Tp, _Alloc>::iterator
1411list<_Tp, _Alloc>::insert(const_iterator __p, size_type __n, const value_type& __x)
1412{
1413    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1414                         "list::insert(iterator, n, x) called with an iterator not referring to this list");
1415    iterator __r(__p.__ptr_, this);
1416    if (__n > 0)
1417    {
1418        size_type __ds = 0;
1419        __node_allocator& __na = base::__node_alloc();
1420        __hold_pointer __hold = __allocate_node(__na);
1421        __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1422        ++__ds;
1423        __r = iterator(__hold->__as_link(), this);
1424        __hold.release();
1425        iterator __e = __r;
1426#ifndef _LIBCPP_NO_EXCEPTIONS
1427        try
1428        {
1429#endif // _LIBCPP_NO_EXCEPTIONS
1430            for (--__n; __n != 0; --__n, (void) ++__e, ++__ds)
1431            {
1432                __hold.reset(__node_alloc_traits::allocate(__na, 1));
1433                __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1434                __e.__ptr_->__next_ = __hold->__as_link();
1435                __hold->__prev_ = __e.__ptr_;
1436                __hold.release();
1437            }
1438#ifndef _LIBCPP_NO_EXCEPTIONS
1439        }
1440        catch (...)
1441        {
1442            while (true)
1443            {
1444                __node_alloc_traits::destroy(__na, _VSTD::addressof(*__e));
1445                __link_pointer __prev = __e.__ptr_->__prev_;
1446                __node_alloc_traits::deallocate(__na, __e.__ptr_->__as_node(), 1);
1447                if (__prev == 0)
1448                    break;
1449                __e = iterator(__prev, this);
1450            }
1451            throw;
1452        }
1453#endif // _LIBCPP_NO_EXCEPTIONS
1454        __link_nodes(__p.__ptr_, __r.__ptr_, __e.__ptr_);
1455        base::__sz() += __ds;
1456    }
1457    return __r;
1458}
1459
1460template <class _Tp, class _Alloc>
1461template <class _InpIter>
1462typename list<_Tp, _Alloc>::iterator
1463list<_Tp, _Alloc>::insert(const_iterator __p, _InpIter __f, _InpIter __l,
1464                          __enable_if_t<__is_cpp17_input_iterator<_InpIter>::value>*)
1465{
1466    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1467                         "list::insert(iterator, range) called with an iterator not referring to this list");
1468    iterator __r(__p.__ptr_, this);
1469    if (__f != __l)
1470    {
1471        size_type __ds = 0;
1472        __node_allocator& __na = base::__node_alloc();
1473        __hold_pointer __hold = __allocate_node(__na);
1474        __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), *__f);
1475        ++__ds;
1476        __r = iterator(__hold.get()->__as_link(), this);
1477        __hold.release();
1478        iterator __e = __r;
1479#ifndef _LIBCPP_NO_EXCEPTIONS
1480        try
1481        {
1482#endif // _LIBCPP_NO_EXCEPTIONS
1483            for (++__f; __f != __l; ++__f, (void) ++__e, ++__ds)
1484            {
1485                __hold.reset(__node_alloc_traits::allocate(__na, 1));
1486                __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), *__f);
1487                __e.__ptr_->__next_ = __hold.get()->__as_link();
1488                __hold->__prev_ = __e.__ptr_;
1489                __hold.release();
1490            }
1491#ifndef _LIBCPP_NO_EXCEPTIONS
1492        }
1493        catch (...)
1494        {
1495            while (true)
1496            {
1497                __node_alloc_traits::destroy(__na, _VSTD::addressof(*__e));
1498                __link_pointer __prev = __e.__ptr_->__prev_;
1499                __node_alloc_traits::deallocate(__na, __e.__ptr_->__as_node(), 1);
1500                if (__prev == 0)
1501                    break;
1502                __e = iterator(__prev, this);
1503            }
1504            throw;
1505        }
1506#endif // _LIBCPP_NO_EXCEPTIONS
1507        __link_nodes(__p.__ptr_, __r.__ptr_, __e.__ptr_);
1508        base::__sz() += __ds;
1509    }
1510    return __r;
1511}
1512
1513template <class _Tp, class _Alloc>
1514void
1515list<_Tp, _Alloc>::push_front(const value_type& __x)
1516{
1517    __node_allocator& __na = base::__node_alloc();
1518    __hold_pointer __hold = __allocate_node(__na);
1519    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1520    __link_pointer __nl = __hold->__as_link();
1521    __link_nodes_at_front(__nl, __nl);
1522    ++base::__sz();
1523    __hold.release();
1524}
1525
1526template <class _Tp, class _Alloc>
1527void
1528list<_Tp, _Alloc>::push_back(const value_type& __x)
1529{
1530    __node_allocator& __na = base::__node_alloc();
1531    __hold_pointer __hold = __allocate_node(__na);
1532    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1533    __link_nodes_at_back(__hold.get()->__as_link(), __hold.get()->__as_link());
1534    ++base::__sz();
1535    __hold.release();
1536}
1537
1538#ifndef _LIBCPP_CXX03_LANG
1539
1540template <class _Tp, class _Alloc>
1541void
1542list<_Tp, _Alloc>::push_front(value_type&& __x)
1543{
1544    __node_allocator& __na = base::__node_alloc();
1545    __hold_pointer __hold = __allocate_node(__na);
1546    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::move(__x));
1547    __link_nodes_at_front(__hold.get()->__as_link(), __hold.get()->__as_link());
1548    ++base::__sz();
1549    __hold.release();
1550}
1551
1552template <class _Tp, class _Alloc>
1553void
1554list<_Tp, _Alloc>::push_back(value_type&& __x)
1555{
1556    __node_allocator& __na = base::__node_alloc();
1557    __hold_pointer __hold = __allocate_node(__na);
1558    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::move(__x));
1559    __link_nodes_at_back(__hold.get()->__as_link(), __hold.get()->__as_link());
1560    ++base::__sz();
1561    __hold.release();
1562}
1563
1564template <class _Tp, class _Alloc>
1565template <class... _Args>
1566#if _LIBCPP_STD_VER > 14
1567typename list<_Tp, _Alloc>::reference
1568#else
1569void
1570#endif
1571list<_Tp, _Alloc>::emplace_front(_Args&&... __args)
1572{
1573    __node_allocator& __na = base::__node_alloc();
1574    __hold_pointer __hold = __allocate_node(__na);
1575    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::forward<_Args>(__args)...);
1576    __link_nodes_at_front(__hold.get()->__as_link(), __hold.get()->__as_link());
1577    ++base::__sz();
1578#if _LIBCPP_STD_VER > 14
1579    return __hold.release()->__value_;
1580#else
1581    __hold.release();
1582#endif
1583}
1584
1585template <class _Tp, class _Alloc>
1586template <class... _Args>
1587#if _LIBCPP_STD_VER > 14
1588typename list<_Tp, _Alloc>::reference
1589#else
1590void
1591#endif
1592list<_Tp, _Alloc>::emplace_back(_Args&&... __args)
1593{
1594    __node_allocator& __na = base::__node_alloc();
1595    __hold_pointer __hold = __allocate_node(__na);
1596    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::forward<_Args>(__args)...);
1597    __link_pointer __nl = __hold->__as_link();
1598    __link_nodes_at_back(__nl, __nl);
1599    ++base::__sz();
1600#if _LIBCPP_STD_VER > 14
1601    return __hold.release()->__value_;
1602#else
1603    __hold.release();
1604#endif
1605}
1606
1607template <class _Tp, class _Alloc>
1608template <class... _Args>
1609typename list<_Tp, _Alloc>::iterator
1610list<_Tp, _Alloc>::emplace(const_iterator __p, _Args&&... __args)
1611{
1612    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1613                         "list::emplace(iterator, args...) called with an iterator not referring to this list");
1614    __node_allocator& __na = base::__node_alloc();
1615    __hold_pointer __hold = __allocate_node(__na);
1616    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::forward<_Args>(__args)...);
1617    __link_pointer __nl = __hold.get()->__as_link();
1618    __link_nodes(__p.__ptr_, __nl, __nl);
1619    ++base::__sz();
1620    __hold.release();
1621    return iterator(__nl, this);
1622}
1623
1624template <class _Tp, class _Alloc>
1625typename list<_Tp, _Alloc>::iterator
1626list<_Tp, _Alloc>::insert(const_iterator __p, value_type&& __x)
1627{
1628    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1629                         "list::insert(iterator, x) called with an iterator not referring to this list");
1630    __node_allocator& __na = base::__node_alloc();
1631    __hold_pointer __hold = __allocate_node(__na);
1632    __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), _VSTD::move(__x));
1633    __link_pointer __nl = __hold->__as_link();
1634    __link_nodes(__p.__ptr_, __nl, __nl);
1635    ++base::__sz();
1636    __hold.release();
1637    return iterator(__nl, this);
1638}
1639
1640#endif // _LIBCPP_CXX03_LANG
1641
1642template <class _Tp, class _Alloc>
1643void
1644list<_Tp, _Alloc>::pop_front()
1645{
1646    _LIBCPP_ASSERT(!empty(), "list::pop_front() called with empty list");
1647    __node_allocator& __na = base::__node_alloc();
1648    __link_pointer __n = base::__end_.__next_;
1649    base::__unlink_nodes(__n, __n);
1650    --base::__sz();
1651#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1652    __c_node* __c = __get_db()->__find_c_and_lock(this);
1653    for (__i_node** __p = __c->end_; __p != __c->beg_; )
1654    {
1655        --__p;
1656        iterator* __i = static_cast<iterator*>((*__p)->__i_);
1657        if (__i->__ptr_ == __n)
1658        {
1659            (*__p)->__c_ = nullptr;
1660            if (--__c->end_ != __p)
1661                _VSTD::memmove(__p, __p+1, (__c->end_ - __p)*sizeof(__i_node*));
1662        }
1663    }
1664    __get_db()->unlock();
1665#endif
1666    __node_pointer __np = __n->__as_node();
1667    __node_alloc_traits::destroy(__na, _VSTD::addressof(__np->__value_));
1668    __node_alloc_traits::deallocate(__na, __np, 1);
1669}
1670
1671template <class _Tp, class _Alloc>
1672void
1673list<_Tp, _Alloc>::pop_back()
1674{
1675    _LIBCPP_ASSERT(!empty(), "list::pop_back() called on an empty list");
1676    __node_allocator& __na = base::__node_alloc();
1677    __link_pointer __n = base::__end_.__prev_;
1678    base::__unlink_nodes(__n, __n);
1679    --base::__sz();
1680#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1681    __c_node* __c = __get_db()->__find_c_and_lock(this);
1682    for (__i_node** __p = __c->end_; __p != __c->beg_; )
1683    {
1684        --__p;
1685        iterator* __i = static_cast<iterator*>((*__p)->__i_);
1686        if (__i->__ptr_ == __n)
1687        {
1688            (*__p)->__c_ = nullptr;
1689            if (--__c->end_ != __p)
1690                _VSTD::memmove(__p, __p+1, (__c->end_ - __p)*sizeof(__i_node*));
1691        }
1692    }
1693    __get_db()->unlock();
1694#endif
1695    __node_pointer __np = __n->__as_node();
1696    __node_alloc_traits::destroy(__na, _VSTD::addressof(__np->__value_));
1697    __node_alloc_traits::deallocate(__na, __np, 1);
1698}
1699
1700template <class _Tp, class _Alloc>
1701typename list<_Tp, _Alloc>::iterator
1702list<_Tp, _Alloc>::erase(const_iterator __p)
1703{
1704    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1705                         "list::erase(iterator) called with an iterator not referring to this list");
1706    _LIBCPP_ASSERT(__p != end(),
1707        "list::erase(iterator) called with a non-dereferenceable iterator");
1708    __node_allocator& __na = base::__node_alloc();
1709    __link_pointer __n = __p.__ptr_;
1710    __link_pointer __r = __n->__next_;
1711    base::__unlink_nodes(__n, __n);
1712    --base::__sz();
1713#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1714    __c_node* __c = __get_db()->__find_c_and_lock(this);
1715    for (__i_node** __ip = __c->end_; __ip != __c->beg_; )
1716    {
1717        --__ip;
1718        iterator* __i = static_cast<iterator*>((*__ip)->__i_);
1719        if (__i->__ptr_ == __n)
1720        {
1721            (*__ip)->__c_ = nullptr;
1722            if (--__c->end_ != __ip)
1723                _VSTD::memmove(__ip, __ip+1, (__c->end_ - __ip)*sizeof(__i_node*));
1724        }
1725    }
1726    __get_db()->unlock();
1727#endif
1728    __node_pointer __np = __n->__as_node();
1729    __node_alloc_traits::destroy(__na, _VSTD::addressof(__np->__value_));
1730    __node_alloc_traits::deallocate(__na, __np, 1);
1731    return iterator(__r, this);
1732}
1733
1734template <class _Tp, class _Alloc>
1735typename list<_Tp, _Alloc>::iterator
1736list<_Tp, _Alloc>::erase(const_iterator __f, const_iterator __l)
1737{
1738    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__f)) == this,
1739                         "list::erase(iterator, iterator) called with an iterator not referring to this list");
1740    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__l)) == this,
1741                         "list::erase(iterator, iterator) called with an iterator not referring to this list");
1742    if (__f != __l)
1743    {
1744        __node_allocator& __na = base::__node_alloc();
1745        base::__unlink_nodes(__f.__ptr_, __l.__ptr_->__prev_);
1746        while (__f != __l)
1747        {
1748            __link_pointer __n = __f.__ptr_;
1749            ++__f;
1750            --base::__sz();
1751#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1752            __c_node* __c = __get_db()->__find_c_and_lock(this);
1753            for (__i_node** __p = __c->end_; __p != __c->beg_; )
1754            {
1755                --__p;
1756                iterator* __i = static_cast<iterator*>((*__p)->__i_);
1757                if (__i->__ptr_ == __n)
1758                {
1759                    (*__p)->__c_ = nullptr;
1760                    if (--__c->end_ != __p)
1761                        _VSTD::memmove(__p, __p+1, (__c->end_ - __p)*sizeof(__i_node*));
1762                }
1763            }
1764            __get_db()->unlock();
1765#endif
1766            __node_pointer __np = __n->__as_node();
1767            __node_alloc_traits::destroy(__na, _VSTD::addressof(__np->__value_));
1768            __node_alloc_traits::deallocate(__na, __np, 1);
1769        }
1770    }
1771    return iterator(__l.__ptr_, this);
1772}
1773
1774template <class _Tp, class _Alloc>
1775void
1776list<_Tp, _Alloc>::resize(size_type __n)
1777{
1778    if (__n < base::__sz())
1779        erase(__iterator(__n), end());
1780    else if (__n > base::__sz())
1781    {
1782        __n -= base::__sz();
1783        size_type __ds = 0;
1784        __node_allocator& __na = base::__node_alloc();
1785        __hold_pointer __hold = __allocate_node(__na);
1786        __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_));
1787        ++__ds;
1788        iterator __r = iterator(__hold.release()->__as_link(), this);
1789        iterator __e = __r;
1790#ifndef _LIBCPP_NO_EXCEPTIONS
1791        try
1792        {
1793#endif // _LIBCPP_NO_EXCEPTIONS
1794            for (--__n; __n != 0; --__n, (void) ++__e, ++__ds)
1795            {
1796                __hold.reset(__node_alloc_traits::allocate(__na, 1));
1797                __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_));
1798                __e.__ptr_->__next_ = __hold.get()->__as_link();
1799                __hold->__prev_ = __e.__ptr_;
1800                __hold.release();
1801            }
1802#ifndef _LIBCPP_NO_EXCEPTIONS
1803        }
1804        catch (...)
1805        {
1806            while (true)
1807            {
1808                __node_alloc_traits::destroy(__na, _VSTD::addressof(*__e));
1809                __link_pointer __prev = __e.__ptr_->__prev_;
1810                __node_alloc_traits::deallocate(__na, __e.__ptr_->__as_node(), 1);
1811                if (__prev == 0)
1812                    break;
1813                __e = iterator(__prev, this);
1814            }
1815            throw;
1816        }
1817#endif // _LIBCPP_NO_EXCEPTIONS
1818        __link_nodes_at_back(__r.__ptr_, __e.__ptr_);
1819        base::__sz() += __ds;
1820    }
1821}
1822
1823template <class _Tp, class _Alloc>
1824void
1825list<_Tp, _Alloc>::resize(size_type __n, const value_type& __x)
1826{
1827    if (__n < base::__sz())
1828        erase(__iterator(__n), end());
1829    else if (__n > base::__sz())
1830    {
1831        __n -= base::__sz();
1832        size_type __ds = 0;
1833        __node_allocator& __na = base::__node_alloc();
1834        __hold_pointer __hold = __allocate_node(__na);
1835        __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1836        ++__ds;
1837        __link_pointer __nl = __hold.release()->__as_link();
1838        iterator __r = iterator(__nl, this);
1839        iterator __e = __r;
1840#ifndef _LIBCPP_NO_EXCEPTIONS
1841        try
1842        {
1843#endif // _LIBCPP_NO_EXCEPTIONS
1844            for (--__n; __n != 0; --__n, (void) ++__e, ++__ds)
1845            {
1846                __hold.reset(__node_alloc_traits::allocate(__na, 1));
1847                __node_alloc_traits::construct(__na, _VSTD::addressof(__hold->__value_), __x);
1848                __e.__ptr_->__next_ = __hold.get()->__as_link();
1849                __hold->__prev_ = __e.__ptr_;
1850                __hold.release();
1851            }
1852#ifndef _LIBCPP_NO_EXCEPTIONS
1853        }
1854        catch (...)
1855        {
1856            while (true)
1857            {
1858                __node_alloc_traits::destroy(__na, _VSTD::addressof(*__e));
1859                __link_pointer __prev = __e.__ptr_->__prev_;
1860                __node_alloc_traits::deallocate(__na, __e.__ptr_->__as_node(), 1);
1861                if (__prev == 0)
1862                    break;
1863                __e = iterator(__prev, this);
1864            }
1865            throw;
1866        }
1867#endif // _LIBCPP_NO_EXCEPTIONS
1868        __link_nodes(base::__end_as_link(), __r.__ptr_, __e.__ptr_);
1869        base::__sz() += __ds;
1870    }
1871}
1872
1873template <class _Tp, class _Alloc>
1874void
1875list<_Tp, _Alloc>::splice(const_iterator __p, list& __c)
1876{
1877    _LIBCPP_ASSERT(this != _VSTD::addressof(__c),
1878                   "list::splice(iterator, list) called with this == &list");
1879    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1880                         "list::splice(iterator, list) called with an iterator not referring to this list");
1881    if (!__c.empty())
1882    {
1883        __link_pointer __f = __c.__end_.__next_;
1884        __link_pointer __l = __c.__end_.__prev_;
1885        base::__unlink_nodes(__f, __l);
1886        __link_nodes(__p.__ptr_, __f, __l);
1887        base::__sz() += __c.__sz();
1888        __c.__sz() = 0;
1889#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1890        if (_VSTD::addressof(__c) != this) {
1891            __libcpp_db* __db = __get_db();
1892            __c_node* __cn1 = __db->__find_c_and_lock(this);
1893            __c_node* __cn2 = __db->__find_c(_VSTD::addressof(__c));
1894            for (__i_node** __ip = __cn2->end_; __ip != __cn2->beg_;)
1895            {
1896                --__ip;
1897                iterator* __i = static_cast<iterator*>((*__ip)->__i_);
1898                if (__i->__ptr_ != __c.__end_as_link())
1899                {
1900                    __cn1->__add(*__ip);
1901                    (*__ip)->__c_ = __cn1;
1902                    if (--__cn2->end_ != __ip)
1903                        _VSTD::memmove(__ip, __ip+1, (__cn2->end_ - __ip)*sizeof(__i_node*));
1904                }
1905            }
1906            __db->unlock();
1907        }
1908#endif
1909    }
1910}
1911
1912template <class _Tp, class _Alloc>
1913void
1914list<_Tp, _Alloc>::splice(const_iterator __p, list& __c, const_iterator __i)
1915{
1916    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1917        "list::splice(iterator, list, iterator) called with the first iterator not referring to this list");
1918    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__i)) == _VSTD::addressof(__c),
1919        "list::splice(iterator, list, iterator) called with the second iterator not referring to the list argument");
1920    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__dereferenceable(_VSTD::addressof(__i)),
1921        "list::splice(iterator, list, iterator) called with the second iterator not dereferenceable");
1922
1923    if (__p.__ptr_ != __i.__ptr_ && __p.__ptr_ != __i.__ptr_->__next_)
1924    {
1925        __link_pointer __f = __i.__ptr_;
1926        base::__unlink_nodes(__f, __f);
1927        __link_nodes(__p.__ptr_, __f, __f);
1928        --__c.__sz();
1929        ++base::__sz();
1930#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1931        if (_VSTD::addressof(__c) != this) {
1932            __libcpp_db* __db = __get_db();
1933            __c_node* __cn1 = __db->__find_c_and_lock(this);
1934            __c_node* __cn2 = __db->__find_c(_VSTD::addressof(__c));
1935            for (__i_node** __ip = __cn2->end_; __ip != __cn2->beg_;)
1936            {
1937                --__ip;
1938                iterator* __j = static_cast<iterator*>((*__ip)->__i_);
1939                if (__j->__ptr_ == __f)
1940                {
1941                    __cn1->__add(*__ip);
1942                    (*__ip)->__c_ = __cn1;
1943                    if (--__cn2->end_ != __ip)
1944                        _VSTD::memmove(__ip, __ip+1, (__cn2->end_ - __ip)*sizeof(__i_node*));
1945                }
1946            }
1947            __db->unlock();
1948        }
1949#endif
1950    }
1951}
1952
1953template <class _Iterator>
1954_LIBCPP_HIDE_FROM_ABI
1955bool __iterator_in_range(_Iterator __first, _Iterator __last, _Iterator __it) {
1956    for (_Iterator __p = __first; __p != __last; ++__p) {
1957        if (__p == __it) {
1958            return true;
1959        }
1960    }
1961    return false;
1962}
1963
1964template <class _Tp, class _Alloc>
1965void
1966list<_Tp, _Alloc>::splice(const_iterator __p, list& __c, const_iterator __f, const_iterator __l)
1967{
1968    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__p)) == this,
1969        "list::splice(iterator, list, iterator, iterator) called with first iterator not referring to this list");
1970    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__f)) == _VSTD::addressof(__c),
1971        "list::splice(iterator, list, iterator, iterator) called with second iterator not referring to the list argument");
1972    _LIBCPP_DEBUG_ASSERT(__get_const_db()->__find_c_from_i(_VSTD::addressof(__l)) == _VSTD::addressof(__c),
1973        "list::splice(iterator, list, iterator, iterator) called with third iterator not referring to the list argument");
1974    _LIBCPP_DEBUG_ASSERT(this != std::addressof(__c) || !std::__iterator_in_range(__f, __l, __p),
1975        "list::splice(iterator, list, iterator, iterator)"
1976        " called with the first iterator within the range of the second and third iterators");
1977
1978    if (__f != __l)
1979    {
1980        __link_pointer __first = __f.__ptr_;
1981        --__l;
1982        __link_pointer __last = __l.__ptr_;
1983        if (this != _VSTD::addressof(__c))
1984        {
1985            size_type __s = _VSTD::distance(__f, __l) + 1;
1986            __c.__sz() -= __s;
1987            base::__sz() += __s;
1988        }
1989        base::__unlink_nodes(__first, __last);
1990        __link_nodes(__p.__ptr_, __first, __last);
1991#ifdef _LIBCPP_ENABLE_DEBUG_MODE
1992        if (_VSTD::addressof(__c) != this) {
1993            __libcpp_db* __db = __get_db();
1994            __c_node* __cn1 = __db->__find_c_and_lock(this);
1995            __c_node* __cn2 = __db->__find_c(_VSTD::addressof(__c));
1996            for (__i_node** __ip = __cn2->end_; __ip != __cn2->beg_;)
1997            {
1998                --__ip;
1999                iterator* __j = static_cast<iterator*>((*__ip)->__i_);
2000                for (__link_pointer __k = __f.__ptr_;
2001                                              __k != __l.__ptr_; __k = __k->__next_)
2002                {
2003                    if (__j->__ptr_ == __k)
2004                    {
2005                        __cn1->__add(*__ip);
2006                        (*__ip)->__c_ = __cn1;
2007                        if (--__cn2->end_ != __ip)
2008                            _VSTD::memmove(__ip, __ip+1, (__cn2->end_ - __ip)*sizeof(__i_node*));
2009                    }
2010                }
2011            }
2012            __db->unlock();
2013        }
2014#endif
2015    }
2016}
2017
2018template <class _Tp, class _Alloc>
2019typename list<_Tp, _Alloc>::__remove_return_type
2020list<_Tp, _Alloc>::remove(const value_type& __x)
2021{
2022    list<_Tp, _Alloc> __deleted_nodes(get_allocator()); // collect the nodes we're removing
2023    for (const_iterator __i = begin(), __e = end(); __i != __e;)
2024    {
2025        if (*__i == __x)
2026        {
2027            const_iterator __j = _VSTD::next(__i);
2028            for (; __j != __e && *__j == __x; ++__j)
2029                ;
2030            __deleted_nodes.splice(__deleted_nodes.end(), *this, __i, __j);
2031            __i = __j;
2032            if (__i != __e)
2033                ++__i;
2034        }
2035        else
2036            ++__i;
2037    }
2038
2039    return (__remove_return_type) __deleted_nodes.size();
2040}
2041
2042template <class _Tp, class _Alloc>
2043template <class _Pred>
2044typename list<_Tp, _Alloc>::__remove_return_type
2045list<_Tp, _Alloc>::remove_if(_Pred __pred)
2046{
2047    list<_Tp, _Alloc> __deleted_nodes(get_allocator()); // collect the nodes we're removing
2048    for (iterator __i = begin(), __e = end(); __i != __e;)
2049    {
2050        if (__pred(*__i))
2051        {
2052            iterator __j = _VSTD::next(__i);
2053            for (; __j != __e && __pred(*__j); ++__j)
2054                ;
2055            __deleted_nodes.splice(__deleted_nodes.end(), *this, __i, __j);
2056            __i = __j;
2057            if (__i != __e)
2058                ++__i;
2059        }
2060        else
2061            ++__i;
2062    }
2063
2064    return (__remove_return_type) __deleted_nodes.size();
2065}
2066
2067template <class _Tp, class _Alloc>
2068template <class _BinaryPred>
2069typename list<_Tp, _Alloc>::__remove_return_type
2070list<_Tp, _Alloc>::unique(_BinaryPred __binary_pred)
2071{
2072    list<_Tp, _Alloc> __deleted_nodes(get_allocator()); // collect the nodes we're removing
2073    for (iterator __i = begin(), __e = end(); __i != __e;)
2074    {
2075        iterator __j = _VSTD::next(__i);
2076        for (; __j != __e && __binary_pred(*__i, *__j); ++__j)
2077            ;
2078        if (++__i != __j) {
2079            __deleted_nodes.splice(__deleted_nodes.end(), *this, __i, __j);
2080            __i = __j;
2081            }
2082    }
2083
2084    return (__remove_return_type) __deleted_nodes.size();
2085}
2086
2087template <class _Tp, class _Alloc>
2088inline
2089void
2090list<_Tp, _Alloc>::merge(list& __c)
2091{
2092    merge(__c, __less<value_type>());
2093}
2094
2095template <class _Tp, class _Alloc>
2096template <class _Comp>
2097void
2098list<_Tp, _Alloc>::merge(list& __c, _Comp __comp)
2099{
2100    if (this != _VSTD::addressof(__c))
2101    {
2102        iterator __f1 = begin();
2103        iterator __e1 = end();
2104        iterator __f2 = __c.begin();
2105        iterator __e2 = __c.end();
2106        while (__f1 != __e1 && __f2 != __e2)
2107        {
2108            if (__comp(*__f2, *__f1))
2109            {
2110                size_type __ds = 1;
2111                iterator __m2 = _VSTD::next(__f2);
2112                for (; __m2 != __e2 && __comp(*__m2, *__f1); ++__m2, (void) ++__ds)
2113                    ;
2114                base::__sz() += __ds;
2115                __c.__sz() -= __ds;
2116                __link_pointer __f = __f2.__ptr_;
2117                __link_pointer __l = __m2.__ptr_->__prev_;
2118                __f2 = __m2;
2119                base::__unlink_nodes(__f, __l);
2120                __m2 = _VSTD::next(__f1);
2121                __link_nodes(__f1.__ptr_, __f, __l);
2122                __f1 = __m2;
2123            }
2124            else
2125                ++__f1;
2126        }
2127        splice(__e1, __c);
2128#ifdef _LIBCPP_ENABLE_DEBUG_MODE
2129        __libcpp_db* __db = __get_db();
2130        __c_node* __cn1 = __db->__find_c_and_lock(this);
2131        __c_node* __cn2 = __db->__find_c(_VSTD::addressof(__c));
2132        for (__i_node** __p = __cn2->end_; __p != __cn2->beg_;)
2133        {
2134            --__p;
2135            iterator* __i = static_cast<iterator*>((*__p)->__i_);
2136            if (__i->__ptr_ != __c.__end_as_link())
2137            {
2138                __cn1->__add(*__p);
2139                (*__p)->__c_ = __cn1;
2140                if (--__cn2->end_ != __p)
2141                    _VSTD::memmove(__p, __p+1, (__cn2->end_ - __p)*sizeof(__i_node*));
2142            }
2143        }
2144        __db->unlock();
2145#endif
2146    }
2147}
2148
2149template <class _Tp, class _Alloc>
2150inline
2151void
2152list<_Tp, _Alloc>::sort()
2153{
2154    sort(__less<value_type>());
2155}
2156
2157template <class _Tp, class _Alloc>
2158template <class _Comp>
2159inline
2160void
2161list<_Tp, _Alloc>::sort(_Comp __comp)
2162{
2163    __sort(begin(), end(), base::__sz(), __comp);
2164}
2165
2166template <class _Tp, class _Alloc>
2167template <class _Comp>
2168typename list<_Tp, _Alloc>::iterator
2169list<_Tp, _Alloc>::__sort(iterator __f1, iterator __e2, size_type __n, _Comp& __comp)
2170{
2171    switch (__n)
2172    {
2173    case 0:
2174    case 1:
2175        return __f1;
2176    case 2:
2177        if (__comp(*--__e2, *__f1))
2178        {
2179            __link_pointer __f = __e2.__ptr_;
2180            base::__unlink_nodes(__f, __f);
2181            __link_nodes(__f1.__ptr_, __f, __f);
2182            return __e2;
2183        }
2184        return __f1;
2185    }
2186    size_type __n2 = __n / 2;
2187    iterator __e1 = _VSTD::next(__f1, __n2);
2188    iterator  __r = __f1 = __sort(__f1, __e1, __n2, __comp);
2189    iterator __f2 = __e1 = __sort(__e1, __e2, __n - __n2, __comp);
2190    if (__comp(*__f2, *__f1))
2191    {
2192        iterator __m2 = _VSTD::next(__f2);
2193        for (; __m2 != __e2 && __comp(*__m2, *__f1); ++__m2)
2194            ;
2195        __link_pointer __f = __f2.__ptr_;
2196        __link_pointer __l = __m2.__ptr_->__prev_;
2197        __r = __f2;
2198        __e1 = __f2 = __m2;
2199        base::__unlink_nodes(__f, __l);
2200        __m2 = _VSTD::next(__f1);
2201        __link_nodes(__f1.__ptr_, __f, __l);
2202        __f1 = __m2;
2203    }
2204    else
2205        ++__f1;
2206    while (__f1 != __e1 && __f2 != __e2)
2207    {
2208        if (__comp(*__f2, *__f1))
2209        {
2210            iterator __m2 = _VSTD::next(__f2);
2211            for (; __m2 != __e2 && __comp(*__m2, *__f1); ++__m2)
2212                ;
2213            __link_pointer __f = __f2.__ptr_;
2214            __link_pointer __l = __m2.__ptr_->__prev_;
2215            if (__e1 == __f2)
2216                __e1 = __m2;
2217            __f2 = __m2;
2218            base::__unlink_nodes(__f, __l);
2219            __m2 = _VSTD::next(__f1);
2220            __link_nodes(__f1.__ptr_, __f, __l);
2221            __f1 = __m2;
2222        }
2223        else
2224            ++__f1;
2225    }
2226    return __r;
2227}
2228
2229template <class _Tp, class _Alloc>
2230void
2231list<_Tp, _Alloc>::reverse() _NOEXCEPT
2232{
2233    if (base::__sz() > 1)
2234    {
2235        iterator __e = end();
2236        for (iterator __i = begin(); __i.__ptr_ != __e.__ptr_;)
2237        {
2238            _VSTD::swap(__i.__ptr_->__prev_, __i.__ptr_->__next_);
2239            __i.__ptr_ = __i.__ptr_->__prev_;
2240        }
2241        _VSTD::swap(__e.__ptr_->__prev_, __e.__ptr_->__next_);
2242    }
2243}
2244
2245template <class _Tp, class _Alloc>
2246bool
2247list<_Tp, _Alloc>::__invariants() const
2248{
2249    return size() == _VSTD::distance(begin(), end());
2250}
2251
2252#ifdef _LIBCPP_ENABLE_DEBUG_MODE
2253
2254template <class _Tp, class _Alloc>
2255bool
2256list<_Tp, _Alloc>::__dereferenceable(const const_iterator* __i) const
2257{
2258    return __i->__ptr_ != this->__end_as_link();
2259}
2260
2261template <class _Tp, class _Alloc>
2262bool
2263list<_Tp, _Alloc>::__decrementable(const const_iterator* __i) const
2264{
2265    return !empty() &&  __i->__ptr_ != base::__end_.__next_;
2266}
2267
2268template <class _Tp, class _Alloc>
2269bool
2270list<_Tp, _Alloc>::__addable(const const_iterator*, ptrdiff_t) const
2271{
2272    return false;
2273}
2274
2275template <class _Tp, class _Alloc>
2276bool
2277list<_Tp, _Alloc>::__subscriptable(const const_iterator*, ptrdiff_t) const
2278{
2279    return false;
2280}
2281
2282#endif // _LIBCPP_ENABLE_DEBUG_MODE
2283
2284template <class _Tp, class _Alloc>
2285inline _LIBCPP_INLINE_VISIBILITY
2286bool
2287operator==(const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2288{
2289    return __x.size() == __y.size() && _VSTD::equal(__x.begin(), __x.end(), __y.begin());
2290}
2291
2292template <class _Tp, class _Alloc>
2293inline _LIBCPP_INLINE_VISIBILITY
2294bool
2295operator< (const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2296{
2297    return _VSTD::lexicographical_compare(__x.begin(), __x.end(), __y.begin(), __y.end());
2298}
2299
2300template <class _Tp, class _Alloc>
2301inline _LIBCPP_INLINE_VISIBILITY
2302bool
2303operator!=(const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2304{
2305    return !(__x == __y);
2306}
2307
2308template <class _Tp, class _Alloc>
2309inline _LIBCPP_INLINE_VISIBILITY
2310bool
2311operator> (const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2312{
2313    return __y < __x;
2314}
2315
2316template <class _Tp, class _Alloc>
2317inline _LIBCPP_INLINE_VISIBILITY
2318bool
2319operator>=(const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2320{
2321    return !(__x < __y);
2322}
2323
2324template <class _Tp, class _Alloc>
2325inline _LIBCPP_INLINE_VISIBILITY
2326bool
2327operator<=(const list<_Tp, _Alloc>& __x, const list<_Tp, _Alloc>& __y)
2328{
2329    return !(__y < __x);
2330}
2331
2332template <class _Tp, class _Alloc>
2333inline _LIBCPP_INLINE_VISIBILITY
2334void
2335swap(list<_Tp, _Alloc>& __x, list<_Tp, _Alloc>& __y)
2336    _NOEXCEPT_(_NOEXCEPT_(__x.swap(__y)))
2337{
2338    __x.swap(__y);
2339}
2340
2341#if _LIBCPP_STD_VER > 17
2342template <class _Tp, class _Allocator, class _Predicate>
2343inline _LIBCPP_INLINE_VISIBILITY typename list<_Tp, _Allocator>::size_type
2344erase_if(list<_Tp, _Allocator>& __c, _Predicate __pred) {
2345  return __c.remove_if(__pred);
2346}
2347
2348template <class _Tp, class _Allocator, class _Up>
2349inline _LIBCPP_INLINE_VISIBILITY typename list<_Tp, _Allocator>::size_type
2350erase(list<_Tp, _Allocator>& __c, const _Up& __v) {
2351  return _VSTD::erase_if(__c, [&](auto& __elem) { return __elem == __v; });
2352}
2353
2354template <>
2355inline constexpr bool __format::__enable_insertable<std::list<char>> = true;
2356#ifndef _LIBCPP_HAS_NO_WIDE_CHARACTERS
2357template <>
2358inline constexpr bool __format::__enable_insertable<std::list<wchar_t>> = true;
2359#endif
2360
2361#endif // _LIBCPP_STD_VER > 17
2362
2363_LIBCPP_END_NAMESPACE_STD
2364
2365_LIBCPP_POP_MACROS
2366
2367#endif // _LIBCPP_LIST
2368