vecmem 1.14.0
Loading...
Searching...
No Matches
copy.ipp
1/*
2 * VecMem project, part of the ACTS project (R&D line)
3 *
4 * (c) 2021-2024 CERN for the benefit of the ACTS project
5 *
6 * Mozilla Public License Version 2.0
7 */
8#pragma once
9
10// VecMem include(s).
11#include "vecmem/containers/details/resize_jagged_vector.hpp"
12#include "vecmem/containers/jagged_vector.hpp"
13#include "vecmem/edm/details/schema_traits.hpp"
14#include "vecmem/memory/host_memory_resource.hpp"
15#include "vecmem/utils/debug.hpp"
16#include "vecmem/utils/type_traits.hpp"
17
18// System include(s).
19#include <algorithm>
20#include <cassert>
21#include <numeric>
22#include <sstream>
23#include <stdexcept>
24#include <type_traits>
25
26namespace vecmem {
27
28template <typename TYPE>
29copy::event_type copy::setup(data::vector_view<TYPE> data) const {
30
31 // Check if anything needs to be done.
32 if ((data.size_ptr() == nullptr) || (data.capacity() == 0)) {
34 }
35
36 // Initialize the "size variable" correctly on the buffer.
38 data.size_ptr(), 0);
39 VECMEM_DEBUG_MSG(2,
40 "Prepared a device vector buffer of capacity %u "
41 "for use on a device (ptr: %p)",
42 data.capacity(), static_cast<void*>(data.size_ptr()));
43
44 // Return a new event.
45 return create_event();
46}
47
48template <typename TYPE>
49copy::event_type copy::memset(data::vector_view<TYPE> data, int value) const {
50
51 // Check if anything needs to be done.
52 if (data.capacity() == 0) {
54 }
55
56 // Call memset with the correct arguments.
57 do_memset(data.capacity() * sizeof(TYPE), data.ptr(), value);
58 VECMEM_DEBUG_MSG(2, "Set %u vector elements to %i at ptr: %p",
59 data.capacity(), value, static_cast<void*>(data.ptr()));
60
61 // Return a new event.
62 return create_event();
63}
64
65template <typename TYPE>
67 const vecmem::data::vector_view<TYPE>& data, memory_resource& resource,
68 type::copy_type cptype) const {
69
70 // Set up the result buffer.
72 resource);
73 setup(result)->wait();
74
75 // Copy the payload of the vector. Explicitly waiting for the copy to finish
76 // before returning the buffer.
77 operator()(data, result, cptype)->wait();
78
79 // Return the buffer.
80 return result;
81}
82
83template <typename TYPE>
85 const data::vector_view<std::add_const_t<TYPE>>& from_view,
86 data::vector_view<TYPE> to_view, type::copy_type cptype) const {
87
88 // Perform the copy. Depending on whether an actual copy has been set up /
89 // performed, return either "an actual", or just a dummy event.
90 if (copy_view_impl(from_view, to_view, cptype)) {
91 return create_event();
92 } else {
94 }
95}
96
97template <typename TYPE, typename ALLOC>
99 const data::vector_view<std::add_const_t<TYPE>>& from_view,
100 std::vector<TYPE, ALLOC>& to_vec, type::copy_type cptype) const {
101
102 // Figure out the size of the buffer.
103 const typename data::vector_view<std::add_const_t<TYPE>>::size_type size =
105
106 // Make the target vector the correct size.
107 to_vec.resize(size);
108 // Perform the memory copy.
109 do_copy(size * sizeof(TYPE), from_view.ptr(), to_vec.data(), cptype);
110
111 // Return a new event.
112 return create_event();
113}
114
115template <typename TYPE>
117 const data::vector_view<TYPE>& data) const {
118
119 // Handle the simple case, when the view/buffer is not resizable.
120 if (data.size_ptr() == nullptr) {
121 return data.capacity();
122 }
123
124 // If it *is* resizable, don't assume that the size is host-accessible.
125 // Explicitly copy it for access.
128 data.size_ptr(), &result, type::unknown);
129
130 // Wait for the copy operation to finish. With some backends
131 // (khm... SYCL... khm...) copies can be asynchronous even into
132 // non-pinned host memory.
133 create_event()->wait();
134
135 // Return what we got.
136 return result;
137}
138
139template <typename TYPE>
141
142 // Check if anything needs to be done.
143 if (data.size() == 0) {
145 }
146
147 // "Set up" the inner vector descriptors, using the host-accessible data.
148 // But only if the jagged vector buffer is resizable.
149 if (data.host_ptr()[0].size_ptr() != nullptr) {
150 do_memset(
151 sizeof(typename data::vector_buffer<TYPE>::size_type) * data.size(),
152 data.host_ptr()[0].size_ptr(), 0);
153 }
154
155 // Check if anything else needs to be done.
156 if (data.ptr() == data.host_ptr()) {
157 return create_event();
158 }
159
160 // Copy the description of the inner vectors of the buffer.
161 do_copy(
162 data.size() *
163 sizeof(
165 data.host_ptr(), data.ptr(), type::host_to_device);
166 VECMEM_DEBUG_MSG(2,
167 "Prepared a jagged device vector buffer of size %lu "
168 "for use on a device",
169 data.size());
170
171 // Return a new event.
172 return create_event();
173}
174
175template <typename TYPE>
176copy::event_type copy::memset(data::jagged_vector_view<TYPE> data,
177 int value) const {
178
179 // Do different things for jagged vectors that are contiguous in memory.
180 if (is_contiguous(data.host_ptr(), data.capacity())) {
181 // For contiguous jagged vectors we can just do a single memset
182 // operation. First, let's calculate the "total size" of the jagged
183 // vector.
184 const std::size_t total_size = std::accumulate(
185 data.host_ptr(), data.host_ptr() + data.size(),
186 static_cast<std::size_t>(0u),
187 [](std::size_t sum, const data::vector_view<TYPE>& iv) {
188 return sum + iv.capacity();
189 });
190 // Find the first "inner vector" that has a non-zero capacity.
191 for (std::size_t i = 0; i < data.size(); ++i) {
192 data::vector_view<TYPE>& iv = data.host_ptr()[i];
193 if ((iv.capacity() != 0u) && (iv.ptr() != nullptr)) {
194 // Call memset with its help.
195 do_memset(total_size * sizeof(TYPE), iv.ptr(), value);
196 return create_event();
197 }
198 }
199 // If we are still here, apparently we didn't need to do anything.
201 } else {
202 // For non-contiguous jagged vectors call memset one-by-one on the
203 // inner vectors. Note that we don't use vecmem::copy::memset here,
204 // as that would require us to wait for each memset individually.
205 for (std::size_t i = 0; i < data.size(); ++i) {
206 data::vector_view<TYPE>& iv = data.host_ptr()[i];
207 do_memset(iv.capacity() * sizeof(TYPE), iv.ptr(), value);
208 }
209 }
210
211 // Return a new event.
212 return create_event();
213}
214
215template <typename TYPE>
217 const data::jagged_vector_view<TYPE>& data, memory_resource& resource,
218 memory_resource* host_access_resource, type::copy_type cptype) const {
219
220 // Create the result buffer object.
222 data, resource, host_access_resource);
223 assert(result.size() == data.size());
224
225 // Copy the description of the "inner vectors" if necessary.
226 setup(result)->wait();
227
228 // Copy the payload of the inner vectors. Explicitly waiting for the copy to
229 // finish before returning the buffer.
230 operator()(data, result, cptype)->wait();
231
232 // Return the newly created object.
233 return result;
234}
235
236template <typename TYPE>
238 const data::jagged_vector_view<std::add_const_t<TYPE>>& from_view,
239 data::jagged_vector_view<TYPE> to_view, type::copy_type cptype) const {
240
241 // Perform the copy. Depending on whether an actual copy has been set up /
242 // performed, return either "an actual", or just a dummy event.
243 if (copy_view_impl(from_view, to_view, cptype)) {
244 return create_event();
245 } else {
247 }
248}
249
250template <typename TYPE, typename ALLOC1, typename ALLOC2>
252 const data::jagged_vector_view<std::add_const_t<TYPE>>& from_view,
253 std::vector<std::vector<TYPE, ALLOC2>, ALLOC1>& to_vec,
254 type::copy_type cptype) const {
255
256 // Resize the output object to the correct size.
258 const auto sizes = get_sizes(from_view);
259 assert(sizes.size() == to_vec.size());
260 for (typename data::jagged_vector_view<std::add_const_t<TYPE>>::size_type
261 i = 0;
262 i < from_view.size(); ++i) {
263 to_vec[i].resize(sizes[i]);
264 }
265
266 // Perform the memory copy.
268}
269
270template <typename TYPE>
271std::vector<typename data::vector_view<TYPE>::size_type> copy::get_sizes(
272 const data::jagged_vector_view<TYPE>& data) const {
273
274 // Perform the operation using the private function.
275 return get_sizes_impl(data.host_ptr(), data.size());
276}
277
278template <typename TYPE>
280 const std::vector<typename data::vector_view<TYPE>::size_type>& sizes,
282
283 // Finish early if possible.
284 if ((sizes.size() == 0) && (data.size() == 0)) {
286 }
287 // Make sure that the sizes match up.
288 if (sizes.size() != data.size()) {
289 std::ostringstream msg;
290 msg << "sizes.size() (" << sizes.size() << ") != data.size() ("
291 << data.size() << ")";
292 throw std::length_error(msg.str());
293 }
294 // Make sure that the target jagged vector is either resizable, or it has
295 // the correct sizes/capacities already.
296 bool perform_copy = true;
298 i < data.size(); ++i) {
299 // Perform a capacity check in all cases. Whether or not the target is
300 // resizable.
301 if (data.host_ptr()[i].capacity() < sizes[i]) {
302 std::ostringstream msg;
303 msg << "data.host_ptr()[" << i << "].capacity() ("
304 << data.host_ptr()[i].capacity() << ") < sizes[" << i << "] ("
305 << sizes[i] << ")";
306 throw std::length_error(msg.str());
307 }
308 // Make sure that the inner vectors are consistently either all
309 // resizable, or none of them are.
310 if (data.host_ptr()[i].size_ptr() == nullptr) {
311 perform_copy = false;
312 } else if (perform_copy == false) {
313 throw std::runtime_error(
314 "Inconsistent target jagged vector view received for resizing");
315 }
316 }
317 // If no copy is necessary, we're done.
318 if (perform_copy == false) {
320 }
321 // Perform the copy with some internal knowledge of how resizable jagged
322 // vector buffers work.
323 do_copy(sizeof(typename data::vector_view<TYPE>::size_type) * sizes.size(),
324 sizes.data(), data.host_ptr()->size_ptr(), type::unknown);
325
326 // Return a new event.
327 return create_event();
328}
329
330template <typename SCHEMA>
331copy::event_type copy::setup(edm::view<SCHEMA> data) const {
332
333 // For empty containers nothing needs to be done.
334 if (data.capacity() == 0) {
336 }
337
338 // Copy the data layout to the device, if needed.
339 if (data.layout().ptr() != data.host_layout().ptr()) {
340 assert(data.layout().capacity() > 0u);
341 [[maybe_unused]] bool did_copy =
342 copy_view_impl(data.host_layout(), data.layout(), type::unknown);
344 }
345
346 // Initialize the "size variable(s)" correctly on the buffer.
347 if (data.size().ptr() != nullptr) {
348 assert(data.size().capacity() > 0u);
349 do_memset(data.size().capacity() * sizeof(char), data.size().ptr(), 0);
350 }
351 VECMEM_DEBUG_MSG(3,
352 "Prepared an SoA container of capacity %u "
353 "for use on a device (layout: {%u, %p}, size: {%u, %p})",
354 data.capacity(), data.layout().size(),
355 static_cast<void*>(data.layout().ptr()),
356 data.size().size(), static_cast<void*>(data.size().ptr()));
357
358 // Return a new event.
359 return create_event();
360}
361
362template <typename... VARTYPES>
363copy::event_type copy::memset(edm::view<edm::schema<VARTYPES...>> data,
364 int value) const {
365
366 // For buffers, we can do this in one go.
367 if (data.payload().ptr() != nullptr) {
368 memset(data.payload(), value);
369 } else {
370 // Do the operation using the helper function, recursively.
371 memset_impl<0>(data, value);
372 }
373
374 // Return a new event.
375 return create_event();
376}
377
378template <typename... VARTYPES>
380 const edm::view<edm::details::add_const_t<edm::schema<VARTYPES...>>>&
381 from_view,
382 edm::view<edm::schema<VARTYPES...>> to_view, type::copy_type cptype) const {
383
384 // First, handle the simple case, when both views have a contiguous memory
385 // layout.
386 if ((from_view.payload().ptr() != nullptr) &&
387 (to_view.payload().ptr() != nullptr) &&
388 (from_view.payload().capacity() == to_view.payload().capacity())) {
389
390 // If the "common size" is zero, we're done.
391 if (from_view.payload().capacity() == 0) {
393 }
394
395 // Let the user know what's happening.
396 VECMEM_DEBUG_MSG(2, "Performing simple SoA copy of %u bytes",
397 from_view.payload().size());
398
399 // Copy the payload with a single copy operation.
400 copy_view_impl(from_view.payload(), to_view.payload(), cptype);
401
402 // If the target view is resizable, set its size.
403 if (to_view.size().ptr() != nullptr) {
404 // If the source is also resizable, the situation should be simple.
405 if (from_view.size().ptr() != nullptr) {
406 // Check that the sizes are the same.
407 if (from_view.size().capacity() != to_view.size().capacity()) {
408 std::ostringstream msg;
409 msg << "from_view.size().capacity() ("
410 << from_view.size().capacity()
411 << ") != to_view.size().capacity() ("
412 << to_view.size().capacity() << ")";
413 throw std::length_error(msg.str());
414 }
415 // Perform a dumb copy.
416 copy_view_impl(from_view.size(), to_view.size(), cptype);
417 } else {
418 // If not, then copy the size(s) recursively.
420 }
421 }
422
423 // Create a synchronization event.
424 return create_event();
425 }
426
427 // If not, then do an un-optimized copy, variable-by-variable.
429
430 // Return a new event.
431 return create_event();
432}
433
434template <typename... VARTYPES, template <typename> class INTERFACE>
436 const edm::view<edm::details::add_const_t<edm::schema<VARTYPES...>>>&
437 from_view,
438 edm::host<edm::schema<VARTYPES...>, INTERFACE>& to_vec,
439 type::copy_type cptype) const {
440
441 // Resize the output object to the correct size(s).
443
444 // Perform the memory copy.
446}
447
448template <typename... VARTYPES>
450 const edm::view<edm::schema<VARTYPES...>>& data) const {
451
452 // Start by taking the capacity of the container.
453 typename edm::view<edm::schema<VARTYPES...>>::size_type size =
454 data.capacity();
455
456 // If there are jagged vectors in the container and/or the container is not
457 // resizable, we're done. It is done in two separate if statements to avoid
458 // MSVC trying to be too smart, and giving a warning...
459 if constexpr (std::disjunction_v<
461 return size;
462 } else {
463 // All the rest is put into an else block, to avoid MSVC trying to be
464 // too smart, and giving a warning about unreachable code...
465 if (data.size().ptr() == nullptr) {
466 return size;
467 }
468
469 // A small security check.
470 assert(data.size().size() ==
471 sizeof(typename edm::view<edm::schema<VARTYPES...>>::size_type));
472
473 // Get the exact size of the container.
474 do_copy(sizeof(typename edm::view<edm::schema<VARTYPES...>>::size_type),
475 data.size().ptr(), &size, type::unknown);
476 // We have to wait for this to finish, since the "size" variable is
477 // not going to be available outside of this function. And
478 // asynchronous SYCL memory copies can happen from variables on the
479 // stack as well...
480 create_event()->wait();
481
482 // Return what we got.
483 return size;
484 }
485}
486
487template <typename TYPE>
488bool copy::copy_view_impl(
489 const data::vector_view<std::add_const_t<TYPE>>& from_view,
490 data::vector_view<TYPE> to_view, type::copy_type cptype) const {
491
492 // Get the size of the source view.
493 const typename data::vector_view<std::add_const_t<TYPE>>::size_type size =
495 // If it's zero, don't do an actual copy.
496 if (size == 0u) {
497 return false;
498 }
499
500 // Make sure that the copy can happen.
501 if (to_view.capacity() < size) {
502 std::ostringstream msg;
503 msg << "Target capacity (" << to_view.capacity() << ") < source size ("
504 << size << ")";
505 throw std::length_error(msg.str());
506 }
507
508 // Make sure that if the target view is resizable, that it would be set up
509 // for the correct size.
510 if (to_view.size_ptr() != nullptr) {
511 // Select what type of copy this should be. Keeping in mind that we copy
512 // from a variable on the host stack. So the question is just whether
513 // the target is the host, or a device.
515 switch (cptype) {
519 break;
523 break;
524 default:
525 break;
526 }
527 // Perform the copy.
528 do_copy(sizeof(typename data::vector_view<TYPE>::size_type), &size,
529 to_view.size_ptr(), size_cptype);
530 // We have to wait for this to finish, since the "size" variable is not
531 // going to be available outside of this function. And asynchronous SYCL
532 // memory copies can happen from variables on the stack as well...
533 create_event()->wait();
534 }
535
536 // Copy the payload.
537 do_copy(size * sizeof(TYPE), from_view.ptr(), to_view.ptr(), cptype);
538 return true;
539}
540
541template <typename TYPE>
542bool copy::copy_view_impl(
543 const data::jagged_vector_view<std::add_const_t<TYPE>>& from_view,
544 data::jagged_vector_view<TYPE> to_view, type::copy_type cptype) const {
545
546 // Sanity checks.
547 if (from_view.size() > to_view.size()) {
548 std::ostringstream msg;
549 msg << "from_view.size() (" << from_view.size()
550 << ") > to_view.size() (" << to_view.size() << ")";
551 throw std::length_error(msg.str());
552 }
553
554 // Get the "outer size" of the container.
555 const typename data::jagged_vector_view<std::add_const_t<TYPE>>::size_type
556 size = from_view.size();
557 if (size == 0u) {
558 return false;
559 }
560
561 // Calculate the contiguous-ness of the memory allocations.
562 const bool from_is_contiguous = is_contiguous(from_view.host_ptr(), size);
563 const bool to_is_contiguous = is_contiguous(to_view.host_ptr(), size);
564 VECMEM_DEBUG_MSG(3, "from_is_contiguous = %d, to_is_contiguous = %d",
566
567 // Get the sizes of the source jagged vector.
568 const auto sizes = get_sizes(from_view);
569
570 // Before even attempting the copy, make sure that the target view either
571 // has the correct sizes, or can be resized correctly. Captire the event
572 // marking the end of the operation. We need to wait for the copy to finish
573 // before returning from this function.
575
576 // Check whether the source and target capacities match up. We can only
577 // perform the "optimised copy" if they do.
578 std::vector<typename data::vector_view<std::add_const_t<TYPE>>::size_type>
579 capacities(size);
580 bool capacities_match = true;
581 for (std::size_t i = 0; i < size; ++i) {
582 if (from_view.host_ptr()[i].capacity() !=
583 to_view.host_ptr()[i].capacity()) {
584 capacities_match = false;
585 break;
586 }
587 capacities[i] = from_view.host_ptr()[i].capacity();
588 }
589
590 // Perform the copy as best as we can.
592 // Perform the copy in one go.
593 copy_views_contiguous_impl(capacities, from_view.host_ptr(),
594 to_view.host_ptr(), cptype);
595 } else {
596 // Do the copy as best as we can. Note that since they are not
597 // contiguous anyway, we use the sizes of the vectors here, not their
598 // capcities.
599 copy_views_impl(sizes, from_view.host_ptr(), to_view.host_ptr(),
600 cptype);
601 }
602
603 // Before returning, make sure that the "size set" operation has finished.
604 // Because the "sizes" variable is just about to go out of scope.
605 set_sizes_event->wait();
606 return true;
607}
608
609template <typename TYPE>
610void copy::copy_views_impl(
611 const std::vector<typename data::vector_view<TYPE>::size_type>& sizes,
612 const data::vector_view<std::add_const_t<TYPE>>* from_view,
613 data::vector_view<TYPE>* to_view, type::copy_type cptype) const {
614
615 // Some security checks.
616 assert(from_view != nullptr);
617 assert(to_view != nullptr);
618
619 // Helper variable(s) used in the copy.
620 const std::size_t size = sizes.size();
621 [[maybe_unused]] std::size_t copy_ops = 0;
622
623 // Perform the copy in multiple steps.
624 for (std::size_t i = 0; i < size; ++i) {
625
626 // Skip empty "inner vectors".
627 if (sizes[i] == 0) {
628 continue;
629 }
630
631 // Some sanity checks.
632 assert(from_view[i].ptr() != nullptr);
633 assert(to_view[i].ptr() != nullptr);
634 assert(sizes[i] <= from_view[i].capacity());
635 assert(sizes[i] <= to_view[i].capacity());
636
637 // Perform the copy.
638 do_copy(sizes[i] * sizeof(TYPE), from_view[i].ptr(), to_view[i].ptr(),
639 cptype);
640 ++copy_ops;
641 }
642
643 // Let the user know what happened.
644 VECMEM_DEBUG_MSG(2,
645 "Copied the payload of a jagged vector of type "
646 "\"%s\" with %lu copy operation(s)",
647 typeid(TYPE).name(), copy_ops);
648}
649
650template <typename TYPE>
651void copy::copy_views_contiguous_impl(
652 const std::vector<typename data::vector_view<TYPE>::size_type>& sizes,
653 const data::vector_view<std::add_const_t<TYPE>>* from_view,
654 data::vector_view<TYPE>* to_view, type::copy_type cptype) const {
655
656 // Some security checks.
657 assert(from_view != nullptr);
658 assert(to_view != nullptr);
659 assert(is_contiguous(from_view, sizes.size()));
660 assert(is_contiguous(to_view, sizes.size()));
661
662 // Helper variable(s) used in the copy.
663 const std::size_t size = sizes.size();
664 const std::size_t total_size =
665 std::accumulate(sizes.begin(), sizes.end(),
666 static_cast<std::size_t>(0)) *
667 sizeof(TYPE);
668
669 // Find the first non-empty element.
670 for (std::size_t i = 0; i < size; ++i) {
671
672 // Jump over empty elements.
673 if (sizes[i] == 0) {
674 continue;
675 }
676
677 // Some sanity checks.
678 assert(from_view[i].ptr() != nullptr);
679 assert(to_view[i].ptr() != nullptr);
680
681 // Perform the copy.
682 do_copy(total_size, from_view[i].ptr(), to_view[i].ptr(), cptype);
683 break;
684 }
685
686 // Let the user know what happened.
687 VECMEM_DEBUG_MSG(2,
688 "Copied the payload of a jagged vector of type "
689 "\"%s\" with 1 copy operation(s)",
690 typeid(TYPE).name());
691}
692
693template <typename TYPE>
694std::vector<typename data::vector_view<TYPE>::size_type> copy::get_sizes_impl(
695 const data::vector_view<TYPE>* data, std::size_t size) const {
696
697 // Create the result vector.
698 std::vector<typename data::vector_view<TYPE>::size_type> result(size, 0);
699
700 // Try to get the "resizable sizes" first.
701 for (std::size_t i = 0; i < size; ++i) {
702 // Find the first "inner vector" that has a non-zero capacity, and is
703 // resizable.
704 if ((data[i].capacity() != 0) && (data[i].size_ptr() != nullptr)) {
705 // Copy the sizes of the inner vectors into the result vector.
707 (size - i),
708 data[i].size_ptr(), result.data() + i, type::unknown);
709 // Wait for the copy operation to finish. With some backends
710 // (khm... SYCL... khm...) copies can be asynchronous even into
711 // non-pinned host memory.
712 create_event()->wait();
713 // At this point the result vector should have been set up
714 // correctly.
715 return result;
716 }
717 }
718
719 // If we're still here, then the buffer is not resizable. So let's just
720 // collect the capacity of each of the inner vectors.
721 for (std::size_t i = 0; i < size; ++i) {
722 result[i] = data[i].capacity();
723 }
724 return result;
725}
726
727template <typename TYPE>
728bool copy::is_contiguous(const data::vector_view<TYPE>* data,
729 std::size_t size) {
730
731 // We should never call this function for an empty jagged vector.
732 assert(size > 0);
733
734 // Check whether all memory blocks are contiguous.
735 auto ptr = data[0].ptr();
736 for (std::size_t i = 1; i < size; ++i) {
737 if ((ptr + data[i - 1].capacity()) != data[i].ptr()) {
738 return false;
739 }
740 ptr = data[i].ptr();
741 }
742 return true;
743}
744
745template <std::size_t INDEX, typename... VARTYPES>
746void copy::memset_impl(edm::view<edm::schema<VARTYPES...>> data,
747 int value) const {
748
749 // Scalars do not have their own dedicated @c memset functions.
750 if constexpr (edm::type::details::is_scalar<typename std::tuple_element<
751 INDEX, std::tuple<VARTYPES...>>::type>::value) {
752 do_memset(sizeof(typename std::tuple_element<
753 INDEX, std::tuple<VARTYPES...>>::type::type),
754 data.template get<INDEX>(), value);
755 } else {
756 // But vectors and jagged vectors do.
757 memset(data.template get<INDEX>(), value);
758 }
759 // Recurse into the next variable.
760 if constexpr (sizeof...(VARTYPES) > (INDEX + 1)) {
761 memset_impl<INDEX + 1>(data, value);
762 }
763}
764
765template <std::size_t INDEX, typename... VARTYPES,
766 template <typename> class INTERFACE>
767void copy::resize_impl(
768 const edm::view<edm::details::add_const_t<edm::schema<VARTYPES...>>>&
769 from_view,
770 edm::host<edm::schema<VARTYPES...>, INTERFACE>& to_vec,
771 [[maybe_unused]] type::copy_type cptype) const {
772
773 // The target is a host container, so the copy type can't be anything
774 // targeting a device.
776 (cptype == type::unknown));
777
778 // First, handle containers with no jagged vectors in them.
779 if constexpr (std::disjunction_v<
780 edm::type::details::is_jagged_vector<VARTYPES>...> ==
781 false) {
782 // The single size of the source container.
783 typename edm::view<
784 edm::details::add_const_t<edm::schema<VARTYPES...>>>::size_type
785 size = from_view.capacity();
786 // If the container is resizable, take its size.
787 if (from_view.size().ptr() != nullptr) {
788 // A small security check.
789 assert(from_view.size().size() ==
790 sizeof(typename edm::view<edm::details::add_const_t<
791 edm::schema<VARTYPES...>>>::size_type));
792 // Get the exact size of the container.
793 do_copy(sizeof(typename edm::view<edm::details::add_const_t<
794 edm::schema<VARTYPES...>>>::size_type),
795 from_view.size().ptr(), &size, cptype);
796 // We have to wait for this to finish, since the "size" variable is
797 // not going to be available outside of this function. And
798 // asynchronous SYCL memory copies can happen from variables on the
799 // stack as well...
800 create_event()->wait();
801 }
802 // Resize the target container.
803 VECMEM_DEBUG_MSG(4, "Resizing a (non-jagged) container to size %u",
804 size);
805 to_vec.resize(size);
806 } else {
807 // Resize vector and jagged vector variables one by one. Note that
808 // evaluation order matters here. Since all jagged vectors are vectors,
809 // but not all vectors are jagged vectors. ;-)
810 if constexpr (edm::type::details::is_jagged_vector<
811 typename std::tuple_element<
812 INDEX, std::tuple<VARTYPES...>>::type>::value) {
813 // Get the sizes of this jagged vector.
814 auto sizes = get_sizes(from_view.template get<INDEX>());
815 // Set the "outer size" of the jagged vector.
816 VECMEM_DEBUG_MSG(
817 4, "Resizing jagged vector variable at index %lu to size %lu",
818 INDEX, sizes.size());
820 sizes.size());
821 // Set the "inner sizes" of the jagged vector.
822 for (std::size_t i = 0; i < sizes.size(); ++i) {
823 to_vec.template get<INDEX>()[i].resize(sizes[i]);
824 }
825 } else if constexpr (edm::type::details::is_vector<
826 typename std::tuple_element<
827 INDEX,
828 std::tuple<VARTYPES...>>::type>::value) {
829 // Get the size of this vector.
830 auto size = get_size(from_view.template get<INDEX>());
831 // Resize the target vector.
832 VECMEM_DEBUG_MSG(4,
833 "Resizing vector variable at index %lu to size %u",
834 INDEX, size);
835 to_vec.template get<INDEX>().resize(size);
836 }
837 // Call this function recursively.
838 if constexpr (sizeof...(VARTYPES) > (INDEX + 1)) {
840 }
841 }
842}
843
844template <std::size_t INDEX, typename... VARTYPES>
845void copy::copy_sizes_impl(
846 [[maybe_unused]] const edm::view<
847 edm::details::add_const_t<edm::schema<VARTYPES...>>>& from_view,
848 [[maybe_unused]] edm::view<edm::schema<VARTYPES...>> to_view,
849 [[maybe_unused]] type::copy_type cptype) const {
850
851 // This should only be called for a resizable target container, with
852 // a non-resizable source container.
853 assert(to_view.size().ptr() != nullptr);
854 assert(from_view.size().ptr() == nullptr);
855
856 // First, handle containers with no jagged vectors in them.
857 if constexpr (std::disjunction_v<
858 edm::type::details::is_jagged_vector<VARTYPES>...> ==
859 false) {
860 // Size of the source container.
861 typename edm::view<
862 edm::details::add_const_t<edm::schema<VARTYPES...>>>::size_type
863 size = from_view.capacity();
864 // Choose the copy type.
866 switch (cptype) {
870 break;
874 break;
875 default:
876 break;
877 }
878 // Set the size of the target container.
879 do_copy(sizeof(typename edm::view<edm::details::add_const_t<
880 edm::schema<VARTYPES...>>>::size_type),
881 &size, to_view.size().ptr(), size_cptype);
882 // We have to wait for this to finish, since the "size" variable is not
883 // going to be available outside of this function. And asynchronous SYCL
884 // memory copies can happen from variables on the stack as well...
885 create_event()->wait();
886 } else {
887 // For the jagged vector case we recursively copy the sizes of every
888 // jagged vector variable. The rest of the variables are not resizable
889 // in such a container, so they are ignored here.
890 if constexpr (edm::type::details::is_jagged_vector<
891 typename std::tuple_element<
892 INDEX, std::tuple<VARTYPES...>>::type>::value) {
893 // Copy the sizes for this variable.
894 const auto sizes = get_sizes(from_view.template get<INDEX>());
895 set_sizes(sizes, to_view.template get<INDEX>())->wait();
896 }
897 // Call this function recursively.
898 if constexpr (sizeof...(VARTYPES) > (INDEX + 1)) {
900 }
901 }
902}
903
904template <std::size_t INDEX, typename... VARTYPES>
905void copy::copy_payload_impl(
906 const edm::view<edm::details::add_const_t<edm::schema<VARTYPES...>>>&
907 from_view,
908 edm::view<edm::schema<VARTYPES...>> to_view, type::copy_type cptype) const {
909
910 // Scalars do not have their own dedicated @c copy functions.
911 if constexpr (edm::type::details::is_scalar<typename std::tuple_element<
912 INDEX, std::tuple<VARTYPES...>>::type>::value) {
913 do_copy(sizeof(typename std::tuple_element<
914 INDEX, std::tuple<VARTYPES...>>::type::type),
915 from_view.template get<INDEX>(), to_view.template get<INDEX>(),
916 cptype);
917 } else {
918 // But vectors and jagged vectors do.
919 copy_view_impl(from_view.template get<INDEX>(),
920 to_view.template get<INDEX>(), cptype);
921 }
922 // Recurse into the next variable.
923 if constexpr (sizeof...(VARTYPES) > (INDEX + 1)) {
925 }
926}
927
928} // namespace vecmem
VECMEM_NODISCARD event_type set_sizes(const std::vector< typename data::vector_view< TYPE >::size_type > &sizes, data::jagged_vector_view< TYPE > data) const
Helper function for setting the sizes of a resizable jagged vector.
VECMEM_NODISCARD event_type operator()(const data::vector_view< std::add_const_t< TYPE > > &from, data::vector_view< TYPE > to, type::copy_type cptype=type::unknown) const
Copy a 1-dimensional vector's data between two existing memory blocks.
VECMEM_NODISCARD event_type setup(data::vector_view< TYPE > data) const
Set up the internal state of a vector buffer correctly on a device.
virtual void do_copy(std::size_t size, const void *from, void *to, type::copy_type cptype) const
Perform a "low level" memory copy.
Definition copy.cpp:27
std::unique_ptr< abstract_event > event_type
Event type used by the copy class.
Definition copy.hpp:69
data::vector_view< TYPE >::size_type get_size(const data::vector_view< TYPE > &data) const
Helper function for getting the size of a resizable 1D buffer.
Definition copy.ipp:116
virtual void do_memset(std::size_t size, void *ptr, int value) const
Perform a "low level" memory filling operation.
Definition copy.cpp:40
virtual VECMEM_NODISCARD event_type create_event() const
Create an event for synchronization.
Definition copy.cpp:50
std::vector< typename data::vector_view< TYPE >::size_type > get_sizes(const data::jagged_vector_view< TYPE > &data) const
Helper function for getting the sizes of a resizable jagged vector.
Definition copy.ipp:271
VECMEM_NODISCARD event_type memset(data::vector_view< TYPE > data, int value) const
Set all bytes of the vector to some value.
data::vector_buffer< std::remove_cv_t< TYPE > > to(const data::vector_view< TYPE > &data, memory_resource &resource, type::copy_type cptype=type::unknown) const
Copy a 1-dimensional vector to the specified memory resource.
Definition copy.ipp:66
Object owning all the data of a jagged vector.
Definition jagged_vector_buffer.hpp:30
typename base_type::value_type value_type
Use the base class's value_type.
Definition jagged_vector_buffer.hpp:38
A view for jagged vectors.
Definition jagged_vector_view.hpp:44
VECMEM_HOST_AND_DEVICE pointer host_ptr() const
Access the host accessible array describing the inner vectors.
Definition jagged_vector_view.ipp:102
std::size_t size_type
We cannot use boolean types.
Definition jagged_vector_view.hpp:52
VECMEM_HOST_AND_DEVICE pointer ptr() const
Get a pointer to the vector elements.
Definition jagged_vector_view.ipp:96
VECMEM_HOST_AND_DEVICE size_type size() const
Get the "outer" size of the jagged vector.
Definition jagged_vector_view.ipp:83
Object owning the data held by it.
Definition vector_buffer.hpp:29
typename base_type::size_type size_type
Size type definition coming from the base class.
Definition vector_buffer.hpp:35
Class holding data about a 1 dimensional vector/array.
Definition vector_view.hpp:38
unsigned int size_type
We cannot use boolean types.
Definition vector_view.hpp:47
VECMEM_HOST_AND_DEVICE size_pointer size_ptr() const
Get a pointer to the size of the vector.
Definition vector_view.ipp:84
VECMEM_HOST_AND_DEVICE size_type capacity() const
Get the maximum capacity of the vector.
Definition vector_view.ipp:78
VECMEM_HOST_AND_DEVICE size_type size() const
Get the size of the vector.
Definition vector_view.ipp:72
Technical base type for view<schema<VARTYPES...>>
Definition view.hpp:28
void resize_jagged_vector(std::vector< std::vector< T, ALLOC1 >, ALLOC2 > &vec, std::size_t size)
Resize a generic jagged vector.
Definition resize_jagged_vector.hpp:23
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
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
copy_type
Types of memory copies to handle.
Definition copy.hpp:52
@ device_to_host
Copy operation between a device and the host.
Definition copy.hpp:56
@ device_to_device
Copy operation between two devices.
Definition copy.hpp:60
@ unknown
Unknown copy type, determined at runtime.
Definition copy.hpp:62
@ host_to_host
Copy operation on the host.
Definition copy.hpp:58
@ host_to_device
Copy operation between the host and a device.
Definition copy.hpp:54
Meta type describing the "schema" of an SoA container.
Definition schema.hpp:46
Definition schema_traits.hpp:82