On Thu, Sep 11, 2025 at 2:18 PM Jonathan Wakely <[email protected]> wrote:

> Most of the basis operations for ranges such as ranges::begin and
> ranges::next are trivial one-line function bodies, so can be made
> always_inline to reduce the abstraction penalty for -O0 code.
>
> Now that we no longer need to support the -fconcepts-ts grammar, we can
> also move some [[nodiscard]] attributes to the more natural position
> before the function declaration, instead of between the declarator-id
> and the function parameters, e.g. we can use:
>
>   template<typename T> requires C<T> [[nodiscard]] auto operator()(T&&)
>
> instead of:
>
>   template<typename T> requires C<T> auto operator() [[nodiscard]] (T&&)
>
> The latter form was necessary because -fconcepts-ts used a different
> grammar for the requires-clause, parsing 'C<T>[[x]]' as a subscripting
> operator with an ill-formed argument '[x]'. In the C++20 grammar you
> would need to use parentheses to use a subscript in a constraint, so
> without parentheses it's parsed as an attribute.
>
> libstdc++-v3/ChangeLog:
>
>         * include/bits/ranges_base.h (__detail::__to_unsigned_like)
>         (__access::__possible_const_range, __access::__as_const)
>         (__distance_fn::operator(), __next_fn::operator())
>         (__prev_fn::operator()): Add always_inline attribute.
>         (_Begin::operator(), _End::operator(), _RBegin::operator())
>         (_REnd::operator(), _Size::operator(), _SSize::operator())
>         (_Empty::operator(), _Data::operator(), _SSize::operator()):
>         Likewise. Move nodiscard attribute to start of declaration.
> ---
>
> This change was inspired when I was adding more uses of ranges::next and
> ranges::prev to fix the iterator arithmetic not using difference_type.
>
LGTM.

>
> Tested powerpc64-linux.
>
>  libstdc++-v3/include/bits/ranges_base.h | 47 ++++++++++++++++---------
>  1 file changed, 30 insertions(+), 17 deletions(-)
>
> diff --git a/libstdc++-v3/include/bits/ranges_base.h
> b/libstdc++-v3/include/bits/ranges_base.h
> index 02e13a2027d1..c1a9f6a90009 100644
> --- a/libstdc++-v3/include/bits/ranges_base.h
> +++ b/libstdc++-v3/include/bits/ranges_base.h
> @@ -61,29 +61,32 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
>  namespace ranges
>  {
>    template<typename>
>      inline constexpr bool disable_sized_range = false;
>
>    template<typename _Tp>
>      inline constexpr bool enable_borrowed_range = false;
>
>    namespace __detail
>    {
> +    [[__gnu__::__always_inline__]]
>      constexpr __max_size_type
>      __to_unsigned_like(__max_size_type __t) noexcept
>      { return __t; }
>
> +    [[__gnu__::__always_inline__]]
>      constexpr __max_size_type
>      __to_unsigned_like(__max_diff_type __t) noexcept
>      { return __max_size_type(__t); }
>
>      template<integral _Tp>
> +      [[__gnu__::__always_inline__]]
>        constexpr auto
>        __to_unsigned_like(_Tp __t) noexcept
>        { return static_cast<make_unsigned_t<_Tp>>(__t); }
>
>      template<typename _Tp>
>        using __make_unsigned_like_t
>         = decltype(__detail::__to_unsigned_like(std::declval<_Tp>()));
>
>      // Part of the constraints of ranges::borrowed_range
>      template<typename _Tp>
> @@ -111,22 +114,23 @@ namespace ranges
>           else if constexpr (__member_begin<_Tp>)
>             return
> noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().begin()));
>           else
>             return
> noexcept(_GLIBCXX_AUTO_CAST(begin(std::declval<_Tp&>())));
>         }
>
>      public:
>        template<__maybe_borrowed_range _Tp>
>         requires is_array_v<remove_reference_t<_Tp>> || __member_begin<_Tp>
>           || __adl_begin<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(_S_noexcept<_Tp&>())
> +       operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (is_array_v<remove_reference_t<_Tp>>)
>             {
>               static_assert(is_lvalue_reference_v<_Tp>);
>               return __t + 0;
>             }
>           else if constexpr (__member_begin<_Tp>)
>             return __t.begin();
>           else
>             return begin(__t);
> @@ -161,22 +165,23 @@ namespace ranges
>           else if constexpr (__member_end<_Tp>)
>             return
> noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().end()));
>           else
>             return noexcept(_GLIBCXX_AUTO_CAST(end(std::declval<_Tp&>())));
>         }
>
>      public:
>        template<__maybe_borrowed_range _Tp>
>         requires is_bounded_array_v<remove_reference_t<_Tp>>
>           || __member_end<_Tp> || __adl_end<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(_S_noexcept<_Tp&>())
> +       operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
>             {
>               static_assert(is_lvalue_reference_v<_Tp>);
>               return __t + extent_v<remove_reference_t<_Tp>>;
>             }
>           else if constexpr (__member_end<_Tp>)
>             return __t.end();
>           else
>             return end(__t);
> @@ -225,22 +230,23 @@ namespace ranges
>                   return is_nothrow_copy_constructible_v<_It>;
>                 }
>               else
>                 return false;
>             }
>         }
>
>      public:
>        template<__maybe_borrowed_range _Tp>
>         requires __member_rbegin<_Tp> || __adl_rbegin<_Tp> ||
> __reversable<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> +       operator()(_Tp&& __t) const
>         noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (__member_rbegin<_Tp>)
>             return __t.rbegin();
>           else if constexpr (__adl_rbegin<_Tp>)
>             return rbegin(__t);
>           else
>             return std::make_reverse_iterator(_End{}(__t));
>         }
>      };
> @@ -282,22 +288,23 @@ namespace ranges
>                   return is_nothrow_copy_constructible_v<_It>;
>                 }
>               else
>                 return false;
>             }
>         }
>
>      public:
>        template<__maybe_borrowed_range _Tp>
>         requires __member_rend<_Tp> || __adl_rend<_Tp> || __reversable<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> +       operator()(_Tp&& __t) const
>         noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (__member_rend<_Tp>)
>             return __t.rend();
>           else if constexpr (__adl_rend<_Tp>)
>             return rend(__t);
>           else
>             return std::make_reverse_iterator(_Begin{}(__t));
>         }
>      };
> @@ -346,42 +353,44 @@ namespace ranges
>             return
> noexcept(_GLIBCXX_AUTO_CAST(size(std::declval<_Tp&>())));
>           else if constexpr (__sentinel_size<_Tp>)
>             return noexcept(_End{}(std::declval<_Tp&>())
>                             - _Begin{}(std::declval<_Tp&>()));
>         }
>
>      public:
>        template<typename _Tp>
>         requires is_bounded_array_v<remove_reference_t<_Tp>>
>           || __member_size<_Tp> || __adl_size<_Tp> || __sentinel_size<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(_S_noexcept<_Tp&>())
> +       operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
>             return extent_v<remove_reference_t<_Tp>>;
>           else if constexpr (__member_size<_Tp>)
>             return __t.size();
>           else if constexpr (__adl_size<_Tp>)
>             return size(__t);
>           else if constexpr (__sentinel_size<_Tp>)
>             return __detail::__to_unsigned_like(_End{}(__t) -
> _Begin{}(__t));
>         }
>      };
>
>      struct _SSize
>      {
>        // _GLIBCXX_RESOLVE_LIB_DEFECTS
>        // 3403. Domain of ranges::ssize(E) doesn't match ranges::size(E)
>        template<typename _Tp>
>         requires requires (_Tp& __t) { _Size{}(__t); }
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(noexcept(_Size{}(__t)))
> +       operator()(_Tp&& __t) const noexcept(noexcept(_Size{}(__t)))
>         {
>           auto __size = _Size{}(__t);
>           using __size_type = decltype(__size);
>           // Return the wider of ptrdiff_t and
> make-signed-like-t<__size_type>.
>           if constexpr (integral<__size_type>)
>             {
>               using __gnu_cxx::__int_traits;
>               if constexpr (__int_traits<__size_type>::__digits
>                             < __int_traits<ptrdiff_t>::__digits)
>                 return static_cast<ptrdiff_t>(__size);
> @@ -422,22 +431,23 @@ namespace ranges
>             return noexcept(_Size{}(std::declval<_Tp&>()) == 0);
>           else
>             return noexcept(bool(_Begin{}(std::declval<_Tp&>())
>                 == _End{}(std::declval<_Tp&>())));
>         }
>
>      public:
>        template<typename _Tp>
>         requires __member_empty<_Tp> || __size0_empty<_Tp>
>           || __eq_iter_empty<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr bool
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(_S_noexcept<_Tp&>())
> +       operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
>         {
>           if constexpr (__member_empty<_Tp>)
>             return bool(__t.empty());
>           else if constexpr (__size0_empty<_Tp>)
>             return _Size{}(__t) == 0;
>           else
>             return bool(_Begin{}(__t) == _End{}(__t));
>         }
>      };
>
> @@ -463,22 +473,23 @@ namespace ranges
>         {
>           if constexpr (__member_data<_Tp>)
>             return
> noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().data()));
>           else
>             return noexcept(_Begin{}(std::declval<_Tp&>()));
>         }
>
>      public:
>        template<__maybe_borrowed_range _Tp>
>         requires __member_data<_Tp> || __begin_data<_Tp>
> +       [[nodiscard, __gnu__::__always_inline__]]
>         constexpr auto
> -       operator()[[nodiscard]](_Tp&& __t) const
> noexcept(_S_noexcept<_Tp>())
> +       operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp>())
>         {
>           if constexpr (__member_data<_Tp>)
>             return __t.data();
>           else
>             return std::to_address(_Begin{}(__t));
>         }
>      };
>
>    } // namespace __access
>
> @@ -625,33 +636,35 @@ namespace ranges
>  #if __glibcxx_ranges_as_const // >= C++23
>    template<typename _Tp>
>      concept constant_range
>        = input_range<_Tp> &&
> std::__detail::__constant_iterator<iterator_t<_Tp>>;
>  #endif
>
>    namespace __access
>    {
>  #if __glibcxx_ranges_as_const // >= C++23
>      template<input_range _Range>
> +      [[__gnu__::__always_inline__]]
>        constexpr auto&
>        __possibly_const_range(_Range& __r) noexcept
>        {
>         // _GLIBCXX_RESOLVE_LIB_DEFECTS
>         // 4027. possibly-const-range should prefer returning const R&
>         if constexpr (input_range<const _Range>)
>           return const_cast<const _Range&>(__r);
>         else
>           return __r;
>        }
>  #else
>      // If _To is an lvalue-reference, return const _Tp&, otherwise const
> _Tp&&.
>      template<typename _To, typename _Tp>
> +      [[__gnu__::__always_inline__]]
>        constexpr decltype(auto)
>        __as_const(_Tp& __t) noexcept
>        {
>         static_assert(std::is_same_v<_To&, _Tp&>);
>
>         if constexpr (is_lvalue_reference_v<_To>)
>           return const_cast<const _Tp&>(__t);
>         else
>           return static_cast<const _Tp&&>(__t);
>        }
> @@ -960,106 +973,106 @@ namespace ranges
>         iter_difference_t<_It> __n = 0;
>         while (__first != __last)
>           {
>             ++__first;
>             ++__n;
>           }
>         return __n;
>        }
>
>      template<typename _It, sized_sentinel_for<decay_t<_It>> _Sent>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr iter_difference_t<decay_t<_It>>
>        operator()(_It&& __first, _Sent __last) const
>        { return __last - static_cast<const decay_t<_It>&>(__first); }
>
>      template<range _Range>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr range_difference_t<_Range>
>        operator()(_Range&& __r) const
>        {
>         if constexpr (sized_range<_Range>)
>           return
> static_cast<range_difference_t<_Range>>(ranges::size(__r));
>         else
>           return (*this)(ranges::begin(__r), ranges::end(__r));
>        }
>
>      void operator&() const = delete;
>    };
>
>    inline constexpr __distance_fn distance{};
>
>    struct __next_fn final
>    {
>      template<input_or_output_iterator _It>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x) const
>        {
>         ++__x;
>         return __x;
>        }
>
>      template<input_or_output_iterator _It>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x, iter_difference_t<_It> __n) const
>        {
>         ranges::advance(__x, __n);
>         return __x;
>        }
>
>      template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x, _Sent __bound) const
>        {
>         ranges::advance(__x, __bound);
>         return __x;
>        }
>
>      template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x, iter_difference_t<_It> __n, _Sent __bound) const
>        {
>         ranges::advance(__x, __n, __bound);
>         return __x;
>        }
>
>      void operator&() const = delete;
>    };
>
>    inline constexpr __next_fn next{};
>
>    struct __prev_fn final
>    {
>      template<bidirectional_iterator _It>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x) const
>        {
>         --__x;
>         return __x;
>        }
>
>      template<bidirectional_iterator _It>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x, iter_difference_t<_It> __n) const
>        {
>         ranges::advance(__x, -__n);
>         return __x;
>        }
>
>      template<bidirectional_iterator _It>
> -      [[nodiscard]]
> +      [[nodiscard, __gnu__::__always_inline__]]
>        constexpr _It
>        operator()(_It __x, iter_difference_t<_It> __n, _It __bound) const
>        {
>         ranges::advance(__x, -__n, __bound);
>         return __x;
>        }
>
>      void operator&() const = delete;
>    };
>
> --
> 2.51.0
>
>

Reply via email to