vecmem 1.14.0
Loading...
Searching...
No Matches
device_vector.ipp
1/* VecMem project, part of the ACTS project (R&D line)
2 *
3 * (c) 2021-2025 CERN for the benefit of the ACTS project
4 *
5 * Mozilla Public License Version 2.0
6 */
7#pragma once
8
9// Local include(s).
10#include "vecmem/utils/debug.hpp"
11
12// System include(s).
13#include <cassert>
14#include <memory>
15#include <type_traits>
16
17namespace vecmem {
18
19template <typename TYPE>
22 : m_capacity(data.capacity()), m_size(data.size_ptr()), m_ptr(data.ptr()) {
23
24 VECMEM_DEBUG_MSG(5,
25 "Created vecmem::device_vector with capacity %u and "
26 "size pointer %p from pointer %p",
27 m_capacity, static_cast<const void*>(m_size),
28 static_cast<const void*>(m_ptr));
29}
30
31template <typename TYPE>
32template <typename OTHERTYPE,
33 std::enable_if_t<std::is_convertible<OTHERTYPE, TYPE>::value, bool>>
35 const device_vector<OTHERTYPE>& parent)
36 : m_capacity(parent.m_capacity),
37 m_size(parent.m_size),
38 m_ptr(parent.m_ptr) {
39
40 VECMEM_DEBUG_MSG(5,
41 "Created vecmem::device_vector with capacity %u and "
42 "size pointer %p from pointer %p",
43 m_capacity, static_cast<const void*>(m_size),
44 static_cast<const void*>(m_ptr));
45}
46
47template <typename TYPE>
49 const device_vector& rhs) {
50
51 // Prevent self-assignment.
52 if (this == &rhs) {
53 return *this;
54 }
55
56 // Copy the other object's payload.
57 resize(rhs.size());
58 for (size_type i = 0; i < size(); ++i) {
59 this->at(i) = rhs.at(i);
60 }
61
62 // Return a reference to this object.
63 return *this;
64}
65
66template <typename TYPE>
67template <typename OTHERTYPE,
68 std::enable_if_t<std::is_convertible<OTHERTYPE, TYPE>::value, bool>>
71
72 // Copy the other object's payload.
73 resize(rhs.size());
74 for (size_type i = 0; i < size(); ++i) {
75 this->at(i) = rhs.at(i);
76 }
77
78 // Return a reference to this object.
79 return *this;
80}
81
82template <typename TYPE>
83VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::at(size_type pos)
84 -> reference {
85
86 // Check if the index is valid.
87 assert(pos < size());
88
89 // Return a reference to the vector element.
90 return m_ptr[pos];
91}
92
93template <typename TYPE>
94VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::at(size_type pos) const
96
97 // Check if the index is valid.
98 assert(pos < size());
99
100 // Return a reference to the vector element.
101 return m_ptr[pos];
102}
103
104template <typename TYPE>
106 -> reference {
107
108 // Return a reference to the vector element.
109 return m_ptr[pos];
110}
111
112template <typename TYPE>
113VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::operator[](size_type pos) const
114 -> const_reference {
115
116 // Return a reference to the vector element.
117 return m_ptr[pos];
118}
119
120template <typename TYPE>
121VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::front() -> reference {
122
123 // Make sure that there is at least one element in the vector.
124 assert(size() > 0);
125
126 // Return a reference to the first element of the vector.
127 return m_ptr[0];
128}
129
130template <typename TYPE>
131VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::front() const
132 -> const_reference {
133
134 // Make sure that there is at least one element in the vector.
135 assert(size() > 0);
136
137 // Return a reference to the first element of the vector.
138 return m_ptr[0];
139}
140
141template <typename TYPE>
142VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::back() -> reference {
143
144 // Make sure that there is at least one element in the vector.
145 assert(size() > 0);
146
147 // Return a reference to the last element of the vector.
148 return m_ptr[size() - 1];
149}
150
151template <typename TYPE>
152VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::back() const
153 -> const_reference {
154
155 // Make sure that there is at least one element in the vector.
156 assert(size() > 0);
157
158 // Return a reference to the last element of the vector.
159 return m_ptr[size() - 1];
160}
161
162template <typename TYPE>
163VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::data() -> pointer {
164
165 return m_ptr;
166}
167
168template <typename TYPE>
169VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::data() const -> const_pointer {
170
171 return m_ptr;
172}
173
174template <typename TYPE>
175VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::assign(size_type count,
176 const_reference value) {
177
178 // This can only be done on a sufficiently large, resizable vector.
179 assert(m_size != nullptr);
180 assert(m_capacity >= count);
181
182 // Remove all previous elements.
183 clear();
184 // Set the assigned size of the vector.
186 asize.store(count);
187
188 // Create the required number of identical elements.
189 for (size_type i = 0; i < count; ++i) {
190 construct(i, value);
191 }
192}
193
194template <typename TYPE>
195template <typename... Args>
196VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::emplace_back(Args&&... args)
197 -> reference {
198
199 // This can only be done on a resizable vector.
200 assert(m_size != nullptr);
201
202 // Increment the size of the vector at first. So that we would "claim" the
203 // index from other threads.
205 const size_type index = asize.fetch_add(1u);
206 assert(index < m_capacity);
207
208 // Instantiate the new value.
209 new (m_ptr + index) value_type(std::forward<Args>(args)...);
210
211 // Return a reference to the newly created object.
212 return m_ptr[index];
213}
214
215template <typename TYPE>
216VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::push_back(
217 const_reference value) -> size_type {
218
219 // This can only be done on a resizable vector.
220 assert(m_size != nullptr);
221
222 // Increment the size of the vector at first. So that we would "claim" the
223 // index from other threads.
225 const size_type index = asize.fetch_add(1u);
226 assert(index < m_capacity);
227
228 // Instantiate the new value.
229 construct(index, value);
230
231 // Return the index under which the element was inserted:
232 return index;
233}
234
235template <typename TYPE>
236VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::bulk_append(size_type n)
237 -> size_type {
238 // This can only be done on a resizable vector.
239 assert(m_size != nullptr);
240
241 static_assert(std::is_default_constructible<TYPE>::value,
242 "Type `T` in `device_vector<T>::bulk_append` is not a "
243 "default-constructible type.");
244
246 const size_type index = asize.fetch_add(n);
247 assert((index + n) <= m_capacity);
248
249 for (size_type i = 0; i < n; ++i) {
250 construct(index + i, value_type());
251 }
252
253 return index;
254}
255
256template <typename TYPE>
257VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::bulk_append(size_type n,
259 -> size_type {
260 // This can only be done on a resizable vector.
261 assert(m_size != nullptr);
262
263 static_assert(std::is_copy_constructible<TYPE>::value,
264 "Type `TYPE` in device_vector<T>::bulk_append(SIZE, TYPE)` "
265 "must be copy-constructible.");
266
268 const size_type index = asize.fetch_add(n);
269 assert((index + n) <= m_capacity);
270
271 for (size_type i = 0; i < n; ++i) {
272 construct(index + i, value_type(v));
273 }
274
275 return index;
276}
277
278template <typename TYPE>
280 size_type n) -> size_type {
281 // This can only be done on a resizable vector.
282 assert(m_size != nullptr);
283
285 "Type `TYPE` in `device_vector<T>::bulk_append_implicit` is "
286 "not an implicit lifetype type, so slots cannot be safely "
287 "reserved. Note that the definition of implicit lifetimes "
288 "differs between C++<=17, C++20, and C++>=23.");
289
291 const size_type index = asize.fetch_add(n);
292 assert((index + n) <= m_capacity);
293
294#if defined(__cpp_lib_start_lifetime_as) && \
295 __cpp_lib_start_lifetime_as >= 202207L
296 std::start_lifetime_as_array<TYPE>(m_ptr[index], n);
297#endif
298
299 return index;
300}
301
302template <typename TYPE>
304 size_type n) -> size_type {
305 // This can only be done on a resizable vector.
306 assert(m_size != nullptr);
307
309 const size_type index = asize.fetch_add(n);
310 assert((index + n) <= m_capacity);
311
312 return index;
313}
314
315template <typename TYPE>
316VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::pop_back() -> size_type {
317
318 // This can only be done on a resizable vector.
319 assert(m_size != nullptr);
320
321 // Decrement the size of the vector, and remember this new size.
323 const size_type new_size = asize.fetch_sub(1u) - 1;
324
325 // Remove the last element.
326 destruct(new_size);
327
328 // Return the vector's new size to the user.
329 return new_size;
330}
331
332template <typename TYPE>
333VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::clear() {
334
335 // This can only be done on a resizable vector.
336 assert(m_size != nullptr);
337
338 // Destruct all of the elements that the vector has "at the moment".
340 const size_type current_size = asize.load();
341 for (size_type i = 0; i < current_size; ++i) {
342 destruct(i);
343 }
344
345 // Set the vector to be empty now.
346 asize.store(0);
347}
348
349template <typename TYPE>
350VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::resize(size_type new_size) {
351
352 resize(new_size, value_type());
353}
354
355template <typename TYPE>
357 const_reference value) {
358
359 // This can only be done on a resizable vector.
360 assert(m_size != nullptr);
361
362 // Get the current size of the vector.
364 const size_type current_size = asize.load();
365
366 // Check if anything needs to be done.
367 if (new_size == current_size) {
368 return;
369 }
370
371 // If the new size is smaller than the current size, remove the unwanted
372 // elements.
373 if (new_size < current_size) {
374 for (size_type i = new_size; i < current_size; ++i) {
375 destruct(i);
376 }
377 }
378 // If the new size is larger than the current size, insert extra elements.
379 else {
380 for (size_type i = current_size; i < new_size; ++i) {
381 construct(i, value);
382 }
383 }
384
385 // Set the new size for the vector.
386 asize.store(new_size);
387}
388
389template <typename TYPE>
390VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::resize_implicit(
392 // This can only be done on a resizable vector.
393 assert(m_size != nullptr);
394
396 "Type `TYPE` in `device_vector<T>::resize_implicit` is not "
397 "an implicit lifetype type, so slots cannot be safely "
398 "reserved. Note that the definition of implicit lifetimes "
399 "differs between C++<=17, C++20, and C++>=23.");
400
402 assert(new_size <= m_capacity);
403 asize.store(new_size);
404}
405
406template <typename TYPE>
409 // This can only be done on a resizable vector.
410 assert(m_size != nullptr);
411
413 assert(new_size <= m_capacity);
414 asize.store(new_size);
415}
416
417template <typename TYPE>
418VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::begin() -> iterator {
419
420 return iterator(m_ptr);
421}
422
423template <typename TYPE>
424VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::begin() const
425 -> const_iterator {
426
427 return const_iterator(m_ptr);
428}
429
430template <typename TYPE>
431VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::cbegin() const
432 -> const_iterator {
433
434 return begin();
435}
436
437template <typename TYPE>
438VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::end() -> iterator {
439
440 return iterator(m_ptr + size());
441}
442
443template <typename TYPE>
444VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::end() const -> const_iterator {
445
446 return const_iterator(m_ptr + size());
447}
448
449template <typename TYPE>
450VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::cend() const
451 -> const_iterator {
452
453 return end();
454}
455
456template <typename TYPE>
457VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::rbegin() -> reverse_iterator {
458
459 return reverse_iterator(end());
460}
461
462template <typename TYPE>
463VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::rbegin() const
465
466 return const_reverse_iterator(end());
467}
468
469template <typename TYPE>
470VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::crbegin() const
472
473 return rbegin();
474}
475
476template <typename TYPE>
477VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::rend() -> reverse_iterator {
478
479 return reverse_iterator(begin());
480}
481
482template <typename TYPE>
483VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::rend() const
485
486 return const_reverse_iterator(begin());
487}
488
489template <typename TYPE>
490VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::crend() const
492
493 return rend();
494}
495
496template <typename TYPE>
497VECMEM_HOST_AND_DEVICE bool device_vector<TYPE>::empty() const {
498
499 return (size() == 0);
500}
501
502template <typename TYPE>
503VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::size() const -> size_type {
504
505 if (m_size == nullptr) {
506 return m_capacity;
507 } else {
508 // For the host, CUDA and HIP it would be possible to use the type
509 // atomic<const size_type> here, and avoid any const-casting. But with
510 // SYCL we must pass a non-const pointer to the sycl::atomic object
511 // that performs the load operation. And for that we need a non-const
512 // pointer...
513 device_atomic_ref<size_type> asize(*(const_cast<size_type*>(m_size)));
514 return asize.load();
515 }
516}
517
518template <typename TYPE>
519VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::max_size() const -> size_type {
520
521 return capacity();
522}
523
524template <typename TYPE>
525VECMEM_HOST_AND_DEVICE auto device_vector<TYPE>::capacity() const -> size_type {
526
527 return m_capacity;
528}
529
530template <typename TYPE>
531VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::construct(
532 size_type pos, const_reference value) {
533
534 // Make sure that the position is available.
535 assert(pos < m_capacity);
536
537 // Use the constructor of the type.
538 new (m_ptr + pos) value_type(value);
539}
540
541template <typename TYPE>
542VECMEM_HOST_AND_DEVICE void device_vector<TYPE>::destruct(size_type pos) {
543
544 // Make sure that the position is available.
545 assert(pos < m_capacity);
546
547 // Use the destructor of the type.
548 pointer ptr = m_ptr + pos;
549 ptr->~value_type();
550}
551
552} // namespace vecmem
Class holding data about a 1 dimensional vector/array.
Definition vector_view.hpp:38
Type mimicking std::reverse_iterator.
Definition reverse_iterator.hpp:25
Class mimicking an std::vector in "device code".
Definition device_vector.hpp:36
VECMEM_HOST_AND_DEVICE size_type bulk_append(size_type n)
Default-construct a given number of elements at the end of the vector.
Definition device_vector.ipp:236
VECMEM_HOST_AND_DEVICE reference back()
Return the last element of the vector (non-const)
Definition device_vector.ipp:142
VECMEM_HOST_AND_DEVICE void clear()
Clear the vector (not thread-safe)
Definition device_vector.ipp:333
VECMEM_HOST_AND_DEVICE iterator begin()
Return a forward iterator pointing at the beginning of the vector.
Definition device_vector.ipp:418
VECMEM_HOST_AND_DEVICE reference operator[](size_type pos)
Return a specific element of the vector (non-const)
Definition device_vector.ipp:105
VECMEM_HOST_AND_DEVICE void resize_implicit_unsafe(size_type new_size)
Resize a vector in constant time, unsafely deallocating non-implicit lifetime types.
Definition device_vector.ipp:407
VECMEM_HOST_AND_DEVICE pointer data()
Access the underlying memory array (non-const)
Definition device_vector.ipp:163
std::add_pointer_t< value_type > pointer
Value pointer type.
Definition device_vector.hpp:64
VECMEM_HOST_AND_DEVICE size_type bulk_append_implicit_unsafe(size_type n)
Reserve a fixed number of slots in the array in a way that technically is ill-formed.
Definition device_vector.ipp:303
VECMEM_HOST_AND_DEVICE reference emplace_back(Args &&... args)
Add a new element at the end of the vector (thread-safe)
VECMEM_HOST_AND_DEVICE device_vector & operator=(const device_vector &rhs)
Copy assignment operator from an identical type.
Definition device_vector.ipp:48
std::add_pointer_t< std::add_const_t< value_type > > const_pointer
Constant value pointer type.
Definition device_vector.hpp:66
VECMEM_HOST_AND_DEVICE size_type capacity() const
Return the current (fixed) capacity of the vector.
Definition device_vector.ipp:525
TYPE value_type
Type of the array elements.
Definition device_vector.hpp:47
VECMEM_HOST_AND_DEVICE const_reverse_iterator crend() const
Return a constant reverse iterator pointing at the beginning of the vector.
Definition device_vector.ipp:490
VECMEM_HOST_AND_DEVICE size_type size() const
Return the number of elements in the vector.
Definition device_vector.ipp:503
const_pointer const_iterator
Constant forward iterator type.
Definition device_vector.hpp:71
VECMEM_HOST_AND_DEVICE bool empty() const
Check whether the vector is empty.
Definition device_vector.ipp:497
VECMEM_HOST_AND_DEVICE void assign(size_type count, const_reference value)
Assign new values to the vector (not thread-safe)
Definition device_vector.ipp:175
VECMEM_HOST_AND_DEVICE reference front()
Return the first element of the vector (non-const)
Definition device_vector.ipp:121
VECMEM_HOST_AND_DEVICE void resize(size_type new_size)
Resize the vector (not thread-safe)
Definition device_vector.ipp:350
VECMEM_HOST_AND_DEVICE const_reverse_iterator crbegin() const
Return a constant reverse iterator pointing at the end of the vector.
Definition device_vector.ipp:470
VECMEM_HOST_AND_DEVICE size_type push_back(const_reference value)
Add a new element at the end of the vector (thread-safe)
Definition device_vector.ipp:216
VECMEM_HOST_AND_DEVICE size_type bulk_append_implicit(size_type n)
Reserve a fixed number of slots in the array in a standards-conformant way.
Definition device_vector.ipp:279
unsigned int size_type
Size type for the array.
Definition device_vector.hpp:49
VECMEM_HOST_AND_DEVICE iterator end()
Return a forward iterator pointing at the end of the vector.
Definition device_vector.ipp:438
pointer iterator
Forward iterator type.
Definition device_vector.hpp:69
std::add_lvalue_reference_t< value_type > reference
Value reference type.
Definition device_vector.hpp:59
VECMEM_HOST_AND_DEVICE size_type max_size() const
Return the maximum (fixed) number of elements in the vector.
Definition device_vector.ipp:519
std::add_lvalue_reference_t< std::add_const_t< value_type > > const_reference
Constant value reference type.
Definition device_vector.hpp:62
VECMEM_HOST_AND_DEVICE const_iterator cend() const
Return a constant forward iterator pointing at the end of the vector.
Definition device_vector.ipp:450
VECMEM_HOST_AND_DEVICE reference at(size_type pos)
Return a specific element of the vector in a "safe way" (non-const)
Definition device_vector.ipp:83
VECMEM_HOST_AND_DEVICE const_iterator cbegin() const
Return a constant forward iterator pointing at the beginning of the vector.
Definition device_vector.ipp:431
VECMEM_HOST_AND_DEVICE size_type pop_back()
Remove the last element of the vector (not thread-safe)
Definition device_vector.ipp:316
VECMEM_HOST_AND_DEVICE reverse_iterator rend()
Return a reverse iterator pointing at the beginning of the vector.
Definition device_vector.ipp:477
VECMEM_HOST_AND_DEVICE device_vector(const data::vector_view< value_type > &data)
Constructor, on top of a previously allocated/filled block of memory.
Definition device_vector.ipp:20
VECMEM_HOST_AND_DEVICE reverse_iterator rbegin()
Return a reverse iterator pointing at the end of the vector.
Definition device_vector.ipp:457
VECMEM_HOST_AND_DEVICE void resize_implicit(size_type new_size)
Resize a vector of implicit lifetime types.
Definition device_vector.ipp:390
std::true_type is_implicit_lifetime
Type trait that indicates whether a given type is an implicit lifetime type.
Definition type_traits.hpp:150
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