TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
node.hpp
1#ifndef LIBTEDDY_DETAILS_NODE_HPP
2#define LIBTEDDY_DETAILS_NODE_HPP
3
4#include <libteddy/details/tools.hpp>
5#include <libteddy/details/types.hpp>
6
7#include <cassert>
8#include <cstdlib>
9
10namespace teddy
11{
12namespace degrees
13{
14struct mixed
15{
16 /*
17 * Just a dummy value to simplify ifs, this one should be never used
18 */
19 static constexpr int32 value = 1;
20};
21
22template<int32 N>
23struct fixed
24{
25 static_assert(N > 1);
26 static constexpr int32 value = N;
27};
28
29template<class T>
31{
32 static constexpr bool value = false;
33};
34
35template<int32 N>
36struct is_fixed<fixed<N>>
37{
38 static constexpr bool value = true;
39};
40
41template<class T>
43{
44 static constexpr bool value = false;
45};
46
47template<>
49{
50 static constexpr bool value = true;
51};
52} // namespace degrees
53
54namespace details
55{
56template<std::size_t Count>
57struct bytes
58{
59 char bytes_[Count];
60};
61
62template<>
63struct bytes<0>
64{
65};
66} // namespace details
67
68template<class Data, class Degree>
69class node;
70
71template<class Data, class Degree>
73{
74 node<Data, Degree>* sons_[Degree::value];
75
76 auto operator[] (int64 const index) -> node<Data, Degree>*&
77 {
78 return sons_[index];
79 }
80
81 auto operator[] (int64 const index) const -> node<Data, Degree>* const&
82 {
83 return sons_[index];
84 }
85};
86
87template<class Data, class Degree>
88// ^^^^
89// byte count, byte align
90class node
91{
92public:
93 template<int32 N>
94 static auto make_son_container (int32, degrees::fixed<N>)
96 {
98 }
99
100 static auto make_son_container (int32 const domain, degrees::mixed)
101 -> node**
102 {
103 return static_cast<node**>(std::malloc(as_usize(domain) * sizeof(node*))
104 );
105 }
106
107 static auto delete_son_container (node** sons) -> void
108 {
109 std::free(sons);
110 }
111
112public:
113 using son_container = decltype(make_son_container(int32(), Degree()));
114
115public:
116 explicit node(int32 value);
117 node(int32 index, son_container sons);
118 ~node() = default;
119 ~node()
121
122 node() = delete;
123 node(node const&) = delete;
124 node(node&&) = delete;
125 auto operator= (node const&) = delete;
126 auto operator= (node&&) = delete;
127
128 // TODO get_data_as<double>() -> double&
129 // get_data_as<expr>() -> expr&
130 // asi uz nebude treba Foo
131 template<class Foo = void>
132 requires(not utils::is_void<Data>::value)
133 auto get_data () -> utils::second_t<Foo, Data>&;
134
135 template<class Foo = void>
136 requires(not utils::is_void<Data>::value)
137 auto get_data () const -> utils::second_t<Foo, Data> const&;
138
139 [[nodiscard]] auto is_internal () const -> bool;
140 [[nodiscard]] auto is_terminal () const -> bool;
141 [[nodiscard]] auto is_used () const -> bool;
142 [[nodiscard]] auto is_marked () const -> bool;
143 [[nodiscard]] auto get_next () const -> node*;
144 [[nodiscard]] auto get_ref_count () const -> int32;
145 [[nodiscard]] auto get_index () const -> int32;
146 [[nodiscard]] auto get_sons () const -> son_container const&;
147 [[nodiscard]] auto get_son (int32 sonOrder) const -> node*;
148 [[nodiscard]] auto get_value () const -> int32;
149 auto set_next (node* next) -> void;
150 auto set_unused () -> void;
151 auto set_marked () -> void;
152 auto set_notmarked () -> void;
153 auto set_index (int32 index) -> void;
154 auto set_sons (son_container const& sons) -> void;
155 auto toggle_marked () -> void;
156 auto inc_ref_count () -> void;
157 auto dec_ref_count () -> void;
158
159private:
160 struct internal
161 {
162 son_container sons_;
163 int32 index_;
164 };
165
166 struct terminal
167 {
168 int32 value_;
169 };
170
171private:
172 [[nodiscard]] auto is_or_was_internal () const -> bool;
173
174private:
175 static constexpr uint32 MarkM = 1U << (8 * sizeof(uint32) - 1);
176 static constexpr uint32 UsedM = 1U << (8 * sizeof(uint32) - 2);
177 static constexpr uint32 LeafM = 1U << (8 * sizeof(uint32) - 3);
178 static constexpr uint32 RefsM = ~(MarkM | UsedM | LeafM);
179 static constexpr uint32 RefsMax = RefsM + 1;
180
181private:
182 union
183 {
184 internal internal_;
185 terminal terminal_;
186 };
187
188 [[no_unique_address]] utils::optional_member<Data> data_;
189 node* next_;
190 /*
191 * 1b -> is marked flag (highest bit)
192 * 1b -> is used flag
193 * 1b -> is leaf flag
194 * 29b -> reference count (lowest bits)
195 */
196 uint32 bits_;
197};
198
199template<class Data, class Degree>
200node<Data, Degree>::node(int32 const value) :
201 terminal_ {value},
202 next_ {nullptr},
203 bits_ {LeafM | UsedM}
204{
205}
206
207template<class Data, class Degree>
208node<Data, Degree>::node(int32 const index, son_container sons) :
209 internal_ {sons, index},
210 next_ {nullptr},
211 bits_ {UsedM}
212{
213}
214
215template<class Data, class Degree>
216node<Data, Degree>::~node()
217requires(degrees::is_mixed<Degree>::value)
218{
219 if constexpr (degrees::is_mixed<Degree>::value)
220 {
221 if (this->is_or_was_internal())
222 {
223 std::free(internal_.sons_);
224 }
225 }
226}
227
228template<class Data, class Degree>
229template<class Foo>
230requires(not utils::is_void<Data>::value)
231auto node<Data, Degree>::get_data() -> utils::second_t<Foo, Data>&
232{
233 assert(this->is_used());
234 return data_.member_;
235}
236
237template<class Data, class Degree>
238template<class Foo>
239requires(not utils::is_void<Data>::value)
240auto node<Data, Degree>::get_data() const -> utils::second_t<Foo, Data> const&
241{
242 assert(this->is_used());
243 return data_.member_;
244}
245
246template<class Data, class Degree>
247auto node<Data, Degree>::get_next() const -> node*
248{
249 return next_;
250}
251
252template<class Data, class Degree>
253auto node<Data, Degree>::set_next(node* const next) -> void
254{
255 next_ = next;
256}
257
258template<class Data, class Degree>
259auto node<Data, Degree>::get_sons() const -> son_container const&
260{
261 assert(this->is_internal());
262 return internal_.sons_;
263}
264
265template<class Data, class Degree>
266auto node<Data, Degree>::get_son(int32 const sonOrder) const -> node*
267{
268 assert(this->is_internal());
269 return internal_.sons_[sonOrder];
270}
271
272template<class Data, class Degree>
273auto node<Data, Degree>::set_sons(son_container const& sons) -> void
274{
275 assert(this->is_internal());
276 if constexpr (degrees::is_mixed<Degree>::value)
277 {
278 std::free(internal_.sons_);
279 }
280 internal_.sons_ = sons;
281}
282
283template<class Data, class Degree>
284auto node<Data, Degree>::get_value() const -> int32
285{
286 assert(this->is_terminal());
287 return terminal_.value_;
288}
289
290template<class Data, class Degree>
291auto node<Data, Degree>::is_internal() const -> bool
292{
293 return this->is_used() && not this->is_terminal();
294}
295
296template<class Data, class Degree>
297auto node<Data, Degree>::is_terminal() const -> bool
298{
299 return this->is_used() && (bits_ & LeafM);
300}
301
302template<class Data, class Degree>
303auto node<Data, Degree>::is_used() const -> bool
304{
305 return static_cast<bool>(bits_ & UsedM);
306}
307
308template<class Data, class Degree>
309auto node<Data, Degree>::set_unused() -> void
310{
311 bits_ &= ~UsedM;
312}
313
314template<class Data, class Degree>
315auto node<Data, Degree>::is_marked() const -> bool
316{
317 return static_cast<bool>(bits_ & MarkM);
318}
319
320template<class Data, class Degree>
321auto node<Data, Degree>::toggle_marked() -> void
322{
323 bits_ ^= MarkM;
324}
325
326template<class Data, class Degree>
327auto node<Data, Degree>::set_marked() -> void
328{
329 bits_ |= MarkM;
330}
331
332template<class Data, class Degree>
333auto node<Data, Degree>::set_notmarked() -> void
334{
335 bits_ &= ~MarkM;
336}
337
338template<class Data, class Degree>
339auto node<Data, Degree>::get_ref_count() const -> int32
340{
341 return static_cast<int32>(bits_ & RefsM);
342}
343
344template<class Data, class Degree>
345auto node<Data, Degree>::inc_ref_count() -> void
346{
347 assert(this->get_ref_count() < static_cast<int32>(RefsMax));
348 ++bits_;
349}
350
351template<class Data, class Degree>
352auto node<Data, Degree>::dec_ref_count() -> void
353{
354 assert(this->get_ref_count() > 0);
355 --bits_;
356}
357
358template<class Data, class Degree>
359auto node<Data, Degree>::get_index() const -> int32
360{
361 assert(this->is_internal());
362 return internal_.index_;
363}
364
365template<class Data, class Degree>
366auto node<Data, Degree>::set_index(int32 const index) -> void
367{
368 assert(this->is_internal());
369 internal_.index_ = index;
370}
371
372template<class Data, class Degree>
373auto node<Data, Degree>::is_or_was_internal() const -> bool
374{
375 return not static_cast<bool>(bits_ & LeafM);
376}
377} // namespace teddy
378
379#endif
Definition node.hpp:91
Definition node.hpp:24
Definition node.hpp:31
Definition node.hpp:43
Definition node.hpp:15
Definition node.hpp:58
Definition node.hpp:73
Definition tools.hpp:283
Definition tools.hpp:348