On Wed, 11 Jun 2025 at 12:42, Tomasz Kaminski <tkami...@redhat.com> wrote: > > > > On Tue, Jun 10, 2025 at 3:05 AM Patrick Palka <ppa...@redhat.com> wrote: >> >> ranges::push_heap, ranges::pop_heap, ranges::make_heap and >> ranges::sort_heap are currently defined in terms of the corresponding >> STL-style algorithms, but this is incorrect because the STL-style >> algorithms rely on the legacy iterator system, and so misbehave when >> passed a narrowly C++20 random access iterator. The other ranges heap >> algos, ranges::is_heap and ranges::is_heap_until, are implemented >> directly already and have no known issues. >> >> This patch reimplements these ranges:: algos directly instead, based >> closely on the legacy stl_algo.h implementation, with the following >> changes for compatibility with the C++20 iterator system: >> >> - handle non-common ranges by computing the corresponding end iterator >> - pass and invoke a projection function as necessary >> - use std::__invoke when invoking the comparator >> - use ranges::iter_move instead of std::move(*iter) >> - use iter_value_t / iter_difference_t instead of iterator_traits >> >> Besides these changes, the implementation of these algorithms is >> intended to mirror the stl_algo.h implementations, for ease of >> maintenance and review. >> >> PR libstdc++/100795 >> >> libstdc++-v3/ChangeLog: >> >> * include/bits/ranges_algo.h (__detail::__push_heap): New, >> based on the stl_algo.h implementation. >> (__push_heap_fn::operator()): Reimplement in terms of the above. >> (__detail::__adjust_heap): New, based on the stl_algo.h >> implementation. >> (__deatil::__pop_heap): Likewise. >> (__pop_heap_fn::operator()): Reimplement in terms of the above. >> (__make_heap_fn::operator()): Likewise. >> (__sort_heap_fn::operator()): Likewise. >> * testsuite/25_algorithms/heap/constrained.cc (test03): New >> test. >> --- >> libstdc++-v3/include/bits/ranges_algo.h | 138 ++++++++++++++++-- >> .../25_algorithms/heap/constrained.cc | 46 ++++++ >> 2 files changed, 168 insertions(+), 16 deletions(-) > > LGTM, only requests to pass the predicates by reference to match stl_heap.
That was done by r7-6137-g437f43cc78d460 for PR67085 > I have validated the impl against the stl_heap, and implementations are the > same modulo described changes. >> >> >> diff --git a/libstdc++-v3/include/bits/ranges_algo.h >> b/libstdc++-v3/include/bits/ranges_algo.h >> index a62c3cd3954d..f84e0c42d76a 100644 >> --- a/libstdc++-v3/include/bits/ranges_algo.h >> +++ b/libstdc++-v3/include/bits/ranges_algo.h >> @@ -1881,6 +1881,30 @@ namespace ranges >> >> inline constexpr __shuffle_fn shuffle{}; >> >> + namespace __detail >> + { >> + template<typename _Iter, typename _Comp, typename _Proj> >> + constexpr void >> + __push_heap(_Iter __first, >> + iter_difference_t<_Iter> __holeIndex, >> + iter_difference_t<_Iter> __topIndex, >> + iter_value_t<_Iter> __value, >> + _Comp __comp, _Proj __proj) > > stl_heap implementation does not copy __comp and __proj, and uses > _Comp& instead. I think I would do that also here. >> >> + { >> + auto __parent = (__holeIndex - 1) / 2; >> + while (__holeIndex > __topIndex >> + && std::__invoke(__comp, >> + std::__invoke(__proj, *(__first + __parent)), >> + std::__invoke(__proj, __value))) >> + { >> + *(__first + __holeIndex) = ranges::iter_move(__first + __parent); >> + __holeIndex = __parent; >> + __parent = (__holeIndex - 1) / 2; >> + } >> + *(__first + __holeIndex) = std::move(__value); >> + } >> + } // namespace __detail >> + >> struct __push_heap_fn >> { >> template<random_access_iterator _Iter, sentinel_for<_Iter> _Sent, >> @@ -1890,10 +1914,16 @@ namespace ranges >> operator()(_Iter __first, _Sent __last, >> _Comp __comp = {}, _Proj __proj = {}) const >> { >> - auto __lasti = ranges::next(__first, __last); >> - std::push_heap(__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 >> + { >> + __detail::__push_heap(__first, (__last - __first) - 1, >> + 0, ranges::iter_move(__last - 1), >> + __comp, __proj); >> + return __last; >> + } >> } >> >> template<random_access_range _Range, >> @@ -1909,6 +1939,50 @@ namespace ranges >> >> inline constexpr __push_heap_fn push_heap{}; >> >> + namespace __detail >> + { >> + template<typename _Iter, typename _Comp, typename _Proj> >> + constexpr void >> + __adjust_heap(_Iter __first, >> + iter_difference_t<_Iter> __holeIndex, >> + iter_difference_t<_Iter> __len, >> + iter_value_t<_Iter> __value, >> + _Comp __comp, _Proj __proj) > > Same comment about references. >> >> + { >> + auto __topIndex = __holeIndex; >> + auto __secondChild = __holeIndex; >> + while (__secondChild < (__len - 1) / 2) >> + { >> + __secondChild = 2 * (__secondChild + 1); >> + if (std::__invoke(__comp, >> + std::__invoke(__proj, *(__first + >> __secondChild)), >> + std::__invoke(__proj, *(__first + >> (__secondChild - 1))))) >> + __secondChild--; >> + *(__first + __holeIndex) = ranges::iter_move(__first + >> __secondChild); >> + __holeIndex = __secondChild; >> + } >> + if ((__len & 1) == 0 && __secondChild == (__len - 2) / 2) >> + { >> + __secondChild = 2 * (__secondChild + 1); >> + *(__first + __holeIndex) = ranges::iter_move(__first + >> (__secondChild - 1)); >> + __holeIndex = __secondChild - 1; >> + } >> + __detail::__push_heap(__first, __holeIndex, __topIndex, >> + std::move(__value), __comp, __proj); >> + } >> + >> + template<typename _Iter, typename _Comp, typename _Proj> >> + constexpr void >> + __pop_heap(_Iter __first, _Iter __last, _Iter __result, >> + _Comp __comp, _Proj __proj) > > Also here. >> >> + { >> + iter_value_t<_Iter> __value = ranges::iter_move(__result); >> + *__result = ranges::iter_move(__first); >> + __detail::__adjust_heap(__first, 0, __last - __first, >> + std::move(__value), __comp, __proj); >> + } >> + } // namespace __detail >> + >> struct __pop_heap_fn >> { >> template<random_access_iterator _Iter, sentinel_for<_Iter> _Sent, >> @@ -1918,10 +1992,15 @@ namespace ranges >> operator()(_Iter __first, _Sent __last, >> _Comp __comp = {}, _Proj __proj = {}) const >> { >> - auto __lasti = ranges::next(__first, __last); >> - std::pop_heap(__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 (__last - __first > 1) >> + __detail::__pop_heap(__first, __last - 1, __last - 1, __comp, >> __proj); >> + return __last; >> + } >> } >> >> template<random_access_range _Range, >> @@ -1946,10 +2025,28 @@ namespace ranges >> operator()(_Iter __first, _Sent __last, >> _Comp __comp = {}, _Proj __proj = {}) const >> { >> - auto __lasti = ranges::next(__first, __last); >> - std::make_heap(__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 (__last - __first > 1) > > I would prefer early if, matching the stl_heap closer: > const auto __len = __last - __first; > if (__len < 2) > return __last; >> >> + { >> + const auto __len = __last - __first; >> + auto __parent = (__len - 2) / 2; >> + while (true) >> + { >> + iter_value_t<_Iter> __value = ranges::iter_move(__first >> + __parent); >> + __detail::__adjust_heap(__first, __parent, __len, >> + std::move(__value), >> + __comp, __proj); >> + if (__parent == 0) >> + break; >> + __parent--; >> + } >> + } >> + return __last; >> + } >> } >> >> template<random_access_range _Range, >> @@ -1974,10 +2071,19 @@ namespace ranges >> operator()(_Iter __first, _Sent __last, >> _Comp __comp = {}, _Proj __proj = {}) const >> { >> - auto __lasti = ranges::next(__first, __last); >> - std::sort_heap(__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 >> + { >> + _Iter __ret = __last; >> + while (__last - __first > 1) >> + { >> + --__last; >> + __detail::__pop_heap(__first, __last, __last, __comp, >> __proj); >> + } >> + return __ret; >> + } >> } >> >> template<random_access_range _Range, >> diff --git a/libstdc++-v3/testsuite/25_algorithms/heap/constrained.cc >> b/libstdc++-v3/testsuite/25_algorithms/heap/constrained.cc >> index 8037a2db6b80..5486c8552d03 100644 >> --- a/libstdc++-v3/testsuite/25_algorithms/heap/constrained.cc >> +++ b/libstdc++-v3/testsuite/25_algorithms/heap/constrained.cc >> @@ -20,6 +20,7 @@ >> >> #include <algorithm> >> #include <random> >> +#include <ranges> >> #include <testsuite_hooks.h> >> #include <testsuite_iterators.h> >> >> @@ -97,10 +98,55 @@ test02() >> return ok; >> } >> >> +constexpr bool >> +test03() >> +{ >> + // PR libstdc++/100795 - ranges::heap algos should not use std::heap >> 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> ); >> + >> + for (int i = 1; i < 20; i++) >> + ranges::push_heap(w.begin(), w.begin() + i); >> + ranges::sort_heap(w); >> + VERIFY( ranges::equal(w, v) ); >> + ranges::make_heap(w); >> + auto it = ranges::pop_heap(w); >> + VERIFY( it[-1] == 19 ); >> + >> + for (int i = 1; i < 20; i++) >> + ranges::push_heap(w.begin(), w.begin() + i, std::ranges::greater{}); >> + ranges::sort_heap(w, std::ranges::greater{}); >> + VERIFY( ranges::equal(w, v | std::views::reverse) ); >> + ranges::make_heap(w, std::ranges::greater{}); >> + it = ranges::pop_heap(w, std::ranges::greater{}); >> + VERIFY( it[-1] == 0 ); >> + >> + for (int i = 1; i < 20; i++) >> + ranges::push_heap(w.begin(), w.begin() + i, std::ranges::greater{}, >> std::negate{}); >> + ranges::sort_heap(w, std::ranges::greater{}, std::negate{}); >> + VERIFY( ranges::equal(w, v) ); >> + ranges::make_heap(w, std::ranges::greater{}, std::negate{}); >> + it = ranges::pop_heap(w, std::ranges::greater{}, std::negate{}); >> + VERIFY( it[-1] == 19 ); >> + >> + return true; >> +} >> + >> int >> main() >> { >> test01<test_range>(); >> test01<test_container>(); >> static_assert(test02()); >> + static_assert(test03()); >> } >> -- >> 2.50.0.rc1.89.g8db3019401 >>