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