PR121061 consists of two bugs for mdspan related code. This commit fixes the first one. Namely, when passing custom IndexType as an array or span, the conversion to int must be const. Prior to this commit the constraint incorrectly also allowed non-const conversion. This commit updates all related constraints to check
__valid_index_type<const OtherIndexType&, index_type> in those cases. Also adds a MutatingInt to int_like.h which only supports non-const conversion to int and updates the tests. PR libstdc++/121061 libstdc++-v3/ChangeLog: * include/std/mdspan (extents::extents): Fix constraint to prevent non-const conversion to index_type. (layout_stride::mapping::mapping): Ditto. (mdspan::mdspan): Ditto. (mdspan::operator[]): Ditto. * testsuite/23_containers/mdspan/extents/custom_integer.cc: Add test for MutatingInt. * testsuite/23_containers/mdspan/int_like.h (MutatingInt): Add. * testsuite/23_containers/mdspan/layouts/mapping.cc: Add test for MutatingInt. * testsuite/23_containers/mdspan/layouts/stride.cc: Ditto. * testsuite/23_containers/mdspan/mdspan.cc: Ditto. Signed-off-by: Luc Grosheintz <luc.groshei...@gmail.com> --- libstdc++-v3/include/std/mdspan | 34 +++++++++++-------- .../mdspan/extents/custom_integer.cc | 1 + .../testsuite/23_containers/mdspan/int_like.h | 7 ++++ .../23_containers/mdspan/layouts/mapping.cc | 3 ++ .../23_containers/mdspan/layouts/stride.cc | 1 + .../testsuite/23_containers/mdspan/mdspan.cc | 2 ++ 6 files changed, 34 insertions(+), 14 deletions(-) diff --git a/libstdc++-v3/include/std/mdspan b/libstdc++-v3/include/std/mdspan index b34116a85e6..930997e9536 100644 --- a/libstdc++-v3/include/std/mdspan +++ b/libstdc++-v3/include/std/mdspan @@ -288,15 +288,17 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION initializer_list{_S_storage::_S_int_cast(__exts)...})) { } - template<__mdspan::__valid_index_type<index_type> _OIndexType, size_t _Nm> - requires (_Nm == rank() || _Nm == rank_dynamic()) + template<typename _OIndexType, size_t _Nm> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> + && (_Nm == rank() || _Nm == rank_dynamic()) constexpr explicit(_Nm != rank_dynamic()) extents(span<_OIndexType, _Nm> __exts) noexcept : _M_exts(span<const _OIndexType, _Nm>(__exts)) { } - template<__mdspan::__valid_index_type<index_type> _OIndexType, size_t _Nm> - requires (_Nm == rank() || _Nm == rank_dynamic()) + template<typename _OIndexType, size_t _Nm> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> + && (_Nm == rank() || _Nm == rank_dynamic()) constexpr explicit(_Nm != rank_dynamic()) extents(const array<_OIndexType, _Nm>& __exts) noexcept : _M_exts(span<const _OIndexType, _Nm>(__exts)) @@ -878,7 +880,8 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION constexpr mapping(const mapping&) noexcept = default; - template<__mdspan::__valid_index_type<index_type> _OIndexType> + template<typename _OIndexType> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> constexpr mapping(const extents_type& __exts, span<_OIndexType, extents_type::rank()> __strides) noexcept @@ -888,7 +891,8 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION _M_strides[__i] = index_type(as_const(__strides[__i])); } - template<__mdspan::__valid_index_type<index_type> _OIndexType> + template<typename _OIndexType> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> constexpr mapping(const extents_type& __exts, const array<_OIndexType, extents_type::rank()>& __strides) @@ -1134,9 +1138,9 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION _M_handle(std::move(__handle)) { } - template<__mdspan::__valid_index_type<index_type> _OIndexType, - size_t _Nm> - requires (_Nm == rank() || _Nm == rank_dynamic()) + template<typename _OIndexType, size_t _Nm> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> + && (_Nm == rank() || _Nm == rank_dynamic()) && is_constructible_v<mapping_type, extents_type> && is_default_constructible_v<accessor_type> constexpr explicit(_Nm != rank_dynamic()) @@ -1145,9 +1149,9 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION _M_handle(std::move(__handle)) { } - template<__mdspan::__valid_index_type<index_type> _OIndexType, - size_t _Nm> - requires (_Nm == rank() || _Nm == rank_dynamic()) + template<typename _OIndexType, size_t _Nm> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> + && (_Nm == rank() || _Nm == rank_dynamic()) && is_constructible_v<mapping_type, extents_type> && is_default_constructible_v<accessor_type> constexpr explicit(_Nm != rank_dynamic()) @@ -1218,7 +1222,8 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION return _M_accessor.access(_M_handle, __index); } - template<__mdspan::__valid_index_type<index_type> _OIndexType> + template<typename _OIndexType> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> constexpr reference operator[](span<_OIndexType, rank()> __indices) const { @@ -1228,7 +1233,8 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION return __call(make_index_sequence<rank()>()); } - template<__mdspan::__valid_index_type<index_type> _OIndexType> + template<typename _OIndexType> + requires __mdspan::__valid_index_type<const _OIndexType&, index_type> constexpr reference operator[](const array<_OIndexType, rank()>& __indices) const { return (*this)[span<const _OIndexType, rank()>(__indices)]; } diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/extents/custom_integer.cc b/libstdc++-v3/testsuite/23_containers/mdspan/extents/custom_integer.cc index 4f631815b10..92c2ebb46e1 100644 --- a/libstdc++-v3/testsuite/23_containers/mdspan/extents/custom_integer.cc +++ b/libstdc++-v3/testsuite/23_containers/mdspan/extents/custom_integer.cc @@ -84,6 +84,7 @@ main() test_shape_all<int, true>(); test_shape_all<IntLike, true>(); test_shape_all<ThrowingInt, false>(); + test_shape_all<MutatingInt, false>(); test_pack_all<int, true>(); test_pack_all<IntLike, true>(); diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/int_like.h b/libstdc++-v3/testsuite/23_containers/mdspan/int_like.h index ed45375f986..f4f4a773193 100644 --- a/libstdc++-v3/testsuite/23_containers/mdspan/int_like.h +++ b/libstdc++-v3/testsuite/23_containers/mdspan/int_like.h @@ -5,6 +5,7 @@ enum class CustomIndexKind { Const, Throwing, + Mutating, }; template<CustomIndexKind Kind> @@ -36,12 +37,18 @@ template<CustomIndexKind Kind> requires (Kind == CustomIndexKind::Throwing) { return _M_i; } + constexpr + operator int() noexcept + requires (Kind == CustomIndexKind::Mutating) + { return _M_i; } + private: int _M_i; }; using IntLike = CustomIndexType<CustomIndexKind::Const>; using ThrowingInt = CustomIndexType<CustomIndexKind::Throwing>; +using MutatingInt = CustomIndexType<CustomIndexKind::Mutating>; struct NotIntLike { }; diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/layouts/mapping.cc b/libstdc++-v3/testsuite/23_containers/mdspan/layouts/mapping.cc index 17f0c00acf2..6742fa11a51 100644 --- a/libstdc++-v3/testsuite/23_containers/mdspan/layouts/mapping.cc +++ b/libstdc++-v3/testsuite/23_containers/mdspan/layouts/mapping.cc @@ -180,6 +180,7 @@ template<typename Layout> static_assert(has_linear_index<Mapping, int>); static_assert(has_linear_index<Mapping, double>); static_assert(has_linear_index<Mapping, IntLike>); + static_assert(has_linear_index<Mapping, MutatingInt>); static_assert(!has_linear_index<Mapping, ThrowingInt>); static_assert(!has_linear_index<Mapping, NotIntLike>); static_assert(!has_linear_index<Mapping, int, int>); @@ -194,6 +195,7 @@ template<typename Layout> static_assert(has_linear_index<Mapping, int, int>); static_assert(has_linear_index<Mapping, double, double>); static_assert(has_linear_index<Mapping, IntLike, int>); + static_assert(has_linear_index<Mapping, MutatingInt, int>); static_assert(!has_linear_index<Mapping, ThrowingInt, int>); static_assert(!has_linear_index<Mapping, NotIntLike, int>); static_assert(!has_linear_index<Mapping, int, int, int>); @@ -524,6 +526,7 @@ template<typename Layout> if !consteval { test_linear_index_all<Layout, IntLike>(); + test_linear_index_all<Layout, MutatingInt>(); } test_required_span_size_all<Layout>(); diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/layouts/stride.cc b/libstdc++-v3/testsuite/23_containers/mdspan/layouts/stride.cc index 1267306fb5c..8d2fad2936f 100644 --- a/libstdc++-v3/testsuite/23_containers/mdspan/layouts/stride.cc +++ b/libstdc++-v3/testsuite/23_containers/mdspan/layouts/stride.cc @@ -65,6 +65,7 @@ test_stride_constructible_all() test_stride_constructible<E0, E0, int, 0, true>(); test_stride_constructible<E0, E0, IntLike, 0, true>(); test_stride_constructible<E0, E0, ThrowingInt, 0, false>(); + test_stride_constructible<E0, E0, MutatingInt, 0, false>(); test_stride_constructible<E0, E0, NotIntLike, 0, false>(); test_stride_constructible<E1, E1, int, 1, true>(); test_stride_constructible<E2, E1, int, 1, true>(); diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/mdspan.cc b/libstdc++-v3/testsuite/23_containers/mdspan/mdspan.cc index adabb0c6639..22ec68ea2d1 100644 --- a/libstdc++-v3/testsuite/23_containers/mdspan/mdspan.cc +++ b/libstdc++-v3/testsuite/23_containers/mdspan/mdspan.cc @@ -693,6 +693,7 @@ main() static_assert(test_from_int_like<int, true, true>()); test_from_int_like<IntLike, true, true>(); test_from_int_like<ThrowingInt, false, false>(); + test_from_int_like<MutatingInt, true, false>(); test_from_opaque_accessor(); test_from_base_class_accessor(); @@ -703,6 +704,7 @@ main() static_assert(test_access<int, true, true>()); test_access<IntLike, true, true>(); test_access<ThrowingInt, false, false>(); + test_access<MutatingInt, true, false>(); test_swap(); static_assert(test_swap()); -- 2.50.0