TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
diagram_manager.hpp
1#ifndef LIBTEDDY_DETAILS_DIAGRAM_MANAGER_HPP
2#define LIBTEDDY_DETAILS_DIAGRAM_MANAGER_HPP
3
4#include <libteddy/details/diagram.hpp>
5#include <libteddy/details/node_manager.hpp>
6#include <libteddy/details/operators.hpp>
7#include <libteddy/details/pla_file.hpp>
8#include <libteddy/details/stats.hpp>
9#include <libteddy/details/tools.hpp>
10#include <libteddy/details/types.hpp>
11
12#include <cmath>
13#include <concepts>
14#include <initializer_list>
15#include <iterator>
16#include <optional>
17#include <ranges>
18#include <vector>
19
20namespace teddy
21{
22template<class Vars>
23concept in_var_values = requires(Vars values, int32 index) {
24 {
25 values[index]
26 } -> std::convertible_to<int32>;
27 };
28
29template<class Vars>
30concept out_var_values = requires(Vars values, int32 index, int32 value) {
31 values[index] = value;
32 };
33
34template<class Node>
35concept expression_node = requires(Node node, int32 value) {
36 {
37 node.is_variable()
38 } -> std::same_as<bool>;
39 {
40 node.is_constant()
41 } -> std::same_as<bool>;
42 {
43 node.is_operation()
44 } -> std::same_as<bool>;
45 {
46 node.get_index()
47 } -> std::same_as<int32>;
48 {
49 node.get_value()
50 } -> std::same_as<int32>;
51 {
52 node.evaluate(value, value)
53 } -> std::same_as<int32>;
54 {
55 node.get_left()
56 } -> std::same_as<Node const&>;
57 {
58 node.get_right()
59 } -> std::same_as<Node const&>;
60 };
61
62template<class Degree>
63concept is_bdd = std::same_as<degrees::fixed<2>, Degree>;
64
65template<class F>
66concept int_to_int = requires(F function, int32 value) {
67 {
68 function(value)
69 } -> std::convertible_to<int32>;
70 };
71
72enum class fold_type
73{
74 Left,
75 Tree
76};
77
79{
80 int32 index_;
81 int32 value_;
82};
83
89template<class Data, class Degree, class Domain>
91{
92public:
98
99public:
105 auto constant (int32 val) -> diagram_t;
106
112 auto variable (int32 index) -> diagram_t;
113
119 template<class Foo = void>
120 requires(is_bdd<Degree>)
121 auto variable_not (int32 index) -> utils::second_t<Foo, diagram_t>;
122
128 auto operator() (int32 index) -> diagram_t;
129
136 template<std::convertible_to<int32> T>
137 auto variables (std::initializer_list<T> indices) -> std::vector<diagram_t>;
138
150 template<std::input_iterator I, std::sentinel_for<I> S>
151 auto variables (I first, S last) -> std::vector<diagram_t>;
152
159 template<std::ranges::input_range Is>
160 auto variables (Is const& indices) -> std::vector<diagram_t>;
161
198 template<std::input_iterator I, std::sentinel_for<I> S>
199 auto from_vector (I first, S last) -> diagram_t;
200
211 template<std::ranges::input_range R>
212 auto from_vector (R&& vector) -> diagram_t;
213
227 auto to_vector (diagram_t const& diagram) const -> std::vector<int32>;
228
235 template<std::output_iterator<teddy::int32> O>
236 auto to_vector_g (diagram_t const& diagram, O out) const -> void;
237
246 template<class Foo = void>
247 requires(is_bdd<Degree>)
248 auto from_pla (pla_file const& file, fold_type foldType = fold_type::Tree)
249 -> utils::second_t<Foo, std::vector<diagram_t>>;
250
259 template<expression_node Node>
260 auto from_expression_tree (Node const& root) -> diagram_t;
261
302 template<teddy_bin_op Op>
303 auto apply (diagram_t const& lhs, diagram_t const& rhs) -> diagram_t;
304
308 template<teddy_bin_op Op, class... Diagram>
309 auto apply_n (Diagram const&... diagrams) -> diagram_t;
310
311 // TODO apply_own
312
330 template<teddy_bin_op Op, std::ranges::input_range R>
331 auto left_fold (R const& diagrams) -> diagram_t;
332
353 template<teddy_bin_op Op, std::input_iterator I, std::sentinel_for<I> S>
354 auto left_fold (I first, S last) -> diagram_t;
355
376 template<teddy_bin_op Op, std::ranges::random_access_range R>
377 auto tree_fold (R& diagrams) -> diagram_t;
378
401 template<
402 teddy_bin_op Op,
403 std::random_access_iterator I,
404 std::sentinel_for<I> S>
405 auto tree_fold (I first, S last) -> diagram_t;
406
419 template<in_var_values Vars>
420 auto evaluate (diagram_t const& diagram, Vars const& values) const -> int32;
421
433 auto satisfy_count (int32 value, diagram_t const& diagram) -> int64;
434
445 template<out_var_values Vars>
446 auto satisfy_one (int32 value, diagram_t const& diagram)
447 -> std::optional<Vars>;
448
467 template<out_var_values Vars>
468 auto satisfy_all (int32 value, diagram_t const& diagram) const
469 -> std::vector<Vars>;
470
491 template<out_var_values Vars, std::output_iterator<Vars> O>
492 auto satisfy_all_g (int32 value, diagram_t const& diagram, O out) const
493 -> void;
494
507 diagram_t const& diagram,
508 int32 varIndex,
509 int32 varValue
510 ) -> diagram_t;
511
512 auto get_cofactor (
513 diagram_t const& diagram,
514 std::vector<var_cofactor> const& vars
515 ) -> diagram_t;
516
534 template<int_to_int F>
535 auto transform (diagram_t const& diagram, F transformer) -> diagram_t;
536
544 template<class Foo = void>
545 requires(is_bdd<Degree>)
546 auto negate (diagram_t const& diagram) -> utils::second_t<Foo, diagram_t>;
547
554 -> std::vector<int32>;
555
562 template<std::output_iterator<int32> O>
563 auto get_dependency_set_g (diagram_t const& diagram, O out) const -> void;
564
574
586 [[nodiscard]] auto get_node_count () const -> int64;
587
594 auto get_node_count (diagram_t const& diagram) const -> int64;
595
604 auto to_dot_graph (std::ostream& out) const -> void;
605
615 auto to_dot_graph (std::ostream& out, diagram_t const& diagram) const
616 -> void;
617
628 auto force_gc () -> void;
629
633 auto force_reorder () -> void;
634
638 auto clear_cache () -> void;
639
645 [[nodiscard]] auto get_var_count () const -> int32;
646
657 [[nodiscard]] auto get_order () const -> std::vector<int32> const&;
658
669 [[nodiscard]] auto get_domains () const -> std::vector<int32>;
670
681 auto set_cache_ratio (double ratio) -> void;
682
693 auto set_gc_ratio (double ratio) -> void;
694
706 auto set_auto_reorder (bool doReorder) -> void;
707
708protected:
709 using node_t = typename diagram<Data, Degree>::node_t;
710 using son_container = typename node_t::son_container;
711
712private:
713 template<int32 Size>
714 struct node_pack
715 {
716 node_t* key_[static_cast<std::size_t>(Size)] {nullptr};
717 node_t* result_ {nullptr};
718 };
719
720 // TODO tmp
721 template<int32 Size, class... Node>
722 static auto pack_equals (node_pack<Size> const& pack, Node... nodes)
723 {
724 node_t* nodeArray[] {nodes...};
725 for (int32 k = 0; k < Size; ++k)
726 {
727 if (pack.key_[k] != nodeArray[k])
728 {
729 return false;
730 }
731 }
732 return true;
733 }
734
735private:
736 // TODO namiesto mema by sa dali pouzit data,
737 // idealne keby data bolo iba pole bytov a dalo by sa tam ulozit cokolvek
738
739 auto variable_impl (int32 index) -> node_t*;
740
741 template<class Op>
742 auto apply_impl (Op operation, node_t* lhs, node_t* rhs) -> node_t*;
743
744 template<class Op, class... Node>
745 auto apply_n_impl (
746 std::vector<node_pack<sizeof...(Node)>>& cache,
747 Op operation,
748 Node... nodes
749 ) -> node_t*;
750
751 template<class Vars>
752 auto satisfy_one_impl (int32 value, Vars& vars, node_t* node) -> bool;
753
754 template<class Vars, class OutputIt>
755 auto satisfy_all_impl (
756 int32 value,
757 Vars& vars,
758 OutputIt out,
759 node_t* node,
760 int32 level
761 ) const -> void;
762
763 auto get_cofactor_impl (
764 std::unordered_map<node_t*, node_t*>& memo,
765 int32 varIndex,
766 int32 varValue,
767 node_t* node
768 ) -> node_t*;
769
770 auto get_cofactor_impl (
771 std::unordered_map<node_t*, node_t*>& memo,
772 std::vector<var_cofactor> const& vars,
773 node_t* node,
774 int32 toCofactor
775 ) -> node_t*;
776
777 template<class F>
778 auto transform_impl (
779 std::unordered_map<node_t*, node_t*>& memo,
780 F transformer,
781 node_t* node
782 ) -> node_t*;
783
784 template<class ExprNode>
785 auto from_expression_tree_impl (
786 std::vector<node_pack<2>>& cache,
787 ExprNode const& exprNode
788 ) -> node_t*;
789
790protected:
803 int32 varCount,
804 int64 nodePoolSize,
805 int64 extraNodePoolSize,
806 std::vector<int32> order
807 )
809
823 int32 varCount,
824 int64 nodePoolSize,
825 int64 extraNodePoolSize,
826 domains::mixed domain,
827 std::vector<int32> order
828 )
830
831public:
832 diagram_manager(diagram_manager&&) noexcept = default;
833 ~diagram_manager() = default;
834 auto operator= (diagram_manager&&) noexcept -> diagram_manager& = default;
835 diagram_manager(diagram_manager const&) = delete;
836 auto operator= (diagram_manager const&) -> diagram_manager& = delete;
837
838protected:
839 node_manager<Data, Degree, Domain> nodes_;
840};
841
842template<class Data, class Degree, class Domain>
843auto diagram_manager<Data, Degree, Domain>::constant(int32 const val)
844 -> diagram_t
845{
846 return diagram_t(nodes_.make_terminal_node(val));
847}
848
849template<class Data, class Degree, class Domain>
851 -> diagram_t
852{
853 return diagram_t(this->variable_impl(index));
854}
855
856template<class Data, class Degree, class Domain>
857template<class Foo>
858requires(is_bdd<Degree>)
860 -> utils::second_t<Foo, diagram_t>
861{
862 son_container sons = nodes_.make_son_container(2);
863 sons[0] = nodes_.make_terminal_node(1);
864 sons[1] = nodes_.make_terminal_node(0);
865 return diagram_t(nodes_.make_internal_node(index, sons));
866}
867
868template<class Data, class Degree, class Domain>
870 -> diagram_t
871{
872 return this->variable(index);
873}
874
875template<class Data, class Degree, class Domain>
876template<std::ranges::input_range Is>
878 -> std::vector<diagram_t>
879{
880 return this->variables(begin(indices), end(indices));
881}
882
883template<class Data, class Degree, class Domain>
884template<std::convertible_to<int32> T>
886 std::initializer_list<T> const indices
887) -> std::vector<diagram_t>
888{
889 return this->variables(begin(indices), end(indices));
890}
891
892template<class Data, class Degree, class Domain>
893template<std::input_iterator I, std::sentinel_for<I> S>
895 -> std::vector<diagram_t>
896{
897 static_assert(std::convertible_to<std::iter_value_t<I>, int32>);
898 std::vector<diagram_t> result;
899 while (first != last)
900 {
901 result.push_back(this->variable(static_cast<int32>(*first)));
902 ++first;
903 }
904 return result;
905}
906
907template<class Data, class Degree, class Domain>
908template<std::input_iterator I, std::sentinel_for<I> S>
910 -> diagram_t
911{
912 if (0 == this->get_var_count())
913 {
914 assert(first != last && ++I(first) == last);
915 return diagram_t(nodes_.make_terminal_node(*first));
916 }
917
918 int32 const lastLevel = this->get_var_count() - 1;
919 int32 const lastIndex = nodes_.get_index(lastLevel);
920
921 if constexpr (std::random_access_iterator<I>)
922 {
923 [[maybe_unused]] int64 const count
924 = nodes_.domain_product(0, lastLevel + 1);
925 [[maybe_unused]] auto const dist = static_cast<int64>(last - first);
926 assert(dist > 0 && dist == count);
927 }
928
929 using stack_frame = struct
930 {
931 node_t* node;
932 int32 level;
933 };
934
935 std::vector<stack_frame> stack;
936 auto const shrink_stack = [this, &stack] ()
937 {
938 for (;;)
939 {
940 int32 const currentLevel = stack.back().level;
941 if (0 == currentLevel)
942 {
943 break;
944 }
945
946 auto const endId = rend(stack);
947 auto stackIt = rbegin(stack);
948 int64 count = 0;
949 while (stackIt != endId and stackIt->level == currentLevel)
950 {
951 ++stackIt;
952 ++count;
953 }
954 int32 const newIndex = nodes_.get_index(currentLevel - 1);
955 int32 const newDomain = nodes_.get_domain(newIndex);
956
957 if (count < newDomain)
958 {
959 break;
960 }
961
962 son_container newSons = nodes_.make_son_container(newDomain);
963 for (int32 k = 0; k < newDomain; ++k)
964 {
965 newSons[k]
966 = stack[as_uindex(ssize(stack) - newDomain + k)].node;
967 }
968 node_t* const newNode
969 = nodes_.make_internal_node(newIndex, newSons);
970 stack.erase(end(stack) - newDomain, end(stack));
971 stack.push_back(stack_frame {newNode, currentLevel - 1});
972 }
973 };
974
975 while (first != last)
976 {
977 int32 const lastDomain = nodes_.get_domain(lastIndex);
978 son_container sons = nodes_.make_son_container(lastDomain);
979 for (int32 k = 0; k < lastDomain; ++k)
980 {
981 sons[k] = nodes_.make_terminal_node(*first++);
982 }
983 node_t* const node = nodes_.make_internal_node(lastIndex, sons);
984 stack.push_back(stack_frame {node, lastLevel});
985 shrink_stack();
986 }
987
988 assert(ssize(stack) == 1);
989 return diagram_t(stack.back().node);
990}
991
992template<class Data, class Degree, class Domain>
993template<std::ranges::input_range R>
995{
996 return this->from_vector(begin(vector), end(vector));
997}
998
999template<class Data, class Degree, class Domain>
1001) const -> std::vector<int32>
1002{
1003 std::vector<int32> vector;
1004 vector.reserve(as_usize(nodes_.domain_product(0, this->get_var_count())));
1005 this->to_vector_g(diagram, std::back_inserter(vector));
1006 return vector;
1007}
1008
1009template<class Data, class Degree, class Domain>
1010template<std::output_iterator<teddy::int32> O>
1012 diagram_t const& diagram,
1013 O out
1014) const -> void
1015{
1016 if (this->get_var_count() == 0)
1017 {
1018 assert(diagram.unsafe_get_root()->is_terminal());
1019 *out++ = diagram.unsafe_get_root()->get_value();
1020 return;
1021 }
1022
1023 std::vector<int32> vars(as_usize(this->get_var_count()));
1024 bool wasLast = false;
1025 do
1026 {
1027 *out++ = this->evaluate(diagram, vars);
1028 bool overflow = true;
1029 int32 level = nodes_.get_leaf_level();
1030 while (level > 0 && overflow)
1031 {
1032 --level;
1033 int32 const index = nodes_.get_index(level);
1034 ++vars[as_uindex(index)];
1035 overflow = vars[as_uindex(index)] == nodes_.get_domain(index);
1036 if (overflow)
1037 {
1038 vars[as_uindex(index)] = 0;
1039 }
1040
1041 wasLast = overflow && 0 == level;
1042 }
1043 } while (not wasLast);
1044}
1045
1046template<class Data, class Degree, class Domain>
1047template<class Foo>
1048requires(is_bdd<Degree>)
1050 pla_file const& file,
1051 fold_type const foldType
1052) -> utils::second_t<Foo, std::vector<diagram_t>>
1053{
1054 auto const product = [this] (auto const& cube)
1055 {
1056 std::vector<diagram_t> variables;
1057 variables.reserve(cube.size());
1058 for (int32 i = 0; i < cube.size(); ++i)
1059 {
1060 if (cube.get(i) == 1)
1061 {
1062 variables.emplace_back(this->variable(i));
1063 }
1064 else if (cube.get(i) == 0)
1065 {
1066 variables.emplace_back(this->variable_not(i));
1067 }
1068 }
1069 return this->left_fold<ops::AND>(variables);
1070 };
1071
1072 auto const orFold = [this, foldType] (auto& diagrams)
1073 {
1074 switch (foldType)
1075 {
1076 case fold_type::Left:
1077 return this->left_fold<ops::OR>(diagrams);
1078
1079 case fold_type::Tree:
1080 return this->tree_fold<ops::OR>(diagrams);
1081
1082 default:
1083 assert(false);
1084 return this->constant(0);
1085 }
1086 };
1087
1088 auto const& plaLines = file.get_lines();
1089 int64 const lineCount = file.get_line_count();
1090 int64 const functionCount = file.get_function_count();
1091
1092 // Create a diagram for each function.
1093 std::vector<diagram_t> functionDiagrams;
1094 functionDiagrams.reserve(functionCount);
1095 for (int32 fi = 0; fi < functionCount; ++fi)
1096 {
1097 // First create a diagram for each product.
1098 std::vector<diagram_t> products;
1099 // products.reserve(lineCount);
1100 for (int32 li = 0; li < lineCount; ++li)
1101 {
1102 // We are doing SOP so we are only interested
1103 // in functions with value 1.
1104 if (plaLines[as_usize(li)].fVals_.get(fi) == 1)
1105 {
1106 products.emplace_back(product(plaLines[li].cube_));
1107 }
1108 }
1109
1110 // In this case we just have a constant function.
1111 if (products.empty())
1112 {
1113 products.emplace_back(this->constant(0));
1114 }
1115
1116 // Then merge products using OR.
1117 functionDiagrams.emplace_back(orFold(products));
1118 }
1119
1120 return functionDiagrams;
1121}
1122
1123template<class Data, class Degree, class Domain>
1124template<expression_node Node>
1126 Node const& root
1127) -> diagram_t
1128{
1129 int64 constexpr CacheCapacity = 100'000;
1130 std::vector<node_pack<2>> cache(as_usize(CacheCapacity));
1131 node_t* newRoot = this->from_expression_tree_impl(cache, root);
1132 nodes_.run_deferred();
1133 return diagram_t(newRoot);
1134}
1135
1136template<class Data, class Degree, class Domain>
1137template<class ExprNode>
1139 std::vector<node_pack<2>>& cache,
1140 ExprNode const& exprNode
1141) -> node_t*
1142{
1143 if (exprNode.is_constant())
1144 {
1145 return nodes_.make_terminal_node(exprNode.get_value());
1146 }
1147
1148 if (exprNode.is_variable())
1149 {
1150 return this->variable_impl(exprNode.get_index());
1151 }
1152
1153 assert(exprNode.is_operation());
1154
1155 node_t* const left
1156 = this->from_expression_tree_impl(cache, exprNode.get_left());
1157 node_t* const right
1158 = this->from_expression_tree_impl(cache, exprNode.get_right());
1159 auto const operation = [&exprNode] (auto const lhs, auto const rhs)
1160 {
1161 if (lhs == Nondetermined || rhs == Nondetermined)
1162 {
1163 return Nondetermined;
1164 }
1165 return static_cast<int32>(exprNode.evaluate(lhs, rhs));
1166 };
1167
1168 node_t* const newRoot = this->apply_n_impl(cache, operation, left, right);
1169 auto const cacheCapacity = cache.size();
1170 cache.clear();
1171 cache.resize(cacheCapacity);
1172 return newRoot;
1173}
1174
1175template<class Data, class Degree, class Domain>
1176auto diagram_manager<Data, Degree, Domain>::variable_impl(int32 const index)
1177 -> node_t*
1178{
1179 int32 const varDomain = nodes_.get_domain(index);
1180 son_container sons = nodes_.make_son_container(varDomain);
1181 for (int32 val = 0; val < varDomain; ++val)
1182 {
1183 sons[val] = nodes_.make_terminal_node(val);
1184 }
1185 return nodes_.make_internal_node(index, sons);
1186}
1187
1188template<class Data, class Degree, class Domain>
1189template<teddy_bin_op Op>
1191 diagram_t const& lhs,
1192 diagram_t const& rhs
1193) -> diagram_t
1194{
1195 /*
1196 * Use bounded MAX if the max value is known.
1197 * This should perform better since it can short-circuit.
1198 */
1199 using OpType = utils::type_if<
1202 Op>::type;
1203
1204 node_t* const newRoot = this->apply_impl(
1205 OpType(),
1206 lhs.unsafe_get_root(),
1207 rhs.unsafe_get_root()
1208 );
1209 nodes_.run_deferred();
1210 return diagram_t(newRoot);
1211}
1212
1213template<class Data, class Degree, class Domain>
1214template<class Op>
1216 Op operation,
1217 node_t* const lhs,
1218 node_t* const rhs
1219) -> node_t*
1220{
1221#ifdef LIBTEDDY_COLLECT_STATS
1222 ++stats::get_stats().applyStepCalls_;
1223#endif
1224
1225 node_t* const cached = nodes_.template cache_find<Op>(lhs, rhs);
1226 if (cached)
1227 {
1228 return cached;
1229 }
1230
1231 int32 const lhsVal = lhs->is_terminal() ? lhs->get_value() : Nondetermined;
1232 int32 const rhsVal = rhs->is_terminal() ? rhs->get_value() : Nondetermined;
1233 int32 const opVal = operation(lhsVal, rhsVal);
1234
1235 if (opVal != Nondetermined)
1236 {
1237 node_t* const result = nodes_.make_terminal_node(opVal);
1238 nodes_.template cache_put<Op>(result, lhs, rhs);
1239 return result;
1240 }
1241
1242 int32 const lhsLevel = nodes_.get_level(lhs);
1243 int32 const rhsLevel = nodes_.get_level(rhs);
1244 int32 const topLevel = utils::min(lhsLevel, rhsLevel);
1245 int32 const topIndex = nodes_.get_index(topLevel);
1246 int32 const domain = nodes_.get_domain(topIndex);
1247 son_container sons = nodes_.make_son_container(domain);
1248 for (int32 k = 0; k < domain; ++k)
1249 {
1250 sons[k] = this->apply_impl(
1251 operation,
1252 lhsLevel == topLevel ? lhs->get_son(k) : lhs,
1253 rhsLevel == topLevel ? rhs->get_son(k) : rhs
1254 );
1255 }
1256
1257 node_t* const result = nodes_.make_internal_node(topIndex, sons);
1258 nodes_.template cache_put<Op>(result, lhs, rhs);
1259 return result;
1260}
1261
1262template<class Data, class Degree, class Domain>
1263template<teddy_bin_op Op, class... Diagram>
1265 -> diagram_t
1266{
1267 /*
1268 * Use bounded MAX if the max value is known.
1269 * This should perform better since it can short-circuit.
1270 */
1271 using OpType = utils::type_if<
1274 Op>::type;
1275
1276 // TODO capacity
1277 std::vector<node_pack<sizeof...(Diagram)>, node_t*> cache(100'000);
1278 node_t* const newRoot
1279 = this->apply_n_impl(cache, OpType(), diagram.unsafe_get_root()...);
1280 nodes_.run_deferred();
1281 return diagram_t(newRoot);
1282}
1283
1284template<class Data, class Degree, class Domain>
1285template<class Op, class... Node>
1287 std::vector<node_pack<sizeof...(Node)>>& cache,
1288 Op operation,
1289 Node... nodes
1290) -> node_t*
1291{
1292 std::size_t const hash = utils::pack_hash(nodes...);
1293 std::size_t const cacheIndex = hash % cache.size();
1294 node_pack<sizeof...(Node)>& cachePack = cache[cacheIndex];
1295 if (pack_equals(cachePack, nodes...))
1296 {
1297 return cachePack.result_;
1298 }
1299
1300 int32 const opVal = operation(
1301 (nodes->is_terminal() ? nodes->get_value() : Nondetermined)...
1302 );
1303 node_t* result = nullptr;
1304
1305 if (opVal != Nondetermined)
1306 {
1307 result = nodes_.make_terminal_node(opVal);
1308 }
1309 else
1310 {
1311 int32 const minLevel = utils::pack_min(nodes_.get_level(nodes)...);
1312 int32 const topIndex = nodes_.get_index(minLevel);
1313 int32 const domain = nodes_.get_domain(topIndex);
1314 son_container sons = nodes_.make_son_container(domain);
1315 for (int32 k = 0; k < domain; ++k)
1316 {
1317 sons[k] = this->apply_n_impl(
1318 cache,
1319 operation,
1320 (nodes_.get_level(nodes) == minLevel ? nodes->get_son(k) : nodes
1321 )...
1322 );
1323 }
1324 result = nodes_.make_internal_node(topIndex, sons);
1325 }
1326
1327 cachePack = {{nodes...}, result};
1328 return result;
1329}
1330
1331template<class Data, class Degree, class Domain>
1332template<teddy_bin_op Op, std::ranges::input_range R>
1334 -> diagram_t
1335{
1336 return this->left_fold<Op>(begin(diagrams), end(diagrams));
1337}
1338
1339template<class Data, class Degree, class Domain>
1340template<teddy_bin_op Op, std::input_iterator I, std::sentinel_for<I> S>
1342 -> diagram_t
1343{
1344 static_assert(std::same_as<std::iter_value_t<I>, diagram_t>);
1345
1346 diagram_t result = *first;
1347 ++first;
1348
1349 while (first != last)
1350 {
1351 result = this->apply<Op>(result, *first);
1352 ++first;
1353 }
1354
1355 return result;
1356}
1357
1358template<class Data, class Degree, class Domain>
1359template<teddy_bin_op Op, std::ranges::random_access_range R>
1361{
1362 return this->tree_fold<Op>(begin(diagrams), end(diagrams));
1363}
1364
1365template<class Data, class Degree, class Domain>
1366template<teddy_bin_op Op, std::random_access_iterator I, std::sentinel_for<I> S>
1368 -> diagram_t
1369{
1370 static_assert(std::same_as<std::iter_value_t<I>, diagram_t>);
1371
1372 int64 const count = std::distance(first, last);
1373 int64 currentCount = count;
1374 auto const numOfSteps = static_cast<int64>(std::ceil(std::log2(count)));
1375
1376 for (auto step = 0; step < numOfSteps; ++step)
1377 {
1378 auto const justMoveLast = static_cast<bool>(currentCount & 1);
1379 currentCount = (currentCount / 2) + justMoveLast;
1380 int64 const pairCount = currentCount - justMoveLast;
1381
1382 for (int64 i = 0; i < pairCount; ++i)
1383 {
1384 *(first + i)
1385 = this->apply<Op>(*(first + 2 * i), *(first + 2 * i + 1));
1386 }
1387
1388 if (justMoveLast)
1389 {
1390 *(first + currentCount - 1)
1391 = static_cast<diagram_t&&>(*(first + 2 * (currentCount - 1)));
1392 }
1393 }
1394
1395 return diagram_t(static_cast<diagram_t&&>(*first));
1396}
1397
1398template<class Data, class Degree, class Domain>
1399template<in_var_values Vars>
1401 diagram_t const& diagram,
1402 Vars const& values
1403) const -> int32
1404{
1405 node_t* node = diagram.unsafe_get_root();
1406
1407 while (not node->is_terminal())
1408 {
1409 int32 const index = node->get_index();
1410 assert(nodes_.is_valid_var_value(index, values[as_uindex(index)]));
1411 node = node->get_son(values[as_uindex(index)]);
1412 }
1413
1414 return node->get_value();
1415}
1416
1417template<class Data, class Degree, class Domain>
1419 int32 const value,
1420 diagram_t const& diagram
1421) -> int64
1422{
1424 {
1425 assert(value < Domain::value);
1426 }
1427
1428 auto constexpr CanUseDataMember = not utils::is_void<Data>::value;
1430
1431 // A function that returns reference to
1432 // the data associated with given node.
1433 auto data = [] ()
1434 {
1435 if constexpr (CanUseDataMember)
1436 {
1437 // Simply return reference to the data member.
1438 return [] (node_t* const node) mutable -> T&
1439 { return (node->get_data()); };
1440 }
1441 else
1442 {
1443 // Return reference to the data that is stored in the map.
1444 return [map = std::unordered_map<node_t*, T>()] (node_t* const node
1445 ) mutable -> T&
1446 {
1447 // If there is no value for given key [] creates new pair
1448 // and value-initializes the value (0 for primitive types).
1449 return map[node];
1450 };
1451 }
1452 }();
1453
1454 node_t* const root = diagram.unsafe_get_root();
1455
1456 // Actual satisfy count algorithm.
1457 nodes_.traverse_post(
1458 root,
1459 [this, value, &data] (node_t* const node) mutable
1460 {
1461 if (node->is_terminal())
1462 {
1463 data(node) = node->get_value() == value ? 1 : 0;
1464 }
1465 else
1466 {
1467 data(node) = 0;
1468 int32 const nodeLevel = nodes_.get_level(node);
1469 int32 const nodeDomain = nodes_.get_domain(node);
1470 for (int32 k = 0; k < nodeDomain; ++k)
1471 {
1472 node_t* const son = node->get_son(k);
1473 int32 const sonLevel = nodes_.get_level(son);
1474 int64 const diff
1475 = nodes_.domain_product(nodeLevel + 1, sonLevel);
1476 data(node) += data(son) * static_cast<T>(diff);
1477 }
1478 }
1479 }
1480 );
1481
1482 auto const rootAlpha = static_cast<int64>(data(root));
1483 int32 const rootLevel = nodes_.get_level(root);
1484 return rootAlpha * nodes_.domain_product(0, rootLevel);
1485}
1486
1487template<class Data, class Degree, class Domain>
1488template<out_var_values Vars>
1490 int32 const value,
1491 diagram_t const& diagram
1492) -> std::optional<Vars>
1493{
1495 {
1496 assert(value < Domain::value);
1497 }
1498
1499 Vars vars;
1500 if constexpr (utils::is_std_vector<Vars>)
1501 {
1502 vars.resize(as_usize(this->get_var_count()));
1503 }
1504
1505 node_t* const root = diagram.unsafe_get_root();
1506 return this->satisfy_one_impl(value, vars, root) ? std::make_optional(vars)
1507 : std::nullopt;
1508}
1509
1510template<class Data, class Degree, class Domain>
1511template<class Vars>
1513 int32 const value,
1514 Vars& vars,
1515 node_t* const node
1516) -> bool
1517{
1518 if (node->is_terminal())
1519 {
1520 return node->get_value() == value;
1521 }
1522
1523 int32 const nodeIndex = node->get_index();
1524 int32 const nodeDomain = nodes_.get_domain(nodeIndex);
1525 for (auto k = 0; k < nodeDomain; ++k)
1526 {
1527 node_t* const son = node->get_son(k);
1528 vars[as_uindex(nodeIndex)] = k;
1529 if (this->satisfy_one_impl(value, vars, son))
1530 {
1531 return true;
1532 }
1533 }
1534
1535 return false;
1536}
1537
1538template<class Data, class Degree, class Domain>
1539template<out_var_values Vars>
1541 int32 const value,
1542 diagram_t const& diagram
1543) const -> std::vector<Vars>
1544{
1545 std::vector<Vars> result;
1546 this->satisfy_all_g<Vars>(value, diagram, std::back_inserter(result));
1547 return result;
1548}
1549
1550template<class Data, class Degree, class Domain>
1551template<out_var_values Vars, std::output_iterator<Vars> OutputIt>
1553 int32 const value,
1554 diagram_t const& diagram,
1555 OutputIt out
1556) const -> void
1557{
1559 {
1560 assert(value < Domain::value);
1561 }
1562
1563 Vars vars;
1564 if constexpr (utils::is_std_vector<Vars>)
1565 {
1566 vars.resize(as_usize(this->get_var_count()));
1567 }
1568
1569 node_t* const root = diagram.unsafe_get_root();
1570 this->satisfy_all_impl(value, vars, out, root, 0);
1571}
1572
1573template<class Data, class Degree, class Domain>
1574template<class Vars, class OutputIt>
1575auto diagram_manager<Data, Degree, Domain>::satisfy_all_impl(
1576 int32 const value,
1577 Vars& vars,
1578 OutputIt out,
1579 node_t* const node,
1580 int32 const level
1581) const -> void
1582{
1583 if (node->is_terminal() && value != node->get_value())
1584 {
1585 return;
1586 }
1587
1588 if (level == nodes_.get_leaf_level() && value == node->get_value())
1589 {
1590 *out++ = vars;
1591 return;
1592 }
1593
1594 if (nodes_.get_level(node) > level)
1595 {
1596 int32 const index = nodes_.get_index(level);
1597 int32 const domain = nodes_.get_domain(index);
1598 for (auto k = 0; k < domain; ++k)
1599 {
1600 vars[as_uindex(index)] = k;
1601 this->satisfy_all_impl(value, vars, out, node, level + 1);
1602 }
1603 }
1604 else
1605 {
1606 int32 const index = node->get_index();
1607 int32 const domain = nodes_.get_domain(index);
1608 for (auto k = 0; k < domain; ++k)
1609 {
1610 vars[as_uindex(index)] = k;
1611 node_t* const son = node->get_son(k);
1612 this->satisfy_all_impl(value, vars, out, son, level + 1);
1613 }
1614 }
1615}
1616
1617template<class Data, class Degree, class Domain>
1619 diagram_t const& diagram,
1620 int32 const varIndex,
1621 int32 const varValue
1622) -> diagram_t
1623{
1624 node_t* const root = diagram.unsafe_get_root();
1625 if (root->is_terminal())
1626 {
1627 return diagram;
1628 }
1629
1630 if (root->get_index() == varIndex)
1631 {
1632 return diagram_t(root->get_son(varValue));
1633 }
1634
1635 std::unordered_map<node_t*, node_t*> memo;
1636 diagram_t result
1637 = diagram_t(this->get_cofactor_impl(memo, varIndex, varValue, root));
1638 nodes_.run_deferred();
1639 return result;
1640}
1641
1642template<class Data, class Degree, class Domain>
1644 diagram_t const& diagram,
1645 std::vector<var_cofactor> const& vars
1646) -> diagram_t
1647{
1648 node_t* root = diagram.unsafe_get_root();
1649 if (root->is_terminal())
1650 {
1651 return diagram;
1652 }
1653
1654 auto const it = utils::find_if(vars.begin(), vars.end(),
1655 [root](var_cofactor var)
1656 {
1657 return var.index_ == root->get_index();
1658 });
1659
1660 int32 toCofactor = static_cast<int32>(vars.size());
1661 if (it != vars.end())
1662 {
1663 root = root->get_son(it->value_);
1664 --toCofactor;
1665 }
1666
1667 std::unordered_map<node_t*, node_t*> memo;
1668 diagram_t result = diagram_t(this->get_cofactor_impl(
1669 memo,
1670 vars,
1671 root,
1672 toCofactor
1673 ));
1674 nodes_.run_deferred();
1675 return result;
1676}
1677
1678template<class Data, class Degree, class Domain>
1679auto diagram_manager<Data, Degree, Domain>::get_cofactor_impl(
1680 std::unordered_map<node_t*, node_t*>& memo,
1681 int32 const varIndex,
1682 int32 const varValue,
1683 node_t* const node
1684) -> node_t*
1685{
1686 auto const memoIt = memo.find(node);
1687 if (memoIt != memo.end())
1688 {
1689 return memoIt->second;
1690 }
1691
1692 if (node->is_terminal())
1693 {
1694 return node;
1695 }
1696
1697 int32 const nodeIndex = node->get_index();
1698 if (nodeIndex == varIndex)
1699 {
1700 return node->get_son(varValue);
1701 }
1702
1703 int32 const nodeDomain = nodes_.get_domain(node);
1704 son_container sons = nodes_.make_son_container(nodeDomain);
1705 for (int32 k = 0; k < nodeDomain; ++k)
1706 {
1707 node_t* const oldSon = node->get_son(k);
1708 sons[k] = this->get_cofactor_impl(memo, varIndex, varValue, oldSon);
1709 }
1710
1711 node_t* const newNode = nodes_.make_internal_node(nodeIndex, sons);
1712 memo.emplace(node, newNode);
1713 return newNode;
1714}
1715
1716template<class Data, class Degree, class Domain>
1717auto diagram_manager<Data, Degree, Domain>::get_cofactor_impl(
1718 std::unordered_map<node_t*, node_t*>& memo,
1719 std::vector<var_cofactor> const& vars,
1720 node_t* const node,
1721 int32 const toCofactor
1722) -> node_t*
1723{
1724 if (toCofactor == 0)
1725 {
1726 return node;
1727 }
1728
1729 auto const memoIt = memo.find(node);
1730 if (memoIt != memo.end())
1731 {
1732 return memoIt->second;
1733 }
1734
1735 if (node->is_terminal())
1736 {
1737 return node;
1738 }
1739
1740 int32 const nodeIndex = node->get_index();
1741 auto const it = utils::find_if(vars.begin(), vars.end(),
1742 [nodeIndex](var_cofactor var)
1743 {
1744 return var.index_ == nodeIndex;
1745 });
1746
1747 node_t* newNode = nullptr;
1748 if (it != vars.end())
1749 {
1750 newNode = this->get_cofactor_impl(
1751 memo,
1752 vars,
1753 node->get_son(it->value_),
1754 toCofactor - 1
1755 );
1756 }
1757 else
1758 {
1759 int32 const nodeDomain = nodes_.get_domain(node);
1760 son_container sons = nodes_.make_son_container(nodeDomain);
1761 for (int32 k = 0; k < nodeDomain; ++k)
1762 {
1763 node_t* const oldSon = node->get_son(k);
1764 sons[k] = this->get_cofactor_impl(memo, vars, oldSon, toCofactor);
1765 }
1766 newNode = nodes_.make_internal_node(nodeIndex, sons);
1767 }
1768
1769 memo.emplace(node, newNode);
1770 return newNode;
1771}
1772
1773template<class Data, class Degree, class Domain>
1774template<int_to_int F>
1776 diagram_t const& diagram,
1777 F transformer
1778) -> diagram_t
1779{
1780 std::unordered_map<node_t*, node_t*> memo;
1781 node_t* const newRoot
1782 = this->transform_impl(memo, transformer, diagram.unsafe_get_root());
1783 nodes_.run_deferred();
1784 return diagram_t(newRoot);
1785}
1786
1787template<class Data, class Degree, class Domain>
1788template<class F>
1790 std::unordered_map<node_t*, node_t*>& memo,
1791 F transformer,
1792 node_t* node
1793) -> node_t*
1794{
1795 auto const memoIt = memo.find(node);
1796 if (memo.end() != memoIt)
1797 {
1798 return memoIt->second;
1799 }
1800
1801 if (node->is_terminal())
1802 {
1803 int32 const newVal = static_cast<int32>(transformer(node->get_value()));
1804 return nodes_.make_terminal_node(newVal);
1805 }
1806
1807 int32 const index = node->get_index();
1808 int32 const domain = nodes_.get_domain(index);
1809 son_container sons = nodes_.make_son_container(domain);
1810 for (int32 k = 0; k < domain; ++k)
1811 {
1812 node_t* const son = node->get_son(k);
1813 sons[k] = this->transform_impl(memo, transformer, son);
1814 }
1815 node_t* const newNode = nodes_.make_internal_node(index, sons);
1816 memo.emplace(node, newNode);
1817 return newNode;
1818}
1819
1820template<class Data, class Degree, class Domain>
1821template<class Foo>
1822requires(is_bdd<Degree>)
1824 -> utils::second_t<Foo, diagram_t>
1825{
1826 return this->transform(
1827 diagram,
1828 [] (int32 const value) { return 1 - value; }
1829 );
1830}
1831
1832template<class Data, class Degree, class Domain>
1834 diagram_t const& diagram
1835) const -> std::vector<int32>
1836{
1837 std::vector<int32> indices;
1838 indices.reserve(this->get_var_count());
1839 this->get_dependency_set_g(diagram, std::back_inserter(indices));
1840 indices.shrink_to_fit();
1841 return indices;
1842}
1843
1844template<class Data, class Degree, class Domain>
1845template<std::output_iterator<int32> O>
1847 diagram_t const& diagram,
1848 O out
1849) const -> void
1850{
1851 std::vector<bool> memo(as_usize(this->get_var_count()), false);
1852 nodes_.traverse_pre(
1854 [&memo, out] (node_t* const node) mutable
1855 {
1856 if (node->is_internal())
1857 {
1858 int32 const index = node->get_index();
1859 if (not memo[as_uindex(index)])
1860 {
1861 *out++ = index;
1862 }
1863 memo[as_uindex(index)] = true;
1864 }
1865 }
1866 );
1867}
1868
1869template<class Data, class Degree, class Domain>
1871 -> diagram_t
1872{
1873 return this->transform(diagram, [] (int32 const val) { return val; });
1874}
1875
1876template<class Data, class Degree, class Domain>
1878{
1879 return nodes_.get_node_count();
1880}
1881
1882template<class Data, class Degree, class Domain>
1884 diagram_t const& diagram
1885) const -> int64
1886{
1887 return nodes_.get_node_count(diagram.unsafe_get_root());
1888}
1889
1890template<class Data, class Degree, class Domain>
1892) const -> void
1893{
1894 nodes_.to_dot_graph(out);
1895}
1896
1897template<class Data, class Degree, class Domain>
1899 std::ostream& out,
1900 diagram_t const& diagram
1901) const -> void
1902{
1903 nodes_.to_dot_graph(out, diagram.unsafe_get_root());
1904}
1905
1906template<class Data, class Degree, class Domain>
1908{
1909 return nodes_.get_var_count();
1910}
1911
1912template<class Data, class Degree, class Domain>
1914 -> std::vector<int32> const&
1915{
1916 return nodes_.get_order();
1917}
1918
1919template<class Data, class Degree, class Domain>
1921 -> std::vector<int32>
1922{
1923 return nodes_.get_domains();
1924}
1925
1926template<class Data, class Degree, class Domain>
1928 -> void
1929{
1930 nodes_.set_cache_ratio(ratio);
1931}
1932
1933template<class Data, class Degree, class Domain>
1935{
1936 nodes_.set_gc_ratio(ratio);
1937}
1938
1939template<class Data, class Degree, class Domain>
1941 bool const doReorder
1942) -> void
1943{
1944 nodes_.set_auto_reorder(doReorder);
1945}
1946
1947template<class Data, class Degree, class Domain>
1949{
1950 nodes_.force_gc();
1951}
1952
1953template<class Data, class Degree, class Domain>
1955{
1956 nodes_.sift_variables();
1957}
1958
1959template<class Data, class Degree, class Domain>
1961{
1962 nodes_.cache_clear();
1963}
1964
1965namespace detail
1966{
1967inline auto default_or_fwd (int32 const varCount, std::vector<int32>& indices)
1968{
1969 if (indices.empty())
1970 {
1971 std::vector<int32> defaultIndices;
1972 defaultIndices.reserve(as_usize(varCount));
1973 for (int32 index = 0; index < varCount; ++index)
1974 {
1975 defaultIndices.push_back(index);
1976 }
1977 return defaultIndices;
1978 }
1979 else
1980 {
1981 return std::vector<int32>(static_cast<std::vector<int32>&&>(indices));
1982 }
1983}
1984} // namespace detail
1985
1986template<class Data, class Degree, class Domain>
1988 int32 const varCount,
1989 int64 const nodePoolSize,
1990 int64 const extraNodePoolSize,
1991 std::vector<int32> order
1992)
1994 :
1995 nodes_(
1996 varCount,
1997 nodePoolSize,
1998 extraNodePoolSize,
1999 detail::default_or_fwd(varCount, order)
2000 )
2001{
2002}
2003
2004template<class Data, class Degree, class Domain>
2006 int32 const varCount,
2007 int64 const nodePoolSize,
2008 int64 const extraNodePoolSize,
2009 domains::mixed domain,
2010 std::vector<int32> order
2011)
2013 :
2014 nodes_(
2015 varCount,
2016 nodePoolSize,
2017 extraNodePoolSize,
2018 detail::default_or_fwd(varCount, order),
2019 static_cast<domains::mixed&&>(domain)
2020 )
2021{
2022}
2023} // namespace teddy
2024
2025#endif
Base class for all diagram managers that generically implements all of the algorithms.
Definition diagram_manager.hpp:91
diagram_manager(int32 varCount, int64 nodePoolSize, int64 extraNodePoolSize, std::vector< int32 > order)
Initializes diagram manager.
Definition diagram_manager.hpp:1987
auto set_cache_ratio(double ratio) -> void
Sets the relative cache size w.r.t the number of nodes.
Definition diagram_manager.hpp:1927
auto set_gc_ratio(double ratio) -> void
Sets ratio used to determine new node pool allocation.
Definition diagram_manager.hpp:1934
auto get_var_count() const -> int32
Returns number of variables for this manager set in the constructor.
Definition diagram_manager.hpp:1907
auto get_dependency_set(diagram_t const &diagram) const -> std::vector< int32 >
Enumerates indices of variables that the function depends on.
Definition diagram_manager.hpp:1833
auto force_gc() -> void
Runs garbage collection.
Definition diagram_manager.hpp:1948
auto get_domains() const -> std::vector< int32 >
Return domains of variables.
Definition diagram_manager.hpp:1920
diagram_manager(int32 varCount, int64 nodePoolSize, int64 extraNodePoolSize, domains::mixed domain, std::vector< int32 > order)
Initializes diagram manager.
Definition diagram_manager.hpp:2005
auto apply_n(Diagram const &... diagrams) -> diagram_t
TODO.
Definition diagram_manager.hpp:1264
auto to_vector(diagram_t const &diagram) const -> std::vector< int32 >
Creates truth vector from the diagram.
Definition diagram_manager.hpp:1000
auto apply(diagram_t const &lhs, diagram_t const &rhs) -> diagram_t
Merges two diagrams using given binary operation.
Definition diagram_manager.hpp:1190
auto from_expression_tree(Node const &root) -> diagram_t
Creates diagram from an expression tree (AST).
Definition diagram_manager.hpp:1125
auto negate(diagram_t const &diagram) -> utils::second_t< Foo, diagram_t >
Negates Boolean function.
Definition diagram_manager.hpp:1823
auto satisfy_count(int32 value, diagram_t const &diagram) -> int64
Calculates number of variable assignments for which the functions evaluates to certain value.
Definition diagram_manager.hpp:1418
auto tree_fold(R &diagrams) -> diagram_t
Merges diagams in the range using the apply function and binary operation.
Definition diagram_manager.hpp:1360
auto variable(int32 index) -> diagram_t
Creates diagram representing function of single variable.
Definition diagram_manager.hpp:850
auto variables(I first, S last) -> std::vector< diagram_t >
Creates vector of diagrams representing functions of single variables.
Definition diagram_manager.hpp:894
auto variables(Is const &indices) -> std::vector< diagram_t >
Creates vector of diagrams representing single variables.
Definition diagram_manager.hpp:877
auto from_vector(I first, S last) -> diagram_t
Creates diagram from a truth vector of a function.
Definition diagram_manager.hpp:909
auto from_vector(R &&vector) -> diagram_t
Creates diagram from a truth vector of a function.
Definition diagram_manager.hpp:994
auto variables(std::initializer_list< T > indices) -> std::vector< diagram_t >
Creates vector of diagrams representing single variables.
Definition diagram_manager.hpp:885
auto left_fold(I first, S last) -> diagram_t
Merges diagams in the range using the apply function and binary operation.
Definition diagram_manager.hpp:1341
auto to_dot_graph(std::ostream &out) const -> void
Prints dot representation of the graph.
Definition diagram_manager.hpp:1891
auto set_auto_reorder(bool doReorder) -> void
Enables or disables automatic variable reordering.
Definition diagram_manager.hpp:1940
auto to_vector_g(diagram_t const &diagram, O out) const -> void
Creates truth vector from the diagram.
Definition diagram_manager.hpp:1011
auto satisfy_one(int32 value, diagram_t const &diagram) -> std::optional< Vars >
Finds variable assignment for which diagram evaluates to value.
Definition diagram_manager.hpp:1489
auto constant(int32 val) -> diagram_t
Creates diagram representing constant function.
Definition diagram_manager.hpp:843
auto reduce(diagram_t const &diagram) -> diagram_t
Reduces diagrams to its canonical form.
Definition diagram_manager.hpp:1870
auto get_order() const -> std::vector< int32 > const &
Returns current order of variables.
Definition diagram_manager.hpp:1913
auto evaluate(diagram_t const &diagram, Vars const &values) const -> int32
Evaluates value of the function represented by the diagram.
Definition diagram_manager.hpp:1400
auto get_dependency_set_g(diagram_t const &diagram, O out) const -> void
Enumerates indices of variables that the function depends on.
Definition diagram_manager.hpp:1846
auto from_pla(pla_file const &file, fold_type foldType=fold_type::Tree) -> utils::second_t< Foo, std::vector< diagram_t > >
Creates BDDs defined by PLA file.
Definition diagram_manager.hpp:1049
auto transform(diagram_t const &diagram, F transformer) -> diagram_t
Transforms values of the function.
Definition diagram_manager.hpp:1775
auto variable_not(int32 index) -> utils::second_t< Foo, diagram_t >
Creates BDD representing function of complemented variable.
Definition diagram_manager.hpp:859
auto operator()(int32 index) -> diagram_t
Creates diagram representing function of single variable.
Definition diagram_manager.hpp:869
auto get_node_count() const -> int64
Returns number of nodes that are currently used by the manager.
Definition diagram_manager.hpp:1877
auto satisfy_all(int32 value, diagram_t const &diagram) const -> std::vector< Vars >
Enumerates all elements of the satisfying set.
Definition diagram_manager.hpp:1540
auto satisfy_all_g(int32 value, diagram_t const &diagram, O out) const -> void
Enumerates all elements of the satisfying set.
auto clear_cache() -> void
Clears apply cache.
Definition diagram_manager.hpp:1960
auto force_reorder() -> void
Runs variable reordering heuristic.
Definition diagram_manager.hpp:1954
auto get_cofactor(diagram_t const &diagram, int32 varIndex, int32 varValue) -> diagram_t
Calculates cofactor of the functions.
Definition diagram_manager.hpp:1618
auto tree_fold(I first, S last) -> diagram_t
Merges diagams in the range using the apply function and binary operation.
Definition diagram_manager.hpp:1367
auto left_fold(R const &diagrams) -> diagram_t
Merges diagams in the range using the apply function and binary operation.
Definition diagram_manager.hpp:1333
diagram< Data, Degree > diagram_t
Alias for the diagram type used in the functions of this manager.
Definition diagram_manager.hpp:97
Cheap wrapper for the internal diagram node type.
Definition diagram.hpp:20
auto unsafe_get_root() const -> node_t *
Returns pointer to internal node type. You should probably don't use this one unless you know what yo...
Definition diagram.hpp:178
Definition node_manager.hpp:82
Definition node.hpp:91
Representation of a PLA file.
Definition pla_file.hpp:52
auto get_line_count() const -> int64
Returns number of lines in the file.
Definition pla_file.hpp:435
auto get_lines() const &-> std::vector< pla_line > const &
Returns reference to a vector holding lines.
Definition pla_file.hpp:440
auto get_function_count() const -> int32
Returns number of functions in the file.
Definition pla_file.hpp:430
Definition diagram_manager.hpp:35
Definition diagram_manager.hpp:23
Definition diagram_manager.hpp:66
Definition diagram_manager.hpp:63
Definition diagram_manager.hpp:30
Definition operators.hpp:293
Definition tools.hpp:309
Definition node_manager.hpp:57
Definition node_manager.hpp:69
Definition node_manager.hpp:25
Same as MAX but short-circuits for M – should be faster.
Definition operators.hpp:242
Definition tools.hpp:295
Definition tools.hpp:283
Provides member typedef based on the value of B Implementation of std::conditional.
Definition tools.hpp:319
Definition diagram_manager.hpp:79