xref: /openbsd-src/gnu/llvm/libcxx/include/__algorithm/nth_element.h (revision 4bdff4bed0e3d54e55670334c7d0077db4170f86)
176d0caaeSpatrick //===----------------------------------------------------------------------===//
276d0caaeSpatrick //
376d0caaeSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
476d0caaeSpatrick // See https://llvm.org/LICENSE.txt for license information.
576d0caaeSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
676d0caaeSpatrick //
776d0caaeSpatrick //===----------------------------------------------------------------------===//
876d0caaeSpatrick 
976d0caaeSpatrick #ifndef _LIBCPP___ALGORITHM_NTH_ELEMENT_H
1076d0caaeSpatrick #define _LIBCPP___ALGORITHM_NTH_ELEMENT_H
1176d0caaeSpatrick 
1276d0caaeSpatrick #include <__algorithm/comp.h>
1376d0caaeSpatrick #include <__algorithm/comp_ref_type.h>
14*4bdff4beSrobert #include <__algorithm/iterator_operations.h>
1576d0caaeSpatrick #include <__algorithm/sort.h>
16*4bdff4beSrobert #include <__config>
17*4bdff4beSrobert #include <__debug>
18*4bdff4beSrobert #include <__debug_utils/randomize_range.h>
1976d0caaeSpatrick #include <__iterator/iterator_traits.h>
20*4bdff4beSrobert #include <__utility/move.h>
2176d0caaeSpatrick 
2276d0caaeSpatrick #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
2376d0caaeSpatrick #  pragma GCC system_header
2476d0caaeSpatrick #endif
2576d0caaeSpatrick 
2676d0caaeSpatrick _LIBCPP_BEGIN_NAMESPACE_STD
2776d0caaeSpatrick 
2876d0caaeSpatrick template<class _Compare, class _RandomAccessIterator>
29*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 bool
__nth_element_find_guard(_RandomAccessIterator & __i,_RandomAccessIterator & __j,_RandomAccessIterator __m,_Compare __comp)3076d0caaeSpatrick __nth_element_find_guard(_RandomAccessIterator& __i, _RandomAccessIterator& __j,
3176d0caaeSpatrick                          _RandomAccessIterator __m, _Compare __comp)
3276d0caaeSpatrick {
3376d0caaeSpatrick     // manually guard downward moving __j against __i
3476d0caaeSpatrick     while (true) {
3576d0caaeSpatrick         if (__i == --__j) {
3676d0caaeSpatrick             return false;
3776d0caaeSpatrick         }
3876d0caaeSpatrick         if (__comp(*__j, *__m)) {
3976d0caaeSpatrick             return true;  // found guard for downward moving __j, now use unguarded partition
4076d0caaeSpatrick         }
4176d0caaeSpatrick     }
4276d0caaeSpatrick }
4376d0caaeSpatrick 
44*4bdff4beSrobert template <class _AlgPolicy, class _Compare, class _RandomAccessIterator>
45*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 void
__nth_element(_RandomAccessIterator __first,_RandomAccessIterator __nth,_RandomAccessIterator __last,_Compare __comp)4676d0caaeSpatrick __nth_element(_RandomAccessIterator __first, _RandomAccessIterator __nth, _RandomAccessIterator __last, _Compare __comp)
4776d0caaeSpatrick {
48*4bdff4beSrobert     using _Ops = _IterOps<_AlgPolicy>;
49*4bdff4beSrobert 
5076d0caaeSpatrick     // _Compare is known to be a reference type
5176d0caaeSpatrick     typedef typename iterator_traits<_RandomAccessIterator>::difference_type difference_type;
5276d0caaeSpatrick     const difference_type __limit = 7;
5376d0caaeSpatrick     while (true)
5476d0caaeSpatrick     {
5576d0caaeSpatrick         if (__nth == __last)
5676d0caaeSpatrick             return;
5776d0caaeSpatrick         difference_type __len = __last - __first;
5876d0caaeSpatrick         switch (__len)
5976d0caaeSpatrick         {
6076d0caaeSpatrick         case 0:
6176d0caaeSpatrick         case 1:
6276d0caaeSpatrick             return;
6376d0caaeSpatrick         case 2:
6476d0caaeSpatrick             if (__comp(*--__last, *__first))
65*4bdff4beSrobert                 _Ops::iter_swap(__first, __last);
6676d0caaeSpatrick             return;
6776d0caaeSpatrick         case 3:
6876d0caaeSpatrick             {
6976d0caaeSpatrick             _RandomAccessIterator __m = __first;
70*4bdff4beSrobert             std::__sort3<_AlgPolicy, _Compare>(__first, ++__m, --__last, __comp);
7176d0caaeSpatrick             return;
7276d0caaeSpatrick             }
7376d0caaeSpatrick         }
7476d0caaeSpatrick         if (__len <= __limit)
7576d0caaeSpatrick         {
76*4bdff4beSrobert             std::__selection_sort<_AlgPolicy, _Compare>(__first, __last, __comp);
7776d0caaeSpatrick             return;
7876d0caaeSpatrick         }
7976d0caaeSpatrick         // __len > __limit >= 3
8076d0caaeSpatrick         _RandomAccessIterator __m = __first + __len/2;
8176d0caaeSpatrick         _RandomAccessIterator __lm1 = __last;
82*4bdff4beSrobert         unsigned __n_swaps = std::__sort3<_AlgPolicy, _Compare>(__first, __m, --__lm1, __comp);
8376d0caaeSpatrick         // *__m is median
8476d0caaeSpatrick         // partition [__first, __m) < *__m and *__m <= [__m, __last)
8576d0caaeSpatrick         // (this inhibits tossing elements equivalent to __m around unnecessarily)
8676d0caaeSpatrick         _RandomAccessIterator __i = __first;
8776d0caaeSpatrick         _RandomAccessIterator __j = __lm1;
8876d0caaeSpatrick         // j points beyond range to be tested, *__lm1 is known to be <= *__m
8976d0caaeSpatrick         // The search going up is known to be guarded but the search coming down isn't.
9076d0caaeSpatrick         // Prime the downward search with a guard.
9176d0caaeSpatrick         if (!__comp(*__i, *__m))  // if *__first == *__m
9276d0caaeSpatrick         {
9376d0caaeSpatrick             // *__first == *__m, *__first doesn't go in first part
9476d0caaeSpatrick             if (_VSTD::__nth_element_find_guard<_Compare>(__i, __j, __m, __comp)) {
95*4bdff4beSrobert                 _Ops::iter_swap(__i, __j);
9676d0caaeSpatrick                 ++__n_swaps;
9776d0caaeSpatrick             } else {
9876d0caaeSpatrick                 // *__first == *__m, *__m <= all other elements
9976d0caaeSpatrick                 // Partition instead into [__first, __i) == *__first and *__first < [__i, __last)
10076d0caaeSpatrick                 ++__i;  // __first + 1
10176d0caaeSpatrick                 __j = __last;
10276d0caaeSpatrick                 if (!__comp(*__first, *--__j)) {  // we need a guard if *__first == *(__last-1)
10376d0caaeSpatrick                     while (true) {
10476d0caaeSpatrick                         if (__i == __j) {
10576d0caaeSpatrick                             return;  // [__first, __last) all equivalent elements
10676d0caaeSpatrick                         } else if (__comp(*__first, *__i)) {
107*4bdff4beSrobert                             _Ops::iter_swap(__i, __j);
10876d0caaeSpatrick                             ++__n_swaps;
10976d0caaeSpatrick                             ++__i;
11076d0caaeSpatrick                             break;
11176d0caaeSpatrick                         }
11276d0caaeSpatrick                         ++__i;
11376d0caaeSpatrick                     }
11476d0caaeSpatrick                 }
11576d0caaeSpatrick                 // [__first, __i) == *__first and *__first < [__j, __last) and __j == __last - 1
11676d0caaeSpatrick                 if (__i == __j) {
11776d0caaeSpatrick                     return;
11876d0caaeSpatrick                 }
11976d0caaeSpatrick                 while (true) {
12076d0caaeSpatrick                     while (!__comp(*__first, *__i))
12176d0caaeSpatrick                         ++__i;
12276d0caaeSpatrick                     while (__comp(*__first, *--__j))
12376d0caaeSpatrick                         ;
12476d0caaeSpatrick                     if (__i >= __j)
12576d0caaeSpatrick                         break;
126*4bdff4beSrobert                     _Ops::iter_swap(__i, __j);
12776d0caaeSpatrick                     ++__n_swaps;
12876d0caaeSpatrick                     ++__i;
12976d0caaeSpatrick                 }
13076d0caaeSpatrick                 // [__first, __i) == *__first and *__first < [__i, __last)
13176d0caaeSpatrick                 // The first part is sorted,
13276d0caaeSpatrick                 if (__nth < __i) {
13376d0caaeSpatrick                     return;
13476d0caaeSpatrick                 }
13576d0caaeSpatrick                 // __nth_element the second part
13676d0caaeSpatrick                 // _VSTD::__nth_element<_Compare>(__i, __nth, __last, __comp);
13776d0caaeSpatrick                 __first = __i;
13876d0caaeSpatrick                 continue;
13976d0caaeSpatrick             }
14076d0caaeSpatrick         }
14176d0caaeSpatrick         ++__i;
14276d0caaeSpatrick         // j points beyond range to be tested, *__lm1 is known to be <= *__m
14376d0caaeSpatrick         // if not yet partitioned...
14476d0caaeSpatrick         if (__i < __j)
14576d0caaeSpatrick         {
14676d0caaeSpatrick             // known that *(__i - 1) < *__m
14776d0caaeSpatrick             while (true)
14876d0caaeSpatrick             {
14976d0caaeSpatrick                 // __m still guards upward moving __i
15076d0caaeSpatrick                 while (__comp(*__i, *__m))
15176d0caaeSpatrick                     ++__i;
15276d0caaeSpatrick                 // It is now known that a guard exists for downward moving __j
15376d0caaeSpatrick                 while (!__comp(*--__j, *__m))
15476d0caaeSpatrick                     ;
15576d0caaeSpatrick                 if (__i >= __j)
15676d0caaeSpatrick                     break;
157*4bdff4beSrobert                 _Ops::iter_swap(__i, __j);
15876d0caaeSpatrick                 ++__n_swaps;
15976d0caaeSpatrick                 // It is known that __m != __j
16076d0caaeSpatrick                 // If __m just moved, follow it
16176d0caaeSpatrick                 if (__m == __i)
16276d0caaeSpatrick                     __m = __j;
16376d0caaeSpatrick                 ++__i;
16476d0caaeSpatrick             }
16576d0caaeSpatrick         }
16676d0caaeSpatrick         // [__first, __i) < *__m and *__m <= [__i, __last)
16776d0caaeSpatrick         if (__i != __m && __comp(*__m, *__i))
16876d0caaeSpatrick         {
169*4bdff4beSrobert             _Ops::iter_swap(__i, __m);
17076d0caaeSpatrick             ++__n_swaps;
17176d0caaeSpatrick         }
17276d0caaeSpatrick         // [__first, __i) < *__i and *__i <= [__i+1, __last)
17376d0caaeSpatrick         if (__nth == __i)
17476d0caaeSpatrick             return;
17576d0caaeSpatrick         if (__n_swaps == 0)
17676d0caaeSpatrick         {
17776d0caaeSpatrick             // We were given a perfectly partitioned sequence.  Coincidence?
17876d0caaeSpatrick             if (__nth < __i)
17976d0caaeSpatrick             {
18076d0caaeSpatrick                 // Check for [__first, __i) already sorted
18176d0caaeSpatrick                 __j = __m = __first;
18276d0caaeSpatrick                 while (true) {
18376d0caaeSpatrick                     if (++__j == __i) {
18476d0caaeSpatrick                         // [__first, __i) sorted
18576d0caaeSpatrick                         return;
18676d0caaeSpatrick                     }
18776d0caaeSpatrick                     if (__comp(*__j, *__m)) {
18876d0caaeSpatrick                         // not yet sorted, so sort
18976d0caaeSpatrick                         break;
19076d0caaeSpatrick                     }
19176d0caaeSpatrick                     __m = __j;
19276d0caaeSpatrick                 }
19376d0caaeSpatrick             }
19476d0caaeSpatrick             else
19576d0caaeSpatrick             {
19676d0caaeSpatrick                 // Check for [__i, __last) already sorted
19776d0caaeSpatrick                 __j = __m = __i;
19876d0caaeSpatrick                 while (true) {
19976d0caaeSpatrick                     if (++__j == __last) {
20076d0caaeSpatrick                         // [__i, __last) sorted
20176d0caaeSpatrick                         return;
20276d0caaeSpatrick                     }
20376d0caaeSpatrick                     if (__comp(*__j, *__m)) {
20476d0caaeSpatrick                         // not yet sorted, so sort
20576d0caaeSpatrick                         break;
20676d0caaeSpatrick                     }
20776d0caaeSpatrick                     __m = __j;
20876d0caaeSpatrick                 }
20976d0caaeSpatrick             }
21076d0caaeSpatrick         }
21176d0caaeSpatrick         // __nth_element on range containing __nth
21276d0caaeSpatrick         if (__nth < __i)
21376d0caaeSpatrick         {
21476d0caaeSpatrick             // _VSTD::__nth_element<_Compare>(__first, __nth, __i, __comp);
21576d0caaeSpatrick             __last = __i;
21676d0caaeSpatrick         }
21776d0caaeSpatrick         else
21876d0caaeSpatrick         {
21976d0caaeSpatrick             // _VSTD::__nth_element<_Compare>(__i+1, __nth, __last, __comp);
22076d0caaeSpatrick             __first = ++__i;
22176d0caaeSpatrick         }
22276d0caaeSpatrick     }
22376d0caaeSpatrick }
22476d0caaeSpatrick 
225*4bdff4beSrobert template <class _AlgPolicy, class _RandomAccessIterator, class _Compare>
226*4bdff4beSrobert inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX20
__nth_element_impl(_RandomAccessIterator __first,_RandomAccessIterator __nth,_RandomAccessIterator __last,_Compare & __comp)227*4bdff4beSrobert void __nth_element_impl(_RandomAccessIterator __first, _RandomAccessIterator __nth, _RandomAccessIterator __last,
228*4bdff4beSrobert                         _Compare& __comp) {
229*4bdff4beSrobert   if (__nth == __last)
230*4bdff4beSrobert     return;
231*4bdff4beSrobert 
232*4bdff4beSrobert   std::__debug_randomize_range<_AlgPolicy>(__first, __last);
233*4bdff4beSrobert 
234*4bdff4beSrobert   std::__nth_element<_AlgPolicy, __comp_ref_type<_Compare> >(__first, __nth, __last, __comp);
235*4bdff4beSrobert 
236*4bdff4beSrobert   std::__debug_randomize_range<_AlgPolicy>(__first, __nth);
237*4bdff4beSrobert   if (__nth != __last) {
238*4bdff4beSrobert     std::__debug_randomize_range<_AlgPolicy>(++__nth, __last);
239*4bdff4beSrobert   }
240*4bdff4beSrobert }
241*4bdff4beSrobert 
24276d0caaeSpatrick template <class _RandomAccessIterator, class _Compare>
243*4bdff4beSrobert inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX20
nth_element(_RandomAccessIterator __first,_RandomAccessIterator __nth,_RandomAccessIterator __last,_Compare __comp)244*4bdff4beSrobert void nth_element(_RandomAccessIterator __first, _RandomAccessIterator __nth, _RandomAccessIterator __last,
245*4bdff4beSrobert                  _Compare __comp) {
246*4bdff4beSrobert   std::__nth_element_impl<_ClassicAlgPolicy>(std::move(__first), std::move(__nth), std::move(__last), __comp);
24776d0caaeSpatrick }
24876d0caaeSpatrick 
24976d0caaeSpatrick template <class _RandomAccessIterator>
250*4bdff4beSrobert inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX20
nth_element(_RandomAccessIterator __first,_RandomAccessIterator __nth,_RandomAccessIterator __last)251*4bdff4beSrobert void nth_element(_RandomAccessIterator __first, _RandomAccessIterator __nth, _RandomAccessIterator __last) {
252*4bdff4beSrobert   std::nth_element(std::move(__first), std::move(__nth), std::move(__last), __less<typename
253*4bdff4beSrobert       iterator_traits<_RandomAccessIterator>::value_type>());
25476d0caaeSpatrick }
25576d0caaeSpatrick 
25676d0caaeSpatrick _LIBCPP_END_NAMESPACE_STD
25776d0caaeSpatrick 
25876d0caaeSpatrick #endif // _LIBCPP___ALGORITHM_NTH_ELEMENT_H
259