vecmem 1.14.0
Loading...
Searching...
No Matches
host.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/host_traits.hpp"
11#include "vecmem/edm/details/schema_traits.hpp"
12
13// System include(s).
14#include <string>
15#include <tuple>
16#include <type_traits>
17
18namespace vecmem {
19namespace edm {
20
21template <typename... VARTYPES, template <typename> class INTERFACE>
22VECMEM_HOST host<schema<VARTYPES...>, INTERFACE>::host(
23 memory_resource& resource)
24 : m_data{details::host_alloc<VARTYPES>::make(resource)...},
25 m_resource{resource} {}
26
27template <typename... VARTYPES, template <typename> class INTERFACE>
28VECMEM_HOST std::size_t host<schema<VARTYPES...>, INTERFACE>::size() const {
29
30 // Make sure that there are some (jagged) vector types in the container.
31 static_assert(
32 std::disjunction_v<type::details::is_vector<VARTYPES>...>,
33 "This function requires at least one (jagged) vector variable.");
34
35 // Get the size of the vector(s).
36 return details::get_host_size<VARTYPES...>(
37 m_data, std::index_sequence_for<VARTYPES...>{});
38}
39
40template <typename... VARTYPES, template <typename> class INTERFACE>
41VECMEM_HOST void host<schema<VARTYPES...>, INTERFACE>::resize(
42 std::size_t size) {
43
44 // Make sure that there are some (jagged) vector types in the container.
45 static_assert(
46 std::disjunction_v<type::details::is_vector<VARTYPES>...>,
47 "This function requires at least one (jagged) vector variable.");
48
49 // Resize the vector(s).
50 details::host_resize<VARTYPES...>(m_data, size,
51 std::index_sequence_for<VARTYPES...>{});
52}
53
54template <typename... VARTYPES, template <typename> class INTERFACE>
55VECMEM_HOST void host<schema<VARTYPES...>, INTERFACE>::reserve(
56 std::size_t size) {
57
58 // Make sure that there are some (jagged) vector types in the container.
59 static_assert(
60 std::disjunction_v<type::details::is_vector<VARTYPES>...>,
61 "This function requires at least one (jagged) vector variable.");
62
63 // Resize the vector(s).
64 details::host_reserve<VARTYPES...>(m_data, size,
65 std::index_sequence_for<VARTYPES...>{});
66}
67
68template <typename... VARTYPES, template <typename> class INTERFACE>
69VECMEM_HOST void host<schema<VARTYPES...>, INTERFACE>::push_back(
70 const object_type& element) {
71
72 // Make sure that there are some (jagged) vector types in the container.
73 static_assert(
74 std::disjunction_v<type::details::is_vector<VARTYPES>...>,
75 "This function requires at least one (jagged) vector variable.");
76
77 // Add a new element to the container.
78 const size_type index = size();
79 resize(index + 1);
80 // Set the new element.
81 at(index) = element;
82}
83
84template <typename... VARTYPES, template <typename> class INTERFACE>
85template <std::size_t INDEX>
86VECMEM_HOST typename details::host_type_at<INDEX, VARTYPES...>::return_type
88
89 // For scalar types we don't return the array, but rather a
90 // reference to the single scalar held by the array.
91 if constexpr (type::details::is_scalar_v<typename std::tuple_element<
92 INDEX, std::tuple<VARTYPES...>>::type>) {
93 return std::get<INDEX>(m_data)[0];
94 } else {
95 return std::get<INDEX>(m_data);
96 }
97}
98
99template <typename... VARTYPES, template <typename> class INTERFACE>
100template <std::size_t INDEX>
101VECMEM_HOST
102 typename details::host_type_at<INDEX, VARTYPES...>::const_return_type
104
105 // For scalar types we don't return the array, but rather a
106 // reference to the single scalar held by the array.
107 if constexpr (type::details::is_scalar_v<typename std::tuple_element<
108 INDEX, std::tuple<VARTYPES...>>::type>) {
109 return std::get<INDEX>(m_data)[0];
110 } else {
111 return std::get<INDEX>(m_data);
112 }
113}
114
115template <typename... VARTYPES, template <typename> class INTERFACE>
116VECMEM_HOST typename host<schema<VARTYPES...>, INTERFACE>::proxy_type
118
119 // Make sure that the index is within bounds.
120 if (index >= size()) {
121 throw std::out_of_range("index (" + std::to_string(index) +
122 ") >= size (" + std::to_string(size()) + ")");
123 }
124
125 // Use the unprotected function.
126 return this->operator[](index);
127}
128
129template <typename... VARTYPES, template <typename> class INTERFACE>
130VECMEM_HOST typename host<schema<VARTYPES...>, INTERFACE>::const_proxy_type
131host<schema<VARTYPES...>, INTERFACE>::at(size_type index) const {
132
133 // Make sure that the index is within bounds.
134 if (index >= size()) {
135 throw std::out_of_range("index (" + std::to_string(index) +
136 ") >= size (" + std::to_string(size()) + ")");
137 }
138
139 // Use the unprotected function.
140 return this->operator[](index);
141}
142
143template <typename... VARTYPES, template <typename> class INTERFACE>
144VECMEM_HOST typename host<schema<VARTYPES...>, INTERFACE>::proxy_type
146
147 // Create the proxy.
148 return proxy_type{*this, index};
149}
150
151template <typename... VARTYPES, template <typename> class INTERFACE>
152VECMEM_HOST typename host<schema<VARTYPES...>, INTERFACE>::const_proxy_type
154
155 // Create the proxy.
156 return const_proxy_type{*this, index};
157}
158
159template <typename... VARTYPES, template <typename> class INTERFACE>
160VECMEM_HOST typename host<schema<VARTYPES...>, INTERFACE>::tuple_type&
161host<schema<VARTYPES...>, INTERFACE>::variables() {
162
163 return m_data;
164}
165
166template <typename... VARTYPES, template <typename> class INTERFACE>
167VECMEM_HOST const typename host<schema<VARTYPES...>, INTERFACE>::tuple_type&
168host<schema<VARTYPES...>, INTERFACE>::variables() const {
169
170 return m_data;
171}
172
173template <typename... VARTYPES, template <typename> class INTERFACE>
174VECMEM_HOST memory_resource& host<schema<VARTYPES...>, INTERFACE>::resource()
175 const {
176
177 return m_resource;
178}
179
180} // namespace edm
181
183template <typename... VARTYPES, template <typename> class INTERFACE>
186 memory_resource&, std::index_sequence<>) {
187 // For SonarCloud: The terminal node doesn't need to do anything.
188}
189
191template <typename... VARTYPES, template <typename> class INTERFACE,
192 std::size_t I, std::size_t... Is>
193VECMEM_HOST void get_data_impl(
195 edm::data<edm::schema<VARTYPES...>>& data, memory_resource& mr,
196 std::index_sequence<I, Is...>) {
197
198 if constexpr (edm::type::details::is_jagged_vector_v<
199 typename std::tuple_element<
200 I, std::tuple<VARTYPES...>>::type>) {
201 // Make the @c vecmem::edm::data object hold on to the
202 // @c vecmem::data::jagged_vector_data object. Notice that this is a
203 // move assignment here.
204 std::get<I>(data.variables()) =
205 ::vecmem::get_data(host.template get<I>(), &mr);
206 // Set up the @c vecmem::edm::view object to point at the
207 // @c vecmem::data::jagged_vector_data object.
208 data.template get<I>() =
209 ::vecmem::get_data(std::get<I>(data.variables()));
210 } else if constexpr (edm::type::details::is_scalar<
211 typename std::tuple_element<
212 I, std::tuple<VARTYPES...>>::type>::value) {
213 // For scalar variables we just make @c vecmem::edm::view remember
214 // a pointer.
215 data.template get<I>() = &(host.template get<I>());
216 } else {
217 // For 1D vectors it's enough to make @c vecmem::edm::view have a
218 // correct @c vecmem::data::vector_view object.
219 data.template get<I>() = ::vecmem::get_data(host.template get<I>());
220 }
221 // Continue the recursion.
222 get_data_impl(host, data, mr, std::index_sequence<Is...>{});
223}
224
225template <typename... VARTYPES, template <typename> class INTERFACE>
228 memory_resource* resource) {
229
230 // Create the result object.
232 // Decide what memory resource to use for setting it up.
233 memory_resource& mr = (resource != nullptr ? *resource : host.resource());
234 // Set its size, if that's available. Note that if there are no vector
235 // variables in the container, then @c vecmem::edm::data also doesn't need
236 // a memory resource.
237 if constexpr (std::disjunction_v<
239 result = {static_cast<
240 typename edm::data<edm::schema<VARTYPES...>>::size_type>(
241 host.size()),
242 mr};
243 }
244 // Fill it with the helper function.
245 get_data_impl<VARTYPES...>(host, result, mr,
246 std::index_sequence_for<VARTYPES...>{});
247 // Return the filled object.
248 return result;
249}
250
252template <typename... VARTYPES, template <typename> class INTERFACE>
253VECMEM_HOST void get_data_impl(
255 edm::data<edm::details::add_const_t<edm::schema<VARTYPES...>>>&,
256 memory_resource&, std::index_sequence<>) {
257 // For SonarCloud: The terminal node doesn't need to do anything.
258}
259
261template <typename... VARTYPES, template <typename> class INTERFACE,
262 std::size_t I, std::size_t... Is>
263VECMEM_HOST void get_data_impl(
265 edm::data<edm::details::add_const_t<edm::schema<VARTYPES...>>>& data,
266 memory_resource& mr, std::index_sequence<I, Is...>) {
267
269 typename std::tuple_element<
270 I, std::tuple<VARTYPES...>>::type>::value) {
271 // Make the @c vecmem::edm::data object hold on to the
272 // @c vecmem::data::jagged_vector_data object. Notice that this is a
273 // move assignment here.
274 std::get<I>(data.variables()) =
275 ::vecmem::get_data(host.template get<I>(), &mr);
276 // Set up the @c vecmem::edm::view object to point at the
277 // @c vecmem::data::jagged_vector_data object.
278 data.template get<I>() =
279 ::vecmem::get_data(std::get<I>(data.variables()));
280 } else if constexpr (edm::type::details::is_scalar<
281 typename std::tuple_element<
282 I, std::tuple<VARTYPES...>>::type>::value) {
283 // For scalar variables we just make @c vecmem::edm::view remember
284 // a pointer.
285 data.template get<I>() = &(host.template get<I>());
286 } else {
287 // For 1D vectors it's enough to make @c vecmem::edm::view have a
288 // correct @c vecmem::data::vector_view object.
289 data.template get<I>() = ::vecmem::get_data(host.template get<I>());
290 }
291 // Continue the recursion.
292 get_data_impl(host, data, mr, std::index_sequence<Is...>{});
293}
294
295template <typename... VARTYPES, template <typename> class INTERFACE>
296VECMEM_HOST edm::data<edm::details::add_const_t<edm::schema<VARTYPES...>>>
298 memory_resource* resource) {
299
300 // Create the result object.
301 edm::data<edm::details::add_const_t<edm::schema<VARTYPES...>>> result;
302 // Decide what memory resource to use for setting it up.
303 memory_resource& mr = (resource != nullptr ? *resource : host.resource());
304 // Set its size, if that's available. Note that if there are no vector
305 // variables in the container, then @c vecmem::edm::data also doesn't need
306 // a memory resource.
307 if constexpr (std::disjunction_v<
309 result = {static_cast<
310 typename edm::view<edm::schema<VARTYPES...>>::size_type>(
311 host.size()),
312 mr};
313 }
314 // Fill it with the helper function.
315 get_data_impl<VARTYPES...>(host, result, mr,
316 std::index_sequence_for<VARTYPES...>{});
317 // Return the filled object.
318 return result;
319}
320
321} // namespace vecmem
Technical base type for data<schema<VARTYPES...>>
Definition data.hpp:25
interface_type< proxy< schema_type, details::proxy_domain::host, details::proxy_access::non_constant, details::proxy_type::standalone > > object_type
Type type of standalone proxy objects for the container.
Definition host.hpp:66
interface_type< proxy< schema_type, details::proxy_domain::host, details::proxy_access::non_constant, details::proxy_type::reference > > proxy_type
The type of the (non-const) proxy objects for the container elements.
Definition host.hpp:57
interface_type< proxy< schema_type, details::proxy_domain::host, details::proxy_access::constant, details::proxy_type::reference > > const_proxy_type
The type of the (const) proxy objects for the container elements.
Definition host.hpp:61
std::size_t size_type
Size type used for the container.
Definition host.hpp:46
Technical base type for host<schema<VARTYPES...>,INTERFACE>
Definition host.hpp:28
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
VECMEM_HOST void get_data_impl(edm::host< edm::schema< VARTYPES... >, INTERFACE > &, edm::data< edm::schema< VARTYPES... > > &, memory_resource &, std::index_sequence<>)
Helper function terminal node.
Definition host.ipp:184
std::vector< T, vecmem::polymorphic_allocator< T > > vector
Alias type for vectors with our polymorphic allocator.
Definition vector.hpp:35
VECMEM_HOST data::vector_view< T > get_data(array< T, N > &a)
Helper function creating a vecmem::data::vector_view object.
Definition array.ipp:217
Definition host_traits.hpp:56
Meta type describing the "schema" of an SoA container.
Definition schema.hpp:46
Definition schema_traits.hpp:82
Definition schema_traits.hpp:51
Definition schema_traits.hpp:64