TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
node_pool.hpp
1#ifndef LIBTEDDY_DETAILS_NODE_POOL_HPP
2#define LIBTEDDY_DETAILS_NODE_POOL_HPP
3
4#include <libteddy/details/config.hpp>
5#include <libteddy/details/debug.hpp>
6#include <libteddy/details/node.hpp>
7#include <libteddy/details/tools.hpp>
8
9#include <cassert>
10#include <cstdlib>
11#include <new>
12
13namespace teddy
14{
15template<class Data, class Degree>
17{
18public:
20
21public:
22 node_pool(int64 mainPoolSize, int64 extraPoolSize);
23 node_pool(node_pool&& other) noexcept;
25
26 node_pool(node_pool const&) = delete;
27 auto operator= (node_pool const&) = delete;
28 auto operator= (node_pool&&) = delete;
29
30 [[nodiscard]] auto get_available_node_count () const -> int64;
31
32 [[nodiscard]] auto get_main_pool_size () const -> int64;
33
34 template<class... Args>
35 [[nodiscard]] auto create (Args&&... args) -> node_t*;
36
37 auto destroy (node_t* node) -> void;
38
39 auto grow () -> void;
40
41private:
42 struct pool_item
43 {
44 node_t* pool_;
45 pool_item* next_;
46 };
47
48private:
55 [[nodiscard]] static auto allocate_pool (int64 size, pool_item* next)
56 -> pool_item*;
57
62 static auto deallocate_pool (pool_item* poolPtr, node_t* lastNode)
63 -> pool_item*;
64
65private:
66 pool_item* pools_;
67 node_t* nextPoolNode_;
68 node_t* freeNodes_;
69 int64 mainPoolSize_;
70 int64 extraPoolSize_;
71 int64 availableNodeCount_;
72};
73
74template<class Data, class Degree>
75node_pool<Data, Degree>::node_pool(
76 int64 const mainPoolSize,
77 int64 const overflowPoolSize
78) :
79 pools_(allocate_pool(mainPoolSize, nullptr)),
80 nextPoolNode_(pools_->pool_),
81 freeNodes_(nullptr),
82 mainPoolSize_(mainPoolSize),
83 extraPoolSize_(overflowPoolSize),
84 availableNodeCount_(mainPoolSize)
85{
86#ifdef LIBTEDDY_VERBOSE
87 debug::out(
88 "node_pool::node_pool\tAllocating initial pool with size ",
89 mainPoolSize_,
90 "\n"
91 );
92#endif
93}
94
95template<class Data, class Degree>
96node_pool<Data, Degree>::node_pool(node_pool&& other) noexcept :
97 pools_(utils::exchange(other.mainPool_, nullptr)),
98 nextPoolNode_(utils::exchange(other.nextPoolNode_, nullptr)),
99 freeNodes_(utils::exchange(other.freeNodes_, nullptr)),
100 mainPoolSize_(utils::exchange(other.mainPoolSize_, -1)),
101 extraPoolSize_(utils::exchange(other.extraPoolSize_, -1)),
102 availableNodeCount_(utils::exchange(other.availableNodeCount_, -1))
103{
104}
105
106template<class Data, class Degree>
108{
109 /*
110 * This is the currently used pool.
111 */
112 pools_ = deallocate_pool(pools_, nextPoolNode_);
113
114 /*
115 * If there are more pools with next pool they are extra pools.
116 */
117 while (pools_ && pools_->next_)
118 {
119 node_t* const lastNode = pools_->pool_ + extraPoolSize_;
120 pools_ = deallocate_pool(pools_, lastNode);
121 }
122
126 if (pools_)
127 {
128 node_t* const lastNode = pools_->pool_ + mainPoolSize_;
129 pools_ = deallocate_pool(pools_, lastNode);
130 }
131}
132
133template<class Data, class Degree>
135{
136 return availableNodeCount_;
137}
138
139template<class Data, class Degree>
140auto node_pool<Data, Degree>::get_main_pool_size() const -> int64
141{
142 return mainPoolSize_;
143}
144
145template<class Data, class Degree>
146template<class... Args>
147auto node_pool<Data, Degree>::create(Args&&... args) -> node_t*
148{
149 assert(availableNodeCount_ > 0);
150 --availableNodeCount_;
151
152 node_t* node = nullptr;
153 if (freeNodes_)
154 {
155 node = freeNodes_;
156 freeNodes_ = freeNodes_->get_next();
157 node->~node_t();
158 }
159 else
160 {
161 node = nextPoolNode_;
162 ++nextPoolNode_;
163 }
164
165 return static_cast<node_t*>(::new (node) node_t(args...));
166}
167
168template<class Data, class Degree>
169auto node_pool<Data, Degree>::destroy(node_t* const node) -> void
170{
171 ++availableNodeCount_;
172 node->set_next(freeNodes_);
173 freeNodes_ = node;
174}
175
176template<class Data, class Degree>
177auto node_pool<Data, Degree>::grow() -> void
178{
179#ifdef LIBTEDDY_VERBOSE
180 debug::out(
181 "node_pool::grow\tallocating overflow pool with size ",
182 overflowPoolSize_,
183 "\n"
184 );
185#endif
186
187 pools_ = allocate_pool(extraPoolSize_, pools_);
188 nextPoolNode_ = pools_->pool_;
189 availableNodeCount_ += extraPoolSize_;
190}
191
192template<class Data, class Degree>
193auto node_pool<Data, Degree>::allocate_pool(
194 int64 const size,
195 pool_item* const next
196) -> pool_item*
197{
198 return new pool_item {
199 static_cast<node_t*>(std::malloc(as_usize(size) * sizeof(node_t))),
200 next};
201}
202
203template<class Data, class Degree>
204auto node_pool<Data, Degree>::deallocate_pool(
205 pool_item* const pool,
206 node_t* const lastNode
207) -> pool_item*
208{
209 node_t* node = pool->pool_;
210 while (node < lastNode)
211 {
212 node->~node_t();
213 ++node;
214 }
215 pool_item* next = pool->next_;
216 std::free(pool->pool_);
217 delete pool;
218 return next;
219}
220} // namespace teddy
221
222#endif
Definition node_pool.hpp:17
~node_pool()
Definition node_pool.hpp:107
Definition node.hpp:91