Home | History | Annotate | Download | only in include
      1 // -*- C++ -*-
      2 //===---------------------------- set -------------------------------------===//
      3 //
      4 //                     The LLVM Compiler Infrastructure
      5 //
      6 // This file is dual licensed under the MIT and the University of Illinois Open
      7 // Source Licenses. See LICENSE.TXT for details.
      8 //
      9 //===----------------------------------------------------------------------===//
     10 
     11 #ifndef _LIBCPP_SET
     12 #define _LIBCPP_SET
     13 
     14 /*
     15 
     16     set synopsis
     17 
     18 namespace std
     19 {
     20 
     21 template <class Key, class Compare = less<Key>,
     22           class Allocator = allocator<Key>>
     23 class set
     24 {
     25 public:
     26     // types:
     27     typedef Key                                      key_type;
     28     typedef key_type                                 value_type;
     29     typedef Compare                                  key_compare;
     30     typedef key_compare                              value_compare;
     31     typedef Allocator                                allocator_type;
     32     typedef typename allocator_type::reference       reference;
     33     typedef typename allocator_type::const_reference const_reference;
     34     typedef typename allocator_type::size_type       size_type;
     35     typedef typename allocator_type::difference_type difference_type;
     36     typedef typename allocator_type::pointer         pointer;
     37     typedef typename allocator_type::const_pointer   const_pointer;
     38 
     39     typedef implementation-defined                   iterator;
     40     typedef implementation-defined                   const_iterator;
     41     typedef std::reverse_iterator<iterator>          reverse_iterator;
     42     typedef std::reverse_iterator<const_iterator>    const_reverse_iterator;
     43 
     44     // construct/copy/destroy:
     45     set()
     46         noexcept(
     47             is_nothrow_default_constructible<allocator_type>::value &&
     48             is_nothrow_default_constructible<key_compare>::value &&
     49             is_nothrow_copy_constructible<key_compare>::value);
     50     explicit set(const value_compare& comp);
     51     set(const value_compare& comp, const allocator_type& a);
     52     template <class InputIterator>
     53         set(InputIterator first, InputIterator last,
     54             const value_compare& comp = value_compare());
     55     template <class InputIterator>
     56         set(InputIterator first, InputIterator last, const value_compare& comp,
     57             const allocator_type& a);
     58     set(const set& s);
     59     set(set&& s)
     60         noexcept(
     61             is_nothrow_move_constructible<allocator_type>::value &&
     62             is_nothrow_move_constructible<key_compare>::value);
     63     explicit set(const allocator_type& a);
     64     set(const set& s, const allocator_type& a);
     65     set(set&& s, const allocator_type& a);
     66     set(initializer_list<value_type> il, const value_compare& comp = value_compare());
     67     set(initializer_list<value_type> il, const value_compare& comp,
     68         const allocator_type& a);
     69     ~set();
     70 
     71     set& operator=(const set& s);
     72     set& operator=(set&& s)
     73         noexcept(
     74             allocator_type::propagate_on_container_move_assignment::value &&
     75             is_nothrow_move_assignable<allocator_type>::value &&
     76             is_nothrow_move_assignable<key_compare>::value);
     77     set& operator=(initializer_list<value_type> il);
     78 
     79     // iterators:
     80           iterator begin() noexcept;
     81     const_iterator begin() const noexcept;
     82           iterator end() noexcept;
     83     const_iterator end()   const noexcept;
     84 
     85           reverse_iterator rbegin() noexcept;
     86     const_reverse_iterator rbegin() const noexcept;
     87           reverse_iterator rend() noexcept;
     88     const_reverse_iterator rend()   const noexcept;
     89 
     90     const_iterator         cbegin()  const noexcept;
     91     const_iterator         cend()    const noexcept;
     92     const_reverse_iterator crbegin() const noexcept;
     93     const_reverse_iterator crend()   const noexcept;
     94 
     95     // capacity:
     96     bool      empty()    const noexcept;
     97     size_type size()     const noexcept;
     98     size_type max_size() const noexcept;
     99 
    100     // modifiers:
    101     template <class... Args>
    102         pair<iterator, bool> emplace(Args&&... args);
    103     template <class... Args>
    104         iterator emplace_hint(const_iterator position, Args&&... args);
    105     pair<iterator,bool> insert(const value_type& v);
    106     pair<iterator,bool> insert(value_type&& v);
    107     iterator insert(const_iterator position, const value_type& v);
    108     iterator insert(const_iterator position, value_type&& v);
    109     template <class InputIterator>
    110         void insert(InputIterator first, InputIterator last);
    111     void insert(initializer_list<value_type> il);
    112 
    113     iterator  erase(const_iterator position);
    114     size_type erase(const key_type& k);
    115     iterator  erase(const_iterator first, const_iterator last);
    116     void clear() noexcept;
    117 
    118     void swap(set& s)
    119         noexcept(
    120             __is_nothrow_swappable<key_compare>::value &&
    121             (!allocator_type::propagate_on_container_swap::value ||
    122              __is_nothrow_swappable<allocator_type>::value));
    123 
    124     // observers:
    125     allocator_type get_allocator() const noexcept;
    126     key_compare    key_comp()      const;
    127     value_compare  value_comp()    const;
    128 
    129     // set operations:
    130           iterator find(const key_type& k);
    131     const_iterator find(const key_type& k) const;
    132     size_type      count(const key_type& k) const;
    133           iterator lower_bound(const key_type& k);
    134     const_iterator lower_bound(const key_type& k) const;
    135           iterator upper_bound(const key_type& k);
    136     const_iterator upper_bound(const key_type& k) const;
    137     pair<iterator,iterator>             equal_range(const key_type& k);
    138     pair<const_iterator,const_iterator> equal_range(const key_type& k) const;
    139 };
    140 
    141 template <class Key, class Compare, class Allocator>
    142 bool
    143 operator==(const set<Key, Compare, Allocator>& x,
    144            const set<Key, Compare, Allocator>& y);
    145 
    146 template <class Key, class Compare, class Allocator>
    147 bool
    148 operator< (const set<Key, Compare, Allocator>& x,
    149            const set<Key, Compare, Allocator>& y);
    150 
    151 template <class Key, class Compare, class Allocator>
    152 bool
    153 operator!=(const set<Key, Compare, Allocator>& x,
    154            const set<Key, Compare, Allocator>& y);
    155 
    156 template <class Key, class Compare, class Allocator>
    157 bool
    158 operator> (const set<Key, Compare, Allocator>& x,
    159            const set<Key, Compare, Allocator>& y);
    160 
    161 template <class Key, class Compare, class Allocator>
    162 bool
    163 operator>=(const set<Key, Compare, Allocator>& x,
    164            const set<Key, Compare, Allocator>& y);
    165 
    166 template <class Key, class Compare, class Allocator>
    167 bool
    168 operator<=(const set<Key, Compare, Allocator>& x,
    169            const set<Key, Compare, Allocator>& y);
    170 
    171 // specialized algorithms:
    172 template <class Key, class Compare, class Allocator>
    173 void
    174 swap(set<Key, Compare, Allocator>& x, set<Key, Compare, Allocator>& y)
    175     noexcept(noexcept(x.swap(y)));
    176 
    177 template <class Key, class Compare = less<Key>,
    178           class Allocator = allocator<Key>>
    179 class multiset
    180 {
    181 public:
    182     // types:
    183     typedef Key                                      key_type;
    184     typedef key_type                                 value_type;
    185     typedef Compare                                  key_compare;
    186     typedef key_compare                              value_compare;
    187     typedef Allocator                                allocator_type;
    188     typedef typename allocator_type::reference       reference;
    189     typedef typename allocator_type::const_reference const_reference;
    190     typedef typename allocator_type::size_type       size_type;
    191     typedef typename allocator_type::difference_type difference_type;
    192     typedef typename allocator_type::pointer         pointer;
    193     typedef typename allocator_type::const_pointer   const_pointer;
    194 
    195     typedef implementation-defined                   iterator;
    196     typedef implementation-defined                   const_iterator;
    197     typedef std::reverse_iterator<iterator>          reverse_iterator;
    198     typedef std::reverse_iterator<const_iterator>    const_reverse_iterator;
    199 
    200     // construct/copy/destroy:
    201     multiset()
    202         noexcept(
    203             is_nothrow_default_constructible<allocator_type>::value &&
    204             is_nothrow_default_constructible<key_compare>::value &&
    205             is_nothrow_copy_constructible<key_compare>::value);
    206     explicit multiset(const value_compare& comp);
    207     multiset(const value_compare& comp, const allocator_type& a);
    208     template <class InputIterator>
    209         multiset(InputIterator first, InputIterator last,
    210                  const value_compare& comp = value_compare());
    211     template <class InputIterator>
    212         multiset(InputIterator first, InputIterator last,
    213                  const value_compare& comp, const allocator_type& a);
    214     multiset(const multiset& s);
    215     multiset(multiset&& s)
    216         noexcept(
    217             is_nothrow_move_constructible<allocator_type>::value &&
    218             is_nothrow_move_constructible<key_compare>::value);
    219     explicit multiset(const allocator_type& a);
    220     multiset(const multiset& s, const allocator_type& a);
    221     multiset(multiset&& s, const allocator_type& a);
    222     multiset(initializer_list<value_type> il, const value_compare& comp = value_compare());
    223     multiset(initializer_list<value_type> il, const value_compare& comp,
    224              const allocator_type& a);
    225     ~multiset();
    226 
    227     multiset& operator=(const multiset& s);
    228     multiset& operator=(multiset&& s)
    229         noexcept(
    230             allocator_type::propagate_on_container_move_assignment::value &&
    231             is_nothrow_move_assignable<allocator_type>::value &&
    232             is_nothrow_move_assignable<key_compare>::value);
    233     multiset& operator=(initializer_list<value_type> il);
    234 
    235     // iterators:
    236           iterator begin() noexcept;
    237     const_iterator begin() const noexcept;
    238           iterator end() noexcept;
    239     const_iterator end()   const noexcept;
    240 
    241           reverse_iterator rbegin() noexcept;
    242     const_reverse_iterator rbegin() const noexcept;
    243           reverse_iterator rend() noexcept;
    244     const_reverse_iterator rend()   const noexcept;
    245 
    246     const_iterator         cbegin()  const noexcept;
    247     const_iterator         cend()    const noexcept;
    248     const_reverse_iterator crbegin() const noexcept;
    249     const_reverse_iterator crend()   const noexcept;
    250 
    251     // capacity:
    252     bool      empty()    const noexcept;
    253     size_type size()     const noexcept;
    254     size_type max_size() const noexcept;
    255 
    256     // modifiers:
    257     template <class... Args>
    258         iterator emplace(Args&&... args);
    259     template <class... Args>
    260         iterator emplace_hint(const_iterator position, Args&&... args);
    261     iterator insert(const value_type& v);
    262     iterator insert(value_type&& v);
    263     iterator insert(const_iterator position, const value_type& v);
    264     iterator insert(const_iterator position, value_type&& v);
    265     template <class InputIterator>
    266         void insert(InputIterator first, InputIterator last);
    267     void insert(initializer_list<value_type> il);
    268 
    269     iterator  erase(const_iterator position);
    270     size_type erase(const key_type& k);
    271     iterator  erase(const_iterator first, const_iterator last);
    272     void clear() noexcept;
    273 
    274     void swap(multiset& s)
    275         noexcept(
    276             __is_nothrow_swappable<key_compare>::value &&
    277             (!allocator_type::propagate_on_container_swap::value ||
    278              __is_nothrow_swappable<allocator_type>::value));
    279 
    280     // observers:
    281     allocator_type get_allocator() const noexcept;
    282     key_compare    key_comp()      const;
    283     value_compare  value_comp()    const;
    284 
    285     // set operations:
    286           iterator find(const key_type& k);
    287     const_iterator find(const key_type& k) const;
    288     size_type      count(const key_type& k) const;
    289           iterator lower_bound(const key_type& k);
    290     const_iterator lower_bound(const key_type& k) const;
    291           iterator upper_bound(const key_type& k);
    292     const_iterator upper_bound(const key_type& k) const;
    293     pair<iterator,iterator>             equal_range(const key_type& k);
    294     pair<const_iterator,const_iterator> equal_range(const key_type& k) const;
    295 };
    296 
    297 template <class Key, class Compare, class Allocator>
    298 bool
    299 operator==(const multiset<Key, Compare, Allocator>& x,
    300            const multiset<Key, Compare, Allocator>& y);
    301 
    302 template <class Key, class Compare, class Allocator>
    303 bool
    304 operator< (const multiset<Key, Compare, Allocator>& x,
    305            const multiset<Key, Compare, Allocator>& y);
    306 
    307 template <class Key, class Compare, class Allocator>
    308 bool
    309 operator!=(const multiset<Key, Compare, Allocator>& x,
    310            const multiset<Key, Compare, Allocator>& y);
    311 
    312 template <class Key, class Compare, class Allocator>
    313 bool
    314 operator> (const multiset<Key, Compare, Allocator>& x,
    315            const multiset<Key, Compare, Allocator>& y);
    316 
    317 template <class Key, class Compare, class Allocator>
    318 bool
    319 operator>=(const multiset<Key, Compare, Allocator>& x,
    320            const multiset<Key, Compare, Allocator>& y);
    321 
    322 template <class Key, class Compare, class Allocator>
    323 bool
    324 operator<=(const multiset<Key, Compare, Allocator>& x,
    325            const multiset<Key, Compare, Allocator>& y);
    326 
    327 // specialized algorithms:
    328 template <class Key, class Compare, class Allocator>
    329 void
    330 swap(multiset<Key, Compare, Allocator>& x, multiset<Key, Compare, Allocator>& y)
    331     noexcept(noexcept(x.swap(y)));
    332 
    333 }  // std
    334 
    335 */
    336 
    337 #include <__config>
    338 #include <__tree>
    339 #include <functional>
    340 
    341 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
    342 #pragma GCC system_header
    343 #endif
    344 
    345 _LIBCPP_BEGIN_NAMESPACE_STD
    346 
    347 template <class _Key, class _Compare = less<_Key>,
    348           class _Allocator = allocator<_Key> >
    349 class _LIBCPP_TYPE_VIS set
    350 {
    351 public:
    352     // types:
    353     typedef _Key                                     key_type;
    354     typedef key_type                                 value_type;
    355     typedef _Compare                                 key_compare;
    356     typedef key_compare                              value_compare;
    357     typedef _Allocator                               allocator_type;
    358     typedef value_type&                              reference;
    359     typedef const value_type&                        const_reference;
    360 
    361 private:
    362     typedef __tree<value_type, value_compare, allocator_type> __base;
    363     typedef allocator_traits<allocator_type>                  __alloc_traits;
    364     typedef typename __base::__node_holder                    __node_holder;
    365 
    366     __base __tree_;
    367 
    368 public:
    369     typedef typename __base::pointer               pointer;
    370     typedef typename __base::const_pointer         const_pointer;
    371     typedef typename __base::size_type             size_type;
    372     typedef typename __base::difference_type       difference_type;
    373     typedef typename __base::const_iterator        iterator;
    374     typedef typename __base::const_iterator        const_iterator;
    375     typedef _VSTD::reverse_iterator<iterator>       reverse_iterator;
    376     typedef _VSTD::reverse_iterator<const_iterator> const_reverse_iterator;
    377 
    378     _LIBCPP_INLINE_VISIBILITY
    379     explicit set(const value_compare& __comp = value_compare())
    380         _NOEXCEPT_(
    381             is_nothrow_default_constructible<allocator_type>::value &&
    382             is_nothrow_default_constructible<key_compare>::value &&
    383             is_nothrow_copy_constructible<key_compare>::value)
    384         : __tree_(__comp) {}
    385     _LIBCPP_INLINE_VISIBILITY
    386     set(const value_compare& __comp, const allocator_type& __a)
    387         : __tree_(__comp, __a) {}
    388     template <class _InputIterator>
    389         _LIBCPP_INLINE_VISIBILITY
    390         set(_InputIterator __f, _InputIterator __l,
    391             const value_compare& __comp = value_compare())
    392         : __tree_(__comp)
    393         {
    394             insert(__f, __l);
    395         }
    396 
    397     template <class _InputIterator>
    398         _LIBCPP_INLINE_VISIBILITY
    399         set(_InputIterator __f, _InputIterator __l, const value_compare& __comp,
    400             const allocator_type& __a)
    401         : __tree_(__comp, __a)
    402         {
    403             insert(__f, __l);
    404         }
    405 
    406     _LIBCPP_INLINE_VISIBILITY
    407     set(const set& __s)
    408         : __tree_(__s.__tree_)
    409         {
    410             insert(__s.begin(), __s.end());
    411         }
    412 
    413     _LIBCPP_INLINE_VISIBILITY
    414     set& operator=(const set& __s)
    415         {
    416             __tree_ = __s.__tree_;
    417             return *this;
    418         }
    419 
    420 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    421     _LIBCPP_INLINE_VISIBILITY
    422     set(set&& __s)
    423         _NOEXCEPT_(is_nothrow_move_constructible<__base>::value)
    424         : __tree_(_VSTD::move(__s.__tree_)) {}
    425 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    426 
    427     _LIBCPP_INLINE_VISIBILITY
    428     explicit set(const allocator_type& __a)
    429         : __tree_(__a) {}
    430 
    431     _LIBCPP_INLINE_VISIBILITY
    432     set(const set& __s, const allocator_type& __a)
    433         : __tree_(__s.__tree_.value_comp(), __a)
    434         {
    435             insert(__s.begin(), __s.end());
    436         }
    437 
    438 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    439     set(set&& __s, const allocator_type& __a);
    440 #endif
    441 
    442 #ifndef _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    443     _LIBCPP_INLINE_VISIBILITY
    444     set(initializer_list<value_type> __il, const value_compare& __comp = value_compare())
    445         : __tree_(__comp)
    446         {
    447             insert(__il.begin(), __il.end());
    448         }
    449 
    450     _LIBCPP_INLINE_VISIBILITY
    451     set(initializer_list<value_type> __il, const value_compare& __comp,
    452         const allocator_type& __a)
    453         : __tree_(__comp, __a)
    454         {
    455             insert(__il.begin(), __il.end());
    456         }
    457 
    458     _LIBCPP_INLINE_VISIBILITY
    459     set& operator=(initializer_list<value_type> __il)
    460         {
    461             __tree_.__assign_unique(__il.begin(), __il.end());
    462             return *this;
    463         }
    464 #endif  // _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    465 
    466 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    467     _LIBCPP_INLINE_VISIBILITY
    468     set& operator=(set&& __s)
    469         _NOEXCEPT_(is_nothrow_move_assignable<__base>::value)
    470         {
    471             __tree_ = _VSTD::move(__s.__tree_);
    472             return *this;
    473         }
    474 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    475 
    476     _LIBCPP_INLINE_VISIBILITY
    477           iterator begin() _NOEXCEPT       {return __tree_.begin();}
    478     _LIBCPP_INLINE_VISIBILITY
    479     const_iterator begin() const _NOEXCEPT {return __tree_.begin();}
    480     _LIBCPP_INLINE_VISIBILITY
    481           iterator end() _NOEXCEPT         {return __tree_.end();}
    482     _LIBCPP_INLINE_VISIBILITY
    483     const_iterator end()   const _NOEXCEPT {return __tree_.end();}
    484 
    485     _LIBCPP_INLINE_VISIBILITY
    486           reverse_iterator rbegin() _NOEXCEPT
    487             {return reverse_iterator(end());}
    488     _LIBCPP_INLINE_VISIBILITY
    489     const_reverse_iterator rbegin() const _NOEXCEPT
    490         {return const_reverse_iterator(end());}
    491     _LIBCPP_INLINE_VISIBILITY
    492           reverse_iterator rend() _NOEXCEPT
    493             {return reverse_iterator(begin());}
    494     _LIBCPP_INLINE_VISIBILITY
    495     const_reverse_iterator rend() const _NOEXCEPT
    496         {return const_reverse_iterator(begin());}
    497 
    498     _LIBCPP_INLINE_VISIBILITY
    499     const_iterator cbegin()  const _NOEXCEPT {return begin();}
    500     _LIBCPP_INLINE_VISIBILITY
    501     const_iterator cend() const _NOEXCEPT {return end();}
    502     _LIBCPP_INLINE_VISIBILITY
    503     const_reverse_iterator crbegin() const _NOEXCEPT {return rbegin();}
    504     _LIBCPP_INLINE_VISIBILITY
    505     const_reverse_iterator crend() const _NOEXCEPT {return rend();}
    506 
    507     _LIBCPP_INLINE_VISIBILITY
    508     bool empty() const _NOEXCEPT {return __tree_.size() == 0;}
    509     _LIBCPP_INLINE_VISIBILITY
    510     size_type size() const _NOEXCEPT {return __tree_.size();}
    511     _LIBCPP_INLINE_VISIBILITY
    512     size_type max_size() const _NOEXCEPT {return __tree_.max_size();}
    513 
    514     // modifiers:
    515 #if !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
    516     template <class... _Args>
    517         _LIBCPP_INLINE_VISIBILITY
    518         pair<iterator, bool> emplace(_Args&&... __args)
    519             {return __tree_.__emplace_unique(_VSTD::forward<_Args>(__args)...);}
    520     template <class... _Args>
    521         _LIBCPP_INLINE_VISIBILITY
    522         iterator emplace_hint(const_iterator __p, _Args&&... __args)
    523             {return __tree_.__emplace_hint_unique(__p, _VSTD::forward<_Args>(__args)...);}
    524 #endif  // !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
    525     _LIBCPP_INLINE_VISIBILITY
    526     pair<iterator,bool> insert(const value_type& __v)
    527         {return __tree_.__insert_unique(__v);}
    528 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    529     _LIBCPP_INLINE_VISIBILITY
    530     pair<iterator,bool> insert(value_type&& __v)
    531         {return __tree_.__insert_unique(_VSTD::move(__v));}
    532 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    533     _LIBCPP_INLINE_VISIBILITY
    534     iterator insert(const_iterator __p, const value_type& __v)
    535         {return __tree_.__insert_unique(__p, __v);}
    536 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    537     _LIBCPP_INLINE_VISIBILITY
    538     iterator insert(const_iterator __p, value_type&& __v)
    539         {return __tree_.__insert_unique(__p, _VSTD::move(__v));}
    540 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    541     template <class _InputIterator>
    542         _LIBCPP_INLINE_VISIBILITY
    543         void insert(_InputIterator __f, _InputIterator __l)
    544         {
    545             for (const_iterator __e = cend(); __f != __l; ++__f)
    546                 __tree_.__insert_unique(__e, *__f);
    547         }
    548 
    549 #ifndef _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    550     _LIBCPP_INLINE_VISIBILITY
    551     void insert(initializer_list<value_type> __il)
    552         {insert(__il.begin(), __il.end());}
    553 #endif  // _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    554 
    555     _LIBCPP_INLINE_VISIBILITY
    556     iterator  erase(const_iterator __p) {return __tree_.erase(__p);}
    557     _LIBCPP_INLINE_VISIBILITY
    558     size_type erase(const key_type& __k)
    559         {return __tree_.__erase_unique(__k);}
    560     _LIBCPP_INLINE_VISIBILITY
    561     iterator  erase(const_iterator __f, const_iterator __l)
    562         {return __tree_.erase(__f, __l);}
    563     _LIBCPP_INLINE_VISIBILITY
    564     void clear() _NOEXCEPT {__tree_.clear();}
    565 
    566     _LIBCPP_INLINE_VISIBILITY
    567     void swap(set& __s) _NOEXCEPT_(__is_nothrow_swappable<__base>::value)
    568         {__tree_.swap(__s.__tree_);}
    569 
    570     _LIBCPP_INLINE_VISIBILITY
    571     allocator_type get_allocator() const _NOEXCEPT {return __tree_.__alloc();}
    572     _LIBCPP_INLINE_VISIBILITY
    573     key_compare    key_comp()      const {return __tree_.value_comp();}
    574     _LIBCPP_INLINE_VISIBILITY
    575     value_compare  value_comp()    const {return __tree_.value_comp();}
    576 
    577     // set operations:
    578     _LIBCPP_INLINE_VISIBILITY
    579     iterator find(const key_type& __k)             {return __tree_.find(__k);}
    580     _LIBCPP_INLINE_VISIBILITY
    581     const_iterator find(const key_type& __k) const {return __tree_.find(__k);}
    582     _LIBCPP_INLINE_VISIBILITY
    583     size_type      count(const key_type& __k) const
    584         {return __tree_.__count_unique(__k);}
    585     _LIBCPP_INLINE_VISIBILITY
    586     iterator lower_bound(const key_type& __k)
    587         {return __tree_.lower_bound(__k);}
    588     _LIBCPP_INLINE_VISIBILITY
    589     const_iterator lower_bound(const key_type& __k) const
    590         {return __tree_.lower_bound(__k);}
    591     _LIBCPP_INLINE_VISIBILITY
    592     iterator upper_bound(const key_type& __k)
    593         {return __tree_.upper_bound(__k);}
    594     _LIBCPP_INLINE_VISIBILITY
    595     const_iterator upper_bound(const key_type& __k) const
    596         {return __tree_.upper_bound(__k);}
    597     _LIBCPP_INLINE_VISIBILITY
    598     pair<iterator,iterator> equal_range(const key_type& __k)
    599         {return __tree_.__equal_range_unique(__k);}
    600     _LIBCPP_INLINE_VISIBILITY
    601     pair<const_iterator,const_iterator> equal_range(const key_type& __k) const
    602         {return __tree_.__equal_range_unique(__k);}
    603 };
    604 
    605 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    606 
    607 template <class _Key, class _Compare, class _Allocator>
    608 set<_Key, _Compare, _Allocator>::set(set&& __s, const allocator_type& __a)
    609     : __tree_(_VSTD::move(__s.__tree_), __a)
    610 {
    611     if (__a != __s.get_allocator())
    612     {
    613         const_iterator __e = cend();
    614         while (!__s.empty())
    615             insert(__e, _VSTD::move(__s.__tree_.remove(__s.begin())->__value_));
    616     }
    617 }
    618 
    619 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    620 
    621 template <class _Key, class _Compare, class _Allocator>
    622 inline _LIBCPP_INLINE_VISIBILITY
    623 bool
    624 operator==(const set<_Key, _Compare, _Allocator>& __x,
    625            const set<_Key, _Compare, _Allocator>& __y)
    626 {
    627     return __x.size() == __y.size() && _VSTD::equal(__x.begin(), __x.end(), __y.begin());
    628 }
    629 
    630 template <class _Key, class _Compare, class _Allocator>
    631 inline _LIBCPP_INLINE_VISIBILITY
    632 bool
    633 operator< (const set<_Key, _Compare, _Allocator>& __x,
    634            const set<_Key, _Compare, _Allocator>& __y)
    635 {
    636     return _VSTD::lexicographical_compare(__x.begin(), __x.end(), __y.begin(), __y.end());
    637 }
    638 
    639 template <class _Key, class _Compare, class _Allocator>
    640 inline _LIBCPP_INLINE_VISIBILITY
    641 bool
    642 operator!=(const set<_Key, _Compare, _Allocator>& __x,
    643            const set<_Key, _Compare, _Allocator>& __y)
    644 {
    645     return !(__x == __y);
    646 }
    647 
    648 template <class _Key, class _Compare, class _Allocator>
    649 inline _LIBCPP_INLINE_VISIBILITY
    650 bool
    651 operator> (const set<_Key, _Compare, _Allocator>& __x,
    652            const set<_Key, _Compare, _Allocator>& __y)
    653 {
    654     return __y < __x;
    655 }
    656 
    657 template <class _Key, class _Compare, class _Allocator>
    658 inline _LIBCPP_INLINE_VISIBILITY
    659 bool
    660 operator>=(const set<_Key, _Compare, _Allocator>& __x,
    661            const set<_Key, _Compare, _Allocator>& __y)
    662 {
    663     return !(__x < __y);
    664 }
    665 
    666 template <class _Key, class _Compare, class _Allocator>
    667 inline _LIBCPP_INLINE_VISIBILITY
    668 bool
    669 operator<=(const set<_Key, _Compare, _Allocator>& __x,
    670            const set<_Key, _Compare, _Allocator>& __y)
    671 {
    672     return !(__y < __x);
    673 }
    674 
    675 // specialized algorithms:
    676 template <class _Key, class _Compare, class _Allocator>
    677 inline _LIBCPP_INLINE_VISIBILITY
    678 void
    679 swap(set<_Key, _Compare, _Allocator>& __x,
    680      set<_Key, _Compare, _Allocator>& __y)
    681     _NOEXCEPT_(_NOEXCEPT_(__x.swap(__y)))
    682 {
    683     __x.swap(__y);
    684 }
    685 
    686 template <class _Key, class _Compare = less<_Key>,
    687           class _Allocator = allocator<_Key> >
    688 class _LIBCPP_TYPE_VIS multiset
    689 {
    690 public:
    691     // types:
    692     typedef _Key                                      key_type;
    693     typedef key_type                                 value_type;
    694     typedef _Compare                                  key_compare;
    695     typedef key_compare                              value_compare;
    696     typedef _Allocator                                allocator_type;
    697     typedef value_type&                              reference;
    698     typedef const value_type&                        const_reference;
    699 
    700 private:
    701     typedef __tree<value_type, value_compare, allocator_type> __base;
    702     typedef allocator_traits<allocator_type>                  __alloc_traits;
    703     typedef typename __base::__node_holder                    __node_holder;
    704 
    705     __base __tree_;
    706 
    707 public:
    708     typedef typename __base::pointer               pointer;
    709     typedef typename __base::const_pointer         const_pointer;
    710     typedef typename __base::size_type             size_type;
    711     typedef typename __base::difference_type       difference_type;
    712     typedef typename __base::const_iterator        iterator;
    713     typedef typename __base::const_iterator        const_iterator;
    714     typedef _VSTD::reverse_iterator<iterator>       reverse_iterator;
    715     typedef _VSTD::reverse_iterator<const_iterator> const_reverse_iterator;
    716 
    717     // construct/copy/destroy:
    718     _LIBCPP_INLINE_VISIBILITY
    719     explicit multiset(const value_compare& __comp = value_compare())
    720         _NOEXCEPT_(
    721             is_nothrow_default_constructible<allocator_type>::value &&
    722             is_nothrow_default_constructible<key_compare>::value &&
    723             is_nothrow_copy_constructible<key_compare>::value)
    724         : __tree_(__comp) {}
    725     _LIBCPP_INLINE_VISIBILITY
    726     multiset(const value_compare& __comp, const allocator_type& __a)
    727         : __tree_(__comp, __a) {}
    728     template <class _InputIterator>
    729         _LIBCPP_INLINE_VISIBILITY
    730         multiset(_InputIterator __f, _InputIterator __l,
    731                  const value_compare& __comp = value_compare())
    732         : __tree_(__comp)
    733         {
    734             insert(__f, __l);
    735         }
    736 
    737     template <class _InputIterator>
    738         _LIBCPP_INLINE_VISIBILITY
    739         multiset(_InputIterator __f, _InputIterator __l,
    740                  const value_compare& __comp, const allocator_type& __a)
    741         : __tree_(__comp, __a)
    742         {
    743             insert(__f, __l);
    744         }
    745 
    746     _LIBCPP_INLINE_VISIBILITY
    747     multiset(const multiset& __s)
    748         : __tree_(__s.__tree_.value_comp(),
    749           __alloc_traits::select_on_container_copy_construction(__s.__tree_.__alloc()))
    750         {
    751             insert(__s.begin(), __s.end());
    752         }
    753 
    754     _LIBCPP_INLINE_VISIBILITY
    755     multiset& operator=(const multiset& __s)
    756         {
    757             __tree_ = __s.__tree_;
    758             return *this;
    759         }
    760 
    761 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    762     _LIBCPP_INLINE_VISIBILITY
    763     multiset(multiset&& __s)
    764         _NOEXCEPT_(is_nothrow_move_constructible<__base>::value)
    765         : __tree_(_VSTD::move(__s.__tree_)) {}
    766 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    767     _LIBCPP_INLINE_VISIBILITY
    768     explicit multiset(const allocator_type& __a)
    769         : __tree_(__a) {}
    770     _LIBCPP_INLINE_VISIBILITY
    771     multiset(const multiset& __s, const allocator_type& __a)
    772         : __tree_(__s.__tree_.value_comp(), __a)
    773         {
    774             insert(__s.begin(), __s.end());
    775         }
    776 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    777     multiset(multiset&& __s, const allocator_type& __a);
    778 #endif
    779 
    780 #ifndef _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    781     _LIBCPP_INLINE_VISIBILITY
    782     multiset(initializer_list<value_type> __il, const value_compare& __comp = value_compare())
    783         : __tree_(__comp)
    784         {
    785             insert(__il.begin(), __il.end());
    786         }
    787 
    788     _LIBCPP_INLINE_VISIBILITY
    789     multiset(initializer_list<value_type> __il, const value_compare& __comp,
    790         const allocator_type& __a)
    791         : __tree_(__comp, __a)
    792         {
    793             insert(__il.begin(), __il.end());
    794         }
    795 
    796     _LIBCPP_INLINE_VISIBILITY
    797     multiset& operator=(initializer_list<value_type> __il)
    798         {
    799             __tree_.__assign_multi(__il.begin(), __il.end());
    800             return *this;
    801         }
    802 #endif  // _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    803 
    804 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    805     _LIBCPP_INLINE_VISIBILITY
    806     multiset& operator=(multiset&& __s)
    807         _NOEXCEPT_(is_nothrow_move_assignable<__base>::value)
    808         {
    809             __tree_ = _VSTD::move(__s.__tree_);
    810             return *this;
    811         }
    812 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    813 
    814     _LIBCPP_INLINE_VISIBILITY
    815           iterator begin() _NOEXCEPT       {return __tree_.begin();}
    816     _LIBCPP_INLINE_VISIBILITY
    817     const_iterator begin() const _NOEXCEPT {return __tree_.begin();}
    818     _LIBCPP_INLINE_VISIBILITY
    819           iterator end() _NOEXCEPT         {return __tree_.end();}
    820     _LIBCPP_INLINE_VISIBILITY
    821     const_iterator end()   const _NOEXCEPT {return __tree_.end();}
    822 
    823     _LIBCPP_INLINE_VISIBILITY
    824           reverse_iterator rbegin() _NOEXCEPT
    825             {return reverse_iterator(end());}
    826     _LIBCPP_INLINE_VISIBILITY
    827     const_reverse_iterator rbegin() const _NOEXCEPT
    828         {return const_reverse_iterator(end());}
    829     _LIBCPP_INLINE_VISIBILITY
    830           reverse_iterator rend() _NOEXCEPT
    831             {return       reverse_iterator(begin());}
    832     _LIBCPP_INLINE_VISIBILITY
    833     const_reverse_iterator rend() const _NOEXCEPT
    834         {return const_reverse_iterator(begin());}
    835 
    836     _LIBCPP_INLINE_VISIBILITY
    837     const_iterator cbegin()  const _NOEXCEPT {return begin();}
    838     _LIBCPP_INLINE_VISIBILITY
    839     const_iterator cend() const _NOEXCEPT {return end();}
    840     _LIBCPP_INLINE_VISIBILITY
    841     const_reverse_iterator crbegin() const _NOEXCEPT {return rbegin();}
    842     _LIBCPP_INLINE_VISIBILITY
    843     const_reverse_iterator crend() const _NOEXCEPT {return rend();}
    844 
    845     _LIBCPP_INLINE_VISIBILITY
    846     bool empty() const _NOEXCEPT {return __tree_.size() == 0;}
    847     _LIBCPP_INLINE_VISIBILITY
    848     size_type size() const _NOEXCEPT {return __tree_.size();}
    849     _LIBCPP_INLINE_VISIBILITY
    850     size_type max_size() const _NOEXCEPT {return __tree_.max_size();}
    851 
    852     // modifiers:
    853 #if !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
    854     template <class... _Args>
    855         _LIBCPP_INLINE_VISIBILITY
    856         iterator emplace(_Args&&... __args)
    857             {return __tree_.__emplace_multi(_VSTD::forward<_Args>(__args)...);}
    858     template <class... _Args>
    859         _LIBCPP_INLINE_VISIBILITY
    860         iterator emplace_hint(const_iterator __p, _Args&&... __args)
    861             {return __tree_.__emplace_hint_multi(__p, _VSTD::forward<_Args>(__args)...);}
    862 #endif  // !defined(_LIBCPP_HAS_NO_RVALUE_REFERENCES) && !defined(_LIBCPP_HAS_NO_VARIADICS)
    863     _LIBCPP_INLINE_VISIBILITY
    864     iterator insert(const value_type& __v)
    865         {return __tree_.__insert_multi(__v);}
    866 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    867     _LIBCPP_INLINE_VISIBILITY
    868     iterator insert(value_type&& __v)
    869         {return __tree_.__insert_multi(_VSTD::move(__v));}
    870 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    871     _LIBCPP_INLINE_VISIBILITY
    872     iterator insert(const_iterator __p, const value_type& __v)
    873         {return __tree_.__insert_multi(__p, __v);}
    874 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    875     _LIBCPP_INLINE_VISIBILITY
    876     iterator insert(const_iterator __p, value_type&& __v)
    877         {return __tree_.__insert_multi(_VSTD::move(__v));}
    878 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    879     template <class _InputIterator>
    880         _LIBCPP_INLINE_VISIBILITY
    881         void insert(_InputIterator __f, _InputIterator __l)
    882         {
    883             for (const_iterator __e = cend(); __f != __l; ++__f)
    884                 __tree_.__insert_multi(__e, *__f);
    885         }
    886 
    887 #ifndef _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    888     _LIBCPP_INLINE_VISIBILITY
    889     void insert(initializer_list<value_type> __il)
    890         {insert(__il.begin(), __il.end());}
    891 #endif  // _LIBCPP_HAS_NO_GENERALIZED_INITIALIZERS
    892 
    893     _LIBCPP_INLINE_VISIBILITY
    894     iterator  erase(const_iterator __p) {return __tree_.erase(__p);}
    895     _LIBCPP_INLINE_VISIBILITY
    896     size_type erase(const key_type& __k) {return __tree_.__erase_multi(__k);}
    897     _LIBCPP_INLINE_VISIBILITY
    898     iterator  erase(const_iterator __f, const_iterator __l)
    899         {return __tree_.erase(__f, __l);}
    900     _LIBCPP_INLINE_VISIBILITY
    901     void clear() _NOEXCEPT {__tree_.clear();}
    902 
    903     _LIBCPP_INLINE_VISIBILITY
    904     void swap(multiset& __s)
    905         _NOEXCEPT_(__is_nothrow_swappable<__base>::value)
    906         {__tree_.swap(__s.__tree_);}
    907 
    908     _LIBCPP_INLINE_VISIBILITY
    909     allocator_type get_allocator() const _NOEXCEPT {return __tree_.__alloc();}
    910     _LIBCPP_INLINE_VISIBILITY
    911     key_compare    key_comp()      const {return __tree_.value_comp();}
    912     _LIBCPP_INLINE_VISIBILITY
    913     value_compare  value_comp()    const {return __tree_.value_comp();}
    914 
    915     // set operations:
    916     _LIBCPP_INLINE_VISIBILITY
    917     iterator find(const key_type& __k)             {return __tree_.find(__k);}
    918     _LIBCPP_INLINE_VISIBILITY
    919     const_iterator find(const key_type& __k) const {return __tree_.find(__k);}
    920     _LIBCPP_INLINE_VISIBILITY
    921     size_type      count(const key_type& __k) const
    922         {return __tree_.__count_multi(__k);}
    923     _LIBCPP_INLINE_VISIBILITY
    924     iterator lower_bound(const key_type& __k)
    925         {return __tree_.lower_bound(__k);}
    926     _LIBCPP_INLINE_VISIBILITY
    927     const_iterator lower_bound(const key_type& __k) const
    928             {return __tree_.lower_bound(__k);}
    929     _LIBCPP_INLINE_VISIBILITY
    930     iterator upper_bound(const key_type& __k)
    931             {return __tree_.upper_bound(__k);}
    932     _LIBCPP_INLINE_VISIBILITY
    933     const_iterator upper_bound(const key_type& __k) const
    934             {return __tree_.upper_bound(__k);}
    935     _LIBCPP_INLINE_VISIBILITY
    936     pair<iterator,iterator>             equal_range(const key_type& __k)
    937             {return __tree_.__equal_range_multi(__k);}
    938     _LIBCPP_INLINE_VISIBILITY
    939     pair<const_iterator,const_iterator> equal_range(const key_type& __k) const
    940             {return __tree_.__equal_range_multi(__k);}
    941 };
    942 
    943 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
    944 
    945 template <class _Key, class _Compare, class _Allocator>
    946 multiset<_Key, _Compare, _Allocator>::multiset(multiset&& __s, const allocator_type& __a)
    947     : __tree_(_VSTD::move(__s.__tree_), __a)
    948 {
    949     if (__a != __s.get_allocator())
    950     {
    951         const_iterator __e = cend();
    952         while (!__s.empty())
    953             insert(__e, _VSTD::move(__s.__tree_.remove(__s.begin())->__value_));
    954     }
    955 }
    956 
    957 #endif  // _LIBCPP_HAS_NO_RVALUE_REFERENCES
    958 
    959 template <class _Key, class _Compare, class _Allocator>
    960 inline _LIBCPP_INLINE_VISIBILITY
    961 bool
    962 operator==(const multiset<_Key, _Compare, _Allocator>& __x,
    963            const multiset<_Key, _Compare, _Allocator>& __y)
    964 {
    965     return __x.size() == __y.size() && _VSTD::equal(__x.begin(), __x.end(), __y.begin());
    966 }
    967 
    968 template <class _Key, class _Compare, class _Allocator>
    969 inline _LIBCPP_INLINE_VISIBILITY
    970 bool
    971 operator< (const multiset<_Key, _Compare, _Allocator>& __x,
    972            const multiset<_Key, _Compare, _Allocator>& __y)
    973 {
    974     return _VSTD::lexicographical_compare(__x.begin(), __x.end(), __y.begin(), __y.end());
    975 }
    976 
    977 template <class _Key, class _Compare, class _Allocator>
    978 inline _LIBCPP_INLINE_VISIBILITY
    979 bool
    980 operator!=(const multiset<_Key, _Compare, _Allocator>& __x,
    981            const multiset<_Key, _Compare, _Allocator>& __y)
    982 {
    983     return !(__x == __y);
    984 }
    985 
    986 template <class _Key, class _Compare, class _Allocator>
    987 inline _LIBCPP_INLINE_VISIBILITY
    988 bool
    989 operator> (const multiset<_Key, _Compare, _Allocator>& __x,
    990            const multiset<_Key, _Compare, _Allocator>& __y)
    991 {
    992     return __y < __x;
    993 }
    994 
    995 template <class _Key, class _Compare, class _Allocator>
    996 inline _LIBCPP_INLINE_VISIBILITY
    997 bool
    998 operator>=(const multiset<_Key, _Compare, _Allocator>& __x,
    999            const multiset<_Key, _Compare, _Allocator>& __y)
   1000 {
   1001     return !(__x < __y);
   1002 }
   1003 
   1004 template <class _Key, class _Compare, class _Allocator>
   1005 inline _LIBCPP_INLINE_VISIBILITY
   1006 bool
   1007 operator<=(const multiset<_Key, _Compare, _Allocator>& __x,
   1008            const multiset<_Key, _Compare, _Allocator>& __y)
   1009 {
   1010     return !(__y < __x);
   1011 }
   1012 
   1013 template <class _Key, class _Compare, class _Allocator>
   1014 inline _LIBCPP_INLINE_VISIBILITY
   1015 void
   1016 swap(multiset<_Key, _Compare, _Allocator>& __x,
   1017      multiset<_Key, _Compare, _Allocator>& __y)
   1018     _NOEXCEPT_(_NOEXCEPT_(__x.swap(__y)))
   1019 {
   1020     __x.swap(__y);
   1021 }
   1022 
   1023 _LIBCPP_END_NAMESPACE_STD
   1024 
   1025 #endif  // _LIBCPP_SET
   1026