vecmem 1.14.0
Loading...
Searching...
No Matches
static_array.ipp
1/*
2 * VecMem project, part of the ACTS project (R&D line)
3 *
4 * (c) 2021 CERN for the benefit of the ACTS project
5 *
6 * Mozilla Public License Version 2.0
7 */
8
9#pragma once
10
11// Local include(s).
12#include "vecmem/utils/types.hpp"
13
14// System include(s).
15#include <cstddef>
16#include <stdexcept>
17#include <type_traits>
18
19namespace vecmem {
20
21template <typename T, std::size_t N>
22VECMEM_HOST constexpr auto static_array<T, N>::at(size_type i) -> reference {
23 /*
24 * The at function is bounds-checking in the standard library, so we
25 * do a boundary check in our code, too. This makes this method
26 * incompatible with device code.
27 */
28 if (i >= N) {
29 throw std::out_of_range("Index greater than size of static array.");
30 }
31
32 return operator[](i);
33}
34
35template <typename T, std::size_t N>
36VECMEM_HOST constexpr auto static_array<T, N>::at(size_type i) const
38 /*
39 * Same thing as with the other at function, we do a bounds check in
40 * accordance with the standard library.
41 */
42 if (i >= N) {
43 throw std::out_of_range("Index greater than size of static array.");
44 }
45
46 return operator[](i);
47}
48
49template <typename T, std::size_t N>
50VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::operator[](
52 /*
53 * Non-bounds checking access, which could cause a segmentation
54 * violation.
55 */
56 return m_array[i];
57}
58
59template <typename T, std::size_t N>
60VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::operator[](
61 size_type i) const -> const_reference {
62 /*
63 * Return an element as constant.
64 */
65 return m_array[i];
66}
67
68template <typename T, std::size_t N>
69template <std::size_t I,
70 std::enable_if_t<I<N, bool> > VECMEM_HOST_AND_DEVICE constexpr auto
72
73 return m_array[I];
74}
75
76template <typename T, std::size_t N>
77template <std::size_t I,
78 std::enable_if_t<I<N, bool> > VECMEM_HOST_AND_DEVICE constexpr auto
79 static_array<T, N>::get() const noexcept->const_reference {
80
81 return m_array[I];
82}
83
84template <typename T, std::size_t N>
85VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::front(void)
86 -> reference {
87 /*
88 * Return the first element.
89 */
90 return m_array[0];
91}
92
93template <typename T, std::size_t N>
94VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::front(void) const
95 -> const_reference {
96 /*
97 * Return the first element, but it's const.
98 */
99 return m_array[0];
100}
101
102template <typename T, std::size_t N>
103VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::back(void)
104 -> reference {
105 /*
106 * Return the last element.
107 */
108 return m_array[N - 1];
109}
110
111template <typename T, std::size_t N>
112VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::back(void) const
113 -> const_reference {
114 /*
115 * Return the last element, but it's const.
116 */
117 return m_array[N - 1];
118}
119
120template <typename T, std::size_t N>
121VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::data(void)
122 -> pointer {
123 /*
124 * Return a pointer to the underlying data.
125 */
126 return m_array;
127}
128
129template <typename T, std::size_t N>
130VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::data(void) const
131 -> const_pointer {
132 /*
133 * Return a pointer to the underlying data, but the elements are const.
134 */
135 return m_array;
136}
137
138template <typename T, std::size_t N>
139VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::begin() -> iterator {
140
141 return m_array;
142}
143
144template <typename T, std::size_t N>
145VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::begin() const
146 -> const_iterator {
147
148 return m_array;
149}
150
151template <typename T, std::size_t N>
152VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::cbegin() const
153 -> const_iterator {
154
155 return begin();
156}
157
158template <typename T, std::size_t N>
159VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::end() -> iterator {
160
161 return m_array + N;
162}
163
164template <typename T, std::size_t N>
165VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::end() const
166 -> const_iterator {
167
168 return m_array + N;
169}
170
171template <typename T, std::size_t N>
172VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::cend() const
173 -> const_iterator {
174
175 return end();
176}
177
178template <typename T, std::size_t N>
179VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::rbegin()
180 -> reverse_iterator {
181
182 return reverse_iterator(end());
183}
184
185template <typename T, std::size_t N>
186VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::rbegin() const
187 -> const_reverse_iterator {
188
189 return const_reverse_iterator(end());
190}
191
192template <typename T, std::size_t N>
193VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::crbegin() const
194 -> const_reverse_iterator {
195
196 return rbegin();
197}
198
199template <typename T, std::size_t N>
200VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::rend()
201 -> reverse_iterator {
202
203 return reverse_iterator(begin());
204}
205
206template <typename T, std::size_t N>
207VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::rend() const
208 -> const_reverse_iterator {
209
210 return const_reverse_iterator(begin());
211}
212
213template <typename T, std::size_t N>
214VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::crend() const
215 -> const_reverse_iterator {
216
217 return rend();
218}
219
220template <typename T, std::size_t N>
221VECMEM_HOST_AND_DEVICE constexpr bool static_array<T, N>::empty() const {
222
223 return N == 0;
224}
225
226template <typename T, std::size_t N>
227VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::size() const
228 -> size_type {
229
230 return N;
231}
232
233template <typename T, std::size_t N>
234VECMEM_HOST_AND_DEVICE constexpr auto static_array<T, N>::max_size() const
235 -> size_type {
236
237 return N;
238}
239
240template <typename T, std::size_t N>
241VECMEM_HOST_AND_DEVICE void static_array<T, N>::fill(const_reference value) {
242
243 for (std::size_t i = 0; i < N; ++i) {
244 m_array[i] = value;
245 }
246}
247
248template <typename T, std::size_t N>
249VECMEM_HOST_AND_DEVICE bool operator==(const static_array<T, N>& lhs,
250 const static_array<T, N>& rhs) {
251 /*
252 * Iterate over all elements in the arrays, if any of them are unequal
253 * between them, the arrays are not equal.
254 */
255 for (typename static_array<T, N>::size_type i = 0; i < N; ++i) {
256 if (lhs[i] != rhs[i]) {
257 return false;
258 }
259 }
260
261 /*
262 * If we have iterated over the entire array without finding a counter-
263 * example to the equality, the arrays must be equal.
264 */
265 return true;
266}
267
268template <typename T, std::size_t N>
269VECMEM_HOST_AND_DEVICE bool operator!=(const static_array<T, N>& lhs,
270 const static_array<T, N>& rhs) {
271 /*
272 * Same thing as before, we check all element pairs, if any of them are
273 * unequal, the entire array is unequal. We could also implement this as
274 * return !(lhs == rhs).
275 */
276 for (typename static_array<T, N>::size_type i = 0; i < N; ++i) {
277 if (lhs[i] != rhs[i]) {
278 return true;
279 }
280 }
281
282 /*
283 * No counter example, so the arrays are not unequal.
284 */
285 return false;
286}
287
288template <std::size_t I, class T, std::size_t N,
289 std::enable_if_t<I<N, bool> > VECMEM_HOST_AND_DEVICE constexpr T& get(
290 static_array<T, N>& a) noexcept {
291
292 return a.template get<I>();
293}
294
295template <
296 std::size_t I, class T, std::size_t N,
297 std::enable_if_t<I<N, bool> > VECMEM_HOST_AND_DEVICE constexpr const T& get(
298 const static_array<T, N>& a) noexcept {
299
300 return a.template get<I>();
301}
302
303} // namespace vecmem
Main namespace for the vecmem classes/functions.
Definition atomic_ref.hpp:16
VECMEM_HOST_AND_DEVICE bool operator!=(const static_array< T, N > &lhs, const static_array< T, N > &rhs)
Non-equality check on two arrays.
Definition static_array.ipp:269
VECMEM_HOST_AND_DEVICE bool operator==(const static_array< T, N > &lhs, const static_array< T, N > &rhs)
Equality check on two arrays.
Definition static_array.ipp:249
VECMEM_HOST_AND_DEVICE constexpr const auto & get(const tuple< Ts... > &t) noexcept
Get a constant element out of a tuple.
Definition tuple.ipp:58
std::vector< T, vecmem::polymorphic_allocator< T > > vector
Alias type for vectors with our polymorphic allocator.
Definition vector.hpp:35
Simple statically-sized array-like class designed for use in device code.
Definition static_array.hpp:33
value_type & reference
Value reference type.
Definition static_array.hpp:46
const value_type & const_reference
Constant value reference type.
Definition static_array.hpp:48
VECMEM_HOST_AND_DEVICE constexpr reference operator[](size_type i)
Accessor method.
Definition static_array.ipp:50
std::size_t size_type
Size type for the array.
Definition static_array.hpp:41
VECMEM_HOST constexpr reference at(size_type i)
Bounds-checked accessor method.
Definition static_array.ipp:22