ttg 1.0.0
Template Task Graph (TTG): flowgraph-based programming model for high-performance distributed-memory algorithms
Loading...
Searching...
No Matches
span.h
Go to the documentation of this file.
1// SPDX-License-Identifier: BSD-3-Clause
2#ifndef TTG_UTIL_SPAN_H
3#define TTG_UTIL_SPAN_H
4
5/* if TCB_SPAN_NAMESPACE_NAME already defined someone imported TCB span somewhere,
6 * then reset it here to the ttg namespace, don't forget to revert back */
7#ifdef TCB_SPAN_NAMESPACE_NAME
8
9#define TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H TCB_SPAN_NAMESPACE_NAME
10#undef TCB_SPAN_NAMESPACE_NAME
11
12#endif
13
14/* Use the TTG namespace */
15#define TCB_SPAN_NAMESPACE_NAME ttg
16
17/*
18This is an implementation of C++20's std::span
19http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/n4820.pdf
20Code taken from https://github.com/tcbrindle/span/
21*/
22
23// Copyright Tristan Brindle 2018.
24// Distributed under the Boost Software License, Version 1.0.
25// (See accompanying file ../../LICENSE_1_0.txt or copy at
26// https://www.boost.org/LICENSE_1_0.txt)
27
28#include <array>
29#include <cstddef>
30#include <cstdint>
31#include <type_traits>
32#include <exception>
33
34#ifndef TCB_SPAN_NO_EXCEPTIONS
35// Attempt to discover whether we're being compiled with exception support
36#if !(defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND))
37#define TCB_SPAN_NO_EXCEPTIONS
38#endif
39#endif
40
41#ifndef TCB_SPAN_NO_EXCEPTIONS
42#include <cstdio>
43#include <stdexcept>
44#endif
45
46// Various feature test macros
47
48#ifndef TCB_SPAN_NAMESPACE_NAME
49#define TCB_SPAN_NAMESPACE_NAME tcb
50#endif
51
52#if __cplusplus >= 201703L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L)
53#define TCB_SPAN_HAVE_CPP17
54#endif
55
56#if __cplusplus >= 201402L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201402L)
57#define TCB_SPAN_HAVE_CPP14
58#endif
59
61
62// Establish default contract checking behavior
63#if !defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION) && !defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION) && \
64 !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
65#if defined(NDEBUG) || !defined(TCB_SPAN_HAVE_CPP14)
66#define TCB_SPAN_NO_CONTRACT_CHECKING
67#else
68#define TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION
69#endif
70#endif
71
72#if defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION)
73 struct contract_violation_error : std::logic_error {
74 explicit contract_violation_error(const char* msg) : std::logic_error(msg) {}
75 };
76
77 inline void contract_violation(const char* msg) { throw contract_violation_error(msg); }
78
79#elif defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION)
80 [[noreturn]] inline void contract_violation(const char* /*unused*/) { std::terminate(); }
81#endif
82
83#if !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
84#define TCB_SPAN_STRINGIFY(cond) #cond
85#define TCB_SPAN_EXPECT(cond) cond ? (void)0 : contract_violation("Expected " TCB_SPAN_STRINGIFY(cond))
86#else
87#define TCB_SPAN_EXPECT(cond)
88#endif
89
90#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_inline_variables)
91#define TCB_SPAN_INLINE_VAR inline
92#else
93#define TCB_SPAN_INLINE_VAR
94#endif
95
96#if defined(TCB_SPAN_HAVE_CPP14) || (defined(__cpp_constexpr) && __cpp_constexpr >= 201304)
97#define TCB_SPAN_HAVE_CPP14_CONSTEXPR
98#endif
99
100#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR)
101#define TCB_SPAN_CONSTEXPR14 constexpr
102#else
103#define TCB_SPAN_CONSTEXPR14
104#endif
105
106#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR) && (!defined(_MSC_VER) || _MSC_VER > 1900)
107#define TCB_SPAN_CONSTEXPR_ASSIGN constexpr
108#else
109#define TCB_SPAN_CONSTEXPR_ASSIGN
110#endif
111
112#if defined(TCB_SPAN_NO_CONTRACT_CHECKING)
113#define TCB_SPAN_CONSTEXPR11 constexpr
114#else
115#define TCB_SPAN_CONSTEXPR11 TCB_SPAN_CONSTEXPR14
116#endif
117
118#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_deduction_guides)
119#define TCB_SPAN_HAVE_DEDUCTION_GUIDES
120#endif
121
122#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_byte)
123#define TCB_SPAN_HAVE_STD_BYTE
124#endif
125
126#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_array_constexpr)
127#define TCB_SPAN_HAVE_CONSTEXPR_STD_ARRAY_ETC
128#endif
129
130#if defined(TCB_SPAN_HAVE_CONSTEXPR_STD_ARRAY_ETC)
131#define TCB_SPAN_ARRAY_CONSTEXPR constexpr
132#else
133#define TCB_SPAN_ARRAY_CONSTEXPR
134#endif
135
136#ifdef TCB_SPAN_HAVE_STD_BYTE
137 using byte = std::byte;
138#else
139 using byte = unsigned char;
140#endif
141
142#if defined(TCB_SPAN_HAVE_CPP17)
143#define TCB_SPAN_NODISCARD [[nodiscard]]
144#else
145#define TCB_SPAN_NODISCARD
146#endif
147
148 TCB_SPAN_INLINE_VAR constexpr std::size_t dynamic_extent = SIZE_MAX;
149
150 template <typename ElementType, std::size_t Extent = dynamic_extent>
151 class span;
152
153 namespace detail {
154
155 template <typename E, std::size_t S>
157 constexpr span_storage() noexcept = default;
158
159 constexpr span_storage(E* p_ptr, std::size_t /*unused*/) noexcept : ptr(p_ptr) {}
160
161 E* ptr = nullptr;
162 static constexpr std::size_t size = S;
163 };
164
165 template <typename E>
167 constexpr span_storage() noexcept = default;
168
169 constexpr span_storage(E* p_ptr, std::size_t p_size) noexcept : ptr(p_ptr), size(p_size) {}
170
171 E* ptr = nullptr;
172 std::size_t size = 0;
173 };
174
175// Reimplementation of C++17 std::size() and std::data()
176#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_nonmember_container_access)
177 using std::data;
178 using std::size;
179#else
180 template <class C>
181 constexpr auto size(const C& c) -> decltype(c.size()) {
182 return c.size();
183 }
184
185 template <class T, std::size_t N>
186 constexpr std::size_t size(const T (&)[N]) noexcept {
187 return N;
188 }
189
190 template <class C>
191 constexpr auto data(C& c) -> decltype(c.data()) {
192 return c.data();
193 }
194
195 template <class C>
196 constexpr auto data(const C& c) -> decltype(c.data()) {
197 return c.data();
198 }
199
200 template <class T, std::size_t N>
201 constexpr T* data(T (&array)[N]) noexcept {
202 return array;
203 }
204
205 template <class E>
206 constexpr const E* data(std::initializer_list<E> il) noexcept {
207 return il.begin();
208 }
209#endif // TCB_SPAN_HAVE_CPP17
210
211#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_void_t)
212 using std::void_t;
213#else
214 template <typename...>
215 using void_t = void;
216#endif
217
218 template <typename T>
219 using uncvref_t = typename std::remove_cv<typename std::remove_reference<T>::type>::type;
220
221 template <typename>
222 struct is_span : std::false_type {};
223
224 template <typename T, std::size_t S>
225 struct is_span<span<T, S> > : std::true_type {};
226
227 template <typename>
228 struct is_std_array : std::false_type {};
229
230 template <typename T, std::size_t N>
231 struct is_std_array<std::array<T, N> > : std::true_type {};
232
233 template <typename, typename = void>
234 struct has_size_and_data : std::false_type {};
235
236 template <typename T>
238 T, void_t<decltype(detail::size(std::declval<T>())), decltype(detail::data(std::declval<T>()))> >
239 : std::true_type {};
240
241 template <typename C, typename U = uncvref_t<C> >
243 static constexpr bool value =
244 !is_span<U>::value && !is_std_array<U>::value && !std::is_array<U>::value && has_size_and_data<C>::value;
245 };
246
247 template <typename T>
248 using remove_pointer_t = typename std::remove_pointer<T>::type;
249
250 template <typename, typename, typename = void>
251 struct is_container_element_type_compatible : std::false_type {};
252
253 template <typename T, typename E>
255 T, E,
256 typename std::enable_if<
257 !std::is_same<typename std::remove_cv<decltype(detail::data(std::declval<T>()))>::type, void>::value>::type>
258 : std::is_convertible<remove_pointer_t<decltype(detail::data(std::declval<T>()))> (*)[], E (*)[]> {};
259
260 template <typename, typename = size_t>
261 struct is_complete : std::false_type {};
262
263 template <typename T>
264 struct is_complete<T, decltype(sizeof(T))> : std::true_type {};
265
266 } // namespace detail
267
268 template <typename ElementType, std::size_t Extent>
269 class span {
270 static_assert(std::is_object<ElementType>::value,
271 "A span's ElementType must be an object type (not a "
272 "reference type or void)");
274 "A span's ElementType must be a complete type (not a forward "
275 "declaration)");
276 static_assert(!std::is_abstract<ElementType>::value, "A span's ElementType cannot be an abstract class type");
277
279
280 public:
281 // constants and types
282 using element_type = ElementType;
283 using value_type = typename std::remove_cv<ElementType>::type;
284 using size_type = std::size_t;
285 using difference_type = std::ptrdiff_t;
291 using reverse_iterator = std::reverse_iterator<iterator>;
292
293 static constexpr size_type extent = Extent;
294
295 // [span.cons], span constructors, copy, assignment, and destructor
296 template <std::size_t E = Extent, typename std::enable_if<(E == dynamic_extent || E <= 0), int>::type = 0>
297 constexpr span() noexcept {}
298
299 TCB_SPAN_CONSTEXPR11 span(pointer ptr, size_type count) : storage_(ptr, count) {
300 TCB_SPAN_EXPECT(extent == dynamic_extent || count == extent);
301 }
302
303 TCB_SPAN_CONSTEXPR11 span(pointer first_elem, pointer last_elem) : storage_(first_elem, last_elem - first_elem) {
304 TCB_SPAN_EXPECT(extent == dynamic_extent || last_elem - first_elem == static_cast<std::ptrdiff_t>(extent));
305 }
306
307 template <std::size_t N, std::size_t E = Extent,
308 typename std::enable_if<(E == dynamic_extent || N == E) && detail::is_container_element_type_compatible<
309 element_type (&)[N], ElementType>::value,
310 int>::type = 0>
311 constexpr span(element_type (&arr)[N]) noexcept : storage_(arr, N) {}
312
313 template <std::size_t N, std::size_t E = Extent,
314 typename std::enable_if<
315 (E == dynamic_extent || N == E) &&
316 detail::is_container_element_type_compatible<std::array<value_type, N>&, ElementType>::value,
317 int>::type = 0>
318 TCB_SPAN_ARRAY_CONSTEXPR span(std::array<value_type, N>& arr) noexcept : storage_(arr.data(), N) {}
319
320 template <std::size_t N, std::size_t E = Extent,
321 typename std::enable_if<(E == dynamic_extent || N == E) &&
322 detail::is_container_element_type_compatible<const std::array<value_type, N>&,
323 ElementType>::value,
324 int>::type = 0>
325 TCB_SPAN_ARRAY_CONSTEXPR span(const std::array<value_type, N>& arr) noexcept : storage_(arr.data(), N) {}
326
327 template <typename Container, std::size_t E = Extent,
328 typename std::enable_if<E == dynamic_extent && detail::is_container<Container>::value &&
330 int>::type = 0>
331 constexpr span(Container& cont) : storage_(detail::data(cont), detail::size(cont)) {}
332
333 template <
334 typename Container, std::size_t E = Extent,
335 typename std::enable_if<E == dynamic_extent && detail::is_container<Container>::value &&
337 int>::type = 0>
338 constexpr span(const Container& cont) : storage_(detail::data(cont), detail::size(cont)) {}
339
340 constexpr span(const span& other) noexcept = default;
341
342 template <typename OtherElementType, std::size_t OtherExtent,
343 typename std::enable_if<(Extent == OtherExtent || Extent == dynamic_extent) &&
344 std::is_convertible<OtherElementType (*)[], ElementType (*)[]>::value,
345 int>::type = 0>
346 constexpr span(const span<OtherElementType, OtherExtent>& other) noexcept : storage_(other.data(), other.size()) {}
347
348 ~span() noexcept = default;
349
350 TCB_SPAN_CONSTEXPR_ASSIGN span& operator=(const span& other) noexcept = default;
351
352 // [span.sub], span subviews
353 template <std::size_t Count>
354 TCB_SPAN_CONSTEXPR11 span<element_type, Count> first() const {
355 TCB_SPAN_EXPECT(Count <= size());
356 return {data(), Count};
357 }
358
359 template <std::size_t Count>
361 TCB_SPAN_EXPECT(Count <= size());
362 return {data() + (size() - Count), Count};
363 }
364
365 template <std::size_t Offset, std::size_t Count = dynamic_extent>
367 span<ElementType,
368 Count != dynamic_extent ? Count : (Extent != dynamic_extent ? Extent - Offset : dynamic_extent)>;
369
370 template <std::size_t Offset, std::size_t Count = dynamic_extent>
372 TCB_SPAN_EXPECT(Offset <= size() && (Count == dynamic_extent || Offset + Count <= size()));
373 return {data() + Offset, Count != dynamic_extent ? Count : size() - Offset};
374 }
375
377 TCB_SPAN_EXPECT(count <= size());
378 return {data(), count};
379 }
380
382 TCB_SPAN_EXPECT(count <= size());
383 return {data() + (size() - count), count};
384 }
385
387 size_type count = dynamic_extent) const {
388 TCB_SPAN_EXPECT(offset <= size() && (count == dynamic_extent || offset + count <= size()));
389 return {data() + offset, count == dynamic_extent ? size() - offset : count};
390 }
391
392 // [span.obs], span observers
393 constexpr size_type size() const noexcept { return storage_.size; }
394
395 constexpr size_type size_bytes() const noexcept { return size() * sizeof(element_type); }
396
397 TCB_SPAN_NODISCARD constexpr bool empty() const noexcept { return size() == 0; }
398
399 // [span.elem], span element access
401 TCB_SPAN_EXPECT(idx < size());
402 return *(data() + idx);
403 }
404
407 return *data();
408 }
409
412 return *(data() + (size() - 1));
413 }
414
415 constexpr pointer data() const noexcept { return storage_.ptr; }
416
417 // [span.iterators], span iterator support
418 constexpr iterator begin() const noexcept { return data(); }
419
420 constexpr iterator end() const noexcept { return data() + size(); }
421
423
425
426 private:
427 storage_type storage_{};
428 };
429
430#ifdef TCB_SPAN_HAVE_DEDUCTION_GUIDES
431
432 /* Deduction Guides */
433 template <class T, size_t N>
434 span(T (&)[N]) -> span<T, N>;
435
436 template <class T, size_t N>
437 span(std::array<T, N>&) -> span<T, N>;
438
439 template <class T, size_t N>
440 span(const std::array<T, N>&) -> span<const T, N>;
441
442 template <class Container>
443 span(Container&) -> span<typename Container::value_type>;
444
445 template <class Container>
446 span(const Container&) -> span<const typename Container::value_type>;
447
448#endif // TCB_HAVE_DEDUCTION_GUIDES
449
450 template <typename ElementType, std::size_t Extent>
452 return s;
453 }
454
455 template <typename T, std::size_t N>
456 constexpr span<T, N> make_span(T (&arr)[N]) noexcept {
457 return {arr};
458 }
459
460 template <typename T, std::size_t N>
461 TCB_SPAN_ARRAY_CONSTEXPR span<T, N> make_span(std::array<T, N>& arr) noexcept {
462 return {arr};
463 }
464
465 template <typename T, std::size_t N>
466 TCB_SPAN_ARRAY_CONSTEXPR span<const T, N> make_span(const std::array<T, N>& arr) noexcept {
467 return {arr};
468 }
469
470 template <typename Container>
472 return {cont};
473 }
474
475 template <typename Container>
476 constexpr span<const typename Container::value_type> make_span(const Container& cont) {
477 return {cont};
478 }
479
480 template <typename ElementType, std::size_t Extent>
481 span<const byte, ((Extent == dynamic_extent) ? dynamic_extent : sizeof(ElementType) * Extent)> as_bytes(
482 span<ElementType, Extent> s) noexcept {
483 return {reinterpret_cast<const byte*>(s.data()), s.size_bytes()};
484 }
485
486 template <class ElementType, size_t Extent,
487 typename std::enable_if<!std::is_const<ElementType>::value, int>::type = 0>
488 span<byte, ((Extent == dynamic_extent) ? dynamic_extent : sizeof(ElementType) * Extent)> as_writable_bytes(
489 span<ElementType, Extent> s) noexcept {
490 return {reinterpret_cast<byte*>(s.data()), s.size_bytes()};
491 }
492
493 template <std::size_t N, typename E, std::size_t S>
494 constexpr auto get(span<E, S> s) -> decltype(s[N]) {
495 return s[N];
496 }
497
498} // namespace TCB_SPAN_NAMESPACE_NAME
499
500namespace std {
501
502 template <typename ElementType, size_t Extent>
503 class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent> > : public integral_constant<size_t, Extent> {};
504
505 template <typename ElementType>
506 class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<ElementType, TCB_SPAN_NAMESPACE_NAME::dynamic_extent> >; // not
507 // defined
508
509 template <size_t I, typename ElementType, size_t Extent>
510 class tuple_element<I, TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent> > {
511 public:
512 static_assert(Extent != TCB_SPAN_NAMESPACE_NAME::dynamic_extent && I < Extent, "");
513 using type = ElementType;
514 };
515
516} // end namespace std
517
518/* revert TCB_SPAN_NAMESPACE_NAME to the original value, if needed */
519#ifdef TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
520
521#undef TCB_SPAN_NAMESPACE_NAME
522#define TCB_SPAN_NAMESPACE_NAME TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
523#undef TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
524
525#endif
526
527#endif // TTG_UTIL_SPAN_H
constexpr span(Container &cont)
Definition span.h:331
TCB_SPAN_CONSTEXPR11 span< element_type, Count > last() const
Definition span.h:360
TCB_SPAN_ARRAY_CONSTEXPR span(const std::array< value_type, N > &arr) noexcept
Definition span.h:325
constexpr iterator end() const noexcept
Definition span.h:420
~span() noexcept=default
constexpr span(const Container &cont)
Definition span.h:338
std::reverse_iterator< iterator > reverse_iterator
Definition span.h:291
constexpr iterator begin() const noexcept
Definition span.h:418
TCB_SPAN_ARRAY_CONSTEXPR span(std::array< value_type, N > &arr) noexcept
Definition span.h:318
static constexpr size_type extent
Definition span.h:293
const element_type * const_pointer
Definition span.h:287
constexpr span(const span &other) noexcept=default
typename std::remove_cv< ElementType >::type value_type
Definition span.h:283
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rbegin() const noexcept
Definition span.h:422
const element_type & const_reference
Definition span.h:289
std::ptrdiff_t difference_type
Definition span.h:285
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > first(size_type count) const
Definition span.h:376
TCB_SPAN_CONSTEXPR11 reference back() const
Definition span.h:410
constexpr size_type size() const noexcept
Definition span.h:393
TCB_SPAN_CONSTEXPR11 reference front() const
Definition span.h:405
constexpr size_type size_bytes() const noexcept
Definition span.h:395
TCB_SPAN_CONSTEXPR11 reference operator[](size_type idx) const
Definition span.h:400
TCB_SPAN_CONSTEXPR11 subspan_return_t< Offset, Count > subspan() const
Definition span.h:371
constexpr span(element_type(&arr)[N]) noexcept
Definition span.h:311
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > subspan(size_type offset, size_type count=dynamic_extent) const
Definition span.h:386
constexpr pointer data() const noexcept
Definition span.h:415
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > last(size_type count) const
Definition span.h:381
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rend() const noexcept
Definition span.h:424
constexpr span(const span< OtherElementType, OtherExtent > &other) noexcept
Definition span.h:346
element_type * pointer
Definition span.h:286
element_type & reference
Definition span.h:288
TCB_SPAN_NODISCARD constexpr bool empty() const noexcept
Definition span.h:397
typename std::remove_cv< typename std::remove_reference< T >::type >::type uncvref_t
Definition span.h:219
typename std::remove_pointer< T >::type remove_pointer_t
Definition span.h:248
constexpr auto data(C &c) -> decltype(c.data())
Definition span.h:191
span< byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_writable_bytes(span< ElementType, Extent > s) noexcept
Definition span.h:488
constexpr span< ElementType, Extent > make_span(span< ElementType, Extent > s) noexcept
Definition span.h:451
span< const byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_bytes(span< ElementType, Extent > s) noexcept
Definition span.h:481
constexpr auto get(span< E, S > s) -> decltype(s[N])
Definition span.h:494
unsigned char byte
Definition span.h:139
TCB_SPAN_INLINE_VAR constexpr std::size_t dynamic_extent
Definition span.h:148
STL namespace.
#define TCB_SPAN_EXPECT(cond)
Definition span.h:87
#define TCB_SPAN_INLINE_VAR
Definition span.h:93
#define TCB_SPAN_NODISCARD
Definition span.h:145
#define TCB_SPAN_ARRAY_CONSTEXPR
Definition span.h:133
#define TCB_SPAN_CONSTEXPR_ASSIGN
Definition span.h:109
#define TCB_SPAN_CONSTEXPR11
Definition span.h:113
constexpr span_storage() noexcept=default
static constexpr std::size_t size
Definition span.h:162