vecmem 1.14.0
Loading...
Searching...
No Matches
unique_ptr.hpp
1/*
2 * VecMem project, part of the ACTS project (R&D line)
3 *
4 * (c) 2021-2025 CERN for the benefit of the ACTS project
5 *
6 * Mozilla Public License Version 2.0
7 */
8
9#pragma once
10
11#include <memory>
12#include <type_traits>
13
14#include "vecmem/memory/details/unique_alloc_deleter.hpp"
15#include "vecmem/memory/details/unique_obj_deleter.hpp"
16#include "vecmem/memory/memory_resource.hpp"
17#include "vecmem/vecmem_core_export.hpp"
18
19namespace vecmem {
20namespace details {
21
27template <typename T>
29 return reinterpret_cast<T>(0xf000U);
30}
31
32} // namespace details
33
50template <typename T>
51using unique_obj_ptr = std::unique_ptr<T, details::unique_obj_deleter<T>>;
52
68template <typename T>
69using unique_alloc_ptr = std::unique_ptr<T, details::unique_alloc_deleter<T>>;
70
88template <typename T, typename... Args>
89typename std::enable_if_t<!std::is_array_v<T>, unique_obj_ptr<T>>
90make_unique_obj(memory_resource& m, Args&&... a) {
91 /*
92 * Calculate the size of the allocation and use the memory resource to
93 * perform an allocation of the requested size.
94 */
95 std::size_t s = sizeof(T);
96 T* p = static_cast<T*>(m.allocate(s));
97
98 /*
99 * Perform a placement-new with forwarded arguments to construct the
100 * object.
101 */
102 new (p) T(std::forward<Args>(a)...);
103
104 /*
105 * Create a new unique_ptr, with its own deleter, and return it.
106 */
108}
109
127template <typename T>
128typename std::enable_if_t<std::is_array_v<T> && std::extent_v<T> == 0,
130make_unique_obj(memory_resource& m, std::size_t n) {
131 using pointer_t = typename unique_obj_ptr<T>::deleter_type::pointer_t;
132
133 // Handle the case with zero elements.
134 if (n == 0) {
135 return unique_obj_ptr<T>(details::get_nonexistent_pointer<pointer_t>());
136 }
137
138 /*
139 * Calculate the size of the allocation and use the memory resource to
140 * perform an allocation of the requested size.
141 */
142 std::size_t s = n * sizeof(std::remove_extent_t<T>);
143 pointer_t p = static_cast<pointer_t>(m.allocate(s));
144
145 /*
146 * Use placement-new to default-construct each of our elements.
147 */
148 for (std::size_t i = 0; i < n; ++i) {
149 new (&p[i]) std::remove_extent_t<T>();
150 }
151
152 /*
153 * Create a new unique_ptr, with its own deleter, and return it.
154 */
156}
157
184template <typename T>
186 /*
187 * This method only works on non-array types and bounded array types.
188 */
189 static_assert(!(std::is_array_v<T> && std::extent_v<T> == 0),
190 "Allocation pointer type cannot be an unbounded array.");
191
192 /*
193 * Since we cannot (in general) construct objects in the memory we are
194 * about to allocate, we need to make sure that "bare", unallocated memory
195 * is semantically compatible with construction of the requested type.
196 */
197 static_assert(std::is_trivially_constructible_v<std::remove_extent_t<T>>,
198 "Allocation pointer type must be trivially constructible.");
199
200 using pointer_t =
201 std::conditional_t<std::is_array_v<T>, std::decay_t<T>, T*>;
202
203 /*
204 * Calculate the size of the allocation and use the memory resource to
205 * perform an allocation of the requested size.
206 */
207 std::size_t s = sizeof(T);
208 pointer_t p = static_cast<pointer_t>(m.allocate(s));
209
210 /*
211 * Create a new unique_ptr, with its own deleter, and return it.
212 */
214}
215
243template <typename T>
244unique_alloc_ptr<T> make_unique_alloc(memory_resource& m, std::size_t n) {
245 /*
246 * This overload only works for unbounded array types.
247 */
248 static_assert(std::is_array_v<T>,
249 "Allocation pointer type must be an array type.");
250 static_assert(std::extent_v<T> == 0,
251 "Allocation pointer type must be unbounded.");
252
253 /*
254 * Since we cannot (in general) construct objects in the memory we are
255 * about to allocate, we need to make sure that "bare", unallocated memory
256 * is semantically compatible with construction of the requested type.
257 */
258 static_assert(std::is_trivially_constructible_v<std::remove_extent_t<T>>,
259 "Allocation pointer type must be trivially constructible.");
260
261 using pointer_t =
262 std::conditional_t<std::is_array_v<T>, std::decay_t<T>, T*>;
263
264 // Handle the case with zero elements.
265 if (n == 0) {
266 return unique_alloc_ptr<T>(
267 details::get_nonexistent_pointer<pointer_t>());
268 }
269
270 /*
271 * Calculate the size of the allocation and use the memory resource to
272 * perform an allocation of the requested size.
273 */
274 std::size_t s = n * sizeof(std::remove_extent_t<T>);
275 pointer_t p = static_cast<pointer_t>(m.allocate(s));
276
277 /*
278 * Create a new unique_ptr, with its own deleter, and return it.
279 */
281}
282
321template <typename T, typename C>
322unique_alloc_ptr<T> make_unique_alloc(memory_resource& m, const T* f,
323 const C& c) {
324 /*
325 * This method only works on non-array types and bounded array types.
326 */
327 static_assert(!(std::is_array_v<T> && std::extent_v<T> == 0),
328 "Allocation pointer type cannot be an ubounded array.");
329
330 /*
331 * In this case, we are going to immediately copy some (hopefully) live
332 * objects into our new allocation, so trivial constructability is not a
333 * hard requirement. Rather, we must ensure that a memory copy is a valid
334 * way of constructing types, which we do by checking trivial copyability.
335 */
336 static_assert(std::is_trivially_copyable_v<std::remove_extent_t<T>>,
337 "Allocation pointer type must be trivially copyable.");
338
339 using pointer_t =
340 std::conditional_t<std::is_array_v<T>, std::decay_t<T>, T*>;
341
342 /*
343 * Calculate the size of the allocation and use the memory resource to
344 * perform an allocation of the requested size.
345 */
346 std::size_t s = sizeof(T);
347 pointer_t p = static_cast<pointer_t>(m.allocate(s));
348
349 c(p, f, s);
350
351 /*
352 * Create a new unique_ptr, with its own deleter, and return it.
353 */
355}
356
395template <typename T, typename C>
396unique_alloc_ptr<T> make_unique_alloc(memory_resource& m, std::size_t n,
397 const std::remove_extent_t<T>* f,
398 const C& c) {
399 /*
400 * This overload only works for unbounded array types.
401 */
402 static_assert(std::is_array_v<T>,
403 "Allocation pointer type must be an array type.");
404 static_assert(std::extent_v<T> == 0,
405 "Allocation pointer type must be unbounded.");
406
407 /*
408 * In this case, we are going to immediately copy some (hopefully) live
409 * objects into our new allocation, so trivial constructability is not a
410 * hard requirement. Rather, we must ensure that a memory copy is a valid
411 * way of constructing types, which we do by checking trivial copyability.
412 */
413 static_assert(std::is_trivially_copyable_v<std::remove_extent_t<T>>,
414 "Allocation pointer type must be trivially copyable.");
415
416 using pointer_t =
417 std::conditional_t<std::is_array_v<T>, std::decay_t<T>, T*>;
418
419 // Handle the case with zero elements.
420 if (n == 0) {
421 return unique_alloc_ptr<T>(
422 details::get_nonexistent_pointer<pointer_t>());
423 }
424
425 /*
426 * Calculate the size of the allocation and use the memory resource to
427 * perform an allocation of the requested size.
428 */
429 std::size_t s = n * sizeof(std::remove_extent_t<T>);
430 pointer_t p = static_cast<pointer_t>(m.allocate(s));
431
432 c(p, f, s);
433
434 /*
435 * Create a new unique_ptr, with its own deleter, and return it.
436 */
438}
439} // namespace vecmem
T get_nonexistent_pointer()
Helper function for generating a nonexistent, but still "aligned" pointer.
Definition unique_ptr.hpp:28
Main namespace for the vecmem classes/functions.
Definition atomic_ref.hpp:16
std::unique_ptr< T, details::unique_obj_deleter< T > > unique_obj_ptr
A unique pointer type for non-trivial objects.
Definition unique_ptr.hpp:51
std::vector< T, vecmem::polymorphic_allocator< T > > vector
Alias type for vectors with our polymorphic allocator.
Definition vector.hpp:35
unique_alloc_ptr< T > make_unique_alloc(memory_resource &m)
Create a unique allocation pointer to a type.
Definition unique_ptr.hpp:185
std::unique_ptr< T, details::unique_alloc_deleter< T > > unique_alloc_ptr
A unique pointer type for trivial types.
Definition unique_ptr.hpp:69
std::enable_if_t<!std::is_array_v< T >, unique_obj_ptr< T > > make_unique_obj(memory_resource &m, Args &&... a)
Create a unique object pointer to a newly constructed object.
Definition unique_ptr.hpp:90
A deleter class for trivial allocations.
Definition unique_alloc_deleter.hpp:33
A deleter class for non-trivial objects.
Definition unique_obj_deleter.hpp:44