On Wed, 16 Jul 2025, Tomasz Kaminski wrote:

> 
> 
> On Tue, Jul 15, 2025 at 6:13 PM Patrick Palka <ppa...@redhat.com> wrote:
>       Tested on x86_64-pc-linux-gnu, does this look OK for trunk only
>       (since it impacts ABI)?
> 
>       Changes in v2:
> 
>        - Condition on forward_iterator instead of default_initializable.
> 
>       -- >8 --
> 
>       LWG 3569 adjusted join_view's iterator specification to handle non
>       default-constructible iterators by wrapping the corresponding data 
> member
>       in std::optional, which we followed suit in r13-2649-g7aa80c82ecf3a3.
> 
>       But this wrapping is unnecessary for iterators that are already
>       default-constructible.  Rather than unconditionally using std::optional
>       here, which introduces time/space overhead, this patch conditionalizes
>       our LWG 3569 changes on the iterator in question being non-forward (and
>       thus non default-constructible).  We check forwardness instead of
>       default-constructibility in order to accomodate input-only iterators
>       whose default constructor might be underconstrained.
> 
> I would rephrase that sentence a bit, to: 
> We check forwardness instead of
> default-constructibility in order to accomodate input-only iterators that
> satisfies but does not model default_initializable, e.g. whose default 
> constructor
> is underconstrained.
> 
> Outside of that this LGTM.

Thanks.  Is it OK to push (with that sentence rephrased)?

> 
>       libstdc++-v3/ChangeLog:
> 
>               * include/std/ranges (join_view::_Iterator::_M_satisfy):
>               Adjust to handle non-std::optional _M_inner as per before LWG 
> 3569.
>               (join_view::_Iterator::_M_get_inner): New.
>               (join_view::_Iterator::_M_inner): Don't wrap in std::optional if
>               the iterator is forward.  Initialize.
>               (join_view::_Iterator::operator*): Use _M_get_inner instead
>               of *_M_inner.
>               (join_view::_Iterator::operator++): Likewise.
>               (join_view::_Iterator::iter_move): Likewise.
>               (join_view::_Iterator::iter_swap): Likewise.
>       ---
>        libstdc++-v3/include/std/ranges | 49 +++++++++++++++++++++++++--------
>        1 file changed, 37 insertions(+), 12 deletions(-)
> 
>       diff --git a/libstdc++-v3/include/std/ranges 
> b/libstdc++-v3/include/std/ranges
>       index efe62969d657..c9dc25ee52ef 100644
>       --- a/libstdc++-v3/include/std/ranges
>       +++ b/libstdc++-v3/include/std/ranges
>       @@ -2971,7 +2971,12 @@ namespace views::__adaptor
>                     }
> 
>                   if constexpr (_S_ref_is_glvalue)
>       -             _M_inner.reset();
>       +             {
>       +               if constexpr (forward_iterator<_Inner_iter>)
>       +                 _M_inner = _Inner_iter();
>       +               else
>       +                 _M_inner.reset();
>       +             }
>                 }
> 
>                 static constexpr auto
>       @@ -3011,6 +3016,24 @@ namespace views::__adaptor
>                     return *_M_parent->_M_outer;
>                 }
> 
>       +         constexpr _Inner_iter&
>       +         _M_get_inner()
>       +         {
>       +           if constexpr (forward_iterator<_Inner_iter>)
>       +             return _M_inner;
>       +           else
>       +             return *_M_inner;
>       +         }
>       +
>       +         constexpr const _Inner_iter&
>       +         _M_get_inner() const
>       +         {
>       +           if constexpr (forward_iterator<_Inner_iter>)
>       +             return _M_inner;
>       +           else
>       +             return *_M_inner;
>       +         }
>       +
>                 constexpr
>                 _Iterator(_Parent* __parent, _Outer_iter __outer) requires 
> forward_range<_Base>
>                   : _M_outer(std::move(__outer)), _M_parent(__parent)
>       @@ -3024,7 +3047,9 @@ namespace views::__adaptor
>                 [[no_unique_address]]
>                   __detail::__maybe_present_t<forward_range<_Base>, 
> _Outer_iter> _M_outer
>                     = decltype(_M_outer)();
>       -         optional<_Inner_iter> _M_inner;
>       +         __conditional_t<forward_iterator<_Inner_iter>,
>       +                         _Inner_iter, optional<_Inner_iter>> _M_inner
>       +           = decltype(_M_inner)();
>                 _Parent* _M_parent = nullptr;
> 
>               public:
>       @@ -3048,7 +3073,7 @@ namespace views::__adaptor
> 
>                 constexpr decltype(auto)
>                 operator*() const
>       -         { return **_M_inner; }
>       +         { return *_M_get_inner(); }
> 
>                 // _GLIBCXX_RESOLVE_LIB_DEFECTS
>                 // 3500. join_view::iterator::operator->() is bogus
>       @@ -3056,7 +3081,7 @@ namespace views::__adaptor
>                 operator->() const
>                   requires __detail::__has_arrow<_Inner_iter>
>                     && copyable<_Inner_iter>
>       -         { return *_M_inner; }
>       +         { return _M_get_inner(); }
> 
>                 constexpr _Iterator&
>                 operator++()
>       @@ -3067,7 +3092,7 @@ namespace views::__adaptor
>                     else
>                       return *_M_parent->_M_inner;
>                   }();
>       -           if (++*_M_inner == ranges::end(__inner_range))
>       +           if (++_M_get_inner() == ranges::end(__inner_range))
>                     {
>                       ++_M_get_outer();
>                       _M_satisfy();
>       @@ -3097,9 +3122,9 @@ namespace views::__adaptor
>                 {
>                   if (_M_outer == ranges::end(_M_parent->_M_base))
>                     _M_inner = 
> ranges::end(__detail::__as_lvalue(*--_M_outer));
>       -           while (*_M_inner == 
> ranges::begin(__detail::__as_lvalue(*_M_outer)))
>       -             *_M_inner = 
> ranges::end(__detail::__as_lvalue(*--_M_outer));
>       -           --*_M_inner;
>       +           while (_M_get_inner() == 
> ranges::begin(__detail::__as_lvalue(*_M_outer)))
>       +             _M_get_inner() = 
> ranges::end(__detail::__as_lvalue(*--_M_outer));
>       +           --_M_get_inner();
>                   return *this;
>                 }
> 
>       @@ -3126,14 +3151,14 @@ namespace views::__adaptor
> 
>                 friend constexpr decltype(auto)
>                 iter_move(const _Iterator& __i)
>       -         noexcept(noexcept(ranges::iter_move(*__i._M_inner)))
>       -         { return ranges::iter_move(*__i._M_inner); }
>       +         noexcept(noexcept(ranges::iter_move(__i._M_get_inner())))
>       +         { return ranges::iter_move(__i._M_get_inner()); }
> 
>                 friend constexpr void
>                 iter_swap(const _Iterator& __x, const _Iterator& __y)
>       -           noexcept(noexcept(ranges::iter_swap(*__x._M_inner, 
> *__y._M_inner)))
>       +           noexcept(noexcept(ranges::iter_swap(__x._M_get_inner(), 
> __y._M_get_inner())))
>                   requires indirectly_swappable<_Inner_iter>
>       -         { return ranges::iter_swap(*__x._M_inner, *__y._M_inner); }
>       +         { return ranges::iter_swap(__x._M_get_inner(), 
> __y._M_get_inner()); }
> 
>                 friend _Iterator<!_Const>;
>                 template<bool> friend struct _Sentinel;
>       --
>       2.50.1.271.gd30e120486
> 
> 
> 

Reply via email to