vecmem 1.14.0
Loading...
Searching...
No Matches
unique_obj_deleter.hpp
1/*
2 * VecMem project, part of the ACTS project (R&D line)
3 *
4 * (c) 2021-2023 CERN for the benefit of the ACTS project
5 *
6 * Mozilla Public License Version 2.0
7 */
8
9#pragma once
10
11// Local include(s).
12#include "vecmem/memory/memory_resource.hpp"
13
14// System include(s).
15#include <cassert>
16#include <type_traits>
17
18namespace vecmem::details {
43template <typename T>
45public:
46 static_assert(!std::is_array_v<T> ||
47 (std::is_array_v<T> && std::extent_v<T> == 0),
48 "Pointer type of unique object must be either non-array or "
49 "unbound array type.");
50
51 using pointer_t =
52 std::conditional_t<std::is_array_v<T>, std::decay_t<T>, T*>;
53 using storage_t = std::remove_pointer_t<pointer_t>;
54
63 unique_obj_deleter(void) = default;
64
79 unique_obj_deleter(memory_resource& mr, std::size_t s, std::size_t a = 0,
80 std::size_t n = 1)
81 : m_mr(&mr), m_size(s), m_align(a), m_elems(n) {}
82
89
96
105
114
124 void operator()(pointer_t p) const {
125 assert(m_mr != nullptr || m_size == 0u);
126
127 /*
128 * Non-null pointers with a zero size can happen.
129 */
130 if (m_size == 0u) {
131 return;
132 }
133
134 /*
135 * The class exhibits different behaviour for array types and non-array
136 * types.
137 */
138 if constexpr (!std::is_array_v<T>) {
139 /*
140 * If we are deleting a non-array type, we can simply call the
141 * stored object's destructor.
142 */
143 p->~T();
144 } else if constexpr (std::is_array_v<T>) {
145 /*
146 * If we are deleting an array type instead, we need to iterate
147 * over each of our stored elements and destruct them.
148 */
149 for (std::size_t i = 0; i < m_elems; ++i) {
150 p[i].~storage_t();
151 }
152 }
153
154 /*
155 * Finally, we need to deallocate the memory. For this, we use the
156 * memory resource, and we optionally pass it an alignment value.
157 */
158 if (m_align > 0) {
159 m_mr->deallocate(p, m_size, m_align);
160 } else {
161 m_mr->deallocate(p, m_size);
162 }
163 };
164
165private:
166 /*
167 * This should really be a reference, but using a reference here makes the
168 * class non-default destructible. At least - without some messy use of
169 * default memory resources. Using a pointer here is not ideal, but it
170 * allows us to make use of empty unique pointers.
171 */
172 memory_resource* m_mr;
173 std::size_t m_size;
174 std::size_t m_align;
175 std::size_t m_elems;
176};
177} // namespace vecmem::details
Namespace for types that should not be used directly by clients.
Definition array.hpp:23
std::vector< T, vecmem::polymorphic_allocator< T > > vector
Alias type for vectors with our polymorphic allocator.
Definition vector.hpp:35
A deleter class for non-trivial objects.
Definition unique_obj_deleter.hpp:44
void operator()(pointer_t p) const
Activate the deletion mechanism of the deleter.
Definition unique_obj_deleter.hpp:124
unique_obj_deleter(memory_resource &mr, std::size_t s, std::size_t a=0, std::size_t n=1)
Construct a new unique object deleter.
Definition unique_obj_deleter.hpp:79
unique_obj_deleter(unique_obj_deleter &&i)=default
Move a unique object deleter.
unique_obj_deleter & operator=(const unique_obj_deleter &i)=default
Copy-assign a unique object deleter.
unique_obj_deleter(void)=default
Default-construct a new unique object deleter.
unique_obj_deleter(const unique_obj_deleter &i)=default
Copy a unique object deleter.
unique_obj_deleter & operator=(unique_obj_deleter &&i)=default
Move-assign a unique object deleter.