Skip to content

Commit 2971250

Browse files
add option to rewrite ite value trees
1 parent e92ccdd commit 2971250

File tree

3 files changed

+80
-0
lines changed

3 files changed

+80
-0
lines changed

src/ast/rewriter/bool_rewriter.cpp

+76
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ void bool_rewriter::updt_params(params_ref const & _p) {
3434
m_blast_distinct = p.blast_distinct();
3535
m_blast_distinct_threshold = p.blast_distinct_threshold();
3636
m_ite_extra_rules = p.ite_extra_rules();
37+
m_elim_ite_value_tree = p.elim_ite_value_tree();
3738
}
3839

3940
void bool_rewriter::get_param_descrs(param_descrs & r) {
@@ -680,9 +681,84 @@ br_status bool_rewriter::try_ite_value(app * ite, app * val, expr_ref & result)
680681
return BR_REWRITE2;
681682
}
682683

684+
if (m_elim_ite_value_tree) {
685+
result = simplify_eq_ite(val, ite);
686+
if (result)
687+
return BR_REWRITE_FULL;
688+
}
689+
683690
return BR_FAILED;
684691
}
685692

693+
expr_ref bool_rewriter::simplify_eq_ite(expr* value, expr* ite) {
694+
SASSERT(m().is_value(value));
695+
SASSERT(m().is_ite(ite));
696+
expr* c = nullptr, * t = nullptr, * e = nullptr;
697+
ptr_buffer<expr> todo;
698+
ptr_vector<expr> values;
699+
expr_ref_vector pinned(m());
700+
expr_ref r(m());
701+
todo.push_back(ite);
702+
while (!todo.empty()) {
703+
expr* arg = todo.back();
704+
if (m().is_value(arg)) {
705+
todo.pop_back();
706+
if (m().are_equal(arg, value)) {
707+
values.setx(arg->get_id(), m().mk_true(), nullptr);
708+
continue;
709+
}
710+
if (m().are_distinct(arg, value)) {
711+
values.setx(arg->get_id(), m().mk_false(), nullptr);
712+
continue;
713+
}
714+
return expr_ref(nullptr, m());
715+
}
716+
if (m().is_ite(arg, c, t, e)) {
717+
unsigned sz = todo.size();
718+
if (!values.get(t->get_id(), nullptr))
719+
todo.push_back(t);
720+
721+
if (!values.get(e->get_id(), nullptr))
722+
todo.push_back(e);
723+
724+
if (sz < todo.size())
725+
continue;
726+
todo.pop_back();
727+
if (m().is_true(values[t->get_id()])) {
728+
r = m().mk_or(c, values[e->get_id()]);
729+
values.setx(arg->get_id(), r, nullptr);
730+
pinned.push_back(r);
731+
continue;
732+
}
733+
if (m().is_false(values[t->get_id()])) {
734+
r = m().mk_and(m().mk_not(c), values[e->get_id()]);
735+
values.setx(arg->get_id(), r, nullptr);
736+
pinned.push_back(r);
737+
continue;
738+
}
739+
if (m().is_false(values[e->get_id()])) {
740+
r = m().mk_and(c, values[t->get_id()]);
741+
values.setx(arg->get_id(), r, nullptr);
742+
pinned.push_back(r);
743+
continue;
744+
}
745+
if (m().is_true(values[e->get_id()])) {
746+
r = m().mk_or(m().mk_not(c), values[t->get_id()]);
747+
values.setx(arg->get_id(), r, nullptr);
748+
pinned.push_back(r);
749+
continue;
750+
}
751+
r = m().mk_ite(c, values[t->get_id()], values[e->get_id()]);
752+
values.setx(arg->get_id(), r, nullptr);
753+
pinned.push_back(r);
754+
continue;
755+
}
756+
IF_VERBOSE(10, verbose_stream() << "bail " << mk_bounded_pp(arg, m()) << "\n");
757+
return expr_ref(nullptr, m());
758+
}
759+
return expr_ref(values[ite->get_id()], m());
760+
}
761+
686762

687763
app* bool_rewriter::mk_eq_plain(expr* lhs, expr* rhs) {
688764
if (m().are_equal(lhs, rhs))

src/ast/rewriter/bool_rewriter.h

+3
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,7 @@ class bool_rewriter {
6161
unsigned m_local_ctx_limit;
6262
unsigned m_local_ctx_cost;
6363
bool m_elim_ite;
64+
bool m_elim_ite_value_tree;
6465
ptr_vector<expr> m_todo1, m_todo2;
6566
unsigned_vector m_counts1, m_counts2;
6667

@@ -83,6 +84,8 @@ class bool_rewriter {
8384

8485
void push_new_arg(expr* arg, expr_ref_vector& new_args, expr_fast_mark1& neg_lits, expr_fast_mark2& pos_lits);
8586

87+
expr_ref simplify_eq_ite(expr* value, expr* ite);
88+
8689
public:
8790
bool_rewriter(ast_manager & m, params_ref const & p = params_ref()):m_manager(m), m_local_ctx_cost(0) {
8891
updt_params(p);

src/params/bool_rewriter_params.pyg

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ def_module_params(module_name='rewriter',
77
("sort_disjunctions", BOOL, True, "sort subterms in disjunctions"),
88
("elim_and", BOOL, False, "conjunctions are rewritten using negation and disjunctions"),
99
('elim_ite', BOOL, True, "eliminate ite in favor of and/or"),
10+
('elim_ite_value_tree', BOOL, False, "eliminate equations 'v = ite(...)' where v is a value and each leaf in the ite tree is a value"),
1011
("local_ctx", BOOL, False, "perform local (i.e., cheap) context simplifications"),
1112
("local_ctx_limit", UINT, UINT_MAX, "limit for applying local context simplifier"),
1213
("blast_distinct", BOOL, False, "expand a distinct predicate into a quadratic number of disequalities"),

0 commit comments

Comments
 (0)