TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
node_manager.hpp
1#ifndef LIBTEDDY_DETAILS_NODE_MANAGER_HPP
2#define LIBTEDDY_DETAILS_NODE_MANAGER_HPP
3
4#include <libteddy/details/config.hpp>
5#include <libteddy/details/debug.hpp>
6#include <libteddy/details/hash_tables.hpp>
7#include <libteddy/details/node.hpp>
8#include <libteddy/details/node_pool.hpp>
9#include <libteddy/details/operators.hpp>
10#include <libteddy/details/tools.hpp>
11#include <libteddy/details/types.hpp>
12
13#include <cassert>
14#include <concepts>
15#include <cstdint>
16#include <ostream>
17#include <string>
18#include <vector>
19
20namespace teddy
21{
22namespace domains
23{
24struct mixed
25{
26 /*
27 * Just a dummy value to simplify ifs, this one should be never used
28 */
29 static constexpr int32 value = 1;
30
31 std::vector<int32> domains_;
32
33 mixed(std::vector<int32> domains) :
34 domains_(static_cast<std::vector<int32>&&>(domains)) {};
35
36 auto operator[] (int32 const index) const
37 {
38 return domains_[as_uindex(index)];
39 }
40};
41
42template<int32 N>
43struct fixed
44{
45 static_assert(N > 1);
46
47 static constexpr int32 value = N;
48
49 constexpr auto operator[] ([[maybe_unused]] int32 const index) const
50 {
51 return N;
52 }
53};
54
55template<class T>
57{
58 static constexpr bool value = false;
59};
60
61template<int32 N>
62struct is_fixed<fixed<N>>
63{
64 static constexpr bool value = true;
65};
66
67template<class T>
69{
70 static constexpr bool value = false;
71};
72
73template<>
75{
76 static constexpr bool value = true;
77};
78} // namespace domains
79
80template<class Data, class Degree, class Domain>
82{
83public:
85 using son_container = typename node_t::son_container;
86
88 {
89 };
90
91public:
93 int32 varCount,
94 int64 nodePoolSize,
95 int64 extraNodePoolSize,
96 std::vector<int32> order
97 )
99
101 int32 varCount,
102 int64 nodePoolSize,
103 int64 extraNodePoolSize,
104 std::vector<int32> order,
105 domains::mixed domains
106 )
108
109 node_manager(node_manager&&) noexcept = default;
110 ~node_manager() = default;
111 node_manager(node_manager const&) = delete;
112 auto operator= (node_manager const&) = delete;
113 auto operator= (node_manager&&) = delete;
114
115 auto set_cache_ratio (double ratio) -> void;
116 auto set_gc_ratio (double ratio) -> void;
117 auto set_auto_reorder (bool doReorder) -> void;
118
119private:
121 common_init_tag initTag,
122 int32 varCount,
123 int64 nodePoolSize,
124 int64 extraNodePoolSize,
125 std::vector<int32> order,
126 Domain domains
127 );
128
129public:
130 [[nodiscard]] auto get_terminal_node (int32 value) const -> node_t*;
131 [[nodiscard]] auto make_terminal_node (int32 value) -> node_t*;
132 [[nodiscard]] auto make_internal_node (
133 int32 index,
134 son_container const& sons
135 ) -> node_t*;
136 [[nodiscard]] auto make_son_container (int32 domain) -> son_container;
137 [[nodiscard]] auto get_level (int32 index) const -> int32;
138 [[nodiscard]] auto get_level (node_t* node) const -> int32;
139 [[nodiscard]] auto get_leaf_level () const -> int32;
140 [[nodiscard]] auto get_index (int32 level) const -> int32;
141 [[nodiscard]] auto get_domain (int32 index) const -> int32;
142 [[nodiscard]] auto get_domain (node_t* node) const -> int32;
143 [[nodiscard]] auto get_node_count (int32 index) const -> int64;
144 [[nodiscard]] auto get_node_count (node_t* node) const -> int64;
145 [[nodiscard]] auto get_node_count () const -> int64;
146 [[nodiscard]] auto get_var_count () const -> int32;
147 [[nodiscard]] auto get_order () const -> std::vector<int32> const&;
148 [[nodiscard]] auto get_domains () const -> std::vector<int32>;
149 auto force_gc () -> void;
150
151 auto to_dot_graph (std::ostream& ost) const -> void;
152 auto to_dot_graph (std::ostream& ost, node_t* node) const -> void;
153
154 [[nodiscard]] auto domain_product (int32 levelFrom, int32 levelTo) const
155 -> int64;
156
157 template<class NodeOp>
158 auto for_each_son (node_t* node, NodeOp&& operation) const -> void;
159
160 template<class NodeOp>
161 auto for_each_son (
162 int32 index,
163 son_container const& sons,
164 NodeOp&& operation
165 ) const -> void;
166
167 template<class NodeOp>
168 auto for_each_node (NodeOp&& operation) const -> void;
169
170 template<class NodeOp>
171 auto for_each_terminal_node (NodeOp&& operation) const -> void;
172
173 template<teddy_bin_op O>
174 [[nodiscard]] auto cache_find (node_t* lhs, node_t* rhs) -> node_t*;
175
176 template<teddy_bin_op O>
177 auto cache_put (node_t* result, node_t* lhs, node_t* rhs) -> void;
178
179 auto cache_clear () -> void;
180
181 template<class NodeOp>
182 auto traverse_pre (node_t* rootNode, NodeOp operation) const -> void;
183
184 template<class NodeOp>
185 auto traverse_post (node_t* rootNode, NodeOp operation) const -> void;
186
187 template<class NodeOp>
188 auto traverse_level (node_t* rootNode, NodeOp operation) const -> void;
189
190 [[nodiscard]] auto is_valid_var_value (int32 index, int32 value) const
191 -> bool;
192
193 auto run_deferred () -> void;
194
195 static auto dec_ref_count (node_t* node) -> void;
196
197 auto sift_variables () -> void;
198
199private:
200 template<class NodeOp>
201 auto traverse_pre_impl (node_t* node, NodeOp operation) const -> void;
202
203 template<class NodeOp>
204 auto traverse_post_impl (node_t* node, NodeOp operation) const -> void;
205
206 [[nodiscard]] auto is_redundant (int32 index, son_container const& sons)
207 const -> bool;
208
209 auto adjust_tables () -> void;
210 auto adjust_caches () -> void;
211
212 auto swap_variable_with_next (int32 index) -> void;
213 auto swap_node_with_next (node_t* node) -> void;
214 auto dec_ref_try_gc (node_t* node) -> void;
215
216 [[nodiscard]] auto make_special_node (int32 value) -> node_t*;
217
218 template<class... Args>
219 [[nodiscard]] auto make_new_node (Args&&... args) -> node_t*;
220 auto delete_node (node_t* node) -> void;
221
222 template<class ForEachNode>
223 auto to_dot_graph_common (std::ostream& ost, ForEachNode&& forEach) const
224 -> void;
225
226 auto deferr_gc_reorder () -> void;
227
228 auto collect_garbage () -> void;
229
230 [[nodiscard]] static auto check_distinct (std::vector<int32> const& ints)
231 -> bool;
232
233 [[nodiscard]] static auto can_be_gced (node_t* node) -> bool;
234
235private:
236 static constexpr int32 DEFAULT_FIRST_TABLE_ADJUSTMENT = 230;
237 static constexpr double DEFAULT_CACHE_RATIO = 1.0;
238 static constexpr double DEFAULT_GC_RATIO = 0.20;
239
240private:
241 apply_cache<Data, Degree> opCache_;
242 node_pool<Data, Degree> pool_;
243 std::vector<unique_table<Data, Degree>> uniqueTables_;
244 std::vector<node_t*> terminals_;
245 std::vector<node_t*> specials_;
246 std::vector<int32> indexToLevel_;
247 std::vector<int32> levelToIndex_;
248 [[no_unique_address]] Domain domains_;
249 int32 varCount_;
250 int64 nodeCount_;
251 int64 adjustmentNodeCount_;
252 double cacheRatio_;
253 double gcRatio_;
254 bool autoReorderEnabled_;
255 bool gcReorderDeferred_;
256};
257
258template<class Data, class Degree>
259auto id_inc_ref_count (node<Data, Degree>* const node)
260 -> ::teddy::node<Data, Degree>*
261{
262 node->inc_ref_count();
263 return node;
264}
265
266template<class Data, class Degree>
267auto id_set_marked (node<Data, Degree>* const node)
269{
270 node->set_marked();
271 return node;
272}
273
274template<class Data, class Degree>
275auto id_set_notmarked (node<Data, Degree>* const node)
277{
278 node->set_notmarked();
279 return node;
280}
281
282template<class Data, class Degree, class Domain>
283node_manager<Data, Degree, Domain>::node_manager(
284 int32 const varCount,
285 int64 const nodePoolSize,
286 int64 const extraNodePoolSize,
287 std::vector<int32> order
288)
289requires(domains::is_fixed<Domain>::value)
290 :
291 node_manager(
292 common_init_tag(),
293 varCount,
294 nodePoolSize,
295 extraNodePoolSize,
296 static_cast<std::vector<int32>&&>(order),
297 {}
298 )
299{
300}
301
302template<class Data, class Degree, class Domain>
303node_manager<Data, Degree, Domain>::node_manager(
304 int32 const varCount,
305 int64 const nodePoolSize,
306 int64 const extraNodePoolSize,
307 std::vector<int32> order,
308 domains::mixed domains
309)
310requires(domains::is_mixed<Domain>::value)
311 :
312 node_manager(
313 common_init_tag(),
314 varCount,
315 nodePoolSize,
316 extraNodePoolSize,
317 static_cast<std::vector<int32>&&>(order),
318 static_cast<domains::mixed&&>(domains)
319 )
320{
321}
322
323template<class Data, class Degree, class Domain>
324node_manager<Data, Degree, Domain>::node_manager(
325 [[maybe_unused]] common_init_tag initTag,
326 int32 const varCount,
327 int64 const nodePoolSize,
328 int64 const extraNodePoolSize,
329 std::vector<int32> order,
330 Domain domains
331) :
332 opCache_(static_cast<int64>(
333 DEFAULT_CACHE_RATIO * static_cast<double>(nodePoolSize)
334 )),
335 pool_(nodePoolSize, extraNodePoolSize),
336 uniqueTables_(),
337 terminals_(),
338 specials_(),
339 indexToLevel_(as_usize(varCount)),
340 levelToIndex_(static_cast<std::vector<int32>&&>(order)),
341 domains_(static_cast<Domain&&>(domains)),
342 varCount_(varCount),
343 nodeCount_(0),
344 adjustmentNodeCount_(DEFAULT_FIRST_TABLE_ADJUSTMENT),
345 cacheRatio_(DEFAULT_CACHE_RATIO),
346 gcRatio_(DEFAULT_GC_RATIO),
347 autoReorderEnabled_(false),
348 gcReorderDeferred_(false)
349{
350 assert(ssize(levelToIndex_) == varCount_);
351 assert(check_distinct(levelToIndex_));
352
353 if constexpr (domains::is_mixed<Domain>::value)
354 {
355 assert(ssize(domains_.domains_) == varCount_);
356 if constexpr (degrees::is_fixed<Degree>::value)
357 {
358 for ([[maybe_unused]] int32 const domain : domains_.domains_)
359 {
360 assert(domain <= Degree::value);
361 }
362 }
363 }
364
365 // Create reverse mapping from (level -> index) to (index -> level)
366 int32 level = 0;
367 for (int32 const index : levelToIndex_)
368 {
369 indexToLevel_[as_uindex(index)] = level++;
370 }
371
372 // Initialize unique tables with pre-allocated sizes
373 // The sizes follow triangular distribution with
374 // a = 0
375 // c = RelPeakPosition * varCount
376 // b = varCount
377 // f(c) = RelPeakNodeCount * nodePoolSize
378
379 // These two magic numbers are empirically obtained
380 double constexpr RelPeakPosition = 0.71;
381 double constexpr RelPeakNodeCount = 0.05;
382 double const c = RelPeakPosition * (static_cast<double>(varCount) - 1);
383 double const fc = RelPeakNodeCount * static_cast<double>(nodePoolSize);
384
385 for (int32 i = 0; i <= static_cast<int32>(c); ++i)
386 {
387 auto const x = static_cast<double>(i);
388 uniqueTables_.emplace_back(
389 static_cast<int64>(fc * x / c),
390 this->get_domain(i)
391 );
392 }
393
394 for (auto i = static_cast<int32>(c) + 1; i < varCount_; ++i)
395 {
396 auto const n = static_cast<double>(varCount) - 1;
397 auto const x = static_cast<double>(i);
398 uniqueTables_.emplace_back(
399 static_cast<int64>((fc * x) / (c - n) - (fc * n) / (c - n)),
400 this->get_domain(i)
401 );
402 }
403}
404
405template<class Data, class Degree, class Domain>
406auto node_manager<Data, Degree, Domain>::set_cache_ratio(double const ratio)
407 -> void
408{
409 assert(ratio > 0);
410 cacheRatio_ = ratio;
411}
412
413template<class Data, class Degree, class Domain>
414auto node_manager<Data, Degree, Domain>::set_gc_ratio(double const ratio)
415 -> void
416{
417 assert(ratio >= 0.0 && ratio <= 1.0);
418 gcRatio_ = ratio;
419}
420
421template<class Data, class Degree, class Domain>
422auto node_manager<Data, Degree, Domain>::set_auto_reorder(bool const doReorder)
423 -> void
424{
425 autoReorderEnabled_ = doReorder;
426}
427
428template<class Data, class Degree, class Domain>
429auto node_manager<Data, Degree, Domain>::get_terminal_node(int32 const value
430) const -> node_t*
431{
432 return value < ssize(terminals_) ? terminals_[as_uindex(value)] : nullptr;
433}
434
435template<class Data, class Degree, class Domain>
436auto node_manager<Data, Degree, Domain>::make_terminal_node(int32 const value)
437 -> node_t*
438{
439 if constexpr (domains::is_fixed<Domain>::value)
440 {
441 assert(value < Domain::value);
442 }
443
444 if (is_special(value))
445 {
446 return this->make_special_node(value);
447 }
448
449 if (value >= ssize(terminals_))
450 {
451 terminals_.resize(as_usize(value + 1), nullptr);
452 }
453
454 if (not terminals_[as_uindex(value)])
455 {
456 terminals_[as_uindex(value)] = this->make_new_node(value);
457 }
458
459 return id_set_marked(terminals_[as_uindex(value)]);
460}
461
462template<class Data, class Degree, class Domain>
463auto node_manager<Data, Degree, Domain>::make_special_node(
464 [[maybe_unused]] int32 const value
465) -> node_t*
466{
467 assert(value == Undefined);
468
469 if (specials_.empty())
470 {
471 specials_.resize(1, nullptr);
472 }
473
474 if (not specials_[0])
475 {
476 specials_[0] = this->make_new_node(Undefined);
477 }
478
479 return id_set_marked(specials_[0]);
480}
481
482template<class Data, class Degree, class Domain>
483auto node_manager<Data, Degree, Domain>::make_son_container(int32 const domain)
484 -> son_container
485{
486 return node_t::make_son_container(domain, Degree());
487}
488
489template<class Data, class Degree, class Domain>
490auto node_manager<Data, Degree, Domain>::make_internal_node(
491 int32 const index,
492 son_container const& sons
493) -> node_t*
494{
495 // redundant node:
496 if (this->is_redundant(index, sons))
497 {
498 node_t* const son = sons[0];
499 if constexpr (degrees::is_mixed<Degree>::value)
500 {
501 node_t::delete_son_container(sons);
502 }
503 return son;
504 }
505
506 // duplicate node:
507 unique_table<Data, Degree>& table = uniqueTables_[as_uindex(index)];
508 auto const [existing, hash] = table.find(sons);
509 if (existing)
510 {
511 if constexpr (degrees::is_mixed<Degree>::value)
512 {
513 node_t::delete_son_container(sons);
514 }
515 this->for_each_son(existing, id_set_notmarked<Data, Degree>);
516 return id_set_marked(existing);
517 }
518
519 // new unique node:
520 node_t* const newNode = this->make_new_node(index, sons);
521 table.insert(newNode, hash);
522 this->for_each_son(newNode, id_inc_ref_count<Data, Degree>);
523 this->for_each_son(newNode, id_set_notmarked<Data, Degree>);
524
525 return id_set_marked(newNode);
526}
527
528template<class Data, class Degree, class Domain>
529auto node_manager<Data, Degree, Domain>::get_level(int32 const index) const
530 -> int32
531{
532 return indexToLevel_[as_uindex(index)];
533}
534
535template<class Data, class Degree, class Domain>
536auto node_manager<Data, Degree, Domain>::get_level(node_t* const node) const
537 -> int32
538{
539 return node->is_terminal() ? this->get_leaf_level()
540 : this->get_level(node->get_index());
541}
542
543template<class Data, class Degree, class Domain>
544auto node_manager<Data, Degree, Domain>::get_leaf_level() const -> int32
545{
546 return this->get_var_count();
547}
548
549template<class Data, class Degree, class Domain>
550auto node_manager<Data, Degree, Domain>::get_index(int32 const level) const
551 -> int32
552{
553 assert(level < ssize(levelToIndex_));
554 return levelToIndex_[as_uindex(level)];
555}
556
557template<class Data, class Degree, class Domain>
558auto node_manager<Data, Degree, Domain>::get_domain(int32 const index) const
559 -> int32
560{
561 assert(index < this->get_var_count());
562 return domains_[index];
563}
564
565template<class Data, class Degree, class Domain>
566auto node_manager<Data, Degree, Domain>::get_domain(node_t* const node) const
567 -> int32
568{
569 return this->get_domain(node->get_index());
570}
571
572template<class Data, class Degree, class Domain>
573auto node_manager<Data, Degree, Domain>::get_node_count(int32 const index) const
574 -> int64
575{
576 assert(index < this->get_var_count());
577 return uniqueTables_[as_uindex(index)].get_size();
578}
579
580template<class Data, class Degree, class Domain>
581auto node_manager<Data, Degree, Domain>::get_node_count(node_t* const node
582) const -> int64
583{
584 int64 count = 0;
585 this->traverse_pre(node, [&count] (node_t*) { ++count; });
586 return count;
587}
588
589template<class Data, class Degree, class Domain>
590auto node_manager<Data, Degree, Domain>::get_node_count() const -> int64
591{
592 return nodeCount_;
593}
594
595template<class Data, class Degree, class Domain>
596auto node_manager<Data, Degree, Domain>::get_var_count() const -> int32
597{
598 return varCount_;
599}
600
601template<class Data, class Degree, class Domain>
602auto node_manager<Data, Degree, Domain>::get_order() const
603 -> std::vector<int32> const&
604{
605 return levelToIndex_;
606}
607
608template<class Data, class Degree, class Domain>
609auto node_manager<Data, Degree, Domain>::get_domains() const
610 -> std::vector<int32>
611{
612 std::vector<int32> domains;
613 domains.reserve(as_usize(varCount_));
614 for (int32 k = 0; k < varCount_; ++k)
615 {
616 domains.push_back(domains_[k]);
617 }
618 return domains;
619}
620
621template<class Data, class Degree, class Domain>
622auto node_manager<Data, Degree, Domain>::force_gc() -> void
623{
624 this->collect_garbage();
625 opCache_.remove_unused();
626}
627
628template<class Data, class Degree, class Domain>
629auto node_manager<Data, Degree, Domain>::collect_garbage() -> void
630{
631#ifdef LIBTEDDY_VERBOSE
632 debug::out("node_manager::collect_garbage, ");
633 int64 const before = nodeCount_;
634#endif
635
636 for (int32 level = 0; level < this->get_var_count(); ++level)
637 {
638 int32 const index = levelToIndex_[as_uindex(level)];
639 auto& table = uniqueTables_[as_uindex(index)];
640 auto const endIt = table.end();
641 auto tableIt = table.begin();
642
643 while (tableIt != endIt)
644 {
645 node_t* const node = *tableIt;
646 if (can_be_gced(node))
647 {
648 this->for_each_son(node, dec_ref_count);
649 tableIt = table.erase(tableIt);
650 this->delete_node(node);
651 }
652 else
653 {
654 ++tableIt;
655 }
656 }
657 }
658
659 for (node_t*& node : terminals_)
660 {
661 if (node && can_be_gced(node))
662 {
663 this->delete_node(node);
664 node = nullptr;
665 }
666 }
667
668 for (node_t*& node : specials_)
669 {
670 if (node && can_be_gced(node))
671 {
672 this->delete_node(node);
673 node = nullptr;
674 }
675 }
676
677#ifdef LIBTEDDY_VERBOSE
678 debug::out(
679 before - nodeCount_,
680 " nodes collected.",
681 " Now there are ",
682 nodeCount_,
683 " unique nodes\n"
684 );
685#endif
686}
687
688template<class Data, class Degree, class Domain>
689auto node_manager<Data, Degree, Domain>::to_dot_graph(std::ostream& ost) const
690 -> void
691{
692 this->to_dot_graph_common(
693 ost,
694 [this] (auto const& operation) { this->for_each_node(operation); }
695 );
696}
697
698template<class Data, class Degree, class Domain>
699auto node_manager<Data, Degree, Domain>::to_dot_graph(
700 std::ostream& ost,
701 node_t* const node
702) const -> void
703{
704 this->to_dot_graph_common(
705 ost,
706 [this, node] (auto const& operation)
707 { this->traverse_pre(node, operation); }
708 );
709}
710
711template<class Data, class Degree, class Domain>
712auto node_manager<Data, Degree, Domain>::domain_product(
713 int32 const levelFrom,
714 int32 const levelTo
715) const -> int64
716{
717 if constexpr (domains::is_fixed<Domain>::value && Degree::value == 2)
718 {
719 return int64(1) << (levelTo - levelFrom);
720 }
721 else if constexpr (domains::is_fixed<Domain>::value)
722 {
723 return utils::int_pow(Domain::value, levelTo - levelFrom);
724 }
725 else
726 {
727 int64 product = 1;
728 for (int32 level = levelFrom; level < levelTo; ++level)
729 {
730 product *= domains_[levelToIndex_[as_uindex(level)]];
731 }
732 return product;
733 }
734}
735
736template<class Data, class Degree, class Domain>
737template<class NodeOp>
738auto node_manager<Data, Degree, Domain>::for_each_son(
739 node_t* const node,
740 NodeOp&& operation
741) const -> void
742{
743 int32 const index = node->get_index();
744 for (int32 k = 0; k < domains_[index]; ++k)
745 {
746 operation(node->get_son(k));
747 }
748}
749
750template<class Data, class Degree, class Domain>
751template<class NodeOp>
752auto node_manager<Data, Degree, Domain>::for_each_son(
753 int32 const index,
754 son_container const& sons,
755 NodeOp&& operation
756) const -> void
757{
758 for (int32 k = 0; k < domains_[index]; ++k)
759 {
760 operation(sons[as_uindex(k)]);
761 }
762}
763
764template<class Data, class Degree, class Domain>
765template<class NodeOp>
766auto node_manager<Data, Degree, Domain>::for_each_node(NodeOp&& operation) const
767 -> void
768{
769 for (unique_table<Data, Degree> const& table : uniqueTables_)
770 {
771 for (node_t* const node : table)
772 {
773 operation(node);
774 }
775 }
776
777 this->for_each_terminal_node(operation);
778}
779
780template<class Data, class Degree, class Domain>
781template<class NodeOp>
782auto node_manager<Data, Degree, Domain>::for_each_terminal_node(
783 NodeOp&& operation
784) const -> void
785{
786 for (node_t* const node : terminals_)
787 {
788 if (node)
789 {
790 operation(node);
791 }
792 }
793}
794
795template<class Data, class Degree, class Domain>
796template<teddy_bin_op O>
797auto node_manager<Data, Degree, Domain>::cache_find(node_t* lhs, node_t* rhs)
798 -> node_t*
799{
800 if constexpr (O::is_commutative())
801 {
802 if (rhs < lhs)
803 {
804 utils::swap(lhs, rhs);
805 }
806 }
807 node_t* const node = opCache_.find(O::get_id(), lhs, rhs);
808 if (node)
809 {
810 id_set_marked(node);
811 }
812 return node;
813}
814
815template<class Data, class Degree, class Domain>
816template<teddy_bin_op O>
817auto node_manager<Data, Degree, Domain>::cache_put(
818 node_t* const result,
819 node_t* lhs,
820 node_t* rhs
821) -> void
822{
823 if constexpr (O::is_commutative())
824 {
825 if (rhs < lhs)
826 {
827 utils::swap(lhs, rhs);
828 }
829 }
830 opCache_.put(O::get_id(), result, lhs, rhs);
831}
832
833template<class Data, class Degree, class Domain>
834auto node_manager<Data, Degree, Domain>::cache_clear() -> void
835{
836 opCache_.clear();
837}
838
839template<class Data, class Degree, class Domain>
840auto node_manager<Data, Degree, Domain>::is_valid_var_value(
841 int32 const index,
842 int32 const value
843) const -> bool
844{
845 return value < domains_[index];
846}
847
848template<class Data, class Degree, class Domain>
849auto node_manager<Data, Degree, Domain>::run_deferred() -> void
850{
851 if (gcReorderDeferred_)
852 {
853 this->collect_garbage();
854 opCache_.clear();
855 this->sift_variables();
856 }
857}
858
859template<class Data, class Degree, class Domain>
860template<class NodeOp>
861auto node_manager<Data, Degree, Domain>::traverse_pre(
862 node_t* const rootNode,
863 NodeOp const operation
864) const -> void
865{
866 this->traverse_pre_impl(rootNode, operation);
867 this->traverse_pre_impl(rootNode, [] (node_t*) {});
868 // Second traverse to reset marks
869}
870
871template<class Data, class Degree, class Domain>
872template<class NodeOp>
873auto node_manager<Data, Degree, Domain>::traverse_pre_impl(
874 node_t* const node,
875 NodeOp const operation
876) const -> void
877{
878 node->toggle_marked();
879 operation(node);
880 if (node->is_internal())
881 {
882 int32 const nodeDomain = this->get_domain(node);
883 for (int32 k = 0; k < nodeDomain; ++k)
884 {
885 node_t* const son = node->get_son(k);
886 if (node->is_marked() != son->is_marked())
887 {
888 this->traverse_pre_impl(son, operation);
889 }
890 }
891 }
892}
893
894template<class Data, class Degree, class Domain>
895template<class NodeOp>
896auto node_manager<Data, Degree, Domain>::traverse_post(
897 node_t* const rootNode,
898 NodeOp operation
899) const -> void
900{
901 this->traverse_post_impl(rootNode, operation);
902 this->traverse_post_impl(rootNode, [] (node_t*) {});
903 // Second traverse to reset marks.
904}
905
906template<class Data, class Degree, class Domain>
907template<class NodeOp>
908auto node_manager<Data, Degree, Domain>::traverse_post_impl(
909 node_t* const node,
910 NodeOp operation
911) const -> void
912{
913 node->toggle_marked();
914 if (node->is_internal())
915 {
916 int32 const nodeDomain = this->get_domain(node);
917 for (int32 k = 0; k < nodeDomain; ++k)
918 {
919 node_t* const son = node->get_son(k);
920 if (node->is_marked() != son->is_marked())
921 {
922 this->traverse_post_impl(son, operation);
923 }
924 }
925 }
926 operation(node);
927}
928
929template<class Data, class Degree, class Domain>
930template<class NodeOp>
931auto node_manager<Data, Degree, Domain>::traverse_level(
932 node_t* const rootNode,
933 NodeOp operation
934) const -> void
935{
936 std::vector<std::vector<node_t*>> buckets(as_usize(varCount_) + 1);
937 auto const endBucketIt = end(buckets);
938 auto bucketIt = begin(buckets) + this->get_level(rootNode);
939 (*bucketIt).push_back(rootNode);
940 rootNode->toggle_marked();
941
942 while (bucketIt != endBucketIt)
943 {
944 for (node_t* const node : *bucketIt)
945 {
946 operation(node);
947 if (node->is_internal())
948 {
949 int32 const domain = this->get_domain(node);
950 for (int32 k = 0; k < domain; ++k)
951 {
952 node_t* const son = node->get_son(k);
953 if (son->is_marked() != node->is_marked())
954 {
955 int32 const level = this->get_level(son);
956 buckets[as_uindex(level)].push_back(son);
957 son->toggle_marked();
958 }
959 }
960 }
961 }
962
963 do
964 {
965 ++bucketIt;
966 } while (bucketIt != endBucketIt && (*bucketIt).empty());
967 }
968
969 // Second traverse to reset marks.
970 this->traverse_pre_impl(rootNode, [] (node_t*) {});
971}
972
973template<class Data, class Degree, class Domain>
974auto node_manager<Data, Degree, Domain>::dec_ref_count(node_t* const node)
975 -> void
976{
977 node->dec_ref_count();
978}
979
980template<class Data, class Degree, class Domain>
981auto node_manager<Data, Degree, Domain>::is_redundant(
982 int32 const index,
983 son_container const& sons
984) const -> bool
985{
986 for (int32 j = 1; j < domains_[index]; ++j)
987 {
988 if (sons[as_uindex(j - 1)] != sons[as_uindex(j)])
989 {
990 return false;
991 }
992 }
993 return true;
994}
995
996template<class Data, class Degree, class Domain>
997auto node_manager<Data, Degree, Domain>::adjust_tables() -> void
998{
999#ifdef LIBTEDDY_VERBOSE
1000 debug::out(
1001 "node_manager::adjust_tables\tAdjusting unique tables.",
1002 " Node count is ",
1003 nodeCount_,
1004 "\n"
1005 );
1006#endif
1007
1008 for (int32 i = 0; i < ssize(uniqueTables_); ++i)
1009 {
1010 uniqueTables_[as_uindex(i)].adjust_capacity();
1011 }
1012}
1013
1014template<class Data, class Degree, class Domain>
1015auto node_manager<Data, Degree, Domain>::adjust_caches() -> void
1016{
1017 auto const newCapacity = cacheRatio_ * static_cast<double>(nodeCount_);
1018 opCache_.grow_capacity(static_cast<int64>(newCapacity));
1019}
1020
1021template<class Data, class Degree, class Domain>
1022template<class... Args>
1023auto node_manager<Data, Degree, Domain>::make_new_node(Args&&... args)
1024 -> node_t*
1025{
1026 if (autoReorderEnabled_)
1027 {
1028 // GC + reorder will be done after current
1029 // high level operations finishes.
1030 // Until then, just create a new pool.
1031
1032 if (pool_.get_available_node_count() == 0)
1033 {
1034 pool_.grow();
1035 this->deferr_gc_reorder();
1036 }
1037 }
1038 else
1039 {
1040 // Run GC. If not enough nodes are collected,
1041 // preventively create a new pool.
1042
1043 if (pool_.get_available_node_count() == 0)
1044 {
1045 auto const growThreshold = static_cast<int64>(
1046 gcRatio_ * static_cast<double>(nodeCount_)
1047 );
1048
1049 this->force_gc();
1050
1051 if (pool_.get_available_node_count() < growThreshold)
1052 {
1053 pool_.grow();
1054 }
1055 }
1056 }
1057
1058 if (nodeCount_ >= adjustmentNodeCount_)
1059 {
1060 // When the number of nodes doubles,
1061 // adjust cache and table sizes.
1062 this->adjust_tables();
1063 this->adjust_caches();
1064 adjustmentNodeCount_ = 2 * adjustmentNodeCount_ / 1;
1065 // ^ ^
1066 // Possible optimization point here
1067 }
1068
1069 ++nodeCount_;
1070 return pool_.create(args...);
1071}
1072
1073template<class Data, class Degree, class Domain>
1074auto node_manager<Data, Degree, Domain>::delete_node(node_t* const n) -> void
1075{
1076 assert(not n->is_marked());
1077 --nodeCount_;
1078 n->set_unused();
1079 pool_.destroy(n);
1080}
1081
1082template<class Data, class Degree, class Domain>
1083template<class ForEachNode>
1084auto node_manager<Data, Degree, Domain>::to_dot_graph_common(
1085 std::ostream& ost,
1086 ForEachNode&& forEach
1087) const -> void
1088{
1089 auto const make_label = [] (node_t* const node)
1090 {
1091 if (node->is_terminal())
1092 {
1093 using namespace std::literals::string_literals;
1094 int32 const val = node->get_value();
1095 return val == Undefined ? "*"s : std::to_string(val);
1096 }
1097
1098 return "x" + std::to_string(node->get_index());
1099 };
1100
1101 auto const get_id_str = [] (node_t* const n)
1102 { return std::to_string(reinterpret_cast<uint64>(n)); };
1103
1104 auto const output_range = [] (auto& ostr, auto const& range, auto const sep)
1105 {
1106 auto const endIt = end(range);
1107 auto rangeIt = begin(range);
1108 while (rangeIt != endIt)
1109 {
1110 ostr << *rangeIt;
1111 ++rangeIt;
1112 if (rangeIt != endIt)
1113 {
1114 ostr << sep;
1115 }
1116 }
1117 };
1118
1119 auto const levelCount = as_usize(1 + this->get_var_count());
1120 std::vector<std::string> labels;
1121 std::vector<std::vector<std::string>> rankGroups(levelCount);
1122 std::vector<std::string> arcs;
1123 std::vector<std::string> squareShapes;
1124
1125 forEach(
1126 [&, this] (node_t* const node)
1127 {
1128 // Create label.
1129 int32 const level = this->get_level(node);
1130 labels.emplace_back(
1131 get_id_str(node) + R"( [label = ")" + make_label(node)
1132 + R"(", tooltip = ")" + std::to_string(node->get_ref_count())
1133 + R"("];)"
1134 );
1135
1136 if (node->is_terminal())
1137 {
1138 squareShapes.emplace_back(get_id_str(node));
1139 rankGroups.back().emplace_back(get_id_str(node) + ";");
1140 return;
1141 }
1142
1143 // Add to same level.
1144 rankGroups[as_uindex(level)].emplace_back(get_id_str(node) + ";");
1145
1146 // Add arcs.
1147 this->for_each_son(
1148 node,
1149 [&, sonOrder = 0] (node_t* const son) mutable
1150 {
1151 if constexpr (std::is_same_v<Degree, degrees::fixed<2>>)
1152 {
1153 arcs.emplace_back(
1154 get_id_str(node) + " -> " + get_id_str(son)
1155 + " [style = "
1156 + (0 == sonOrder ? "dashed" : "solid") + "];"
1157 );
1158 }
1159 else
1160 {
1161 arcs.emplace_back(
1162 get_id_str(node) + " -> " + get_id_str(son)
1163 + R"( [label = )" + std::to_string(sonOrder) + "];"
1164 );
1165 }
1166 ++sonOrder;
1167 }
1168 );
1169 }
1170 );
1171
1172 // Finally, output everything into the output stream.
1173 ost << "digraph DD {" << '\n';
1174 ost << " node [shape = square] ";
1175 output_range(ost, squareShapes, " ");
1176 ost << ";\n";
1177 ost << " node [shape = circle];"
1178 << "\n\n";
1179
1180 ost << " ";
1181 output_range(ost, labels, "\n ");
1182 ost << "\n\n";
1183 ost << " ";
1184 output_range(ost, arcs, "\n ");
1185 ost << "\n\n";
1186
1187 for (std::vector<std::string> const& ranks : rankGroups)
1188 {
1189 if (not ranks.empty())
1190 {
1191 ost << " { rank = same; ";
1192 output_range(ost, ranks, " ");
1193 ost << " }" << '\n';
1194 }
1195 }
1196 ost << '\n';
1197 ost << "}" << '\n';
1198}
1199
1200template<class Data, class Degree, class Domain>
1201auto node_manager<Data, Degree, Domain>::deferr_gc_reorder() -> void
1202{
1203 gcReorderDeferred_ = true;
1204}
1205
1206template<class Data, class Degree, class Domain>
1207auto node_manager<Data, Degree, Domain>::check_distinct(
1208 std::vector<int32> const& ints
1209) -> bool
1210{
1211 if (ints.empty())
1212 {
1213 return true;
1214 }
1215
1216 int32 const maxElem = *utils::max_elem(ints.begin(), ints.end());
1217 std::vector<bool> bitset(as_usize(maxElem + 1), false);
1218 for (int32 const checkInt : ints)
1219 {
1220 if (bitset[as_uindex(checkInt)])
1221 {
1222 return false;
1223 }
1224 bitset[as_uindex(checkInt)] = true;
1225 }
1226 return true;
1227}
1228
1229template<class Data, class Degree, class Domain>
1230auto node_manager<Data, Degree, Domain>::can_be_gced(node_t* const node) -> bool
1231{
1232 return node->get_ref_count() == 0 && not node->is_marked();
1233}
1234
1235template<class Data, class Degree, class Domain>
1236auto node_manager<Data, Degree, Domain>::swap_node_with_next(node_t* const node)
1237 -> void
1238{
1239 using node_matrix = utils::type_if<
1240 degrees::is_fixed<Degree>::value,
1241 node_t * [Degree::value][Degree::value],
1242 std::vector<std::vector<node_t*>>>::type;
1243
1244 int32 const nodeIndex = node->get_index();
1245 int32 const nextIndex = this->get_index(1 + this->get_level(node));
1246 int32 const nodeDomain = this->get_domain(nodeIndex);
1247 int32 const nextDomain = this->get_domain(nextIndex);
1248 son_container oldSons = this->make_son_container(nodeDomain);
1249 for (int32 k = 0; k < nodeDomain; ++k)
1250 {
1251 oldSons[k] = node->get_son(k);
1252 }
1253
1254 node_matrix cofactorMatrix;
1255 if constexpr (degrees::is_mixed<Degree>::value)
1256 {
1257 cofactorMatrix.resize(
1258 as_usize(nodeDomain),
1259 std::vector<node_t*>(as_usize(nextDomain))
1260 );
1261 }
1262
1263 for (auto nk = 0; nk < nodeDomain; ++nk)
1264 {
1265 node_t* const son = node->get_son(nk);
1266 for (auto sk = 0; sk < nextDomain; ++sk)
1267 {
1268 bool const justUseSon
1269 = son->is_terminal() || son->get_index() != nextIndex;
1270 cofactorMatrix[as_uindex(nk)][as_uindex(sk)]
1271 = justUseSon ? son : son->get_son(sk);
1272 }
1273 }
1274
1275 son_container outerSons = this->make_son_container(nextDomain);
1276 for (int32 outerK = 0; outerK < nextDomain; ++outerK)
1277 {
1278 son_container innerSons = this->make_son_container(nodeDomain);
1279 for (int32 innerK = 0; innerK < nodeDomain; ++innerK)
1280 {
1281 innerSons[innerK]
1282 = cofactorMatrix[as_uindex(innerK)][as_uindex(outerK)];
1283 }
1284 outerSons[outerK] = this->make_internal_node(nodeIndex, innerSons);
1285 }
1286
1287 node->set_index(nextIndex);
1288 node->set_sons(outerSons);
1289
1290 for (int32 k = 0; k < nextDomain; ++k)
1291 {
1292 node->get_son(k)->inc_ref_count();
1293 node->get_son(k)->set_notmarked();
1294 }
1295
1296 for (int32 k = 0; k < nodeDomain; ++k)
1297 {
1298 this->dec_ref_try_gc(oldSons[k]);
1299 }
1300
1301 if constexpr (degrees::is_mixed<Degree>::value)
1302 {
1303 node_t::delete_son_container(oldSons);
1304 }
1305}
1306
1307template<class Data, class Degree, class Domain>
1308auto node_manager<Data, Degree, Domain>::dec_ref_try_gc(node_t* const node)
1309 -> void
1310{
1311 node->dec_ref_count();
1312
1313 if (not can_be_gced(node))
1314 {
1315 return;
1316 }
1317
1318 if (node->is_internal())
1319 {
1320 int32 const nodeDomain = this->get_domain(node);
1321 for (int32 k = 0; k < nodeDomain; ++k)
1322 {
1323 this->dec_ref_try_gc(node->get_son(k));
1324 }
1325
1326 uniqueTables_[as_uindex(node->get_index())].erase(node);
1327 }
1328 else
1329 {
1330 if (is_special(node->get_value()))
1331 {
1332 specials_[as_uindex(special_to_index(node->get_value()))] = nullptr;
1333 }
1334 else
1335 {
1336 terminals_[as_uindex(node->get_value())] = nullptr;
1337 }
1338 }
1339
1340 this->delete_node(node);
1341}
1342
1343template<class Data, class Degree, class Domain>
1344auto node_manager<Data, Degree, Domain>::swap_variable_with_next(
1345 int32 const index
1346) -> void
1347{
1348 int32 const level = this->get_level(index);
1349 int32 const nextIndex = this->get_index(1 + level);
1350 unique_table<Data, Degree> tmpTable(uniqueTables_[as_uindex(index)]);
1351 uniqueTables_[as_uindex(index)].clear();
1352 for (node_t* const node : tmpTable)
1353 {
1354 this->swap_node_with_next(node);
1355 }
1356 uniqueTables_[as_uindex(index)].adjust_capacity();
1357 uniqueTables_[as_uindex(nextIndex)].merge(
1358 static_cast<unique_table<Data, Degree>&&>(tmpTable)
1359 );
1360
1361 utils::swap(
1362 levelToIndex_[as_uindex(level)],
1363 levelToIndex_[as_uindex(1 + level)]
1364 );
1365 ++indexToLevel_[as_uindex(index)];
1366 --indexToLevel_[as_uindex(nextIndex)];
1367}
1368
1369template<class Data, class Degree, class Domain>
1370auto node_manager<Data, Degree, Domain>::sift_variables() -> void
1371{
1372 using count_pair = struct
1373 {
1374 int32 index_;
1375 int64 count_;
1376 };
1377
1378 // Sorts indices by number of nodes with given index descending.
1379 auto const determine_sift_order = [this] ()
1380 {
1381 std::vector<count_pair> counts;
1382 counts.reserve(as_usize(varCount_));
1383 for (int32 index = 0; index < varCount_; ++index)
1384 {
1385 counts.push_back(count_pair {index, this->get_node_count(index)});
1386 }
1387 utils::sort(
1388 counts,
1389 [] (count_pair const& lhs, count_pair const& rhs)
1390 { return lhs.count_ > rhs.count_; }
1391 );
1392 return counts;
1393 };
1394
1395 // Moves variable one level down.
1396 auto const move_var_down
1397 = [this] (auto const index) { this->swap_variable_with_next(index); };
1398
1399 // Moves variable one level up.
1400 auto const move_var_up = [this] (auto const index)
1401 {
1402 int32 const level = this->get_level(index);
1403 int32 const prevIndex = this->get_index(level - 1);
1404 this->swap_variable_with_next(prevIndex);
1405 };
1406
1407 // Tries to place variable on each level.
1408 // In the end, restores position with lowest total number of nodes.
1409 auto const place_variable = [&, this] (auto const index)
1410 {
1411 int32 const lastInternalLevel = this->get_var_count() - 1;
1412 int32 currentLevel = this->get_level(index);
1413 int32 optimalLevel = currentLevel;
1414 int64 optimalCount = nodeCount_;
1415
1416 // Sift down.
1417 while (currentLevel != lastInternalLevel)
1418 {
1419 move_var_down(index);
1420 ++currentLevel;
1421 if (nodeCount_ < optimalCount)
1422 {
1423 optimalCount = nodeCount_;
1424 optimalLevel = currentLevel;
1425 }
1426 }
1427
1428 // Sift up.
1429 while (currentLevel != 0)
1430 {
1431 move_var_up(index);
1432 --currentLevel;
1433 if (nodeCount_ < optimalCount)
1434 {
1435 optimalCount = nodeCount_;
1436 optimalLevel = currentLevel;
1437 }
1438 }
1439
1440 // Restore optimal position.
1441 while (currentLevel != optimalLevel)
1442 {
1443 move_var_down(index);
1444 ++currentLevel;
1445 }
1446 };
1447
1448#ifdef LIBTEDDY_VERBOSE
1449 debug::out(
1450 "node_manager: Sifting variables. Node count before ",
1451 nodeCount_,
1452 ".\n"
1453 );
1454#endif
1455
1456 auto const siftOrder = determine_sift_order();
1457 for (auto const pair : siftOrder)
1458 {
1459 place_variable(pair.index_);
1460 }
1461
1462#ifdef LIBTEDDY_VERBOSE
1463 debug::out(
1464 "node_manager: Done sifting. Node count after ",
1465 nodeCount_,
1466 ".\n"
1467 );
1468#endif
1469
1470 gcReorderDeferred_ = false;
1471}
1472} // namespace teddy
1473
1474#endif
Cache for the apply opertaion.
Definition hash_tables.hpp:255
Definition node_manager.hpp:82
Definition node_manager.hpp:88
Definition node_pool.hpp:17
Definition node.hpp:91
Table of unique nodes.
Definition hash_tables.hpp:71
Definition operators.hpp:293
Definition node_manager.hpp:44
Definition node_manager.hpp:57
Definition node_manager.hpp:69
Definition node_manager.hpp:25