mirror of
https://github.com/Z3Prover/z3
synced 2025-04-06 17:44:08 +00:00
Add unification based projection function
Heuristic for EMBPR is to unify terms that occur in uninterpreted contents. Walk partitions that are pure f(t) and an impure f(s) and attempt to unify t with s ensuring that merges from s preserves satisfiability.
This commit is contained in:
parent
b27a2aa7fc
commit
50f9fddfd2
|
@ -7,15 +7,12 @@ Copyright (c) 2025 Microsoft Corporation
|
|||
#include "ast/ast_util.h"
|
||||
#include "ast/for_each_expr.h"
|
||||
#include "qe/mbp/mbp_euf.h"
|
||||
#include "qe/mbp/mbp_term_graph.h"
|
||||
|
||||
namespace mbp {
|
||||
euf_project_plugin::euf_project_plugin(ast_manager& m): project_plugin(m) {
|
||||
|
||||
}
|
||||
|
||||
euf_project_plugin::~euf_project_plugin() {
|
||||
|
||||
}
|
||||
|
||||
bool euf_project_plugin::project1(model& model, app* var, app_ref_vector& vars, expr_ref_vector& lits) {
|
||||
|
@ -26,9 +23,6 @@ namespace mbp {
|
|||
return basic_family_id;
|
||||
}
|
||||
|
||||
bool euf_project_plugin::operator()(model& model, app_ref_vector& vars, expr_ref_vector& lits) {
|
||||
return false;
|
||||
}
|
||||
|
||||
void euf_project_plugin::solve_eqs(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs) {
|
||||
flatten_and(lits);
|
||||
|
@ -52,9 +46,7 @@ namespace mbp {
|
|||
}
|
||||
}
|
||||
|
||||
bool euf_project_plugin::project(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs) {
|
||||
if (vars.empty())
|
||||
return false;
|
||||
bool euf_project_plugin::solve_eqs_saturate(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs) {
|
||||
unsigned sz = defs.size();
|
||||
while (true) {
|
||||
unsigned sz1 = defs.size();
|
||||
|
@ -62,14 +54,51 @@ namespace mbp {
|
|||
if (sz1 == defs.size())
|
||||
break;
|
||||
}
|
||||
return sz < defs.size();
|
||||
}
|
||||
|
||||
if (sz < defs.size())
|
||||
|
||||
bool euf_project_plugin::operator()(model& model, app_ref_vector& vars, expr_ref_vector& lits) {
|
||||
if (vars.empty())
|
||||
return false;
|
||||
// check if there is a variable of uninterp sort
|
||||
if (all_of(vars, [&](expr* v) { return !m.is_uninterp(v->get_sort()); }))
|
||||
return false;
|
||||
|
||||
vector<def> defs;
|
||||
if (solve_eqs_saturate(model, vars, lits, defs))
|
||||
return true;
|
||||
term_graph tg(m);
|
||||
tg.add_lits(lits);
|
||||
for (auto v : vars)
|
||||
if (m.is_uninterp(v->get_sort()))
|
||||
tg.add_var(v);
|
||||
|
||||
auto result = tg.project(model);
|
||||
lits.reset();
|
||||
lits.append(result);
|
||||
|
||||
unsigned j = 0;
|
||||
for (app* v : vars)
|
||||
if (!m.is_uninterp(v->get_sort()))
|
||||
vars[j++] = v;
|
||||
vars.shrink(j);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool euf_project_plugin::project(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs) {
|
||||
if (vars.empty())
|
||||
return false;
|
||||
|
||||
// check if there is a variable of uninterp sort
|
||||
if (all_of(vars, [&](expr* v) { return !m.is_uninterp(v->get_sort()); }))
|
||||
return false;
|
||||
|
||||
if (solve_eqs_saturate(model, vars, lits, defs))
|
||||
return true;
|
||||
|
||||
|
||||
term_graph tg(m);
|
||||
tg.add_lits(lits);
|
||||
for (auto v : vars)
|
||||
|
@ -78,19 +107,36 @@ namespace mbp {
|
|||
|
||||
expr_ref_vector terms(m);
|
||||
for (expr* f : subterms::all(lits))
|
||||
if (!m.is_bool(f))
|
||||
terms.push_back(f);
|
||||
|
||||
|
||||
// try to project on representatives:
|
||||
tg.add_model_based_terms(model, terms);
|
||||
|
||||
// save rep_of. It gets reset by tg.get_partition.
|
||||
struct scoped_reset {
|
||||
euf_project_plugin& p;
|
||||
scoped_reset(euf_project_plugin& p): p(p) {}
|
||||
~scoped_reset() { p.m_reps.reset(); p.m_parents.reset(); }
|
||||
};
|
||||
scoped_reset _reset(*this);
|
||||
expr_ref_vector pinned(m);
|
||||
for (auto t : terms) {
|
||||
m_reps.insert(t, tg.rep_of(t));
|
||||
pinned.push_back(tg.rep_of(t));
|
||||
if (is_app(t)) {
|
||||
for (expr* arg : *to_app(t))
|
||||
m_parents.insert_if_not_there(arg, {}).push_back(t);
|
||||
}
|
||||
}
|
||||
for (auto [key, value] : m_reps)
|
||||
verbose_stream() << mk_pp(key, m) << " -> " << mk_pp(value, m) << "\n";
|
||||
unsigned j = 0;
|
||||
for (unsigned i = 0; i < vars.size(); ++i) {
|
||||
app* v = vars.get(i);
|
||||
bool solved = false;
|
||||
for (app* v : vars) {
|
||||
if (m.is_uninterp(v->get_sort())) {
|
||||
expr* r = tg.rep_of(v);
|
||||
expr* r = m_reps[v];
|
||||
if (r)
|
||||
defs.push_back({ expr_ref(v, m), expr_ref(r, m) });
|
||||
defs.push_back({ expr_ref(v, m), expr_ref(r, m) }), solved = true;
|
||||
else
|
||||
vars[j++] = v;
|
||||
}
|
||||
|
@ -99,18 +145,131 @@ namespace mbp {
|
|||
}
|
||||
vars.shrink(j);
|
||||
|
||||
if (sz < defs.size())
|
||||
if (solved)
|
||||
return true;
|
||||
|
||||
|
||||
// walk equivalence classes.
|
||||
// try to unify with pure classes.
|
||||
// some parent of y contains f(y) and class of f contains f(t) with repr(t) != nullptr
|
||||
// merge y = t, provided parent f(y) is not distinct from f(t)
|
||||
// unification can break distinctness, and should require a full solver call.
|
||||
// a conservative rule is to walk only the equivalence class of f(y) to check if it
|
||||
// contains f(t),
|
||||
// or recursively if there is a term g(f(t)) that is equal to g(f(y))
|
||||
// thus, take equivalence classes that contain a repr and unify
|
||||
//
|
||||
// this can still break satisfiability if f(y) is required distinct from f(t) even
|
||||
// if g(f(t)) = g(f(y))
|
||||
// so as a simplification unification is only allowed if g(f(y)) is the only parent of f(y)
|
||||
|
||||
|
||||
// try to build a fresh term using available signature
|
||||
|
||||
auto Ps = tg.get_partition(model);
|
||||
for (auto const& p : Ps) {
|
||||
expr* r = m_reps[p[0]];
|
||||
if (!r)
|
||||
continue;
|
||||
for (auto e : p) {
|
||||
if (!is_app(e))
|
||||
continue;
|
||||
app* a = to_app(e);
|
||||
func_decl* f = a->get_decl();
|
||||
if (!is_uninterp(f))
|
||||
continue;
|
||||
if (all_of(*a, [&](expr* arg) { return !!m_reps[arg]; }))
|
||||
continue;
|
||||
if (try_unify(tg, a, p, vars, defs))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool euf_project_plugin::try_unify(term_graph& tg, app* a, expr_ref_vector const& partition, app_ref_vector& vars, vector<def>& defs) {
|
||||
|
||||
auto same_decl = [&](expr* x, expr* y) {
|
||||
if (!is_app(x) || !is_app(y))
|
||||
return false;
|
||||
app* a = to_app(x);
|
||||
app* b = to_app(y);
|
||||
if (a->get_decl() != b->get_decl())
|
||||
return false;
|
||||
return a->get_num_args() == b->get_num_args();
|
||||
};
|
||||
|
||||
auto is_var = [&](expr* x) {
|
||||
return is_uninterp_const(x) && m.is_uninterp(x->get_sort()) && vars.contains(to_app(x));
|
||||
};
|
||||
|
||||
for (auto e : partition) {
|
||||
if (a == e)
|
||||
continue;
|
||||
verbose_stream() << "Unify " << mk_pp(a, m) << " with " << mk_pp(e, m) << "\n";
|
||||
if (!same_decl(a, e))
|
||||
continue;
|
||||
app* b = to_app(e);
|
||||
if (!all_of(*b, [&](expr* arg) { return !!m_reps[arg]; }))
|
||||
continue;
|
||||
// same function symbol. All kids of b are defined, some kid of a is not defined.
|
||||
svector<std::pair<expr*, expr*>> todo;
|
||||
obj_map<expr, expr*> soln;
|
||||
for (unsigned i = 0; i < b->get_num_args(); ++i) {
|
||||
expr* x = a->get_arg(i);
|
||||
expr* y = b->get_arg(i);
|
||||
todo.push_back({x, y});
|
||||
}
|
||||
while (!todo.empty()) {
|
||||
auto [x, y] = todo.back();
|
||||
todo.pop_back();
|
||||
auto rx = m_reps[x];
|
||||
auto ry = m_reps[y];
|
||||
SASSERT(ry);
|
||||
if (rx == ry)
|
||||
continue;
|
||||
if (rx)
|
||||
break; // fail
|
||||
if (m_parents[x].size() > 1)
|
||||
// a crude safey hatch to preserve models
|
||||
// we could require that every parent of x
|
||||
// has a pure representative which comes from a correspondnig
|
||||
// parent of y
|
||||
break;
|
||||
expr* s = nullptr;
|
||||
if (soln.find(x, s)) {
|
||||
if (s != ry)
|
||||
break;
|
||||
continue;
|
||||
}
|
||||
if (is_var(x)) {
|
||||
soln.insert(x, ry);
|
||||
continue;
|
||||
}
|
||||
if (!same_decl(x, y))
|
||||
break;
|
||||
app* ax = to_app(x);
|
||||
app* ay = to_app(y);
|
||||
for (unsigned i = 0; i < ax->get_num_args(); ++i)
|
||||
todo.push_back({ ax->get_arg(i), ay->get_arg(i) });
|
||||
}
|
||||
|
||||
TRACE("qe", tout << "unification attempt\n";
|
||||
for (auto [a, b] : todo)
|
||||
tout << mk_pp(a, m) << " == " << mk_pp(b, m) << "\n";
|
||||
for (auto [key, value] : soln)
|
||||
tout << mk_pp(key, m) << " -> " << mk_pp(value, m) << "\n";
|
||||
);
|
||||
|
||||
|
||||
if (todo.empty() && !soln.empty()) {
|
||||
for (auto [key, value] : soln) {
|
||||
vars.erase(to_app(key));
|
||||
defs.push_back( { expr_ref(key, m), expr_ref(value, m) });
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
|
|
@ -9,11 +9,16 @@ Copyright (c) 2025 Microsoft Corporation
|
|||
|
||||
#include "model/model.h"
|
||||
#include "qe/mbp/mbp_plugin.h"
|
||||
#include "qe/mbp/mbp_term_graph.h"
|
||||
|
||||
namespace mbp {
|
||||
|
||||
class euf_project_plugin : public project_plugin {
|
||||
obj_map<expr, expr*> m_reps;
|
||||
obj_map<expr, ptr_vector<expr>> m_parents;
|
||||
void solve_eqs(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs);
|
||||
bool solve_eqs_saturate(model& model, app_ref_vector& vars, expr_ref_vector& lits, vector<def>& defs);
|
||||
bool try_unify(term_graph& g, app* a, expr_ref_vector const& partitions, app_ref_vector& vars, vector<def>& defs);
|
||||
public:
|
||||
euf_project_plugin(ast_manager& m);
|
||||
~euf_project_plugin() override;
|
||||
|
|
|
@ -577,8 +577,9 @@ term *term_graph::internalize_term(expr *t) {
|
|||
}
|
||||
merge_flush();
|
||||
SASSERT(res);
|
||||
if (m.is_not(t) && is_app(to_app(t)->get_arg(0)) && is_partial_eq(to_app(to_app(t)->get_arg(0)))) {
|
||||
term* p = get_term(to_app(t)->get_arg(0));
|
||||
expr* arg = nullptr;
|
||||
if (m.is_not(t, arg) && is_app(arg) && is_partial_eq(to_app(arg))) {
|
||||
term* p = get_term(arg);
|
||||
SASSERT(p);
|
||||
p->set_npeq_child();
|
||||
}
|
||||
|
@ -879,8 +880,7 @@ void term_graph::pick_repr_class(term *t) {
|
|||
void term_graph::pick_repr() {
|
||||
// invalidates cache
|
||||
m_term2app.reset();
|
||||
DEBUG_CODE(for (term *t
|
||||
: m_terms)
|
||||
DEBUG_CODE(for (term *t : m_terms)
|
||||
SASSERT(t->deg() == 0 || !t->all_children_ground() ||
|
||||
t->is_cgr()););
|
||||
for (term *t : m_terms) t->reset_repr();
|
||||
|
@ -1090,25 +1090,27 @@ class term_graph::projector {
|
|||
expr_ref_vector m_pinned; // tracks expr in the maps
|
||||
|
||||
expr *mk_pure(term const &t) {
|
||||
TRACE("qe", t.display(tout););
|
||||
TRACE("qe", t.display(tout));
|
||||
expr *e = nullptr;
|
||||
if (find_term2app(t, e)) return e;
|
||||
if (find_term2app(t, e))
|
||||
return e;
|
||||
e = t.get_expr();
|
||||
if (!is_app(e)) return nullptr;
|
||||
if (!is_app(e))
|
||||
return nullptr;
|
||||
app *a = ::to_app(e);
|
||||
expr_ref_buffer kids(m);
|
||||
for (term *ch : term::children(t)) {
|
||||
// prefer a node that resembles current child,
|
||||
// otherwise, pick a root representative, if present.
|
||||
if (find_term2app(*ch, e)) { kids.push_back(e); }
|
||||
else if (m_root2rep.find(ch->get_root().get_id(), e)) {
|
||||
if (find_term2app(*ch, e))
|
||||
kids.push_back(e);
|
||||
}
|
||||
else { return nullptr; }
|
||||
else if (m_root2rep.find(ch->get_root().get_id(), e))
|
||||
kids.push_back(e);
|
||||
else
|
||||
return nullptr;
|
||||
TRACE("qe_verbose", tout << *ch << " -> " << mk_pp(e, m) << "\n";);
|
||||
}
|
||||
expr_ref pure =
|
||||
m_rewriter.mk_app(a->get_decl(), kids.size(), kids.data());
|
||||
expr_ref pure = m_rewriter.mk_app(a->get_decl(), kids.size(), kids.data());
|
||||
m_pinned.push_back(pure);
|
||||
add_term2app(t, pure);
|
||||
return pure;
|
||||
|
@ -1164,7 +1166,7 @@ class term_graph::projector {
|
|||
m_tg.reset_marks();
|
||||
}
|
||||
|
||||
bool find_app(term &t, expr *&res) {
|
||||
bool find_app(term const &t, expr *&res) {
|
||||
return find_term2app(t, res) ||
|
||||
m_root2rep.find(t.get_root().get_id(), res);
|
||||
}
|
||||
|
@ -1177,9 +1179,9 @@ class term_graph::projector {
|
|||
|
||||
void mk_lits(expr_ref_vector &res) {
|
||||
expr *e = nullptr;
|
||||
for (auto *lit : m_tg.m_lits) {
|
||||
if (!m.is_eq(lit) && find_app(lit, e)) res.push_back(e);
|
||||
}
|
||||
for (auto *lit : m_tg.m_lits)
|
||||
if (!m.is_eq(lit) && find_app(lit, e))
|
||||
res.push_back(e);
|
||||
TRACE("qe", tout << "literals: " << res << "\n";);
|
||||
}
|
||||
|
||||
|
@ -1219,12 +1221,12 @@ class term_graph::projector {
|
|||
}
|
||||
|
||||
void remove_duplicates(expr_ref_vector &v) {
|
||||
obj_hashtable<expr> seen;
|
||||
expr_fast_mark1 seen;
|
||||
unsigned j = 0;
|
||||
for (expr *e : v) {
|
||||
if (!seen.contains(e)) {
|
||||
if (!seen.is_marked(e)) {
|
||||
v[j++] = e;
|
||||
seen.insert(e);
|
||||
seen.mark(e);
|
||||
}
|
||||
}
|
||||
v.shrink(j);
|
||||
|
@ -1439,9 +1441,9 @@ class term_graph::projector {
|
|||
return m_term2app.find(t.get_id(), r);
|
||||
}
|
||||
|
||||
expr *find_term2app(term const &t) {
|
||||
expr* rep_of(term const &t) {
|
||||
expr *r = nullptr;
|
||||
find_term2app(t, r);
|
||||
find_app(t, r);
|
||||
return r;
|
||||
}
|
||||
|
||||
|
@ -1713,7 +1715,7 @@ expr *term_graph::rep_of(expr *e) {
|
|||
SASSERT(m_projector);
|
||||
term *t = get_term(e);
|
||||
SASSERT(t && "only get representatives");
|
||||
return m_projector->find_term2app(*t);
|
||||
return m_projector->rep_of(*t);
|
||||
}
|
||||
|
||||
expr_ref_vector term_graph::dcert(model &mdl, expr_ref_vector const &lits) {
|
||||
|
@ -1834,15 +1836,14 @@ void term_graph::cground_percolate_up(term *t) {
|
|||
}
|
||||
|
||||
void term_graph::cground_percolate_up(ptr_vector<term> &todo) {
|
||||
term *t;
|
||||
|
||||
while (!todo.empty()) {
|
||||
t = todo.back();
|
||||
term* t = todo.back();
|
||||
todo.pop_back();
|
||||
t->set_cgr(true);
|
||||
t->set_class_gr(true);
|
||||
for (auto p : term::parents(t->get_root()))
|
||||
if (!p->is_cgr() && p->all_children_ground()) todo.push_back(p);
|
||||
if (!p->is_cgr() && p->all_children_ground())
|
||||
todo.push_back(p);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1852,12 +1853,11 @@ void term_graph::compute_cground() {
|
|||
t->set_class_gr(false);
|
||||
}
|
||||
ptr_vector<term> todo;
|
||||
for (auto t : m_terms) {
|
||||
if (t->is_gr()) { todo.push_back(t); }
|
||||
}
|
||||
for (auto t : m_terms)
|
||||
if (t->is_gr())
|
||||
todo.push_back(t);
|
||||
cground_percolate_up(todo);
|
||||
DEBUG_CODE(for (auto t
|
||||
: m_terms) {
|
||||
DEBUG_CODE(for (auto t : m_terms) {
|
||||
bool isclsg = true;
|
||||
for (auto c : term::children(t)) isclsg &= c->is_class_gr();
|
||||
SASSERT(t->deg() == 0 || !isclsg || t->is_cgr());
|
||||
|
|
Loading…
Reference in a new issue