vecmem 1.14.0
Loading...
Searching...
No Matches
static_vector.ipp
1/* VecMem project, part of the ACTS project (R&D line)
2 *
3 * (c) 2021-2023 CERN for the benefit of the ACTS project
4 *
5 * Mozilla Public License Version 2.0
6 */
7#pragma once
8
9// System include(s).
10#include <cassert>
11#include <cstring>
12#include <utility>
13
14namespace vecmem {
15
16template <typename TYPE, std::size_t MAX_SIZE>
18 : m_size(0), m_elements() {}
19
20template <typename TYPE, std::size_t MAX_SIZE>
22 size_type size, const_reference value)
23 : m_size(size), m_elements() {
24
25 assign(size, value);
26}
27
28template <typename TYPE, std::size_t MAX_SIZE>
30 const static_vector& parent)
31 : m_size(parent.m_size), m_elements() {
32
33 // Make copies of all of the elements.
34 for (size_type i = 0; i < m_size; ++i) {
35 construct(i, parent[i]);
36 }
37}
38
39template <typename TYPE, std::size_t MAX_SIZE>
41
42 clear();
43}
44
45template <typename TYPE, std::size_t MAX_SIZE>
46VECMEM_HOST_AND_DEVICE typename static_vector<TYPE, MAX_SIZE>::reference
48
49 // Make sure that the element exists.
50 assert(pos < m_size);
51
52 // Return the element.
53 return *(reinterpret_cast<pointer>(m_elements) + pos);
54}
55
56template <typename TYPE, std::size_t MAX_SIZE>
57VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::at(
58 size_type pos) const -> const_reference {
59
60 // Make sure that the element exists.
61 assert(pos < m_size);
62
63 // Return the element.
64 return *(reinterpret_cast<const_pointer>(m_elements) + pos);
65}
66
67template <typename TYPE, std::size_t MAX_SIZE>
70
71 // Return the element.
72 return *(reinterpret_cast<pointer>(m_elements) + pos);
73}
74
75template <typename TYPE, std::size_t MAX_SIZE>
77 size_type pos) const -> const_reference {
78
79 // Return the element.
80 return *(reinterpret_cast<const_pointer>(m_elements) + pos);
81}
82
83template <typename TYPE, std::size_t MAX_SIZE>
84VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::front()
85 -> reference {
86
87 // Make sure that the element exists.
88 assert(m_size > 0);
89
90 // Return the element.
91 return *(reinterpret_cast<pointer>(m_elements));
92}
93
94template <typename TYPE, std::size_t MAX_SIZE>
95VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::front() const
97
98 // Make sure that the element exists.
99 assert(m_size > 0);
100
101 // Return the element.
102 return *(reinterpret_cast<const_pointer>(m_elements));
103}
104
105template <typename TYPE, std::size_t MAX_SIZE>
106VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::back() -> reference {
107
108 // Make sure that the element exists.
109 assert(m_size > 0);
110
111 // Return the element.
112 return *(reinterpret_cast<pointer>(m_elements) + m_size - 1);
113}
114
115template <typename TYPE, std::size_t MAX_SIZE>
116VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::back() const
117 -> const_reference {
118
119 // Make sure that the element exists.
120 assert(m_size > 0);
121
122 // Return the element.
123 return *(reinterpret_cast<const_pointer>(m_elements) + m_size - 1);
124}
125
126template <typename TYPE, std::size_t MAX_SIZE>
127VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::data() -> pointer {
128
129 return reinterpret_cast<pointer>(m_elements);
130}
131
132template <typename TYPE, std::size_t MAX_SIZE>
133VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::data() const
134 -> const_pointer {
135
136 return reinterpret_cast<const_pointer>(m_elements);
137}
138
139template <typename TYPE, std::size_t MAX_SIZE>
140VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::assign(
141 size_type count, const_reference value) {
142
143 // Make sure that the sizes are compatible.
144 assert(array_max_size >= count);
145
146 // Remove all previous elements.
147 clear();
148
149 // Create the required number of identical elements.
150 for (size_type i = 0; i < count; ++i) {
151 construct(i, value);
152 }
153
154 // Set the assigned size of the vector.
155 m_size = count;
156}
157
158template <typename TYPE, std::size_t MAX_SIZE>
159VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::insert(
161
162 // Make sure that one more position is available.
163 assert(m_size < array_max_size);
164
165 // Find the index of this iterator inside of the vector.
166 auto id = element_id(pos);
167
168 // Move the payload of the existing elements after "pos".
169 memmove(static_cast<void*>(id.m_ptr + 1),
170 static_cast<const void*>(id.m_ptr),
171 (m_size - id.m_index) * value_size);
172
173 // Instantiate the new value.
174 construct(id.m_index, value);
175
176 // Increment the size.
177 ++m_size;
178
179 // Return an iterator to the inserted element.
180 return id.m_ptr;
181}
182
183template <typename TYPE, std::size_t MAX_SIZE>
184VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::insert(
186
187 // Make sure that the requested number of positions are still available.
188 assert(m_size + count <= array_max_size);
189
190 // Find the index of this iterator inside of the vector.
191 auto id = element_id(pos);
192
193 // Move the payload of the existing elements after "pos".
194 memmove(static_cast<void*>(id.m_ptr + count),
195 static_cast<const void*>(id.m_ptr),
196 (m_size - id.m_index) * value_size);
197
198 // Instantiate all the new values.
199 for (size_type i = 0; i < count; ++i) {
200 construct(id.m_index + i, value);
201 }
202
203 // Increment the size.
204 m_size += count;
205
206 // Return an iterator to the first inserted element.
207 return id.m_ptr;
208}
209
210template <typename TYPE, std::size_t MAX_SIZE>
211template <typename... Args>
212VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::emplace(
213 const_iterator pos, Args&&... args) -> iterator {
214
215 // Make sure that one more position is available.
216 assert(m_size < array_max_size);
217
218 // Find the index of this iterator inside of the vector.
219 auto id = element_id(pos);
220
221 // Move the payload of the existing elements after "pos".
222 memmove(static_cast<void*>(id.m_ptr + 1),
223 static_cast<const void*>(id.m_ptr),
224 (m_size - id.m_index) * value_size);
225
226 // Instantiate the new value.
227 new (id.m_ptr) value_type(std::forward<Args>(args)...);
228
229 // Increment the size.
230 ++m_size;
231
232 // Return an iterator to the inserted element.
233 return id.m_ptr;
234}
235
236template <typename TYPE, std::size_t MAX_SIZE>
237template <typename... Args>
238VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::emplace_back(
239 Args&&... args) -> reference {
240
241 return *(emplace(end(), std::forward<Args>(args)...));
242}
243
244template <typename TYPE, std::size_t MAX_SIZE>
246 const_reference value) {
247
248 insert(end(), value);
249}
250
251template <typename TYPE, std::size_t MAX_SIZE>
252VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::erase(
254
255 // Find the index of this iterator inside of the vector.
256 auto id = element_id(pos);
257
258 // Destroy the object.
259 destruct(id.m_index);
260
261 // Move up the payload of the elements from after the removed one.
262 memmove(static_cast<void*>(id.m_ptr),
263 static_cast<const void*>(id.m_ptr + 1),
264 (m_size - id.m_index - 1) * value_size);
265
266 // Decrement the size.
267 --m_size;
268
269 // Return an iterator to after the removed element.
270 return id.m_ptr;
271}
272
273template <typename TYPE, std::size_t MAX_SIZE>
274VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::erase(
276
277 // Find the indices and pointers of the iterators.
278 auto first_id = element_id(first);
279 auto last_id = element_id(last);
280 assert(first_id.m_index <= last_id.m_index);
281
282 // Destroy the objects.
283 for (size_type i = first_id.m_index; i < last_id.m_index; ++i) {
284 destruct(i);
285 }
286
287 // Move up the payload of the elements from after the removed range.
288 memmove(static_cast<void*>(first_id.m_ptr),
289 static_cast<const void*>(last_id.m_ptr),
290 (m_size - last_id.m_index) * value_size);
291
292 // Decrease the size.
293 m_size -= (last_id.m_index - first_id.m_index);
294
295 // Return an iterator to after the removed elements.
296 return first_id.m_ptr;
297}
298
299template <typename TYPE, std::size_t MAX_SIZE>
300VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::pop_back() {
301
302 erase(end() - 1);
303}
304
305template <typename TYPE, std::size_t MAX_SIZE>
306VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::clear() {
307
308 for (size_type i = 0; i < m_size; ++i) {
309 destruct(i);
310 }
311 m_size = 0;
312}
313
314template <typename TYPE, std::size_t MAX_SIZE>
315VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::resize(
316 std::size_t new_size) {
317
318 resize(new_size, value_type());
319}
320
321template <typename TYPE, std::size_t MAX_SIZE>
322VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::resize(
323 std::size_t new_size, const_reference value) {
324
325 // Make sure that the request can be done.
326 assert(new_size <= array_max_size);
327
328 // Check if anything even needs to be done.
329 if (new_size == m_size) {
330 return;
331 }
332
333 // If the new size is smaller than the current size, remove the unwanted
334 // elements.
335 if (new_size < m_size) {
336 erase(begin() + new_size, end());
337 }
338 // If the new size is larger than the current size, insert extra elements.
339 else {
340 insert(end(), new_size - m_size, value);
341 }
342}
343
344template <typename TYPE, std::size_t MAX_SIZE>
345VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::begin() -> iterator {
346
347 return reinterpret_cast<iterator>(m_elements);
348}
349
350template <typename TYPE, std::size_t MAX_SIZE>
351VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::begin() const
352 -> const_iterator {
353
354 return reinterpret_cast<const_iterator>(m_elements);
355}
356
357template <typename TYPE, std::size_t MAX_SIZE>
358VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::cbegin() const
359 -> const_iterator {
360
361 return begin();
362}
363
364template <typename TYPE, std::size_t MAX_SIZE>
365VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::end() -> iterator {
366
367 return (reinterpret_cast<iterator>(m_elements) + m_size);
368}
369
370template <typename TYPE, std::size_t MAX_SIZE>
371VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::end() const
372 -> const_iterator {
373
374 return (reinterpret_cast<const_iterator>(m_elements) + m_size);
375}
376
377template <typename TYPE, std::size_t MAX_SIZE>
378VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::cend() const
379 -> const_iterator {
380
381 return end();
382}
383
384template <typename TYPE, std::size_t MAX_SIZE>
385VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::rbegin()
387
388 return reverse_iterator(end());
389}
390
391template <typename TYPE, std::size_t MAX_SIZE>
392VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::rbegin() const
394
395 return const_reverse_iterator(end());
396}
397
398template <typename TYPE, std::size_t MAX_SIZE>
401
402 return rbegin();
403}
404
405template <typename TYPE, std::size_t MAX_SIZE>
406VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::rend()
408
409 return reverse_iterator(begin());
410}
411
412template <typename TYPE, std::size_t MAX_SIZE>
413VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::rend() const
415
416 return const_reverse_iterator(begin());
417}
418
419template <typename TYPE, std::size_t MAX_SIZE>
420VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::crend() const
422
423 return rend();
424}
425
426template <typename TYPE, std::size_t MAX_SIZE>
427VECMEM_HOST_AND_DEVICE bool static_vector<TYPE, MAX_SIZE>::empty() const {
428
429 return m_size == 0;
430}
431
432template <typename TYPE, std::size_t MAX_SIZE>
433VECMEM_HOST_AND_DEVICE auto static_vector<TYPE, MAX_SIZE>::size() const
434 -> size_type {
435
436 return m_size;
437}
438
439template <typename TYPE, std::size_t MAX_SIZE>
441 -> size_type {
442
443 return array_max_size;
444}
445
446template <typename TYPE, std::size_t MAX_SIZE>
448 -> size_type {
449
450 return array_max_size;
451}
452
453template <typename TYPE, std::size_t MAX_SIZE>
454VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::reserve(
456
457 // Make sure that the user didn't ask for too much.
458 assert(new_cap <= array_max_size);
459 (void)new_cap;
460}
461
462template <typename TYPE, std::size_t MAX_SIZE>
463VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::construct(
464 size_type pos, const_reference value) {
465
466 // Make sure that the position is available.
467 assert(pos < array_max_size);
468
469 // Use the constructor of the type.
470 pointer ptr = reinterpret_cast<pointer>(m_elements) + pos;
471 new (ptr) value_type(value);
472}
473
474template <typename TYPE, std::size_t MAX_SIZE>
475VECMEM_HOST_AND_DEVICE void static_vector<TYPE, MAX_SIZE>::destruct(
476 size_type pos) {
477
478 // Make sure that the element has been allocated.
479 assert(pos < m_size);
480
481 // Use the destructor of the type.
482 pointer ptr = reinterpret_cast<pointer>(m_elements) + pos;
483 ptr->~value_type();
484}
485
486template <typename TYPE, std::size_t MAX_SIZE>
487VECMEM_HOST_AND_DEVICE typename static_vector<TYPE, MAX_SIZE>::ElementId
488static_vector<TYPE, MAX_SIZE>::element_id(const_iterator pos) {
489
490 size_type const index = static_cast<size_type>(pos - begin());
491 assert(index <= m_size);
492 pointer const ptr = reinterpret_cast<pointer>(m_elements) + index;
493 return {index, ptr};
494}
495
496} // namespace vecmem
Type mimicking std::reverse_iterator.
Definition reverse_iterator.hpp:25
Class mimicking std::vector on top of a fixed sized array.
Definition static_vector.hpp:30
VECMEM_HOST_AND_DEVICE size_type capacity() const
Return the current (fixed) capacity of the vector.
Definition static_vector.ipp:447
VECMEM_HOST_AND_DEVICE void assign(size_type count, const_reference value)
Assign new values to the vector.
Definition static_vector.ipp:140
VECMEM_HOST_AND_DEVICE const_iterator cend() const
Return a constant forward iterator pointing at the end of the vector.
Definition static_vector.ipp:378
value_type * pointer
Value pointer type.
Definition static_vector.hpp:56
value_type & reference
Value reference type.
Definition static_vector.hpp:52
VECMEM_HOST_AND_DEVICE ~static_vector()
Destructor.
Definition static_vector.ipp:40
VECMEM_HOST_AND_DEVICE reference at(size_type pos)
Return a specific element of the vector in a "safe way" (non-const)
Definition static_vector.ipp:47
VECMEM_HOST_AND_DEVICE void resize(std::size_t new_size)
Resize the vector.
Definition static_vector.ipp:315
VECMEM_HOST_AND_DEVICE bool empty() const
Check whether the vector is empty.
Definition static_vector.ipp:427
VECMEM_HOST_AND_DEVICE size_type size() const
Return the number of elements in the vector.
Definition static_vector.ipp:433
std::size_t size_type
Size type for the array.
Definition static_vector.hpp:39
VECMEM_HOST_AND_DEVICE iterator erase(const_iterator pos)
Remove one element from the vector.
Definition static_vector.ipp:252
VECMEM_HOST_AND_DEVICE static_vector()
Default constructor.
Definition static_vector.ipp:17
VECMEM_HOST_AND_DEVICE iterator begin()
Return a forward iterator pointing at the beginning of the vector.
Definition static_vector.ipp:345
VECMEM_HOST_AND_DEVICE reference operator[](size_type pos)
Return a specific element of the vector (non-const)
Definition static_vector.ipp:68
const value_type * const_pointer
Constant value pointer type.
Definition static_vector.hpp:58
VECMEM_HOST_AND_DEVICE void clear()
Clear the vector.
Definition static_vector.ipp:306
VECMEM_HOST_AND_DEVICE const_reverse_iterator crend() const
Return a constant reverse iterator pointing at the beginning of the vector.
Definition static_vector.ipp:420
VECMEM_HOST_AND_DEVICE pointer data()
Access the underlying memory array (non-const)
Definition static_vector.ipp:127
VECMEM_HOST_AND_DEVICE reference emplace_back(Args &&... args)
Add a new element at the end of the vector.
VECMEM_HOST_AND_DEVICE void push_back(const_reference value)
Add a new element at the end of the vector.
Definition static_vector.ipp:245
VECMEM_HOST_AND_DEVICE iterator emplace(const_iterator pos, Args &&... args)
Insert a new element into the vector.
VECMEM_HOST_AND_DEVICE iterator end()
Return a forward iterator pointing at the end of the vector.
Definition static_vector.ipp:365
VECMEM_HOST_AND_DEVICE const_reverse_iterator crbegin() const
Return a constant reverse iterator pointing at the end of the vector.
Definition static_vector.ipp:399
VECMEM_HOST_AND_DEVICE reference back()
Return the last element of the vector (non-const)
Definition static_vector.ipp:106
const value_type & const_reference
Constant value reference type.
Definition static_vector.hpp:54
VECMEM_HOST_AND_DEVICE reverse_iterator rend()
Return a reverse iterator pointing at the beginning of the vector.
Definition static_vector.ipp:406
VECMEM_HOST_AND_DEVICE void reserve(size_type new_cap)
Reserve additional storage for the vector.
Definition static_vector.ipp:454
VECMEM_HOST_AND_DEVICE const_iterator cbegin() const
Return a constant forward iterator pointing at the beginning of the vector.
Definition static_vector.ipp:358
pointer iterator
Forward iterator type.
Definition static_vector.hpp:61
VECMEM_HOST_AND_DEVICE iterator insert(const_iterator pos, const_reference value)
Insert a new element into the vector.
Definition static_vector.ipp:159
VECMEM_HOST_AND_DEVICE reverse_iterator rbegin()
Return a reverse iterator pointing at the end of the vector.
Definition static_vector.ipp:385
VECMEM_HOST_AND_DEVICE size_type max_size() const
Return the maximum (fixed) number of elements in the vector.
Definition static_vector.ipp:440
TYPE value_type
Type of the array elements.
Definition static_vector.hpp:37
const_pointer const_iterator
Constant forward iterator type.
Definition static_vector.hpp:63
VECMEM_HOST_AND_DEVICE reference front()
Return the first element of the vector (non-const)
Definition static_vector.ipp:84
VECMEM_HOST_AND_DEVICE void pop_back()
Remove the last element of the vector.
Definition static_vector.ipp:300
Main namespace for the vecmem classes/functions.
Definition atomic_ref.hpp:16
std::vector< T, vecmem::polymorphic_allocator< T > > vector
Alias type for vectors with our polymorphic allocator.
Definition vector.hpp:35