1#ifndef LIBTEDDY_DETAILS_DIAGRAM_MANAGER_HPP
2#define LIBTEDDY_DETAILS_DIAGRAM_MANAGER_HPP
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>
14#include <initializer_list>
26 } -> std::convertible_to<int32>;
31 values[index] = value;
38 } -> std::same_as<bool>;
41 } -> std::same_as<bool>;
44 } -> std::same_as<bool>;
47 } -> std::same_as<int32>;
50 } -> std::same_as<int32>;
52 node.evaluate(value, value)
53 } -> std::same_as<int32>;
56 } -> std::same_as<Node const&>;
59 } -> std::same_as<Node const&>;
63concept is_bdd = std::same_as<degrees::fixed<2>, Degree>;
69 } -> std::convertible_to<int32>;
89template<
class Data,
class Degree,
class Domain>
119 template<
class Foo =
void>
136 template<std::convertible_to<
int32> T>
137 auto variables (std::initializer_list<T> indices) -> std::vector<diagram_t>;
150 template<std::input_iterator I, std::sentinel_for<I> S>
151 auto variables (I first, S last) -> std::vector<diagram_t>;
159 template<std::ranges::input_range Is>
160 auto variables (Is
const& indices) -> std::vector<diagram_t>;
198 template<std::input_iterator I, std::sentinel_for<I> S>
211 template<std::ranges::input_range R>
235 template<std::output_iterator<teddy::
int32> O>
246 template<
class Foo =
void>
249 -> utils::second_t<Foo, std::vector<diagram_t>>;
259 template<expression_node Node>
302 template<teddy_bin_op Op>
330 template<teddy_bin_op Op, std::ranges::input_range R>
353 template<teddy_bin_op Op, std::input_iterator I, std::sentinel_for<I> S>
376 template<teddy_bin_op Op, std::ranges::random_access_range R>
403 std::random_access_iterator I,
404 std::sentinel_for<I> S>
419 template<in_var_values Vars>
445 template<out_var_values Vars>
447 -> std::optional<Vars>;
467 template<out_var_values Vars>
469 -> std::vector<Vars>;
491 template<out_var_values Vars, std::output_iterator<Vars> O>
514 std::vector<var_cofactor>
const& vars
534 template<
int_to_
int F>
544 template<
class Foo =
void>
554 -> std::vector<int32>;
562 template<std::output_iterator<
int32> O>
657 [[nodiscard]] auto
get_order () const -> std::vector<int32> const&;
709 using node_t = typename
diagram<Data, Degree>::node_t;
710 using son_container = typename node_t::son_container;
716 node_t* key_[
static_cast<std::size_t
>(Size)] {
nullptr};
717 node_t* result_ {
nullptr};
721 template<int32 Size,
class... Node>
722 static auto pack_equals (node_pack<Size>
const& pack, Node... nodes)
724 node_t* nodeArray[] {nodes...};
725 for (int32 k = 0; k < Size; ++k)
727 if (pack.key_[k] != nodeArray[k])
739 auto variable_impl (int32 index) -> node_t*;
742 auto apply_impl (Op operation, node_t* lhs, node_t* rhs) -> node_t*;
744 template<
class Op,
class... Node>
746 std::vector<node_pack<
sizeof...(Node)>>& cache,
752 auto satisfy_one_impl (int32 value, Vars& vars, node_t* node) -> bool;
754 template<
class Vars,
class OutputIt>
755 auto satisfy_all_impl (
763 auto get_cofactor_impl (
764 std::unordered_map<node_t*, node_t*>& memo,
770 auto get_cofactor_impl (
771 std::unordered_map<node_t*, node_t*>& memo,
772 std::vector<var_cofactor>
const& vars,
778 auto transform_impl (
779 std::unordered_map<node_t*, node_t*>& memo,
784 template<
class ExprNode>
785 auto from_expression_tree_impl (
786 std::vector<node_pack<2>>& cache,
787 ExprNode
const& exprNode
805 int64 extraNodePoolSize,
806 std::vector<int32> order
825 int64 extraNodePoolSize,
827 std::vector<int32> order
842template<class Data, class Degree, class Domain>
846 return diagram_t(nodes_.make_terminal_node(val));
849template<
class Data,
class Degree,
class Domain>
853 return diagram_t(this->variable_impl(index));
856template<
class Data,
class Degree,
class Domain>
860 -> utils::second_t<Foo, diagram_t>
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));
868template<
class Data,
class Degree,
class Domain>
875template<
class Data,
class Degree,
class Domain>
876template<std::ranges::input_range Is>
878 -> std::vector<diagram_t>
880 return this->
variables(begin(indices), end(indices));
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>
889 return this->
variables(begin(indices), end(indices));
892template<
class Data,
class Degree,
class Domain>
893template<std::input_iterator I, std::sentinel_for<I> S>
895 -> std::vector<diagram_t>
897 static_assert(std::convertible_to<std::iter_value_t<I>, int32>);
898 std::vector<diagram_t> result;
899 while (first != last)
901 result.push_back(this->
variable(
static_cast<int32
>(*first)));
907template<
class Data,
class Degree,
class Domain>
908template<std::input_iterator I, std::sentinel_for<I> S>
914 assert(first != last && ++I(first) == last);
915 return diagram_t(nodes_.make_terminal_node(*first));
919 int32
const lastIndex = nodes_.get_index(lastLevel);
921 if constexpr (std::random_access_iterator<I>)
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);
929 using stack_frame =
struct
935 std::vector<stack_frame> stack;
936 auto const shrink_stack = [
this, &stack] ()
940 int32
const currentLevel = stack.back().level;
941 if (0 == currentLevel)
946 auto const endId = rend(stack);
947 auto stackIt = rbegin(stack);
949 while (stackIt != endId and stackIt->level == currentLevel)
954 int32
const newIndex = nodes_.get_index(currentLevel - 1);
955 int32
const newDomain = nodes_.get_domain(newIndex);
957 if (count < newDomain)
962 son_container newSons = nodes_.make_son_container(newDomain);
963 for (int32 k = 0; k < newDomain; ++k)
966 = stack[as_uindex(ssize(stack) - newDomain + k)].node;
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});
975 while (first != last)
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)
981 sons[k] = nodes_.make_terminal_node(*first++);
983 node_t*
const node = nodes_.make_internal_node(lastIndex, sons);
984 stack.push_back(stack_frame {
node, lastLevel});
988 assert(ssize(stack) == 1);
992template<
class Data,
class Degree,
class Domain>
993template<std::ranges::input_range R>
996 return this->
from_vector(begin(vector), end(vector));
999template<
class Data,
class Degree,
class Domain>
1001)
const -> std::vector<int32>
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));
1009template<
class Data,
class Degree,
class Domain>
1010template<std::output_iterator<teddy::
int32> O>
1024 bool wasLast =
false;
1027 *out++ = this->
evaluate(diagram, vars);
1028 bool overflow =
true;
1029 int32 level = nodes_.get_leaf_level();
1030 while (level > 0 && overflow)
1033 int32
const index = nodes_.get_index(level);
1034 ++vars[as_uindex(index)];
1035 overflow = vars[as_uindex(index)] == nodes_.get_domain(index);
1038 vars[as_uindex(index)] = 0;
1041 wasLast = overflow && 0 == level;
1043 }
while (not wasLast);
1046template<
class Data,
class Degree,
class Domain>
1051 fold_type
const foldType
1052) -> utils::second_t<Foo, std::vector<diagram_t>>
1054 auto const product = [
this] (
auto const& cube)
1058 for (int32 i = 0; i < cube.size(); ++i)
1060 if (cube.get(i) == 1)
1064 else if (cube.get(i) == 0)
1069 return this->left_fold<ops::AND>(
variables);
1072 auto const orFold = [
this, foldType] (
auto& diagrams)
1076 case fold_type::Left:
1077 return this->left_fold<ops::OR>(diagrams);
1079 case fold_type::Tree:
1080 return this->tree_fold<ops::OR>(diagrams);
1088 auto const& plaLines = file.
get_lines();
1093 std::vector<diagram_t> functionDiagrams;
1094 functionDiagrams.reserve(functionCount);
1095 for (int32 fi = 0; fi < functionCount; ++fi)
1098 std::vector<diagram_t> products;
1100 for (int32 li = 0; li < lineCount; ++li)
1104 if (plaLines[as_usize(li)].fVals_.get(fi) == 1)
1106 products.emplace_back(product(plaLines[li].cube_));
1111 if (products.empty())
1113 products.emplace_back(this->
constant(0));
1117 functionDiagrams.emplace_back(orFold(products));
1120 return functionDiagrams;
1123template<
class Data,
class Degree,
class Domain>
1124template<expression_node Node>
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();
1136template<
class Data,
class Degree,
class Domain>
1137template<
class ExprNode>
1139 std::vector<node_pack<2>>& cache,
1140 ExprNode
const& exprNode
1143 if (exprNode.is_constant())
1145 return nodes_.make_terminal_node(exprNode.get_value());
1148 if (exprNode.is_variable())
1150 return this->variable_impl(exprNode.get_index());
1153 assert(exprNode.is_operation());
1156 = this->from_expression_tree_impl(cache, exprNode.get_left());
1158 = this->from_expression_tree_impl(cache, exprNode.get_right());
1159 auto const operation = [&exprNode] (
auto const lhs,
auto const rhs)
1161 if (lhs == Nondetermined || rhs == Nondetermined)
1163 return Nondetermined;
1165 return static_cast<int32
>(exprNode.evaluate(lhs, rhs));
1168 node_t*
const newRoot = this->apply_n_impl(cache, operation, left, right);
1169 auto const cacheCapacity = cache.size();
1171 cache.resize(cacheCapacity);
1175template<
class Data,
class Degree,
class Domain>
1176auto diagram_manager<Data, Degree, Domain>::variable_impl(int32
const index)
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)
1183 sons[val] = nodes_.make_terminal_node(val);
1185 return nodes_.make_internal_node(index, sons);
1188template<
class Data,
class Degree,
class Domain>
1189template<teddy_bin_op Op>
1204 node_t*
const newRoot = this->apply_impl(
1206 lhs.unsafe_get_root(),
1207 rhs.unsafe_get_root()
1209 nodes_.run_deferred();
1213template<
class Data,
class Degree,
class Domain>
1221#ifdef LIBTEDDY_COLLECT_STATS
1222 ++stats::get_stats().applyStepCalls_;
1225 node_t*
const cached = nodes_.template cache_find<Op>(lhs, rhs);
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);
1235 if (opVal != Nondetermined)
1237 node_t*
const result = nodes_.make_terminal_node(opVal);
1238 nodes_.template cache_put<Op>(result, lhs, rhs);
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)
1250 sons[k] = this->apply_impl(
1252 lhsLevel == topLevel ? lhs->get_son(k) : lhs,
1253 rhsLevel == topLevel ? rhs->get_son(k) : rhs
1257 node_t*
const result = nodes_.make_internal_node(topIndex, sons);
1258 nodes_.template cache_put<Op>(result, lhs, rhs);
1262template<
class Data,
class Degree,
class Domain>
1263template<teddy_bin_op Op,
class... Diagram>
1277 std::vector<node_pack<
sizeof...(Diagram)>, node_t*> cache(100'000);
1278 node_t*
const newRoot
1280 nodes_.run_deferred();
1284template<
class Data,
class Degree,
class Domain>
1285template<
class Op,
class... Node>
1287 std::vector<node_pack<
sizeof...(Node)>>& cache,
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...))
1297 return cachePack.result_;
1300 int32
const opVal = operation(
1301 (nodes->is_terminal() ? nodes->get_value() : Nondetermined)...
1303 node_t* result =
nullptr;
1305 if (opVal != Nondetermined)
1307 result = nodes_.make_terminal_node(opVal);
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)
1317 sons[k] = this->apply_n_impl(
1320 (nodes_.get_level(nodes) == minLevel ? nodes->get_son(k) : nodes
1324 result = nodes_.make_internal_node(topIndex, sons);
1327 cachePack = {{nodes...}, result};
1331template<
class Data,
class Degree,
class Domain>
1332template<teddy_bin_op Op, std::ranges::input_range R>
1336 return this->left_fold<Op>(begin(diagrams), end(diagrams));
1339template<
class Data,
class Degree,
class Domain>
1340template<teddy_bin_op Op, std::input_iterator I, std::sentinel_for<I> S>
1344 static_assert(std::same_as<std::iter_value_t<I>,
diagram_t>);
1349 while (first != last)
1351 result = this->apply<Op>(result, *first);
1358template<
class Data,
class Degree,
class Domain>
1359template<teddy_bin_op Op, std::ranges::random_access_range R>
1362 return this->tree_fold<Op>(begin(diagrams), end(diagrams));
1365template<
class Data,
class Degree,
class Domain>
1366template<teddy_bin_op Op, std::random_access_iterator I, std::sentinel_for<I> S>
1370 static_assert(std::same_as<std::iter_value_t<I>,
diagram_t>);
1372 int64
const count = std::distance(first, last);
1373 int64 currentCount = count;
1374 auto const numOfSteps =
static_cast<int64
>(std::ceil(std::log2(count)));
1376 for (
auto step = 0; step < numOfSteps; ++step)
1378 auto const justMoveLast =
static_cast<bool>(currentCount & 1);
1379 currentCount = (currentCount / 2) + justMoveLast;
1380 int64
const pairCount = currentCount - justMoveLast;
1382 for (int64 i = 0; i < pairCount; ++i)
1385 = this->apply<Op>(*(first + 2 * i), *(first + 2 * i + 1));
1390 *(first + currentCount - 1)
1391 =
static_cast<diagram_t&&
>(*(first + 2 * (currentCount - 1)));
1398template<
class Data,
class Degree,
class Domain>
1399template<in_var_values Vars>
1407 while (not
node->is_terminal())
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)]);
1414 return node->get_value();
1417template<
class Data,
class Degree,
class Domain>
1425 assert(value < Domain::value);
1435 if constexpr (CanUseDataMember)
1438 return [] (node_t*
const node)
mutable -> T&
1439 {
return (
node->get_data()); };
1444 return [map = std::unordered_map<node_t*, T>()] (node_t*
const node
1457 nodes_.traverse_post(
1459 [
this, value, &data] (node_t*
const node)
mutable
1461 if (
node->is_terminal())
1463 data(
node) =
node->get_value() == value ? 1 : 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)
1472 node_t*
const son =
node->get_son(k);
1473 int32
const sonLevel = nodes_.get_level(son);
1475 = nodes_.domain_product(nodeLevel + 1, sonLevel);
1476 data(
node) += data(son) *
static_cast<T
>(diff);
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);
1487template<
class Data,
class Degree,
class Domain>
1488template<out_var_values Vars>
1492) -> std::optional<Vars>
1496 assert(value < Domain::value);
1506 return this->satisfy_one_impl(value, vars, root) ? std::make_optional(vars)
1510template<
class Data,
class Degree,
class Domain>
1518 if (
node->is_terminal())
1520 return node->get_value() == value;
1523 int32
const nodeIndex = node->get_index();
1524 int32
const nodeDomain = nodes_.get_domain(nodeIndex);
1525 for (
auto k = 0; k < nodeDomain; ++k)
1527 node_t*
const son = node->get_son(k);
1528 vars[as_uindex(nodeIndex)] = k;
1529 if (this->satisfy_one_impl(value, vars, son))
1538template<
class Data,
class Degree,
class Domain>
1539template<out_var_values Vars>
1543)
const -> std::vector<Vars>
1545 std::vector<Vars> result;
1546 this->satisfy_all_g<Vars>(value,
diagram, std::back_inserter(result));
1550template<
class Data,
class Degree,
class Domain>
1551template<out_var_values Vars, std::output_iterator<Vars> OutputIt>
1560 assert(value < Domain::value);
1564 if constexpr (utils::is_std_vector<Vars>)
1570 this->satisfy_all_impl(value, vars, out, root, 0);
1573template<
class Data,
class Degree,
class Domain>
1574template<
class Vars,
class OutputIt>
1575auto diagram_manager<Data, Degree, Domain>::satisfy_all_impl(
1583 if (node->is_terminal() && value != node->get_value())
1588 if (level == nodes_.get_leaf_level() && value == node->get_value())
1594 if (nodes_.get_level(node) > level)
1596 int32
const index = nodes_.get_index(level);
1597 int32
const domain = nodes_.get_domain(index);
1598 for (
auto k = 0; k < domain; ++k)
1600 vars[as_uindex(index)] = k;
1601 this->satisfy_all_impl(value, vars, out, node, level + 1);
1606 int32
const index = node->get_index();
1607 int32
const domain = nodes_.get_domain(index);
1608 for (
auto k = 0; k < domain; ++k)
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);
1617template<
class Data,
class Degree,
class Domain>
1620 int32
const varIndex,
1621 int32
const varValue
1625 if (root->is_terminal())
1630 if (root->get_index() == varIndex)
1632 return diagram_t(root->get_son(varValue));
1635 std::unordered_map<node_t*, node_t*> memo;
1637 =
diagram_t(this->get_cofactor_impl(memo, varIndex, varValue, root));
1638 nodes_.run_deferred();
1642template<
class Data,
class Degree,
class Domain>
1645 std::vector<var_cofactor>
const& vars
1649 if (root->is_terminal())
1654 auto const it = utils::find_if(vars.begin(), vars.end(),
1655 [root](var_cofactor var)
1657 return var.index_ == root->get_index();
1660 int32 toCofactor =
static_cast<int32
>(vars.size());
1661 if (it != vars.end())
1663 root = root->get_son(it->value_);
1667 std::unordered_map<node_t*, node_t*> memo;
1674 nodes_.run_deferred();
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,
1686 auto const memoIt = memo.find(node);
1687 if (memoIt != memo.end())
1689 return memoIt->second;
1692 if (node->is_terminal())
1697 int32
const nodeIndex = node->get_index();
1698 if (nodeIndex == varIndex)
1700 return node->get_son(varValue);
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)
1707 node_t*
const oldSon = node->get_son(k);
1708 sons[k] = this->get_cofactor_impl(memo, varIndex, varValue, oldSon);
1711 node_t*
const newNode = nodes_.make_internal_node(nodeIndex, sons);
1712 memo.emplace(node, newNode);
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,
1721 int32
const toCofactor
1724 if (toCofactor == 0)
1729 auto const memoIt = memo.find(node);
1730 if (memoIt != memo.end())
1732 return memoIt->second;
1735 if (node->is_terminal())
1740 int32
const nodeIndex = node->get_index();
1741 auto const it = utils::find_if(vars.begin(), vars.end(),
1742 [nodeIndex](var_cofactor var)
1744 return var.index_ == nodeIndex;
1747 node_t* newNode =
nullptr;
1748 if (it != vars.end())
1750 newNode = this->get_cofactor_impl(
1753 node->get_son(it->value_),
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)
1763 node_t*
const oldSon = node->get_son(k);
1764 sons[k] = this->get_cofactor_impl(memo, vars, oldSon, toCofactor);
1766 newNode = nodes_.make_internal_node(nodeIndex, sons);
1769 memo.emplace(node, newNode);
1773template<
class Data,
class Degree,
class Domain>
1774template<
int_to_
int F>
1780 std::unordered_map<node_t*, node_t*> memo;
1781 node_t*
const newRoot
1783 nodes_.run_deferred();
1787template<
class Data,
class Degree,
class Domain>
1790 std::unordered_map<node_t*, node_t*>& memo,
1795 auto const memoIt = memo.find(
node);
1796 if (memo.end() != memoIt)
1798 return memoIt->second;
1801 if (node->is_terminal())
1803 int32
const newVal =
static_cast<int32
>(transformer(node->get_value()));
1804 return nodes_.make_terminal_node(newVal);
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)
1812 node_t*
const son = node->get_son(k);
1813 sons[k] = this->transform_impl(memo, transformer, son);
1815 node_t*
const newNode = nodes_.make_internal_node(index, sons);
1816 memo.emplace(node, newNode);
1820template<
class Data,
class Degree,
class Domain>
1822requires(is_bdd<Degree>)
1824 -> utils::second_t<Foo, diagram_t>
1828 [] (int32
const value) {
return 1 - value; }
1832template<
class Data,
class Degree,
class Domain>
1835)
const -> std::vector<int32>
1837 std::vector<int32> indices;
1840 indices.shrink_to_fit();
1844template<
class Data,
class Degree,
class Domain>
1845template<std::output_iterator<
int32> O>
1851 std::vector<bool> memo(as_usize(this->
get_var_count()),
false);
1852 nodes_.traverse_pre(
1854 [&memo, out] (node_t*
const node)
mutable
1856 if (node->is_internal())
1858 int32 const index = node->get_index();
1859 if (not memo[as_uindex(index)])
1863 memo[as_uindex(index)] = true;
1869template<
class Data,
class Degree,
class Domain>
1873 return this->transform(
diagram, [] (int32
const val) {
return val; });
1876template<
class Data,
class Degree,
class Domain>
1879 return nodes_.get_node_count();
1882template<
class Data,
class Degree,
class Domain>
1890template<
class Data,
class Degree,
class Domain>
1894 nodes_.to_dot_graph(out);
1897template<
class Data,
class Degree,
class Domain>
1906template<
class Data,
class Degree,
class Domain>
1909 return nodes_.get_var_count();
1912template<
class Data,
class Degree,
class Domain>
1914 -> std::vector<int32> const&
1916 return nodes_.get_order();
1919template<
class Data,
class Degree,
class Domain>
1921 -> std::vector<int32>
1923 return nodes_.get_domains();
1926template<
class Data,
class Degree,
class Domain>
1930 nodes_.set_cache_ratio(ratio);
1933template<
class Data,
class Degree,
class Domain>
1936 nodes_.set_gc_ratio(ratio);
1939template<
class Data,
class Degree,
class Domain>
1941 bool const doReorder
1944 nodes_.set_auto_reorder(doReorder);
1947template<
class Data,
class Degree,
class Domain>
1953template<
class Data,
class Degree,
class Domain>
1956 nodes_.sift_variables();
1959template<
class Data,
class Degree,
class Domain>
1962 nodes_.cache_clear();
1967inline auto default_or_fwd (int32
const varCount, std::vector<int32>& indices)
1969 if (indices.empty())
1971 std::vector<int32> defaultIndices;
1972 defaultIndices.reserve(as_usize(varCount));
1973 for (int32 index = 0; index < varCount; ++index)
1975 defaultIndices.push_back(index);
1977 return defaultIndices;
1981 return std::vector<int32>(
static_cast<std::vector<int32>&&
>(indices));
1986template<
class Data,
class Degree,
class Domain>
1988 int32
const varCount,
1989 int64
const nodePoolSize,
1990 int64
const extraNodePoolSize,
1991 std::vector<int32> order
1999 detail::default_or_fwd(varCount, order)
2004template<
class Data,
class Degree,
class Domain>
2006 int32
const varCount,
2007 int64
const nodePoolSize,
2008 int64
const extraNodePoolSize,
2010 std::vector<int32> order
2018 detail::default_or_fwd(varCount, order),
2019 static_cast<domains::mixed&&>(domain)
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
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 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
Provides member typedef based on the value of B Implementation of std::conditional.
Definition tools.hpp:319
Definition diagram_manager.hpp:79