On Thu, 18 Jan 2024 at 02:48, Patrick Palka wrote:
>
> Tested on x86_64-pc-linux-gnu, does this look OK for trunk?

Please add PR109536 to the commit message.



>
> -- >8 --
>
> Some _Safe_iterator member functions define a variable of non-literal
> type __gnu_cxx::__scoped_lock, which automatically disqualifies them from
> being constexpr in C++20 mode even if that code path is never constant
> evaluated.  This restriction was lifted by P2242R3 for C++23, but we
> need to work around it in C++20 mode.  To that end this patch defines
> a pair of macros that encapsulate the lambda-based workaround mentioned
> in that paper and uses them to make the functions valid C++20 constexpr
> functions.  The augmented std::vector test element_access/constexpr.cc
> now successfully compiles in C++20 mode with -D_GLIBCXX_DEBUG (and it
> tests all modified member functions).
>
> libstdc++-v3/ChangeLog:
>
>         * include/debug/safe_base.h (_Safe_sequence_base::_M_swap):
>         Remove _GLIBCXX20_CONSTEXPR.
>         * include/debug/safe_iterator.h 
> (_GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN):
>         (_GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END): Define.
>         (_Safe_iterator::operator=): Use them around the code path that
>         defines a variable of type __gnu_cxx::__scoped_lock.
>         (_Safe_iterator::operator++): Likewise.
>         (_Safe_iterator::operator--): Likewise.
>         (_Safe_iterator::operator+=): Likewise.
>         (_Safe_iterator::operator-=): Likewise.
>         * testsuite/23_containers/vector/element_access/constexpr.cc
>         (test_iterators): Also test copy and move assignment.
>         * testsuite/std/ranges/adaptors/all.cc (test08) [_GLIBCXX_DEBUG]:
>         Use std::vector unconditionally.
> ---
>  libstdc++-v3/include/debug/safe_base.h        |  1 -
>  libstdc++-v3/include/debug/safe_iterator.h    | 48 ++++++++++++++-----
>  .../vector/element_access/constexpr.cc        |  2 +
>  .../testsuite/std/ranges/adaptors/all.cc      |  4 --
>  4 files changed, 38 insertions(+), 17 deletions(-)
>
> diff --git a/libstdc++-v3/include/debug/safe_base.h 
> b/libstdc++-v3/include/debug/safe_base.h
> index 107fef3cb02..d5fbe4b1320 100644
> --- a/libstdc++-v3/include/debug/safe_base.h
> +++ b/libstdc++-v3/include/debug/safe_base.h
> @@ -268,7 +268,6 @@ namespace __gnu_debug
>       *  operation is complete all iterators that originally referenced
>       *  one container now reference the other container.
>       */
> -    _GLIBCXX20_CONSTEXPR
>      void
>      _M_swap(_Safe_sequence_base& __x) _GLIBCXX_USE_NOEXCEPT;
>
> diff --git a/libstdc++-v3/include/debug/safe_iterator.h 
> b/libstdc++-v3/include/debug/safe_iterator.h
> index 1bc7c904ee0..929fd9b0ade 100644
> --- a/libstdc++-v3/include/debug/safe_iterator.h
> +++ b/libstdc++-v3/include/debug/safe_iterator.h
> @@ -65,6 +65,20 @@
>    _GLIBCXX_DEBUG_VERIFY_OPERANDS(_Lhs, _Rhs, __msg_distance_bad,       \
>                                  __msg_distance_different)
>
> +// This pair of macros helps with writing valid C++20 constexpr functions 
> that
> +// contain a non-constexpr code path that defines a non-literal variable, 
> which
> +// was otherwise disallowed until P2242R3 for C++23.  We use them below for
> +// __gnu_cxx::__scoped_lock so that the containing functions are still
> +// considered valid C++20 constexpr functions.
> +
> +#if __cplusplus >= 202002L && __cpp_constexpr < 202110L
> +# define _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN [&]() -> void { do
> +# define _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END while(false); }();

Do we need the do-while to create a single statement from the block?
Isn't the lambda body enough to create a single statement from it,
which can't be broken by a dangling else or anything like that?


> +#else
> +# define _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN
> +# define _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
> +#endif
> +
>  namespace __gnu_debug
>  {
>    /** Helper struct to deal with sequence offering a before_begin
> @@ -266,11 +280,11 @@ namespace __gnu_debug
>                               ._M_iterator(__x, "other"));
>
>         if (this->_M_sequence && this->_M_sequence == __x._M_sequence)
> -         {
> +         _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
>             __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
>             base() = __x.base();
>             _M_version = __x._M_sequence->_M_version;
> -         }
> +         } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         else
>           {
>             _M_detach();
> @@ -306,11 +320,11 @@ namespace __gnu_debug
>           return *this;
>
>         if (this->_M_sequence && this->_M_sequence == __x._M_sequence)
> -         {
> +         _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
>             __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
>             base() = __x.base();
>             _M_version = __x._M_sequence->_M_version;
> -         }
> +         } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         else
>           {
>             _M_detach();
> @@ -378,8 +392,10 @@ namespace __gnu_debug
>         _GLIBCXX_DEBUG_VERIFY(this->_M_incrementable(),
>                               _M_message(__msg_bad_inc)
>                               ._M_iterator(*this, "this"));
> -       __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> -       ++base();
> +       _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
> +         __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> +         ++base();
> +       } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         return *this;
>        }
>
> @@ -697,8 +713,10 @@ namespace __gnu_debug
>         _GLIBCXX_DEBUG_VERIFY(this->_M_decrementable(),
>                               _M_message(__msg_bad_dec)
>                               ._M_iterator(*this, "this"));
> -       __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> -       --this->base();
> +       _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
> +         __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> +         --this->base();
> +       } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         return *this;
>        }
>
> @@ -912,8 +930,10 @@ namespace __gnu_debug
>         _GLIBCXX_DEBUG_VERIFY(this->_M_can_advance(__n),
>                               _M_message(__msg_advance_oob)
>                               ._M_iterator(*this)._M_integer(__n));
> -       __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> -       this->base() += __n;
> +       _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
> +         __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> +         this->base() += __n;
> +       } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         return *this;
>        }
>
> @@ -930,8 +950,10 @@ namespace __gnu_debug
>         _GLIBCXX_DEBUG_VERIFY(this->_M_can_advance(-__n),
>                               _M_message(__msg_retreat_oob)
>                               ._M_iterator(*this)._M_integer(__n));
> -       __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> -       this->base() -= __n;
> +       _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN {
> +         __gnu_cxx::__scoped_lock __l(this->_M_get_mutex());
> +         this->base() -= __n;
> +       } _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
>         return *this;
>        }
>
> @@ -1156,6 +1178,8 @@ _GLIBCXX_END_NAMESPACE_VERSION
>  }
>  #endif
>
> +#undef _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_END
> +#undef _GLIBCXX20_CONSTEXPR_NON_LITERAL_SCOPE_BEGIN
>  #undef _GLIBCXX_DEBUG_VERIFY_DIST_OPERANDS
>  #undef _GLIBCXX_DEBUG_VERIFY_REL_OPERANDS
>  #undef _GLIBCXX_DEBUG_VERIFY_EQ_OPERANDS
> diff --git 
> a/libstdc++-v3/testsuite/23_containers/vector/element_access/constexpr.cc 
> b/libstdc++-v3/testsuite/23_containers/vector/element_access/constexpr.cc
> index ee93d2fd95e..ab1e7f1bb70 100644
> --- a/libstdc++-v3/testsuite/23_containers/vector/element_access/constexpr.cc
> +++ b/libstdc++-v3/testsuite/23_containers/vector/element_access/constexpr.cc
> @@ -25,6 +25,8 @@ test_iterators()
>    it -= 2;
>    it += 1;
>    VERIFY( (it + 1) == v.end() );
> +  it = it + 1;
> +  it = it;

I think we also need to test these operators here:

it[n]
n + it
it - it

And also for the reverse iterator.

I think that invokes all the operators. For vector, none of those
operators do anything different for positive or negative arguments, so
we don't need to test cases like it[-1], it+-1, -1+it etc.


>
>    auto rit = v.rbegin();
>    VERIFY( &*rit == &v.back() );
> diff --git a/libstdc++-v3/testsuite/std/ranges/adaptors/all.cc 
> b/libstdc++-v3/testsuite/std/ranges/adaptors/all.cc
> index e7010f80e18..5f7206dc8c3 100644
> --- a/libstdc++-v3/testsuite/std/ranges/adaptors/all.cc
> +++ b/libstdc++-v3/testsuite/std/ranges/adaptors/all.cc
> @@ -156,11 +156,7 @@ test07()
>  constexpr bool
>  test08()
>  {
> -#ifdef _GLIBCXX_DEBUG
> -  using std::_GLIBCXX_STD_C::vector;
> -#else
>    using std::vector;
> -#endif

Oh that's nice to remove.

>
>    // Verify P2415R2 "What is a view?" changes.
>    // In particular, rvalue non-view non-borrowed ranges are now viewable.
> --
> 2.43.0.367.g186b115d30
>

Reply via email to