mirror of
https://github.com/Z3Prover/z3
synced 2025-04-05 17:14:07 +00:00
* qe_lite: cleanup and comment no change to code * mbp_arrays: refactor out partial equality (peq) Partial array equality, PEQ, is used as an intermediate expression during MBP for arrays. We need to factor it out so that it can be shared between MBP-QEL and existing MBP. Partial array equality (peq) is used in MBP for arrays. Factoring this out to be used by multiple MBP implementations. * rewriter: new rewrite rules These rules are specializes for terms that are created in QEL. QEL commit is comming later * datatype_rw: new rewrite rule for ADTs The rule handles this special case: (cons (head x) (tail x)) --> x * array_rewriter rules for rewriting PEQs Special rules to simplify PEQs * th_rewriter: wire PEQ simplifications * spacer_iuc: avoid terms with default in IUC Spacer prfers to not have a term representing default value of an array. This guides IUC from picking such terms in interpolation * mbp_term_graph: replace root with repr * mbp_term_graph: formatting * mbp_term_graph: class_props, getters, setters Class properties allow to keep information for an equivalence class. Getters and setters for terms allow accessing information * mbp_term_graph: auxiliary methods for qel QEL commit is comming later in the history * mbp_term_graph: bug fix * mbp_term_graph: pick, refine repr, compute cgrnd * mbp_term_graph: internalize deq * mbp_term_graph: constructor * mbp_term_graph: optionally internalize equalities Reperesent equalities explicitly by nodes in the term_graph * qel * formatting * comments on term_lt * get terms and other api for mbp_qel * plugins for mbp_qel * mbp_qel_util: utilities for mbp_qel * qe_mbp: QEL-based mbp * qel: expose QEL API * spacer: replace qe_lite in qe_project_spacer by qel This changes the default projection engine that spacer uses. * cmd_context: debug commands for qel and mbp_qel New commands are mbp-qel -- MBP with term graphs qel -- QEL with term graphs qe-lite -- older qelite * qe_mbp: model-based rewriters for arrays * qe_mbp: QEL-based projection functions * qsat: wire in QEL-based mbp * qsat: debug code * qsat: maybe a bug fix Changed the code to follow the paper by adding all predicates above a given level, not just predicates of immediately preceding level. * chore: use new api to create solver in qsat * mbp_term_graph use all_of idiom * feat: solver for integer multiplication * array_peq: formatting, no change to code * mbp_qel_util: block comment + format * mbt_term_graph: clang-format * bug fix. Move dt rewrite to qe_mbp * array_peq: add header * run clang format on mbp plugins * clang format on mul solver * format do-while * format * format do-while * update release notes --------- Co-authored-by: hgvk94 <hgvk94@gmail.com> Co-authored-by: Isabel Garcia <igarciac@uwaterloo.ca>
262 lines
7.7 KiB
C++
262 lines
7.7 KiB
C++
/**++
|
|
Copyright (c) Arie Gurfinkel
|
|
|
|
Module Name:
|
|
|
|
mbp_term_graph.h
|
|
|
|
Abstract:
|
|
|
|
Equivalence graph of terms
|
|
|
|
Author:
|
|
|
|
Arie Gurfinkel
|
|
Hari Govind V K (hgvk94)
|
|
Isabel Garcia (igcontreras)
|
|
|
|
Revision History:
|
|
|
|
Added implementation of qe_lite using term graph
|
|
|
|
Notes:
|
|
|
|
--*/
|
|
#pragma once
|
|
|
|
#include "ast/ast.h"
|
|
#include "ast/expr_functors.h"
|
|
#include "ast/is_variable_test.h"
|
|
#include "model/model.h"
|
|
#include "qe/mbp/mbp_solve_plugin.h"
|
|
#include "util/plugin_manager.h"
|
|
|
|
namespace mbp {
|
|
namespace is_ground_ns {
|
|
struct proc;
|
|
struct found;
|
|
} // namespace is_ground_ns
|
|
class term;
|
|
|
|
class term_graph {
|
|
class projector;
|
|
friend struct is_ground_ns::proc;
|
|
friend struct is_ground_ns::found;
|
|
|
|
class is_variable_proc : public ::is_variable_proc {
|
|
bool m_exclude;
|
|
obj_hashtable<func_decl> m_decls, m_solved;
|
|
|
|
public:
|
|
bool operator()(const expr *e) const override;
|
|
bool operator()(const term &t) const;
|
|
|
|
void set_decls(const func_decl_ref_vector &decls, bool exclude);
|
|
void set_decls(const app_ref_vector &vars, bool exclude);
|
|
void add_decls(const app_ref_vector &vars);
|
|
void add_decl(app *var);
|
|
void mark_solved(const expr *e);
|
|
void reset_solved() { m_solved.reset(); }
|
|
void reset() {
|
|
m_decls.reset();
|
|
m_solved.reset();
|
|
m_exclude = true;
|
|
}
|
|
bool contains(func_decl *f) { return m_decls.contains(f) == m_exclude; }
|
|
};
|
|
|
|
class is_non_core : public i_expr_pred {
|
|
std::function<bool(expr *)> *m_non_core;
|
|
|
|
public:
|
|
is_non_core(std::function<bool(expr *)> *nc) : m_non_core(nc) {}
|
|
bool operator()(expr *n) override {
|
|
if (m_non_core == nullptr) return false;
|
|
return (*m_non_core)(n);
|
|
}
|
|
};
|
|
|
|
struct term_hash {
|
|
unsigned operator()(term const *t) const;
|
|
};
|
|
struct term_eq {
|
|
bool operator()(term const *a, term const *b) const;
|
|
};
|
|
ast_manager &m;
|
|
ptr_vector<term> m_terms;
|
|
expr_ref_vector m_lits; // NSB: expr_ref_vector?
|
|
u_map<term *> m_app2term;
|
|
ast_ref_vector m_pinned;
|
|
projector *m_projector;
|
|
bool m_explicit_eq;
|
|
bool m_repick_repr;
|
|
u_map<expr *>
|
|
m_term2app; // any representative change invalidates this cache
|
|
plugin_manager<solve_plugin> m_plugins;
|
|
ptr_hashtable<term, term_hash, term_eq> m_cg_table;
|
|
vector<std::pair<term *, term *>> m_merge;
|
|
|
|
term_graph::is_variable_proc m_is_var;
|
|
|
|
void merge(term &t1, term &t2);
|
|
void merge_flush();
|
|
|
|
term *mk_term(expr *t);
|
|
term *get_term(expr *t);
|
|
term *get_term(func_decl *f);
|
|
|
|
term *internalize_term(expr *t);
|
|
void internalize_eq(expr *a1, expr *a2);
|
|
void internalize_lit(expr *lit);
|
|
void internalize_distinct(expr *d);
|
|
void internalize_deq(expr *a1, expr *a2);
|
|
|
|
bool is_internalized(expr *a);
|
|
bool is_ground(expr *e);
|
|
|
|
bool term_lt(term const &t1, term const &t2);
|
|
void pick_repr_percolate_up(ptr_vector<term> &todo);
|
|
void pick_repr_class(term *t);
|
|
void pick_repr();
|
|
|
|
void reset_marks();
|
|
void reset_marks2();
|
|
bool marks_are_clear();
|
|
|
|
expr *mk_app_core(expr *a);
|
|
expr_ref mk_app(term &t);
|
|
expr *mk_pure(term &t);
|
|
expr_ref mk_app(expr *a);
|
|
void mk_equalities(term &t, expr_ref_vector &out);
|
|
void mk_all_equalities(term &t, expr_ref_vector &out);
|
|
void mk_qe_lite_equalities(term &t, expr_ref_vector &out,
|
|
check_pred ¬_in_core);
|
|
void display(std::ostream &out);
|
|
|
|
bool is_pure_def(expr *atom, expr *&v);
|
|
void cground_percolate_up(ptr_vector<term> &);
|
|
void cground_percolate_up(term *t);
|
|
void compute_cground();
|
|
|
|
public:
|
|
term_graph(ast_manager &m);
|
|
~term_graph();
|
|
|
|
const expr_ref_vector &get_lits() const { return m_lits; }
|
|
void get_terms(expr_ref_vector &res, bool exclude_cground = true);
|
|
bool is_cgr(expr *e);
|
|
unsigned size() { return m_terms.size(); }
|
|
|
|
void set_vars(func_decl_ref_vector const &decls, bool exclude = true);
|
|
void set_vars(app_ref_vector const &vars, bool exclude = true);
|
|
void add_vars(app_ref_vector const &vars);
|
|
void add_var(app *var);
|
|
|
|
ast_manager &get_ast_manager() const { return m; }
|
|
|
|
void add_lit(expr *lit);
|
|
void add_lits(expr_ref_vector const &lits) {
|
|
for (expr *e : lits) add_lit(e);
|
|
}
|
|
void add_eq(expr *a, expr *b) { internalize_eq(a, b); }
|
|
void add_deq(expr *a, expr *b) { internalize_deq(a, b); }
|
|
|
|
void reset();
|
|
|
|
// deprecate?
|
|
void to_lits(expr_ref_vector &lits, bool all_equalities = false,
|
|
bool repick_repr = true);
|
|
void to_lits_qe_lite(expr_ref_vector &lits,
|
|
std::function<bool(expr *)> *non_core = nullptr);
|
|
expr_ref to_expr(bool repick_repr = true);
|
|
|
|
/**
|
|
* Return literals obtained by projecting added literals
|
|
* onto the vocabulary of decls (if exclude is false) or outside the
|
|
* vocabulary of decls (if exclude is true).
|
|
*/
|
|
expr_ref_vector project();
|
|
expr_ref_vector solve();
|
|
expr_ref_vector project(model &mdl);
|
|
|
|
/**
|
|
* Return disequalities to ensure that disequalities between
|
|
* excluded functions are preserved.
|
|
* For example if f(a) = b, f(c) = d, and b and d are not
|
|
* congruent, then produce the disequality a != c.
|
|
*/
|
|
expr_ref_vector get_ackerman_disequalities();
|
|
|
|
/**
|
|
* Produce model-based disequality
|
|
* certificate corresponding to
|
|
* definition in BGVS 2020.
|
|
* A disequality certificate is a reduced set of
|
|
* disequalities, true under mdl, such that the literals
|
|
* can be satisfied when non-shared symbols are projected.
|
|
*/
|
|
expr_ref_vector dcert(model &mdl, expr_ref_vector const &lits);
|
|
|
|
/**
|
|
* Produce a model-based partition.
|
|
*/
|
|
vector<expr_ref_vector> get_partition(model &mdl);
|
|
|
|
/**
|
|
* Extract shared occurrences of terms whose sort are
|
|
* fid, but appear in a context that is not fid.
|
|
* for example f(x + y) produces the shared occurrence
|
|
* x + y when f is uninterpreted and x + y has sort Int or Real.
|
|
*/
|
|
expr_ref_vector shared_occurrences(family_id fid);
|
|
|
|
/**
|
|
* Map expression that occurs in added literals into representative if it
|
|
* exists.
|
|
*/
|
|
void add_model_based_terms(model &mdl, expr_ref_vector const &terms);
|
|
expr *rep_of(expr *e);
|
|
|
|
using deqs = bit_vector;
|
|
struct add_deq_proc {
|
|
uint64_t m_deq_cnt = 0;
|
|
void operator()(term *t1, term *t2);
|
|
void operator()(ptr_vector<term> &ts);
|
|
};
|
|
|
|
// -- disequalities added for output
|
|
vector<std::pair<term *, term *>> m_deq_pairs;
|
|
// -- maybe they are not necessary since they are in the original formula
|
|
vector<ptr_vector<term>> m_deq_distinct;
|
|
|
|
expr_ref_vector non_ground_terms();
|
|
void gr_terms_to_lits(expr_ref_vector &lits, bool all_equalities);
|
|
// produce a quantifier reduction of the formula stored in the term graph
|
|
// output of qel will not contain expression e s.t. non_core(e) == true
|
|
void qel(app_ref_vector &vars, expr_ref &fml,
|
|
std::function<bool(expr *)> *non_core = nullptr);
|
|
bool has_val_in_class(expr *e);
|
|
app *get_const_in_class(expr *e);
|
|
void set_explicit_eq() { m_explicit_eq = true; }
|
|
|
|
private:
|
|
add_deq_proc m_add_deq;
|
|
void refine_repr_class(term *t);
|
|
void refine_repr();
|
|
bool makes_cycle(term *t);
|
|
};
|
|
|
|
namespace is_ground_ns {
|
|
struct found {};
|
|
struct proc {
|
|
term_graph::is_variable_proc &m_is_var;
|
|
proc(term_graph::is_variable_proc &is_var) : m_is_var(is_var) {}
|
|
void operator()(var *n) const {}
|
|
void operator()(app const *n) const {
|
|
if (m_is_var.contains(n->get_decl())) throw found();
|
|
}
|
|
void operator()(quantifier *n) const {}
|
|
};
|
|
} // namespace is_ground_ns
|
|
} // namespace mbp
|