3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-04-05 17:14:07 +00:00

Add support for initializing variable values in solver and optimize contexts in Z3

This commit is contained in:
Nikolaj Bjorner 2024-09-19 22:44:16 +03:00
parent 342dccdc02
commit 0c48a50d59
12 changed files with 98 additions and 9 deletions

View file

@ -18,7 +18,18 @@ Version 4.13.1
The projection is described in paper by Haokun Li and Bican Xia, [Solving Satisfiability of Polynomial Formulas By Sample - Cell Projection](https://arxiv.org/abs/2003.00409). The code ported from https://github.com/hybridSMT/hybridSMT.git The projection is described in paper by Haokun Li and Bican Xia, [Solving Satisfiability of Polynomial Formulas By Sample - Cell Projection](https://arxiv.org/abs/2003.00409). The code ported from https://github.com/hybridSMT/hybridSMT.git
- Add API for providing hints for the solver/optimize contexts for which initial values to attempt to use for variables. - Add API for providing hints for the solver/optimize contexts for which initial values to attempt to use for variables.
The new API function are Z3_solver_set_initial_value and Z3_optimize_set_initial_value, respectively. Supply these functions with a Boolean or numeric variable, and a value. The solver will then attempt to use these values in the initial phase of search. The feature is aimed at resolving nearly similar problems, or problems with a predicted model and the intent is that restarting the solver based on a near solution can avoid prune the space of constraints that are initially infeasible. The new API function are Z3_solver_set_initial_value and Z3_optimize_set_initial_value, respectively. Supply these functions with a Boolean or numeric variable, and a value. The solver will then attempt to use these values in the initial phase of search. The feature is aimed at resolving nearly similar problems, or problems with a predicted model and the intent is that restarting the solver based on a near solution can avoid prune the space of constraints that are initially infeasible.
The SMTLIB front-end contains the new command (set-initial-value var value). For example,
(declare-const x Int)
(set-initial-value x 10)
(push)
(assert (> x 0))
(check-sat)
(get-model)
produces a model where x = 10. We use (push) to ensure that z3 doesn't run a
specialized pre-processor that eliminates x, which renders the initialization
without effect.
Version 4.13.0 Version 4.13.0
============== ==============

View file

@ -1008,7 +1008,8 @@ sort* basic_decl_plugin::join(unsigned n, expr* const* es) {
} }
sort* basic_decl_plugin::join(sort* s1, sort* s2) { sort* basic_decl_plugin::join(sort* s1, sort* s2) {
if (s1 == s2) return s1; if (s1 == s2)
return s1;
if (s1->get_family_id() == arith_family_id && if (s1->get_family_id() == arith_family_id &&
s2->get_family_id() == arith_family_id) { s2->get_family_id() == arith_family_id) {
if (s1->get_decl_kind() == REAL_SORT) { if (s1->get_decl_kind() == REAL_SORT) {
@ -1016,6 +1017,10 @@ sort* basic_decl_plugin::join(sort* s1, sort* s2) {
} }
return s2; return s2;
} }
if (s1 == m_bool_sort && s2->get_family_id() == arith_family_id)
return s2;
if (s2 == m_bool_sort && s1->get_family_id() == arith_family_id)
return s1;
std::ostringstream buffer; std::ostringstream buffer;
buffer << "Sorts " << mk_pp(s1, *m_manager) << " and " << mk_pp(s2, *m_manager) << " are incompatible"; buffer << "Sorts " << mk_pp(s1, *m_manager) << " and " << mk_pp(s2, *m_manager) << " are incompatible";
throw ast_exception(buffer.str()); throw ast_exception(buffer.str());

View file

@ -130,6 +130,47 @@ generic_model_converter * generic_model_converter::copy(ast_translation & transl
return res; return res;
} }
void generic_model_converter::convert_initialize_value(expr_ref& var, expr_ref& value) {
for (auto const& e : m_entries) {
switch (e.m_instruction) {
case HIDE:
break;
case ADD:
if (is_uninterp_const(var) && e.m_f == to_app(var)->get_decl())
convert_initialize_value(e.m_def, var, value);
break;
}
}
}
void generic_model_converter::convert_initialize_value(expr* def, expr_ref& var, expr_ref& value) {
// var = if(c, th, el) = value
// th = value => c = true
// el = value => c = false
expr* c = nullptr, *th = nullptr, *el = nullptr;
if (m.is_ite(def, c, th, el)) {
if (value == th) {
var = c;
value = m.mk_true();
return;
}
if (value == el) {
var = c;
value = m.mk_false();
return;
}
}
// var = def = value
// => def = value
if (is_uninterp(def))
var = def;
}
void generic_model_converter::set_env(ast_pp_util* visitor) { void generic_model_converter::set_env(ast_pp_util* visitor) {
if (!visitor) { if (!visitor) {

View file

@ -37,6 +37,7 @@ private:
vector<entry> m_entries; vector<entry> m_entries;
expr_ref simplify_def(entry const& e); expr_ref simplify_def(entry const& e);
void convert_initialize_value(expr* def, expr_ref& var, expr_ref& value);
public: public:
generic_model_converter(ast_manager & m, char const* orig) : m(m), m_orig(orig) {} generic_model_converter(ast_manager & m, char const* orig) : m(m), m_orig(orig) {}
@ -61,6 +62,8 @@ public:
model_converter * translate(ast_translation & translator) override { return copy(translator); } model_converter * translate(ast_translation & translator) override { return copy(translator); }
void convert_initialize_value(expr_ref& var, expr_ref& value) override;
generic_model_converter* copy(ast_translation & translator); generic_model_converter* copy(ast_translation & translator);
void set_env(ast_pp_util* visitor) override; void set_env(ast_pp_util* visitor) override;

View file

@ -107,6 +107,12 @@ public:
m_c2->get_units(fmls); m_c2->get_units(fmls);
m_c1->get_units(fmls); m_c1->get_units(fmls);
} }
void convert_initialize_value(expr_ref& var, expr_ref& value) override {
m_c2->convert_initialize_value(var, value);
m_c1->convert_initialize_value(var, value);
}
char const * get_name() const override { return "concat-model-converter"; } char const * get_name() const override { return "concat-model-converter"; }

View file

@ -86,6 +86,8 @@ public:
virtual void set_env(ast_pp_util* visitor); virtual void set_env(ast_pp_util* visitor);
virtual void convert_initialize_value(expr_ref& var, expr_ref& value) { }
/** /**
\brief we are adding a formula to the context of the model converter. \brief we are adding a formula to the context of the model converter.
The operator has as side effect of adding definitions as assertions to the The operator has as side effect of adding definitions as assertions to the

View file

@ -330,10 +330,7 @@ public:
void set_next_arg(cmd_context& ctx, expr* e) override { if (m_var) m_value = e; else m_var = e; } void set_next_arg(cmd_context& ctx, expr* e) override { if (m_var) m_value = e; else m_var = e; }
void execute(cmd_context& ctx) override { void execute(cmd_context& ctx) override {
SASSERT(m_var && m_value); SASSERT(m_var && m_value);
if (ctx.get_opt()) ctx.set_initial_value(m_var, m_value);
ctx.get_opt()->initialize_value(m_var, m_value);
else if (ctx.get_solver())
ctx.get_solver()->user_propagate_initialize_value(m_var, m_value);
} }
}; };

View file

@ -629,6 +629,7 @@ cmd_context::~cmd_context() {
finalize_cmds(); finalize_cmds();
finalize_tactic_manager(); finalize_tactic_manager();
m_proof_cmds = nullptr; m_proof_cmds = nullptr;
m_var2values.reset();
reset(true); reset(true);
m_mcs.reset(); m_mcs.reset();
m_solver = nullptr; m_solver = nullptr;
@ -654,6 +655,8 @@ void cmd_context::set_opt(opt_wrapper* opt) {
m_opt = opt; m_opt = opt;
for (unsigned i = 0; i < m_scopes.size(); ++i) for (unsigned i = 0; i < m_scopes.size(); ++i)
m_opt->push(); m_opt->push();
for (auto const& [var, value] : m_var2values)
m_opt->initialize_value(var, value);
m_opt->set_logic(m_logic); m_opt->set_logic(m_logic);
} }
@ -1874,6 +1877,17 @@ void cmd_context::display_dimacs() {
} }
} }
void cmd_context::set_initial_value(expr* var, expr* value) {
if (get_opt()) {
get_opt()->initialize_value(var, value);
return;
}
if (get_solver())
get_solver()->user_propagate_initialize_value(var, value);
m_var2values.push_back({expr_ref(var, m()), expr_ref(value, m())});
}
void cmd_context::display_model(model_ref& mdl) { void cmd_context::display_model(model_ref& mdl) {
if (mdl) { if (mdl) {
if (mc0()) (*mc0())(mdl); if (mc0()) (*mc0())(mdl);

View file

@ -262,6 +262,7 @@ protected:
scoped_ptr_vector<builtin_decl> m_extra_builtin_decls; // make sure that dynamically allocated builtin_decls are deleted scoped_ptr_vector<builtin_decl> m_extra_builtin_decls; // make sure that dynamically allocated builtin_decls are deleted
dictionary<object_ref*> m_object_refs; // anything that can be named. dictionary<object_ref*> m_object_refs; // anything that can be named.
dictionary<sexpr*> m_user_tactic_decls; dictionary<sexpr*> m_user_tactic_decls;
vector<std::pair<expr_ref, expr_ref>> m_var2values;
dictionary<func_decls> m_func_decls; dictionary<func_decls> m_func_decls;
obj_map<func_decl, symbol> m_func_decl2alias; obj_map<func_decl, symbol> m_func_decl2alias;
@ -421,6 +422,7 @@ public:
solver* get_solver() { return m_solver.get(); } solver* get_solver() { return m_solver.get(); }
void set_solver(solver* s) { m_solver = s; } void set_solver(solver* s) { m_solver = s; }
void set_proof_cmds(proof_cmds* pc) { m_proof_cmds = pc; } void set_proof_cmds(proof_cmds* pc) { m_proof_cmds = pc; }
void set_initial_value(expr* var, expr* value);
void set_solver_factory(solver_factory * s); void set_solver_factory(solver_factory * s);
void set_check_sat_result(check_sat_result * r) { m_check_sat_result = r; } void set_check_sat_result(check_sat_result * r) { m_check_sat_result = r; }

View file

@ -311,7 +311,9 @@ namespace opt {
} }
solver& s = get_solver(); solver& s = get_solver();
s.assert_expr(m_hard_constraints); s.assert_expr(m_hard_constraints);
for (auto const& [var, value] : m_scoped_state.m_values) { for (auto & [var, value] : m_scoped_state.m_values) {
if (m_model_converter)
m_model_converter->convert_initialize_value(var, value);
s.user_propagate_initialize_value(var, value); s.user_propagate_initialize_value(var, value);
} }

View file

@ -703,7 +703,14 @@ public:
} }
void user_propagate_initialize_value(expr* var, expr* value) override { void user_propagate_initialize_value(expr* var, expr* value) override {
ensure_euf()->user_propagate_initialize_value(var, value); expr_ref _var(var, m), _value(value, m);
if (m_mcs.back())
m_mcs.back()->convert_initialize_value(_var, _value);
sat::bool_var b = m_map.to_bool_var(_var);
if (b != sat::null_bool_var)
m_solver.set_phase(sat::literal(b, m.is_false(_value)));
else if (get_euf())
ensure_euf()->user_propagate_initialize_value(_var, _value);
} }

View file

@ -1 +0,0 @@
nbjorner@LAPTOP-04AEAFKH.32880:1726092166