On Thu, 12 Jun 2025, Patrick Palka wrote:

> On Thu, 12 Jun 2025, Patrick Palka wrote:
> 
> > On Thu, 12 Jun 2025, Tomasz Kaminski wrote:
> > 
> > > 
> > > 
> > > 
> > > On Thu, Jun 12, 2025 at 5:06 PM Patrick Palka <ppa...@redhat.com> wrote:
> > >       On Wed, 11 Jun 2025, Tomasz Kaminski wrote:
> > > 
> > >       >
> > >       >
> > >       > On Tue, Jun 10, 2025 at 3:08 AM Patrick Palka <ppa...@redhat.com> 
> > > wrote:
> > >       >       As with the previous patch, this patch reimplements 
> > > ranges::sort
> > >       >       directly instead of incorrectly forwarding to std::sort.  
> > > In addition to
> > >       >       the compatibility changes listed in the previous patch we 
> > > also:
> > >       >
> > >       >         - use ranges::iter_swap instead of std::iter_swap
> > >       >         - use ranges::move_backward instead of std::move_backward
> > >       >         - use __bit_floor and __to_unsigned_like instead of __lg
> > >       >
> > >       >       We also need to extend std::__lg to support integer-class 
> > > types.
> > >       >
> > >       >               PR libstdc++/100795
> > >       >               PR libstdc++/118209
> > >       >
> > >       >       libstdc++-v3/ChangeLog:
> > >       >
> > >       >               * include/bits/ranges_algo.h 
> > > (__detail::__move_median_to_first):
> > >       >               New, based on the stl_algo.h implementation.
> > >       >               (__detail::__unguarded_liner_insert): Likewise.
> > >       >               (__detail::__insertion_sort): Likewise.
> > >       >               (__detail::__sort_threshold): Likewise.
> > >       >               (__detail::__unguarded_insertion_sort): Likewise.
> > >       >               (__detail::__final_insertion_sort): Likewise.
> > >       >               (__detail::__unguarded_partition): Likewise.
> > >       >               (__detail::__unguarded_partition_pivot): Likewise.
> > >       >               (__detail::__heap_select): Likewise.
> > >       >               (__detail::__partial_sort): Likewise.
> > >       >               (__detail::__introsort_loop): Likewise.
> > >       >               (__sort_fn::operator()): Reimplement in terms of 
> > > the above.
> > >       >               * 
> > > libstdc++-v3/testsuite/25_algorithms/sort/118209.cc: New test.
> > >       >               * 
> > > libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc
> > >       >               (test03): New test.
> > >       >       ---
> > >       >        libstdc++-v3/include/bits/ranges_algo.h       | 210 
> > > +++++++++++++++++-
> > >       >        .../testsuite/25_algorithms/sort/118209.cc    |  23 ++
> > >       >        .../25_algorithms/sort/constrained.cc         |  31 +++
> > >       >        3 files changed, 260 insertions(+), 4 deletions(-)
> > >       >        create mode 100644 
> > > libstdc++-v3/testsuite/25_algorithms/sort/118209.cc
> > >       >
> > >       >       diff --git a/libstdc++-v3/include/bits/ranges_algo.h 
> > > b/libstdc++-v3/include/bits/ranges_algo.h
> > >       >       index f84e0c42d76a..67bdf4287241 100644
> > >       >       --- a/libstdc++-v3/include/bits/ranges_algo.h
> > >       >       +++ b/libstdc++-v3/include/bits/ranges_algo.h
> > >       >       @@ -32,6 +32,7 @@
> > >       >
> > >       >        #if __cplusplus > 201703L
> > >       >
> > >       >       +#include <bit> // __bit_floor
> > >       >        #if __cplusplus > 202002L
> > >       >        #include <optional>
> > >       >        #endif
> > >       >       @@ -2166,6 +2167,184 @@ namespace ranges
> > >       >
> > >       >          inline constexpr __is_heap_fn is_heap{};
> > >       >
> > >       >       +  namespace __detail
> > >       >       +  {
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __move_median_to_first(_Iter __result, _Iter __a, 
> > > _Iter __b, _Iter __c,
> > >       >       +                            _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       if (std::__invoke(__comp,
> > >       >       +                         std::__invoke(__proj, *__a),
> > >       >       +                         std::__invoke(__proj, *__b)))
> > >       >
> > >       > See all this __invoke calls, that are the same in each algo, 
> > > maybe we could accept only _Comp here,
> > >       > and then pass following lambda as comp [&](auto& __a, auto& __b).
> > > 
> > >       Seems this sentence was cut off; I guess you mean doing the 
> > > equivalent
> > >       of __detail::__make_comp_proj?  That sounds good to me.  Only
> > >       in __move_median_to_first or throughout all the helpers?
> > > 
> > > In all helpers, and also ones from the previous patch. 
> > 
> > Hmm, some of the helpers such as __heap_select and __partial_sort
> > directly call ranges algos (make_heap, sort_heap) that of course
> > take a projection.  It feels a little iffy to me to pass only a composite
> > comparator+projection function to these algos from these helpers, and
> > omit the projection function.  It should be safe for these particular
> > ranges algos that we're calling (make_heap, sort_heap), but in general
> > I don't think we can assume
> > 
> >   ranges::foo(..., __detail::__make_comp_proj(comp, proj));
> > 
> > is equivalent to
> > 
> >  ranges::foo(..., __comp, __proj);
> > 
> > So I'm not 100% sold on this refactoring anymore.  What do you think?
> 
> i.e. the refactoring makes the implementation of ranges::sort dependent
> on implementation details of ranges::make_heap/sort_heap in a subtle
> way.

I think I'm wrong,

  ranges::foo(..., __detail::__make_comp_proj(comp, proj));

and

  ranges::foo(..., __comp, __proj);

should always be equivalent as long as the constraints of the latter
form are met.  I posted update patches that includes the refactoring to
use __make_comp_proj.

> 
> > 
> > > 
> > >       > }
> > >       >       +         {
> > >       >       +           if (std::__invoke(__comp,
> > >       >       +                             std::__invoke(__proj, *__b),
> > >       >       +                             std::__invoke(__proj, *__c)))
> > >       >       +             ranges::iter_swap(__result, __b);
> > >       >       +           else if (std::__invoke(__comp,
> > >       >       +                                  std::__invoke(__proj, 
> > > *__a),
> > >       >       +                                  std::__invoke(__proj, 
> > > *__c)))
> > >       >       +             ranges::iter_swap(__result, __c);
> > >       >       +           else
> > >       >       +             ranges::iter_swap(__result, __a);
> > >       >       +         }
> > >       >       +       else if (std::__invoke(__comp,
> > >       >       +                              std::__invoke(__proj, *__a),
> > >       >       +                              std::__invoke(__proj, *__c)))
> > >       >       +         ranges::iter_swap(__result, __a);
> > >       >       +       else if (std::__invoke(__comp,
> > >       >       +                              std::__invoke(__proj, *__b),
> > >       >       +                              std::__invoke(__proj, *__c)))
> > >       >       +         ranges::iter_swap(__result, __c);
> > >       >       +       else
> > >       >       +         ranges::iter_swap(__result, __b);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __unguarded_linear_insert(_Iter __last, _Comp 
> > > __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       iter_value_t<_Iter> __val = 
> > > ranges::iter_move(__last);
> > >       >       +       _Iter __next = __last;
> > >       >       +       --__next;
> > >       >       +       while (std::__invoke(__comp,
> > >       >       +                            std::__invoke(__proj, __val),
> > >       >       +                            std::__invoke(__proj, 
> > > *__next)))
> > >       >       +         {
> > >       >       +           *__last = ranges::iter_move(__next);
> > >       >       +           __last = __next;
> > >       >       +           --__next;
> > >       >       +         }
> > >       >       +       *__last = std::move(__val);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __insertion_sort(_Iter __first, _Iter __last, _Comp 
> > > __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       if (__first == __last) return;
> > >       >       +
> > >       >       +       for (_Iter __i = __first + 1; __i != __last; ++__i)
> > >       >       +         {
> > >       >       +           if (std::__invoke(__comp,
> > >       >       +                             std::__invoke(__proj, *__i),
> > >       >       +                             std::__invoke(__proj, 
> > > *__first)))
> > >       >       +             {
> > >       >       +               iter_value_t<_Iter> __val = 
> > > ranges::iter_move(__i);
> > >       >       +               ranges::move_backward(__first, __i, __i + 
> > > 1);
> > >       >       +               *__first = std::move(__val);
> > >       >       +             }
> > >       >       +           else
> > >       >       +             __detail::__unguarded_linear_insert(__i, 
> > > __comp, __proj);
> > >       >       +         }
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __unguarded_insertion_sort(_Iter __first, _Iter 
> > > __last,
> > >       >       +                                _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       for (_Iter __i = __first; __i != __last; ++__i)
> > >       >       +         __detail::__unguarded_linear_insert(__i, __comp, 
> > > __proj);
> > >       >       +      }
> > >       >       +
> > >       >       +    inline constexpr int __sort_threshold = 16;
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __final_insertion_sort(_Iter __first, _Iter __last,
> > >       >       +                            _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       if (__last - __first > __sort_threshold)
> > >       >       +         {
> > >       >       +           __detail::__insertion_sort(__first, __first + 
> > > __sort_threshold,
> > >       >       +                                      __comp, __proj);
> > >       >       +           __detail::__unguarded_insertion_sort(__first + 
> > > __sort_threshold, __last,
> > >       >       +                                                __comp, 
> > > __proj);
> > >       >       +         }
> > >       >       +       else
> > >       >       +         __detail::__insertion_sort(__first, __last, 
> > > __comp, __proj);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr _Iter
> > >       >       +      __unguarded_partition(_Iter __first, _Iter __last, 
> > > _Iter __pivot,
> > >       >       +                           _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       while (true)
> > >       >       +         {
> > >       >       +           while (std::__invoke(__comp,
> > >       >       +                                std::__invoke(__proj, 
> > > *__first),
> > >       >       +                                std::__invoke(__proj, 
> > > *__pivot)))
> > >       >       +             ++__first;
> > >       >       +           --__last;
> > >       >       +           while (std::__invoke(__comp,
> > >       >       +                                std::__invoke(__proj, 
> > > *__pivot),
> > >       >       +                                std::__invoke(__proj, 
> > > *__last)))
> > >       >       +             --__last;
> > >       >       +           if (!(__first < __last))
> > >       >       +             return __first;
> > >       >       +           ranges::iter_swap(__first, __last);
> > >       >       +           ++__first;
> > >       >       +         }
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr _Iter
> > >       >       +      __unguarded_partition_pivot(_Iter __first, _Iter 
> > > __last,
> > >       >       +                                 _Comp __comp, _Proj 
> > > __proj)
> > >       >       +      {
> > >       >       +       _Iter __mid = __first + (__last - __first) / 2;
> > >       >       +       __detail::__move_median_to_first(__first, __first + 
> > > 1, __mid, __last - 1,
> > >       >       +                                        __comp, __proj);
> > >       >       +       return __detail::__unguarded_partition(__first + 1, 
> > > __last, __first,
> > >       >       +                                              __comp, 
> > > __proj);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __heap_select(_Iter __first, _Iter __middle, _Iter 
> > > __last,
> > >       >       +                   _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       ranges::make_heap(__first, __middle, __comp, 
> > > __proj);
> > >       >       +       for (_Iter __i = __middle; __i < __last; ++__i)
> > >       >       +         if (std::__invoke(__comp,
> > >       >       +                           std::__invoke(__proj, *__i),
> > >       >       +                           std::__invoke(__proj, 
> > > *__first)))
> > >       >       +           __detail::__pop_heap(__first, __middle, __i, 
> > > __comp, __proj);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Compare, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __partial_sort(_Iter __first, _Iter __middle, _Iter 
> > > __last,
> > >       >       +                    _Compare __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       __detail::__heap_select(__first, __middle, __last, 
> > > __comp, __proj);
> > >       >       +       ranges::sort_heap(__first, __middle, __comp, 
> > > __proj);
> > >       >       +      }
> > >       >       +
> > >       >       +    template<typename _Iter, typename _Comp, typename 
> > > _Proj>
> > >       >       +      constexpr void
> > >       >       +      __introsort_loop(_Iter __first, _Iter __last, 
> > > unsigned __depth_limit,
> > >       >       +                      _Comp __comp, _Proj __proj)
> > >       >       +      {
> > >       >       +       while (__last - __first > __sort_threshold)
> > >       >       +         {
> > >       >       +           if (__depth_limit == 0)
> > >       >       +             {
> > >       >       +               __detail::__partial_sort(__first, __last, 
> > > __last, __comp, __proj);
> > >       >       +               return;
> > >       >       +             }
> > >       >       +           --__depth_limit;
> > >       >       +           _Iter __cut = 
> > > __detail::__unguarded_partition_pivot(__first, __last,
> > >       >       +                                                           
> > >     __comp, __proj);
> > >       >       +           __detail::__introsort_loop(__cut, __last, 
> > > __depth_limit, __comp, __proj);
> > >       >       +           __last = __cut;
> > >       >       +         }
> > >       >       +      }
> > >       >       +  } // namespace __detail
> > >       >       +
> > >       >          struct __sort_fn
> > >       >          {
> > >       >            template<random_access_iterator _Iter, 
> > > sentinel_for<_Iter> _Sent,
> > >       >       @@ -2175,10 +2354,33 @@ namespace ranges
> > >       >              operator()(_Iter __first, _Sent __last,
> > >       >                        _Comp __comp = {}, _Proj __proj = {}) const
> > >       >              {
> > >       >       -       auto __lasti = ranges::next(__first, __last);
> > >       >       -       _GLIBCXX_STD_A::sort(std::move(__first), __lasti,
> > >       >       -                            
> > > __detail::__make_comp_proj(__comp, __proj));
> > >       >       -       return __lasti;
> > >       >       +       if constexpr (!same_as<_Iter, _Sent>)
> > >       >       +         return (*this)(__first, ranges::next(__first, 
> > > __last),
> > >       >       +                        std::move(__comp), 
> > > std::move(__proj));
> > >       >       +       else
> > >       >       +         {
> > >       >       +           if (__first != __last)
> > >       >       +             {
> > >       >       +               auto __n = 
> > > __detail::__to_unsigned_like(__last - __first);
> > >       >       +               unsigned __log_n;
> > >       >       +               if constexpr 
> > > (is_class_v<iter_difference_t<_Iter>>)
> > >       >       +                 {
> > >       >       +                   // FIXME: <bit> doesn't handle 
> > > integer-class types.
> > >       >       +                   __log_n = 0;
> > >       >       +                   while (__n != 1)
> > >       >       +                     {
> > >       >       +                       __n >>= 1;
> > >       >       +                       ++__log_n;
> > >       >       +                     }
> > >       >
> > >       > I would add overload for std::__bit_width for __max_size_type in 
> > > max_size_type.h,
> > >       > as it should be simple to implement:
> > >       > if (_M_msb)
> > >       >   return numeric_limits<__rep>::digits + 1;
> > >       > else
> > >       >   return std::__bit_width(_M_val);
> > > 
> > >       That's much better, fixed.  I opted to use an explicit 
> > > specialization of
> > >       the existing overload of a new overload.
> > > 
> > >       -- >8 --
> > > 
> > >       Subject: [PATCH 2/2] libstdc++: Directly implement ranges::sort 
> > > [PR100795]
> > > 
> > >       As with the previous patch, this patch reimplements ranges::sort
> > >       directly instead of incorrectly forwarding to std::sort.  In 
> > > addition to
> > >       the compatibility changes listed in the previous patch we also:
> > > 
> > >         - use ranges::iter_swap instead of std::iter_swap
> > >         - use ranges::move_backward instead of std::move_backward
> > >         - use __bit_width and __to_unsigned_like instead of __lg
> > > 
> > >               PR libstdc++/100795
> > >               PR libstdc++/118209
> > > 
> > >       libstdc++-v3/ChangeLog:
> > > 
> > >               * include/bits/max_size_type.h (__bit_width): New explicit
> > >               specializatino for __max_size_type.
> > >               * include/bits/ranges_algo.h 
> > > (__detail::__move_median_to_first):
> > >               New, based on the stl_algo.h implementation.
> > >               (__detail::__unguarded_liner_insert): Likewise.
> > >               (__detail::__insertion_sort): Likewise.
> > >               (__detail::__sort_threshold): Likewise.
> > >               (__detail::__unguarded_insertion_sort): Likewise.
> > >               (__detail::__final_insertion_sort): Likewise.
> > >               (__detail::__unguarded_partition): Likewise.
> > >               (__detail::__unguarded_partition_pivot): Likewise.
> > >               (__detail::__heap_select): Likewise.
> > >               (__detail::__partial_sort): Likewise.
> > >               (__detail::__introsort_loop): Likewise.
> > >               (__sort_fn::operator()): Reimplement in terms of the above.
> > >               * libstdc++-v3/testsuite/25_algorithms/sort/118209.cc: New 
> > > test.
> > >               * libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc
> > >               (test03): New test.
> > >       ---
> > >        libstdc++-v3/include/bits/max_size_type.h     |   6 +
> > >        libstdc++-v3/include/bits/ranges_algo.h       | 198 
> > > +++++++++++++++++-
> > >        .../testsuite/25_algorithms/sort/118209.cc    |  23 ++
> > >        .../25_algorithms/sort/constrained.cc         |  31 +++
> > >        4 files changed, 254 insertions(+), 4 deletions(-)
> > >        create mode 100644 
> > > libstdc++-v3/testsuite/25_algorithms/sort/118209.cc
> > > 
> > >       diff --git a/libstdc++-v3/include/bits/max_size_type.h 
> > > b/libstdc++-v3/include/bits/max_size_type.h
> > >       index e602b1b4bee5..915e59f22d52 100644
> > >       --- a/libstdc++-v3/include/bits/max_size_type.h
> > >       +++ b/libstdc++-v3/include/bits/max_size_type.h
> > >       @@ -36,6 +36,7 @@
> > > 
> > >        #if __cplusplus > 201703L && __cpp_lib_concepts
> > >        #include <ext/numeric_traits.h>
> > >       +#include <bit> // __bit_width
> > >        #include <numbers>
> > > 
> > >        // This header implements unsigned and signed integer-class types 
> > > (as per
> > >       @@ -818,6 +819,11 @@ namespace ranges
> > >              { return min(); }
> > >            };
> > > 
> > >       +  template<>
> > >       +  inline constexpr int
> > >       +  __bit_width(ranges::__detail::__max_size_type __x) noexcept
> > >       +  { return std::__bit_width(__x._M_val) + __x._M_msb; }
> > > 
> > >       +
> > >        _GLIBCXX_END_NAMESPACE_VERSION
> > >        } // namespace
> > > 
> > >       diff --git a/libstdc++-v3/include/bits/ranges_algo.h 
> > > b/libstdc++-v3/include/bits/ranges_algo.h
> > >       index ca86f63812f4..819316fa79c3 100644
> > >       --- a/libstdc++-v3/include/bits/ranges_algo.h
> > >       +++ b/libstdc++-v3/include/bits/ranges_algo.h
> > >       @@ -32,6 +32,7 @@
> > > 
> > >        #if __cplusplus > 201703L
> > > 
> > >       +#include <bit> // __bit_width
> > >        #if __cplusplus > 202002L
> > >        #include <optional>
> > >        #endif
> > >       @@ -2166,6 +2167,184 @@ namespace ranges
> > > 
> > >          inline constexpr __is_heap_fn is_heap{};
> > > 
> > >       +  namespace __detail
> > >       +  {
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __move_median_to_first(_Iter __result, _Iter __a, _Iter __b, 
> > > _Iter __c,
> > >       +                            _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       if (std::__invoke(__comp,
> > >       +                         std::__invoke(__proj, *__a),
> > >       +                         std::__invoke(__proj, *__b)))
> > >       +         {
> > >       +           if (std::__invoke(__comp,
> > >       +                             std::__invoke(__proj, *__b),
> > >       +                             std::__invoke(__proj, *__c)))
> > >       +             ranges::iter_swap(__result, __b);
> > >       +           else if (std::__invoke(__comp,
> > >       +                                  std::__invoke(__proj, *__a),
> > >       +                                  std::__invoke(__proj, *__c)))
> > >       +             ranges::iter_swap(__result, __c);
> > >       +           else
> > >       +             ranges::iter_swap(__result, __a);
> > >       +         }
> > >       +       else if (std::__invoke(__comp,
> > >       +                              std::__invoke(__proj, *__a),
> > >       +                              std::__invoke(__proj, *__c)))
> > >       +         ranges::iter_swap(__result, __a);
> > >       +       else if (std::__invoke(__comp,
> > >       +                              std::__invoke(__proj, *__b),
> > >       +                              std::__invoke(__proj, *__c)))
> > >       +         ranges::iter_swap(__result, __c);
> > >       +       else
> > >       +         ranges::iter_swap(__result, __b);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __unguarded_linear_insert(_Iter __last, _Comp __comp, _Proj 
> > > __proj)
> > >       +      {
> > >       +       iter_value_t<_Iter> __val = ranges::iter_move(__last);
> > >       +       _Iter __next = __last;
> > >       +       --__next;
> > >       +       while (std::__invoke(__comp,
> > >       +                            std::__invoke(__proj, __val),
> > >       +                            std::__invoke(__proj, *__next)))
> > >       +         {
> > >       +           *__last = ranges::iter_move(__next);
> > >       +           __last = __next;
> > >       +           --__next;
> > >       +         }
> > >       +       *__last = std::move(__val);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __insertion_sort(_Iter __first, _Iter __last, _Comp __comp, 
> > > _Proj __proj)
> > >       +      {
> > >       +       if (__first == __last) return;
> > >       +
> > >       +       for (_Iter __i = __first + 1; __i != __last; ++__i)
> > >       +         {
> > >       +           if (std::__invoke(__comp,
> > >       +                             std::__invoke(__proj, *__i),
> > >       +                             std::__invoke(__proj, *__first)))
> > >       +             {
> > >       +               iter_value_t<_Iter> __val = ranges::iter_move(__i);
> > >       +               ranges::move_backward(__first, __i, __i + 1);
> > >       +               *__first = std::move(__val);
> > >       +             }
> > >       +           else
> > >       +             __detail::__unguarded_linear_insert(__i, __comp, 
> > > __proj);
> > >       +         }
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __unguarded_insertion_sort(_Iter __first, _Iter __last,
> > >       +                                _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       for (_Iter __i = __first; __i != __last; ++__i)
> > >       +         __detail::__unguarded_linear_insert(__i, __comp, __proj);
> > >       +      }
> > >       +
> > >       +    inline constexpr int __sort_threshold = 16;
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __final_insertion_sort(_Iter __first, _Iter __last,
> > >       +                            _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       if (__last - __first > __sort_threshold)
> > >       +         {
> > >       +           __detail::__insertion_sort(__first, __first + 
> > > __sort_threshold,
> > >       +                                      __comp, __proj);
> > >       +           __detail::__unguarded_insertion_sort(__first + 
> > > __sort_threshold, __last,
> > >       +                                                __comp, __proj);
> > >       +         }
> > >       +       else
> > >       +         __detail::__insertion_sort(__first, __last, __comp, 
> > > __proj);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr _Iter
> > >       +      __unguarded_partition(_Iter __first, _Iter __last, _Iter 
> > > __pivot,
> > >       +                           _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       while (true)
> > >       +         {
> > >       +           while (std::__invoke(__comp,
> > >       +                                std::__invoke(__proj, *__first),
> > >       +                                std::__invoke(__proj, *__pivot)))
> > >       +             ++__first;
> > >       +           --__last;
> > >       +           while (std::__invoke(__comp,
> > >       +                                std::__invoke(__proj, *__pivot),
> > >       +                                std::__invoke(__proj, *__last)))
> > >       +             --__last;
> > >       +           if (!(__first < __last))
> > >       +             return __first;
> > >       +           ranges::iter_swap(__first, __last);
> > >       +           ++__first;
> > >       +         }
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr _Iter
> > >       +      __unguarded_partition_pivot(_Iter __first, _Iter __last,
> > >       +                                 _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       _Iter __mid = __first + (__last - __first) / 2;
> > >       +       __detail::__move_median_to_first(__first, __first + 1, 
> > > __mid, __last - 1,
> > >       +                                        __comp, __proj);
> > >       +       return __detail::__unguarded_partition(__first + 1, __last, 
> > > __first,
> > >       +                                              __comp, __proj);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __heap_select(_Iter __first, _Iter __middle, _Iter __last,
> > >       +                   _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       ranges::make_heap(__first, __middle, __comp, __proj);
> > >       +       for (_Iter __i = __middle; __i < __last; ++__i)
> > >       +         if (std::__invoke(__comp,
> > >       +                           std::__invoke(__proj, *__i),
> > >       +                           std::__invoke(__proj, *__first)))
> > >       +           __detail::__pop_heap(__first, __middle, __i, __comp, 
> > > __proj);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Compare, typename _Proj>
> > >       +      constexpr void
> > >       +      __partial_sort(_Iter __first, _Iter __middle, _Iter __last,
> > >       +                    _Compare __comp, _Proj __proj)
> > >       +      {
> > >       +       __detail::__heap_select(__first, __middle, __last, __comp, 
> > > __proj);
> > >       +       ranges::sort_heap(__first, __middle, __comp, __proj);
> > >       +      }
> > >       +
> > >       +    template<typename _Iter, typename _Comp, typename _Proj>
> > >       +      constexpr void
> > >       +      __introsort_loop(_Iter __first, _Iter __last, unsigned 
> > > __depth_limit,
> > >       +                      _Comp __comp, _Proj __proj)
> > >       +      {
> > >       +       while (__last - __first > __sort_threshold)
> > >       +         {
> > >       +           if (__depth_limit == 0)
> > >       +             {
> > >       +               __detail::__partial_sort(__first, __last, __last, 
> > > __comp, __proj);
> > >       +               return;
> > >       +             }
> > >       +           --__depth_limit;
> > >       +           _Iter __cut = 
> > > __detail::__unguarded_partition_pivot(__first, __last,
> > >       +                                                               
> > > __comp, __proj);
> > >       +           __detail::__introsort_loop(__cut, __last, 
> > > __depth_limit, __comp, __proj);
> > >       +           __last = __cut;
> > >       +         }
> > >       +      }
> > >       +  } // namespace __detail
> > >       +
> > >          struct __sort_fn
> > >          {
> > >            template<random_access_iterator _Iter, sentinel_for<_Iter> 
> > > _Sent,
> > >       @@ -2175,10 +2354,21 @@ namespace ranges
> > >              operator()(_Iter __first, _Sent __last,
> > >                        _Comp __comp = {}, _Proj __proj = {}) const
> > >              {
> > >       -       auto __lasti = ranges::next(__first, __last);
> > >       -       _GLIBCXX_STD_A::sort(std::move(__first), __lasti,
> > >       -                            __detail::__make_comp_proj(__comp, 
> > > __proj));
> > >       -       return __lasti;
> > >       +       if constexpr (!same_as<_Iter, _Sent>)
> > >       +         return (*this)(__first, ranges::next(__first, __last),
> > >       +                        std::move(__comp), std::move(__proj));
> > >       +       else
> > >       +         {
> > >       +           if (__first != __last)
> > >       +             {
> > >       +               auto __n = __detail::__to_unsigned_like(__last - 
> > > __first);
> > >       +               unsigned __depth_limit = (std::__bit_width(__n) - 
> > > 1) * 2;
> > >       +               __detail::__introsort_loop(__first, __last, 
> > > __depth_limit,
> > >       +                                          __comp, __proj);
> > >       +               __detail::__final_insertion_sort(__first, __last, 
> > > __comp, __proj);
> > >       +             }
> > >       +           return __last;
> > >       +         }
> > >              }
> > > 
> > >            template<random_access_range _Range,
> > >       diff --git a/libstdc++-v3/testsuite/25_algorithms/sort/118209.cc 
> > > b/libstdc++-v3/testsuite/25_algorithms/sort/118209.cc
> > >       new file mode 100644
> > >       index 000000000000..6dedbde75173
> > >       --- /dev/null
> > >       +++ b/libstdc++-v3/testsuite/25_algorithms/sort/118209.cc
> > >       @@ -0,0 +1,23 @@
> > >       +// PR libstdc++ - ranges::sort doesn't use iter_move, cannot sort 
> > > zip of move-only type
> > >       +// { dg-do compile { target c++23 } }
> > >       +
> > >       +#include <algorithm>
> > >       +#include <ranges>
> > >       +#include <vector>
> > >       +
> > >       +struct F {
> > >       +  int a = -1;
> > >       +  explicit F(int d) : a(d) { }
> > >       +  F(const F&) = delete;
> > >       +  F(F&& o) : a(o.a) { }
> > >       +  void operator=(const F&) = delete;
> > >       +  F& operator=(F&& o) { return *this; }
> > >       +  auto operator<=>(const F&) const = default;
> > >       +};
> > >       +
> > >       +int main () {
> > >       +  int a[] = {3,2,1};
> > >       +  std::vector<F> v(a, a+3);
> > >       +  std::ranges::sort(v); // OK
> > >       +  std::ranges::sort(std::views::zip(v)); // didn't compile
> > >       +}
> > >       diff --git 
> > > a/libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc 
> > > b/libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc
> > >       index 82754af96430..930dbd77a376 100644
> > >       --- a/libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc
> > >       +++ b/libstdc++-v3/testsuite/25_algorithms/sort/constrained.cc
> > >       @@ -20,6 +20,7 @@
> > > 
> > >        #include <algorithm>
> > >        #include <random>
> > >       +#include <ranges>
> > >        #include <vector>
> > >        #include <testsuite_hooks.h>
> > >        #include <testsuite_iterators.h>
> > >       @@ -72,9 +73,39 @@ test02()
> > >                 && ranges::equal(x, y, {}, &X::i, &X::i));
> > >        }
> > > 
> > >       +constexpr bool
> > >       +test03()
> > >       +{
> > >       +  // PR libstdc++/100795 - ranges::sort should not use std::sort 
> > > directly
> > >       +#if __SIZEOF_INT128__
> > >       +  auto v = std::views::iota(__int128(0), __int128(20));
> > >       +#else
> > >       +  auto v = std::views::iota(0ll, 20ll);
> > >       +#endif
> > >       +
> > >       +  int storage[20] = 
> > > {2,5,4,3,1,6,7,9,10,8,11,14,12,13,15,16,18,0,19,17};
> > >       +  auto w = v | std::views::transform([&](auto i) -> int& { return 
> > > storage[i]; });
> > >       +  using type = decltype(w);
> > >       +  using cat = 
> > > std::iterator_traits<std::ranges::iterator_t<type>>::iterator_category;
> > >       +  static_assert( std::same_as<cat, std::output_iterator_tag> );
> > >       +  static_assert( std::ranges::random_access_range<type> );
> > >       +
> > >       +  ranges::sort(w);
> > >       +  VERIFY( ranges::equal(w, v) );
> > >       +
> > >       +  ranges::sort(w, std::ranges::greater{});
> > >       +  VERIFY( ranges::equal(w, v | std::views::reverse) );
> > >       +
> > >       +  ranges::sort(w, std::ranges::greater{}, std::negate{});
> > >       +  VERIFY( ranges::equal(w, v) );
> > >       +
> > >       +  return true;
> > >       +}
> > >       +
> > >        int
> > >        main()
> > >        {
> > >          test01();
> > >          static_assert(test02());
> > >       +  static_assert(test03());
> > >        }
> > >       --
> > >       2.50.0.rc2
> > > 
> > > 
> > > 

Reply via email to