On Mon, Aug 4, 2025 at 1:03 PM Luc Grosheintz <luc.groshei...@gmail.com> wrote:
> This commit completes the implementation of P2897R7 by implementing and > testing the template class aligned_accessor. > > PR libstdc++/120994 > > libstdc++-v3/ChangeLog: > > * include/bits/version.def (aligned_accessor): Add. > * include/bits/version.h: Regenerate. > * include/std/mdspan (aligned_accessor): New class. > * src/c++23/std.cc.in (aligned_accessor): Add. > * testsuite/23_containers/mdspan/accessors/generic.cc: Add tests > for aligned_accessor. > * testsuite/23_containers/mdspan/accessors/aligned_neg.cc: New > test. > * testsuite/23_containers/mdspan/version.cc: Add test for > __cpp_lib_aligned_accessor. > > Signed-off-by: Luc Grosheintz <luc.groshei...@gmail.com> > --- > This is also LGTM, thanks for the update. I have added it to my local branch, and am running a test suite now. I will respond on PATCH 0 once the test will be done. > libstdc++-v3/include/bits/version.def | 10 ++++ > libstdc++-v3/include/bits/version.h | 10 ++++ > libstdc++-v3/include/std/mdspan | 54 +++++++++++++++++++ > libstdc++-v3/src/c++23/std.cc.in | 9 ++-- > .../mdspan/accessors/aligned_neg.cc | 33 ++++++++++++ > .../accessors/debug/aligned_access_neg.cc | 23 ++++++++ > .../accessors/debug/aligned_offset_neg.cc | 23 ++++++++ > .../23_containers/mdspan/accessors/generic.cc | 31 +++++++++++ > .../testsuite/23_containers/mdspan/version.cc | 8 +++ > 9 files changed, 198 insertions(+), 3 deletions(-) > create mode 100644 > libstdc++-v3/testsuite/23_containers/mdspan/accessors/aligned_neg.cc > create mode 100644 > libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_access_neg.cc > create mode 100644 > libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_offset_neg.cc > > diff --git a/libstdc++-v3/include/bits/version.def > b/libstdc++-v3/include/bits/version.def > index 56ad9ee9d4b..9d5122fed17 100644 > --- a/libstdc++-v3/include/bits/version.def > +++ b/libstdc++-v3/include/bits/version.def > @@ -1025,6 +1025,16 @@ ftms = { > }; > }; > > +ftms = { > + name = aligned_accessor; > + values = { > + v = 202411; > + cxxmin = 26; > + extra_cond = "__glibcxx_assume_aligned " > + "&& __glibcxx_is_sufficiently_aligned"; > + }; > +}; > + > ftms = { > name = ssize; > values = { > diff --git a/libstdc++-v3/include/bits/version.h > b/libstdc++-v3/include/bits/version.h > index 51805d292f0..6feeaa4d15a 100644 > --- a/libstdc++-v3/include/bits/version.h > +++ b/libstdc++-v3/include/bits/version.h > @@ -1149,6 +1149,16 @@ > #endif /* !defined(__cpp_lib_mdspan) && defined(__glibcxx_want_mdspan) */ > #undef __glibcxx_want_mdspan > > +#if !defined(__cpp_lib_aligned_accessor) > +# if (__cplusplus > 202302L) && (__glibcxx_assume_aligned && > __glibcxx_is_sufficiently_aligned) > +# define __glibcxx_aligned_accessor 202411L > +# if defined(__glibcxx_want_all) || > defined(__glibcxx_want_aligned_accessor) > +# define __cpp_lib_aligned_accessor 202411L > +# endif > +# endif > +#endif /* !defined(__cpp_lib_aligned_accessor) && > defined(__glibcxx_want_aligned_accessor) */ > +#undef __glibcxx_want_aligned_accessor > + > #if !defined(__cpp_lib_ssize) > # if (__cplusplus >= 202002L) > # define __glibcxx_ssize 201902L > diff --git a/libstdc++-v3/include/std/mdspan > b/libstdc++-v3/include/std/mdspan > index 691d4836262..9ac283347ce 100644 > --- a/libstdc++-v3/include/std/mdspan > +++ b/libstdc++-v3/include/std/mdspan > @@ -39,7 +39,12 @@ > #include <limits> > #include <utility> > > +#if __cplusplus > 202302L > +#include <bits/align.h> > +#endif > + > #define __glibcxx_want_mdspan > +#define __glibcxx_want_aligned_accessor > #include <bits/version.h> > > #ifdef __glibcxx_mdspan > @@ -1055,6 +1060,55 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION > { return __p + __i; } > }; > > +#ifdef __glibcxx_aligned_accessor > + template<typename _ElementType, size_t _ByteAlignment> > + struct aligned_accessor > + { > + static_assert(has_single_bit(_ByteAlignment), > + "ByteAlignment must be a power of two"); > + static_assert(_ByteAlignment >= alignof(_ElementType), > + "ByteAlignment is too small for ElementType"); > + > + using offset_policy = default_accessor<_ElementType>; > + using element_type = _ElementType; > + using reference = element_type&; > + using data_handle_type = element_type*; > + > + static constexpr size_t byte_alignment = _ByteAlignment; > + > + constexpr > + aligned_accessor() noexcept = default; > + > + template<typename _OElementType, size_t _OByteAlignment> > + requires (is_convertible_v<_OElementType(*)[], element_type(*)[]> > + && _OByteAlignment >= byte_alignment) > + constexpr > + aligned_accessor(aligned_accessor<_OElementType, _OByteAlignment>) > + noexcept > + { } > + > + template<typename _OElementType> > + requires is_convertible_v<_OElementType(*)[], element_type(*)[]> > + constexpr explicit > + aligned_accessor(default_accessor<_OElementType>) noexcept > + { } > + > + template<typename _OElementType> > + requires is_convertible_v<element_type(*)[], _OElementType(*)[]> > + constexpr > + operator default_accessor<_OElementType>() const noexcept > + { return {}; } > + > + constexpr reference > + access(data_handle_type __p, size_t __i) const noexcept > + { return std::assume_aligned<byte_alignment>(__p)[__i]; } > + > + constexpr typename offset_policy::data_handle_type > + offset(data_handle_type __p, size_t __i) const noexcept > + { return std::assume_aligned<byte_alignment>(__p) + __i; } > + }; > +#endif > + > namespace __mdspan > { > template<typename _Extents, typename _IndexType, size_t _Nm> > diff --git a/libstdc++-v3/src/c++23/std.cc.in b/libstdc++-v3/src/c++23/ > std.cc.in > index 24411476ba0..40d2fa03c5a 100644 > --- a/libstdc++-v3/src/c++23/std.cc.in > +++ b/libstdc++-v3/src/c++23/std.cc.in > @@ -1858,10 +1858,13 @@ export namespace std > using std::layout_right; > using std::layout_stride; > using std::default_accessor; > +#if __glibcxx_aligned_accessor > + using std::aligned_accessor; > +#endif > using std::mdspan; > - // FIXME layout_left_padded, layout_right_padded, aligned_accessor, > - // strided_slice, submdspan_mapping_result, full_extent_t, full_extent, > - // submdspan_extents, mdsubspan > + // FIXME layout_left_padded, layout_right_padded, strided_slice, > + // submdspan_mapping_result, full_extent_t, full_extent, > submdspan_extents, > + // mdsubspan > } > #endif > > diff --git > a/libstdc++-v3/testsuite/23_containers/mdspan/accessors/aligned_neg.cc > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/aligned_neg.cc > new file mode 100644 > index 00000000000..46d80bf4083 > --- /dev/null > +++ b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/aligned_neg.cc > @@ -0,0 +1,33 @@ > +// { dg-do compile { target c++26 } } > +#include<mdspan> > + > +#include <cstdint> > + > +std::aligned_accessor<uint32_t, 0> a; // { dg-error "required > from here" } > +std::aligned_accessor<uint32_t, 7> b; // { dg-error "required > from here" } > +std::aligned_accessor<uint32_t, size_t(-1)> c; // { dg-error "required > from here" } > + > +std::aligned_accessor<uint32_t, 2> d; // { dg-error "required > from here" } > + > +std::aligned_accessor<int[2], 32> e; // { dg-error "required > from here" } > + > +class Abstract > +{ > + virtual void > + foo() const = 0; > +}; > + > +class Derived : public Abstract > +{ > + void > + foo() const override > + { } > +}; > + > +std::aligned_accessor<Derived, alignof(int)> f_ok; > +std::aligned_accessor<Abstract, alignof(int)> f_err; // { dg-error > "required from here" } > + > +// { dg-prune-output "ByteAlignment must be a power of two" } > +// { dg-prune-output "ByteAlignment is too small for ElementType" } > +// { dg-prune-output "ElementType must not be an array type" } > +// { dg-prune-output "ElementType must not be an abstract" } > diff --git > a/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_access_neg.cc > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_access_neg.cc > new file mode 100644 > index 00000000000..3511cef1c3a > --- /dev/null > +++ > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_access_neg.cc > @@ -0,0 +1,23 @@ > +// { dg-do run { target c++26 xfail *-*-* } } > +// { dg-require-debug-mode "" } > + > +#include <mdspan> > +#include <array> > + > +void > +test_unaligned_access() > +{ > + constexpr size_t N = 4; > + alignas(N) std::array<char, 128> buffer{}; > + auto* unaligned = buffer.data() + 1; > + auto a = std::aligned_accessor<char, N>{}; > + > + [[maybe_unused]] char x = a.access(unaligned, 0); > +} > + > +int > +main() > +{ > + test_unaligned_access(); > + return 0; > +}; > diff --git > a/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_offset_neg.cc > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_offset_neg.cc > new file mode 100644 > index 00000000000..319da5ffef3 > --- /dev/null > +++ > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/debug/aligned_offset_neg.cc > @@ -0,0 +1,23 @@ > +// { dg-do run { target c++26 xfail *-*-* } } > +// { dg-require-debug-mode "" } > + > +#include <mdspan> > +#include <array> > + > +void > +test_unaligned_offset() > +{ > + constexpr size_t N = 4; > + alignas(N) std::array<char, 128> buffer{}; > + auto* unaligned = buffer.data() + 1; > + auto a = std::aligned_accessor<char, N>{}; > + > + [[maybe_unused]] char* x = a.offset(unaligned, 0); > +} > + > +int > +main() > +{ > + test_unaligned_offset(); > + return 0; > +}; > diff --git > a/libstdc++-v3/testsuite/23_containers/mdspan/accessors/generic.cc > b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/generic.cc > index 66009ad89c0..31cb13ed431 100644 > --- a/libstdc++-v3/testsuite/23_containers/mdspan/accessors/generic.cc > +++ b/libstdc++-v3/testsuite/23_containers/mdspan/accessors/generic.cc > @@ -97,10 +97,36 @@ template<template<typename T> typename Accessor> > > static_assert(test_properties<std::default_accessor>()); > > +#ifdef __glibcxx_aligned_accessor > +template<size_t Mult> > +struct OverAlignedAccessorTrait > +{ > + template<typename T> > + using type = std::aligned_accessor<T, Mult*alignof(T)>; > +}; > + > +static_assert(test_properties<OverAlignedAccessorTrait<1>::type>()); > +static_assert(test_properties<OverAlignedAccessorTrait<2>::type>()); > +static_assert(test_ctor<OverAlignedAccessorTrait<2>::type, > + std::default_accessor, false>()); > +static_assert(test_ctor<OverAlignedAccessorTrait<2>::type, > + OverAlignedAccessorTrait<4>::type>()); > +static_assert(test_ctor<std::default_accessor, > + OverAlignedAccessorTrait<2>::type>()); > +static_assert(!std::is_constructible_v<std::aligned_accessor<char, 4>, > + std::aligned_accessor<char, 2>>); > +#endif > + > template<typename A> > constexpr size_t > accessor_alignment = alignof(typename A::element_type); > > +#ifdef __glibcxx_aligned_accessor > +template<typename T, size_t N> > + constexpr size_t > + accessor_alignment<std::aligned_accessor<T, N>> = N; > +#endif > + > template<typename Accessor> > constexpr void > test_access(Accessor accessor) > @@ -136,5 +162,10 @@ main() > { > test_all<std::default_accessor<double>>(); > static_assert(test_all<std::default_accessor<double>>()); > + > +#ifdef __glibcxx_aligned_accessor > + test_all<typename OverAlignedAccessorTrait<4>::type<double>>(); > + static_assert(test_all<typename > OverAlignedAccessorTrait<4>::type<double>>()); > +#endif > return 0; > } > diff --git a/libstdc++-v3/testsuite/23_containers/mdspan/version.cc > b/libstdc++-v3/testsuite/23_containers/mdspan/version.cc > index 752060262a0..18826005971 100644 > --- a/libstdc++-v3/testsuite/23_containers/mdspan/version.cc > +++ b/libstdc++-v3/testsuite/23_containers/mdspan/version.cc > @@ -10,3 +10,11 @@ > #elif __cplusplus > 202302L && __cpp_lib_mdspan != 202406L > #error "Feature test macro __cpp_lib_mdspan has the wrong value for C++26" > #endif > + > +#if __cplusplus > 202302L > +#ifndef __cpp_lib_aligned_accessor > +#error "Feature test macro __cpp_lib_aligned_accessor is missing for > <mdspan>" > +#elif __cpp_lib_aligned_accessor != 202411L > +#error "Feature test macro __cpp_lib_aligned_accessor has the wrong value" > +#endif > +#endif > -- > 2.50.0 > >