1#ifndef LIBTEDDY_DETAILS_NODE_HPP
2#define LIBTEDDY_DETAILS_NODE_HPP
4#include <libteddy/details/tools.hpp>
5#include <libteddy/details/types.hpp>
19 static constexpr int32 value = 1;
26 static constexpr int32 value = N;
32 static constexpr bool value =
false;
38 static constexpr bool value =
true;
44 static constexpr bool value =
false;
50 static constexpr bool value =
true;
56template<std::
size_t Count>
68template<
class Data,
class Degree>
71template<
class Data,
class Degree>
87template<
class Data,
class Degree>
100 static auto make_son_container (int32
const domain,
degrees::mixed)
103 return static_cast<node**
>(std::malloc(as_usize(domain) *
sizeof(
node*))
107 static auto delete_son_container (
node** sons) ->
void
113 using son_container =
decltype(make_son_container(int32(), Degree()));
116 explicit node(int32 value);
117 node(int32 index, son_container sons);
125 auto operator= (
node const&) =
delete;
126 auto operator= (
node&&) =
delete;
131 template<
class Foo =
void>
133 auto get_data () -> utils::second_t<Foo, Data>&;
135 template<
class Foo =
void>
137 auto get_data ()
const -> utils::second_t<Foo, Data>
const&;
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;
172 [[nodiscard]]
auto is_or_was_internal ()
const -> bool;
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;
199template<
class Data,
class Degree>
203 bits_ {LeafM | UsedM}
207template<
class Data,
class Degree>
208node<Data, Degree>::node(int32
const index, son_container sons) :
209 internal_ {sons, index},
215template<
class Data,
class Degree>
216node<Data, Degree>::~node()
217requires(degrees::is_mixed<Degree>::value)
219 if constexpr (degrees::is_mixed<Degree>::value)
221 if (this->is_or_was_internal())
223 std::free(internal_.sons_);
228template<
class Data,
class Degree>
230requires(not utils::is_void<Data>::value)
231auto node<Data, Degree>::get_data() -> utils::second_t<Foo, Data>&
233 assert(this->is_used());
234 return data_.member_;
237template<
class Data,
class Degree>
239requires(not utils::is_void<Data>::value)
240auto node<Data, Degree>::get_data()
const -> utils::second_t<Foo, Data>
const&
242 assert(this->is_used());
243 return data_.member_;
246template<
class Data,
class Degree>
247auto node<Data, Degree>::get_next() const -> node*
252template<
class Data,
class Degree>
253auto node<Data, Degree>::set_next(node*
const next) ->
void
258template<
class Data,
class Degree>
259auto node<Data, Degree>::get_sons() const -> son_container const&
261 assert(this->is_internal());
262 return internal_.sons_;
265template<
class Data,
class Degree>
266auto node<Data, Degree>::get_son(int32
const sonOrder)
const -> node*
268 assert(this->is_internal());
269 return internal_.sons_[sonOrder];
272template<
class Data,
class Degree>
273auto node<Data, Degree>::set_sons(son_container
const& sons) ->
void
275 assert(this->is_internal());
276 if constexpr (degrees::is_mixed<Degree>::value)
278 std::free(internal_.sons_);
280 internal_.sons_ = sons;
283template<
class Data,
class Degree>
284auto node<Data, Degree>::get_value() const -> int32
286 assert(this->is_terminal());
287 return terminal_.value_;
290template<
class Data,
class Degree>
291auto node<Data, Degree>::is_internal() const ->
bool
293 return this->is_used() && not this->is_terminal();
296template<
class Data,
class Degree>
297auto node<Data, Degree>::is_terminal() const ->
bool
299 return this->is_used() && (bits_ & LeafM);
302template<
class Data,
class Degree>
303auto node<Data, Degree>::is_used() const ->
bool
305 return static_cast<bool>(bits_ & UsedM);
308template<
class Data,
class Degree>
309auto node<Data, Degree>::set_unused() ->
void
314template<
class Data,
class Degree>
315auto node<Data, Degree>::is_marked() const ->
bool
317 return static_cast<bool>(bits_ & MarkM);
320template<
class Data,
class Degree>
321auto node<Data, Degree>::toggle_marked() ->
void
326template<
class Data,
class Degree>
327auto node<Data, Degree>::set_marked() ->
void
332template<
class Data,
class Degree>
333auto node<Data, Degree>::set_notmarked() ->
void
338template<
class Data,
class Degree>
339auto node<Data, Degree>::get_ref_count() const -> int32
341 return static_cast<int32
>(bits_ & RefsM);
344template<
class Data,
class Degree>
345auto node<Data, Degree>::inc_ref_count() ->
void
347 assert(this->get_ref_count() <
static_cast<int32
>(RefsMax));
351template<
class Data,
class Degree>
352auto node<Data, Degree>::dec_ref_count() ->
void
354 assert(this->get_ref_count() > 0);
358template<
class Data,
class Degree>
359auto node<Data, Degree>::get_index() const -> int32
361 assert(this->is_internal());
362 return internal_.index_;
365template<
class Data,
class Degree>
366auto node<Data, Degree>::set_index(int32
const index) ->
void
368 assert(this->is_internal());
369 internal_.index_ = index;
372template<
class Data,
class Degree>
373auto node<Data, Degree>::is_or_was_internal() const ->
bool
375 return not
static_cast<bool>(bits_ & LeafM);