mirror of
https://github.com/Z3Prover/z3
synced 2025-04-06 17:44:08 +00:00
Add a sharp throttle to lia2card tactic to control overhead in default tactic mode
lia2card was added to qfuflia tactic based on a user scenario, but it overshot: lia2card is by default harmful. It is here tamed to only convert binary variables and throttle on nested ite terms
This commit is contained in:
parent
fd2a8a554d
commit
99cbfa715c
|
@ -9,6 +9,8 @@ def_module_params('tactic',
|
||||||
('blast_term_ite.max_inflation', UINT, UINT_MAX, "multiplicative factor of initial term size."),
|
('blast_term_ite.max_inflation', UINT, UINT_MAX, "multiplicative factor of initial term size."),
|
||||||
('blast_term_ite.max_steps', UINT, UINT_MAX, "maximal number of steps allowed for tactic."),
|
('blast_term_ite.max_steps', UINT, UINT_MAX, "maximal number of steps allowed for tactic."),
|
||||||
('propagate_values.max_rounds', UINT, 4, "maximal number of rounds to propagate values."),
|
('propagate_values.max_rounds', UINT, 4, "maximal number of rounds to propagate values."),
|
||||||
|
('lia2card.max_range', UINT, 100, "maximal range of integers to compilation into Booleans"),
|
||||||
|
('lia2card.max_ite_nesting', UINT, 4, "maximal nesting depth for ite expressions to be compiled into PB constraints"),
|
||||||
('default_tactic', SYMBOL, '', "overwrite default tactic in strategic solver"),
|
('default_tactic', SYMBOL, '', "overwrite default tactic in strategic solver"),
|
||||||
|
|
||||||
# ('aig.per_assertion', BOOL, True, "process one assertion at a time"),
|
# ('aig.per_assertion', BOOL, True, "process one assertion at a time"),
|
||||||
|
|
|
@ -25,6 +25,7 @@ Notes:
|
||||||
#include "ast/ast_util.h"
|
#include "ast/ast_util.h"
|
||||||
#include "ast/ast_pp_util.h"
|
#include "ast/ast_pp_util.h"
|
||||||
#include "tactic/tactical.h"
|
#include "tactic/tactical.h"
|
||||||
|
#include "params/tactic_params.hpp"
|
||||||
#include "ast/simplifiers/bound_manager.h"
|
#include "ast/simplifiers/bound_manager.h"
|
||||||
#include "ast/converters/generic_model_converter.h"
|
#include "ast/converters/generic_model_converter.h"
|
||||||
|
|
||||||
|
@ -116,7 +117,8 @@ public:
|
||||||
mutable ptr_vector<expr>* m_todo;
|
mutable ptr_vector<expr>* m_todo;
|
||||||
bounds_map m_bounds;
|
bounds_map m_bounds;
|
||||||
bool m_compile_equality;
|
bool m_compile_equality;
|
||||||
unsigned m_max_ub;
|
unsigned m_max_range = 101;
|
||||||
|
unsigned m_max_ite_nesting = 1;
|
||||||
ref<generic_model_converter> m_mc;
|
ref<generic_model_converter> m_mc;
|
||||||
|
|
||||||
lia2card_tactic(ast_manager & _m, params_ref const & p):
|
lia2card_tactic(ast_manager & _m, params_ref const & p):
|
||||||
|
@ -126,7 +128,7 @@ public:
|
||||||
m_pb(m),
|
m_pb(m),
|
||||||
m_todo(alloc(ptr_vector<expr>)),
|
m_todo(alloc(ptr_vector<expr>)),
|
||||||
m_compile_equality(true) {
|
m_compile_equality(true) {
|
||||||
m_max_ub = 100;
|
updt_params(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
~lia2card_tactic() override {
|
~lia2card_tactic() override {
|
||||||
|
@ -137,7 +139,11 @@ public:
|
||||||
|
|
||||||
void updt_params(params_ref const & p) override {
|
void updt_params(params_ref const & p) override {
|
||||||
m_params.append(p);
|
m_params.append(p);
|
||||||
|
tactic_params tp(p);
|
||||||
|
|
||||||
m_compile_equality = m_params.get_bool("compile_equality", true);
|
m_compile_equality = m_params.get_bool("compile_equality", true);
|
||||||
|
m_max_range = tp.lia2card_max_range();
|
||||||
|
m_max_ite_nesting = tp.lia2card_max_ite_nesting();
|
||||||
}
|
}
|
||||||
|
|
||||||
expr_ref mk_bounded(expr_ref_vector& axioms, app* x, unsigned lo, unsigned hi) {
|
expr_ref mk_bounded(expr_ref_vector& axioms, app* x, unsigned lo, unsigned hi) {
|
||||||
|
@ -152,7 +158,6 @@ public:
|
||||||
if (lo > 0) {
|
if (lo > 0) {
|
||||||
xs.push_back(a.mk_int(lo));
|
xs.push_back(a.mk_int(lo));
|
||||||
}
|
}
|
||||||
verbose_stream() << "bounded " << lo << " " << hi << "\n";
|
|
||||||
for (unsigned i = lo; i < hi; ++i) {
|
for (unsigned i = lo; i < hi; ++i) {
|
||||||
checkpoint();
|
checkpoint();
|
||||||
|
|
||||||
|
@ -193,7 +198,7 @@ public:
|
||||||
if (a.is_int(x) &&
|
if (a.is_int(x) &&
|
||||||
is_uninterp_const(x) &&
|
is_uninterp_const(x) &&
|
||||||
bounds.has_lower(x, lo, s1) && !s1 && lo.is_unsigned() &&
|
bounds.has_lower(x, lo, s1) && !s1 && lo.is_unsigned() &&
|
||||||
bounds.has_upper(x, hi, s2) && !s2 && hi.is_unsigned() && hi.get_unsigned() - lo.get_unsigned() <= m_max_ub) {
|
bounds.has_upper(x, hi, s2) && !s2 && hi.is_unsigned() && hi.get_unsigned() - lo.get_unsigned() <= m_max_range) {
|
||||||
expr_ref b = mk_bounded(axioms, to_app(x), lo.get_unsigned(), hi.get_unsigned());
|
expr_ref b = mk_bounded(axioms, to_app(x), lo.get_unsigned(), hi.get_unsigned());
|
||||||
rep.insert(x, b);
|
rep.insert(x, b);
|
||||||
m_bounds.insert(x, bound(lo.get_unsigned(), hi.get_unsigned(), b));
|
m_bounds.insert(x, bound(lo.get_unsigned(), hi.get_unsigned(), b));
|
||||||
|
@ -288,11 +293,14 @@ public:
|
||||||
rational r, q;
|
rational r, q;
|
||||||
if (!is_app(x))
|
if (!is_app(x))
|
||||||
return false;
|
return false;
|
||||||
if (nesting > 8)
|
if (nesting > m_max_ite_nesting && !is_numeral(x, r))
|
||||||
return false;
|
return false;
|
||||||
app* f = to_app(x);
|
app* f = to_app(x);
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
if (a.is_add(x)) {
|
if (is_numeral(x, r)) {
|
||||||
|
insert_arg(mul * r, conds, m.mk_true(), args, coeffs, coeff);
|
||||||
|
}
|
||||||
|
else if (a.is_add(x)) {
|
||||||
for (unsigned i = 0; ok && i < f->get_num_args(); ++i) {
|
for (unsigned i = 0; ok && i < f->get_num_args(); ++i) {
|
||||||
ok = get_sum(nesting, f->get_arg(i), mul, conds, args, coeffs, coeff);
|
ok = get_sum(nesting, f->get_arg(i), mul, conds, args, coeffs, coeff);
|
||||||
}
|
}
|
||||||
|
@ -321,9 +329,6 @@ public:
|
||||||
ok &= get_sum(nesting + 1, u, mul, conds, args, coeffs, coeff);
|
ok &= get_sum(nesting + 1, u, mul, conds, args, coeffs, coeff);
|
||||||
conds.pop_back();
|
conds.pop_back();
|
||||||
}
|
}
|
||||||
else if (is_numeral(x, r)) {
|
|
||||||
insert_arg(mul*r, conds, m.mk_true(), args, coeffs, coeff);
|
|
||||||
}
|
|
||||||
else {
|
else {
|
||||||
TRACE("pb", tout << "Can't handle " << mk_pp(x, m) << "\n";);
|
TRACE("pb", tout << "Can't handle " << mk_pp(x, m) << "\n";);
|
||||||
ok = false;
|
ok = false;
|
||||||
|
|
|
@ -192,6 +192,11 @@ tactic * mk_preamble_tactic(ast_manager& m) {
|
||||||
ctx_simp_p.set_uint("max_depth", 30);
|
ctx_simp_p.set_uint("max_depth", 30);
|
||||||
ctx_simp_p.set_uint("max_steps", 5000000);
|
ctx_simp_p.set_uint("max_steps", 5000000);
|
||||||
|
|
||||||
|
// only binary integer variables are converted to PB
|
||||||
|
params_ref lia2card_p;
|
||||||
|
lia2card_p.set_uint("lia2card.max_range", 1);
|
||||||
|
lia2card_p.set_uint("lia2card.max_ite_nesting", 1);
|
||||||
|
|
||||||
return
|
return
|
||||||
and_then(
|
and_then(
|
||||||
mk_simplify_tactic(m),
|
mk_simplify_tactic(m),
|
||||||
|
@ -199,7 +204,7 @@ tactic * mk_preamble_tactic(ast_manager& m) {
|
||||||
using_params(mk_ctx_simplify_tactic(m), ctx_simp_p),
|
using_params(mk_ctx_simplify_tactic(m), ctx_simp_p),
|
||||||
using_params(mk_simplify_tactic(m), pull_ite_p),
|
using_params(mk_simplify_tactic(m), pull_ite_p),
|
||||||
mk_solve_eqs_tactic(m),
|
mk_solve_eqs_tactic(m),
|
||||||
mk_lia2card_tactic(m),
|
mk_lia2card_tactic(m, lia2card_p),
|
||||||
mk_elim_uncnstr_tactic(m));
|
mk_elim_uncnstr_tactic(m));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -209,6 +214,8 @@ tactic * mk_qflia_tactic(ast_manager & m, params_ref const & p) {
|
||||||
main_p.set_bool("som", true);
|
main_p.set_bool("som", true);
|
||||||
main_p.set_bool("blast_distinct", true);
|
main_p.set_bool("blast_distinct", true);
|
||||||
main_p.set_uint("blast_distinct_threshold", 128);
|
main_p.set_uint("blast_distinct_threshold", 128);
|
||||||
|
main_p.set_uint("lia2card.max_range", 1);
|
||||||
|
main_p.set_uint("lia2card.max_ite_nesting", 1);
|
||||||
// main_p.set_bool("push_ite_arith", true);
|
// main_p.set_bool("push_ite_arith", true);
|
||||||
|
|
||||||
params_ref quasi_pb_p;
|
params_ref quasi_pb_p;
|
||||||
|
|
Loading…
Reference in a new issue