vecmem 1.14.0
Loading...
Searching...
No Matches
device.ipp
1/* VecMem project, part of the ACTS project (R&D line)
2 *
3 * (c) 2023-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/edm/details/device_traits.hpp"
11#include "vecmem/edm/details/schema_traits.hpp"
12#include "vecmem/memory/device_atomic_ref.hpp"
13
14// System include(s).
15#include <cassert>
16
17namespace vecmem {
18namespace edm {
19
20template <typename... VARTYPES, template <typename> class INTERFACE>
21VECMEM_HOST_AND_DEVICE device<schema<VARTYPES...>, INTERFACE>::device(
23 : m_capacity{view.capacity()},
24 m_size{details::device_size_pointer<vecmem::details::disjunction_v<
25 type::details::is_jagged_vector<VARTYPES>...>>::get(view.size())},
26 m_data{view.variables()} {
27
28 // Check that all variables have the correct capacities.
29 assert(details::device_capacities_match<VARTYPES...>(
30 m_capacity, m_data, std::index_sequence_for<VARTYPES...>{}));
31}
32
33template <typename... VARTYPES, template <typename> class INTERFACE>
34VECMEM_HOST_AND_DEVICE auto device<schema<VARTYPES...>, INTERFACE>::size() const
35 -> size_type {
36
37 // Check that all variables have the correct capacities.
38 assert(details::device_capacities_match<VARTYPES...>(
39 m_capacity, m_data, std::index_sequence_for<VARTYPES...>{}));
40
41 return (m_size == nullptr ? m_capacity : *m_size);
42}
43
44template <typename... VARTYPES, template <typename> class INTERFACE>
45VECMEM_HOST_AND_DEVICE auto device<schema<VARTYPES...>, INTERFACE>::capacity()
46 const -> size_type {
47
48 // Check that all variables have the correct capacities.
49 assert(details::device_capacities_match<VARTYPES...>(
50 m_capacity, m_data, std::index_sequence_for<VARTYPES...>{}));
51
52 return m_capacity;
53}
54
55template <typename... VARTYPES, template <typename> class INTERFACE>
56VECMEM_HOST_AND_DEVICE auto
57device<schema<VARTYPES...>, INTERFACE>::push_back_default() -> size_type {
58
59 // There must be no jagged vector variables for this to work.
60 static_assert(!vecmem::details::disjunction<
62 "Containers with jagged vector variables cannot be resized!");
63 // There must be at least one vector variable in the container.
64 static_assert(vecmem::details::disjunction<
66 "This function requires at least one vector variable.");
67 // This can only be done on a resizable container.
68 assert(m_size != nullptr);
69 // Check that all variables have the correct capacities.
70 assert(details::device_capacities_match<VARTYPES...>(
71 m_capacity, m_data, std::index_sequence_for<VARTYPES...>{}));
72
73 // Increment the size of the container at first. So that we would "claim"
74 // the index from other threads.
76 const size_type index = asize.fetch_add(1u);
77 assert(index < m_capacity);
78
79 // Construct the new elements in all of the vector variables.
80 construct_default(index, std::index_sequence_for<VARTYPES...>{});
81
82 // Return the position of the new variable(s).
83 return index;
84}
85
86template <typename... VARTYPES, template <typename> class INTERFACE>
87VECMEM_HOST_AND_DEVICE auto device<schema<VARTYPES...>, INTERFACE>::push_back(
88 const object_type& element) -> size_type {
89
90 // Add a new default element to the container.
91 const size_type index = push_back_default();
92
93 // Set it to the given value.
94 at(index) = element;
95
96 // Return the position of the new variable(s).
97 return index;
98}
99
100template <typename... VARTYPES, template <typename> class INTERFACE>
101template <std::size_t INDEX>
102VECMEM_HOST_AND_DEVICE
103 typename details::device_type_at<INDEX, VARTYPES...>::return_type
109
110template <typename... VARTYPES, template <typename> class INTERFACE>
111template <std::size_t INDEX>
112VECMEM_HOST_AND_DEVICE
113 typename details::device_type_at<INDEX, VARTYPES...>::const_return_type
119
120template <typename... VARTYPES, template <typename> class INTERFACE>
121VECMEM_HOST_AND_DEVICE
122 typename device<schema<VARTYPES...>, INTERFACE>::proxy_type
124
125 // Make sure that the index is within bounds.
126 assert(index < size());
127
128 // Use the unprotected function.
129 return this->operator[](index);
130}
131
132template <typename... VARTYPES, template <typename> class INTERFACE>
133VECMEM_HOST_AND_DEVICE
134 typename device<schema<VARTYPES...>, INTERFACE>::const_proxy_type
135 device<schema<VARTYPES...>, INTERFACE>::at(size_type index) const {
136
137 // Make sure that the index is within bounds.
138 assert(index < size());
139
140 // Use the unprotected function.
141 return this->operator[](index);
142}
143
144template <typename... VARTYPES, template <typename> class INTERFACE>
145VECMEM_HOST_AND_DEVICE
146 typename device<schema<VARTYPES...>, INTERFACE>::proxy_type
148
149 // Create the proxy.
150 return proxy_type{*this, index};
151}
152
153template <typename... VARTYPES, template <typename> class INTERFACE>
154VECMEM_HOST_AND_DEVICE
155 typename device<schema<VARTYPES...>, INTERFACE>::const_proxy_type
157
158 // Create the proxy.
159 return const_proxy_type{*this, index};
160}
161
162template <typename... VARTYPES, template <typename> class INTERFACE>
163VECMEM_HOST_AND_DEVICE auto device<schema<VARTYPES...>, INTERFACE>::variables()
164 -> tuple_type& {
165
166 return m_data;
167}
168
169template <typename... VARTYPES, template <typename> class INTERFACE>
170VECMEM_HOST_AND_DEVICE auto device<schema<VARTYPES...>, INTERFACE>::variables()
172
173 return m_data;
174}
175
176template <typename... VARTYPES, template <typename> class INTERFACE>
177template <std::size_t INDEX, std::size_t... Is>
178VECMEM_HOST_AND_DEVICE void
179device<schema<VARTYPES...>, INTERFACE>::construct_default(
180 size_type index, std::index_sequence<INDEX, Is...>) {
181
182 // Construct the new element in this variable, if it's a vector.
183 construct_vector(index, vecmem::get<INDEX>(m_data));
184 // Continue the recursion.
185 construct_default(index, std::index_sequence<Is...>{});
186}
187
188template <typename... VARTYPES, template <typename> class INTERFACE>
189VECMEM_HOST_AND_DEVICE void
190device<schema<VARTYPES...>, INTERFACE>::construct_default(
191 size_type, std::index_sequence<>) {}
192
193template <typename... VARTYPES, template <typename> class INTERFACE>
194template <typename T>
195VECMEM_HOST_AND_DEVICE void
196device<schema<VARTYPES...>, INTERFACE>::construct_vector(size_type, T&) {}
197
198template <typename... VARTYPES, template <typename> class INTERFACE>
199template <typename T>
200VECMEM_HOST_AND_DEVICE void
201device<schema<VARTYPES...>, INTERFACE>::construct_vector(
202 size_type index, device_vector<T>& vec) {
203
204 vec.construct(index, {});
205}
206
207} // namespace edm
208} // namespace vecmem
interface_type< proxy< schema_type, details::proxy_domain::device, details::proxy_access::non_constant, details::proxy_type::reference > > proxy_type
The type of the (non-const) proxy objects for the container elements.
Definition device.hpp:58
interface_type< proxy< schema_type, details::proxy_domain::device, details::proxy_access::non_constant, details::proxy_type::standalone > > object_type
Type type of standalone proxy objects for the container.
Definition device.hpp:67
interface_type< proxy< schema_type, details::proxy_domain::device, details::proxy_access::constant, details::proxy_type::reference > > const_proxy_type
The type of the (const) proxy objects for the container elements.
Definition device.hpp:62
typename view< schema_type >::size_type size_type
Size type used for the container.
Definition device.hpp:46
Technical base type for device<schema<VARTYPES...>,INTERFACE>
Definition device_vector.hpp:25
Technical base type for view<schema<VARTYPES...>>
Definition view.hpp:28
Main namespace for the vecmem classes/functions.
Definition atomic_ref.hpp:16
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
typename tuple_element< I, T >::type tuple_element_t
Convenience accessor for the I-th element of a tuple.
Definition tuple.hpp:211
Implementation for std::disjunction.
Definition type_traits.hpp:72
Definition device_traits.hpp:70
Definition device_traits.hpp:55
Meta type describing the "schema" of an SoA container.
Definition schema.hpp:46
Definition schema_traits.hpp:82
Definition schema_traits.hpp:64
Default tuple type.
Definition tuple.hpp:24