Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/contrib/llvm-project/libcxx/include/__mdspan/layout_stride.h
35262 views
1
// -*- C++ -*-
2
//===----------------------------------------------------------------------===//
3
//
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
7
//
8
// Kokkos v. 4.0
9
// Copyright (2022) National Technology & Engineering
10
// Solutions of Sandia, LLC (NTESS).
11
//
12
// Under the terms of Contract DE-NA0003525 with NTESS,
13
// the U.S. Government retains certain rights in this software.
14
//
15
//===---------------------------------------------------------------------===//
16
17
#ifndef _LIBCPP___MDSPAN_LAYOUT_STRIDE_H
18
#define _LIBCPP___MDSPAN_LAYOUT_STRIDE_H
19
20
#include <__assert>
21
#include <__config>
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>
30
#include <array>
31
#include <cinttypes>
32
#include <cstddef>
33
#include <limits>
34
35
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
36
# pragma GCC system_header
37
#endif
38
39
_LIBCPP_PUSH_MACROS
40
#include <__undef_macros>
41
42
_LIBCPP_BEGIN_NAMESPACE_STD
43
44
#if _LIBCPP_STD_VER >= 23
45
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>;
50
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;
61
};
62
} // namespace __mdspan_detail
63
64
template <class _Extents>
65
class layout_stride::mapping {
66
public:
67
static_assert(__mdspan_detail::__is_extents<_Extents>::value,
68
"layout_stride::mapping template argument must be a specialization of extents.");
69
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;
75
76
private:
77
static constexpr rank_type __rank_ = extents_type::rank();
78
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)
82
return true;
83
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);
87
if (__overflowed)
88
return false;
89
}
90
return true;
91
}
92
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)
97
return true;
98
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()))
105
return false;
106
}
107
if (__ext.extent(__r) == static_cast<index_type>(0))
108
return true;
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)
112
return false;
113
bool __overflowed_add = __builtin_add_overflow(__size, __prod, &__size);
114
if (__overflowed_add)
115
return false;
116
}
117
return true;
118
}
119
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);
127
} else {
128
return [&]<size_t... _Pos>(index_sequence<_Pos...>) {
129
return static_cast<index_type>(__mapping((_Pos ? 0 : 0)...));
130
}(make_index_sequence<__rank_>());
131
}
132
}
133
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]);
141
} else {
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]);
147
}
148
}
149
}
150
}
151
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.");
154
155
public:
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);
165
}
166
__strides_[0] = __stride;
167
}
168
}
169
170
_LIBCPP_HIDE_FROM_ABI constexpr mapping(const mapping&) noexcept = default;
171
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);
185
} else {
186
return ((static_cast<index_type>(__strides[_Pos]) > static_cast<index_type>(0)) && ... && true);
187
}
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);
199
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]))
204
return false;
205
return true;
206
}(make_index_sequence<__rank_>())),
207
"layout_stride::mapping ctor: the provided extents and strides lead to a non-unique mapping");
208
}
209
}
210
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)) {}
217
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))...};
233
} else {
234
return __mdspan_detail::__possibly_empty_array<index_type, 0>{};
235
}
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");
244
}
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.");
250
}
251
252
_LIBCPP_HIDE_FROM_ABI constexpr mapping& operator=(const mapping&) noexcept = default;
253
254
// [mdspan.layout.stride.obs], observers
255
_LIBCPP_HIDE_FROM_ABI constexpr const extents_type& extents() const noexcept { return __extents_; }
256
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_>());
261
}
262
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);
266
} else {
267
return [&]<size_t... _Pos>(index_sequence<_Pos...>) {
268
if ((__extents_.extent(_Pos) * ... * 1) == 0)
269
return static_cast<index_type>(0);
270
else
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_>());
276
}
277
}
278
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)>());
292
}
293
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; }
297
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
300
// extents are zero.
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)
305
return true;
306
else {
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;
311
else {
312
rank_type __r_largest = 0;
313
for (rank_type __r = 1; __r < __rank_; __r++)
314
if (__strides_[__r] > __strides_[__r_largest])
315
__r_largest = __r;
316
for (rank_type __r = 0; __r < __rank_; __r++)
317
if (__extents_.extent(__r) == 0 && __r != __r_largest)
318
return false;
319
return true;
320
}
321
} else {
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_>());
325
}
326
}
327
}
328
_LIBCPP_HIDE_FROM_ABI static constexpr bool is_strided() noexcept { return true; }
329
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];
335
}
336
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 {
341
if (__offset(__rhs))
342
return false;
343
if constexpr (__rank_ == 0)
344
return true;
345
else {
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))) && ... &&
350
true);
351
}(make_index_sequence<__rank_>());
352
}
353
}
354
355
private:
356
_LIBCPP_NO_UNIQUE_ADDRESS extents_type __extents_{};
357
_LIBCPP_NO_UNIQUE_ADDRESS __mdspan_detail::__possibly_empty_array<index_type, __rank_> __strides_{};
358
};
359
360
#endif // _LIBCPP_STD_VER >= 23
361
362
_LIBCPP_END_NAMESPACE_STD
363
364
_LIBCPP_POP_MACROS
365
366
#endif // _LIBCPP___MDSPAN_LAYOUT_STRIDE_H
367
368