2 //===----------------------------------------------------------------------===//
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
9 // Copyright (2022) National Technology & Engineering
10 // Solutions of Sandia, LLC (NTESS).
12 // Under the terms of Contract DE-NA0003525 with NTESS,
13 // the U.S. Government retains certain rights in this software.
15 //===---------------------------------------------------------------------===//
17 #ifndef _LIBCPP___MDSPAN_LAYOUT_STRIDE_H
18 #define _LIBCPP___MDSPAN_LAYOUT_STRIDE_H
22 #include <__fwd/mdspan.h>
23 #include <__mdspan/extents.h>
24 #include <__type_traits/is_constructible.h>
25 #include <__type_traits/is_convertible.h>
26 #include <__type_traits/is_nothrow_constructible.h>
27 #include <__utility/as_const.h>
28 #include <__utility/integer_sequence.h>
29 #include <__utility/swap.h>
35 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
36 # pragma GCC system_header
40 #include <__undef_macros>
42 _LIBCPP_BEGIN_NAMESPACE_STD
44 #if _LIBCPP_STD_VER >= 23
46 namespace __mdspan_detail
{
47 template <class _Layout
, class _Mapping
>
48 constexpr bool __is_mapping_of
=
49 is_same_v
<typename
_Layout::template mapping
<typename
_Mapping::extents_type
>, _Mapping
>;
51 template <class _Mapping
>
52 concept __layout_mapping_alike
= requires
{
53 requires __is_mapping_of
<typename
_Mapping::layout_type
, _Mapping
>;
54 requires __is_extents_v
<typename
_Mapping::extents_type
>;
55 { _Mapping::is_always_strided() } -> same_as
<bool>;
56 { _Mapping::is_always_exhaustive() } -> same_as
<bool>;
57 { _Mapping::is_always_unique() } -> same_as
<bool>;
58 bool_constant
<_Mapping::is_always_strided()>::value
;
59 bool_constant
<_Mapping::is_always_exhaustive()>::value
;
60 bool_constant
<_Mapping::is_always_unique()>::value
;
62 } // namespace __mdspan_detail
64 template <class _Extents
>
65 class layout_stride::mapping
{
67 static_assert(__mdspan_detail::__is_extents
<_Extents
>::value
,
68 "layout_stride::mapping template argument must be a specialization of extents.");
70 using extents_type
= _Extents
;
71 using index_type
= typename
extents_type::index_type
;
72 using size_type
= typename
extents_type::size_type
;
73 using rank_type
= typename
extents_type::rank_type
;
74 using layout_type
= layout_stride
;
77 static constexpr rank_type __rank_
= extents_type::rank();
79 // Used for default construction check and mandates
80 _LIBCPP_HIDE_FROM_ABI
static constexpr bool __required_span_size_is_representable(const extents_type
& __ext
) {
81 if constexpr (__rank_
== 0)
84 index_type __prod
= __ext
.extent(0);
85 for (rank_type __r
= 1; __r
< __rank_
; __r
++) {
86 bool __overflowed
= __builtin_mul_overflow(__prod
, __ext
.extent(__r
), &__prod
);
93 template <class _OtherIndexType
>
94 _LIBCPP_HIDE_FROM_ABI
static constexpr bool
95 __required_span_size_is_representable(const extents_type
& __ext
, span
<_OtherIndexType
, __rank_
> __strides
) {
96 if constexpr (__rank_
== 0)
99 index_type __size
= 1;
100 for (rank_type __r
= 0; __r
< __rank_
; __r
++) {
101 // We can only check correct conversion of _OtherIndexType if it is an integral
102 if constexpr (is_integral_v
<_OtherIndexType
>) {
103 using _CommonType
= common_type_t
<index_type
, _OtherIndexType
>;
104 if (static_cast<_CommonType
>(__strides
[__r
]) > static_cast<_CommonType
>(numeric_limits
<index_type
>::max()))
107 if (__ext
.extent(__r
) == static_cast<index_type
>(0))
109 index_type __prod
= (__ext
.extent(__r
) - 1);
110 bool __overflowed_mul
= __builtin_mul_overflow(__prod
, static_cast<index_type
>(__strides
[__r
]), &__prod
);
111 if (__overflowed_mul
)
113 bool __overflowed_add
= __builtin_add_overflow(__size
, __prod
, &__size
);
114 if (__overflowed_add
)
120 // compute offset of a strided layout mapping
121 template <class _StridedMapping
>
122 _LIBCPP_HIDE_FROM_ABI
static constexpr index_type
__offset(const _StridedMapping
& __mapping
) {
123 if constexpr (_StridedMapping::extents_type::rank() == 0) {
124 return static_cast<index_type
>(__mapping());
125 } else if (__mapping
.required_span_size() == static_cast<typename
_StridedMapping::index_type
>(0)) {
126 return static_cast<index_type
>(0);
128 return [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
129 return static_cast<index_type
>(__mapping((_Pos
? 0 : 0)...));
130 }(make_index_sequence
<__rank_
>());
134 // compute the permutation for sorting the stride array
135 // we never actually sort the stride array
136 _LIBCPP_HIDE_FROM_ABI
constexpr void __bubble_sort_by_strides(array
<rank_type
, __rank_
>& __permute
) const {
137 for (rank_type __i
= __rank_
- 1; __i
> 0; __i
--) {
138 for (rank_type __r
= 0; __r
< __i
; __r
++) {
139 if (__strides_
[__permute
[__r
]] > __strides_
[__permute
[__r
+ 1]]) {
140 swap(__permute
[__r
], __permute
[__r
+ 1]);
142 // if two strides are the same then one of the associated extents must be 1 or 0
143 // both could be, but you can't have one larger than 1 come first
144 if ((__strides_
[__permute
[__r
]] == __strides_
[__permute
[__r
+ 1]]) &&
145 (__extents_
.extent(__permute
[__r
]) > static_cast<index_type
>(1)))
146 swap(__permute
[__r
], __permute
[__r
+ 1]);
152 static_assert((extents_type::rank_dynamic() > 0) || __required_span_size_is_representable(extents_type()),
153 "layout_stride::mapping product of static extents must be representable as index_type.");
156 // [mdspan.layout.stride.cons], constructors
157 _LIBCPP_HIDE_FROM_ABI
constexpr mapping() noexcept
: __extents_(extents_type()) {
158 // Note the nominal precondition is covered by above static assert since
159 // if rank_dynamic is != 0 required_span_size is zero for default construction
160 if constexpr (__rank_
> 0) {
161 index_type __stride
= 1;
162 for (rank_type __r
= __rank_
- 1; __r
> static_cast<rank_type
>(0); __r
--) {
163 __strides_
[__r
] = __stride
;
164 __stride
*= __extents_
.extent(__r
);
166 __strides_
[0] = __stride
;
170 _LIBCPP_HIDE_FROM_ABI
constexpr mapping(const mapping
&) noexcept
= default;
172 template <class _OtherIndexType
>
173 requires(is_convertible_v
<const _OtherIndexType
&, index_type
> &&
174 is_nothrow_constructible_v
<index_type
, const _OtherIndexType
&>)
175 _LIBCPP_HIDE_FROM_ABI
constexpr mapping(const extents_type
& __ext
, span
<_OtherIndexType
, __rank_
> __strides
) noexcept
176 : __extents_(__ext
), __strides_([&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
177 return __mdspan_detail::__possibly_empty_array
<index_type
, __rank_
>{
178 static_cast<index_type
>(std::as_const(__strides
[_Pos
]))...};
179 }(make_index_sequence
<__rank_
>())) {
180 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
181 ([&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
182 // For integrals we can do a pre-conversion check, for other types not
183 if constexpr (is_integral_v
<_OtherIndexType
>) {
184 return ((__strides
[_Pos
] > static_cast<_OtherIndexType
>(0)) && ... && true);
186 return ((static_cast<index_type
>(__strides
[_Pos
]) > static_cast<index_type
>(0)) && ... && true);
188 }(make_index_sequence
<__rank_
>())),
189 "layout_stride::mapping ctor: all strides must be greater than 0");
190 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
191 __required_span_size_is_representable(__ext
, __strides
),
192 "layout_stride::mapping ctor: required span size is not representable as index_type.");
193 if constexpr (__rank_
> 1) {
194 _LIBCPP_ASSERT_UNCATEGORIZED(
195 ([&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
196 // basically sort the dimensions based on strides and extents, sorting is represented in permute array
197 array
<rank_type
, __rank_
> __permute
{_Pos
...};
198 __bubble_sort_by_strides(__permute
);
200 // check that this permutations represents a growing set
201 for (rank_type __i
= 1; __i
< __rank_
; __i
++)
202 if (static_cast<index_type
>(__strides
[__permute
[__i
]]) <
203 static_cast<index_type
>(__strides
[__permute
[__i
- 1]]) * __extents_
.extent(__permute
[__i
- 1]))
206 }(make_index_sequence
<__rank_
>())),
207 "layout_stride::mapping ctor: the provided extents and strides lead to a non-unique mapping");
211 template <class _OtherIndexType
>
212 requires(is_convertible_v
<const _OtherIndexType
&, index_type
> &&
213 is_nothrow_constructible_v
<index_type
, const _OtherIndexType
&>)
214 _LIBCPP_HIDE_FROM_ABI
constexpr mapping(const extents_type
& __ext
,
215 const array
<_OtherIndexType
, __rank_
>& __strides
) noexcept
216 : mapping(__ext
, span(__strides
)) {}
218 template <class _StridedLayoutMapping
>
219 requires(__mdspan_detail::__layout_mapping_alike
<_StridedLayoutMapping
> &&
220 is_constructible_v
<extents_type
, typename
_StridedLayoutMapping::extents_type
> &&
221 _StridedLayoutMapping::is_always_unique() && _StridedLayoutMapping::is_always_strided())
222 _LIBCPP_HIDE_FROM_ABI
constexpr explicit(
223 !(is_convertible_v
<typename
_StridedLayoutMapping::extents_type
, extents_type
> &&
224 (__mdspan_detail::__is_mapping_of
<layout_left
, _StridedLayoutMapping
> ||
225 __mdspan_detail::__is_mapping_of
<layout_right
, _StridedLayoutMapping
> ||
226 __mdspan_detail::__is_mapping_of
<layout_stride
, _StridedLayoutMapping
>)))
227 mapping(const _StridedLayoutMapping
& __other
) noexcept
228 : __extents_(__other
.extents()), __strides_([&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
229 // stride() only compiles for rank > 0
230 if constexpr (__rank_
> 0) {
231 return __mdspan_detail::__possibly_empty_array
<index_type
, __rank_
>{
232 static_cast<index_type
>(__other
.stride(_Pos
))...};
234 return __mdspan_detail::__possibly_empty_array
<index_type
, 0>{};
236 }(make_index_sequence
<__rank_
>())) {
237 // stride() only compiles for rank > 0
238 if constexpr (__rank_
> 0) {
239 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
240 ([&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
241 return ((static_cast<index_type
>(__other
.stride(_Pos
)) > static_cast<index_type
>(0)) && ... && true);
242 }(make_index_sequence
<__rank_
>())),
243 "layout_stride::mapping converting ctor: all strides must be greater than 0");
245 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
246 __mdspan_detail::__is_representable_as
<index_type
>(__other
.required_span_size()),
247 "layout_stride::mapping converting ctor: other.required_span_size() must be representable as index_type.");
248 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(static_cast<index_type
>(0) == __offset(__other
),
249 "layout_stride::mapping converting ctor: base offset of mapping must be zero.");
252 _LIBCPP_HIDE_FROM_ABI
constexpr mapping
& operator=(const mapping
&) noexcept
= default;
254 // [mdspan.layout.stride.obs], observers
255 _LIBCPP_HIDE_FROM_ABI
constexpr const extents_type
& extents() const noexcept
{ return __extents_
; }
257 _LIBCPP_HIDE_FROM_ABI
constexpr array
<index_type
, __rank_
> strides() const noexcept
{
258 return [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
259 return array
<index_type
, __rank_
>{__strides_
[_Pos
]...};
260 }(make_index_sequence
<__rank_
>());
263 _LIBCPP_HIDE_FROM_ABI
constexpr index_type
required_span_size() const noexcept
{
264 if constexpr (__rank_
== 0) {
265 return static_cast<index_type
>(1);
267 return [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
268 if ((__extents_
.extent(_Pos
) * ... * 1) == 0)
269 return static_cast<index_type
>(0);
271 return static_cast<index_type
>(
272 static_cast<index_type
>(1) +
273 (((__extents_
.extent(_Pos
) - static_cast<index_type
>(1)) * __strides_
[_Pos
]) + ... +
274 static_cast<index_type
>(0)));
275 }(make_index_sequence
<__rank_
>());
279 template <class... _Indices
>
280 requires((sizeof...(_Indices
) == __rank_
) && (is_convertible_v
<_Indices
, index_type
> && ...) &&
281 (is_nothrow_constructible_v
<index_type
, _Indices
> && ...))
282 _LIBCPP_HIDE_FROM_ABI
constexpr index_type
operator()(_Indices
... __idx
) const noexcept
{
283 // Mappings are generally meant to be used for accessing allocations and are meant to guarantee to never
284 // return a value exceeding required_span_size(), which is used to know how large an allocation one needs
285 // Thus, this is a canonical point in multi-dimensional data structures to make invalid element access checks
286 // However, mdspan does check this on its own, so for now we avoid double checking in hardened mode
287 _LIBCPP_ASSERT_UNCATEGORIZED(__mdspan_detail::__is_multidimensional_index_in(__extents_
, __idx
...),
288 "layout_stride::mapping: out of bounds indexing");
289 return [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
290 return ((static_cast<index_type
>(__idx
) * __strides_
[_Pos
]) + ... + index_type(0));
291 }(make_index_sequence
<sizeof...(_Indices
)>());
294 _LIBCPP_HIDE_FROM_ABI
static constexpr bool is_always_unique() noexcept
{ return true; }
295 _LIBCPP_HIDE_FROM_ABI
static constexpr bool is_always_exhaustive() noexcept
{ return false; }
296 _LIBCPP_HIDE_FROM_ABI
static constexpr bool is_always_strided() noexcept
{ return true; }
298 _LIBCPP_HIDE_FROM_ABI
static constexpr bool is_unique() noexcept
{ return true; }
299 // The answer of this function is fairly complex in the case where one or more
301 // Technically it is meaningless to query is_exhaustive() in that case, but unfortunately
302 // the way the standard defines this function, we can't give a simple true or false then.
303 _LIBCPP_HIDE_FROM_ABI
constexpr bool is_exhaustive() const noexcept
{
304 if constexpr (__rank_
== 0)
307 index_type __span_size
= required_span_size();
308 if (__span_size
== static_cast<index_type
>(0)) {
309 if constexpr (__rank_
== 1)
310 return __strides_
[0] == 1;
312 rank_type __r_largest
= 0;
313 for (rank_type __r
= 1; __r
< __rank_
; __r
++)
314 if (__strides_
[__r
] > __strides_
[__r_largest
])
316 for (rank_type __r
= 0; __r
< __rank_
; __r
++)
317 if (__extents_
.extent(__r
) == 0 && __r
!= __r_largest
)
322 return required_span_size() == [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
323 return (__extents_
.extent(_Pos
) * ... * static_cast<index_type
>(1));
324 }(make_index_sequence
<__rank_
>());
328 _LIBCPP_HIDE_FROM_ABI
static constexpr bool is_strided() noexcept
{ return true; }
330 // according to the standard layout_stride does not have a constraint on stride(r) for rank>0
331 // it still has the precondition though
332 _LIBCPP_HIDE_FROM_ABI
constexpr index_type
stride(rank_type __r
) const noexcept
{
333 _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(__r
< __rank_
, "layout_stride::mapping::stride(): invalid rank index");
334 return __strides_
[__r
];
337 template <class _OtherMapping
>
338 requires(__mdspan_detail::__layout_mapping_alike
<_OtherMapping
> &&
339 (_OtherMapping::extents_type::rank() == __rank_
) && _OtherMapping::is_always_strided())
340 _LIBCPP_HIDE_FROM_ABI
friend constexpr bool operator==(const mapping
& __lhs
, const _OtherMapping
& __rhs
) noexcept
{
343 if constexpr (__rank_
== 0)
346 return __lhs
.extents() == __rhs
.extents() && [&]<size_t... _Pos
>(index_sequence
<_Pos
...>) {
347 // avoid warning when comparing signed and unsigner integers and pick the wider of two types
348 using _CommonType
= common_type_t
<index_type
, typename
_OtherMapping::index_type
>;
349 return ((static_cast<_CommonType
>(__lhs
.stride(_Pos
)) == static_cast<_CommonType
>(__rhs
.stride(_Pos
))) && ... &&
351 }(make_index_sequence
<__rank_
>());
356 _LIBCPP_NO_UNIQUE_ADDRESS extents_type __extents_
{};
357 _LIBCPP_NO_UNIQUE_ADDRESS
__mdspan_detail::__possibly_empty_array
<index_type
, __rank_
> __strides_
{};
360 #endif // _LIBCPP_STD_VER >= 23
362 _LIBCPP_END_NAMESPACE_STD
366 #endif // _LIBCPP___MDSPAN_LAYOUT_STRIDE_H