3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-04-06 17:44:08 +00:00

wip - arith local search

This commit is contained in:
Nikolaj Bjorner 2023-02-20 12:17:14 -08:00
parent 4aa05b2b57
commit 146f0eae06
3 changed files with 49 additions and 33 deletions

View file

@ -62,7 +62,7 @@ namespace sat {
void ddfw::check_with_plugin() {
m_plugin->init_search();
m_steps_since_progress = 0;
while (m_min_sz > 0 && m_steps_since_progress++ <= 1500000) {
while (m_min_sz > 0 && m_steps_since_progress++ <= 150000) {
if (should_reinit_weights()) do_reinit_weights();
else if (do_flip<true>());
else if (do_literal_flip<true>());

View file

@ -1302,6 +1302,12 @@ namespace sat {
return l_undef;
}
if (false && m_config.m_phase == PS_LOCAL_SEARCH && m_ext) {
IF_VERBOSE(0, verbose_stream() << "WARNING: local search with theories is in testing mode\n");
bounded_local_search();
exit(0);
}
log_stats();
if (m_config.m_max_conflicts > 0 && m_config.m_burst_search > 0) {
m_restart_threshold = m_config.m_burst_search;

View file

@ -207,41 +207,59 @@ namespace arith {
return false;
}
bool sls::cm(bool sign, ineq const& ineq, var_t v, int64_t coeff, int64_t& new_value) {
VERIFY(ineq.is_true() == sign);
verbose_stream() << "cm " << ineq << " for " << v << " sign " << sign << "\n";
bool sls::cm(bool new_sign, ineq const& ineq, var_t v, int64_t coeff, int64_t& new_value) {
SASSERT(ineq.is_true() == new_sign);
VERIFY(ineq.is_true() == new_sign);
auto bound = ineq.m_bound;
auto argsv = ineq.m_args_value;
bool solved = false;
int64_t delta = argsv - bound;
if (sign) {
auto make_eq = [&]() {
SASSERT(delta != 0);
if (delta < 0)
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
else
new_value = value(v) - (delta + abs(coeff) - 1) / coeff;
solved = argsv + coeff * (new_value - value(v)) == bound;
if (!solved && abs(coeff) == 1) {
verbose_stream() << "did not solve equality " << ineq << " for " << v << "\n";
verbose_stream() << new_value << " " << value(v) << " delta " << delta << " lhs " << (argsv + coeff * (new_value - value(v))) << " bound " << bound << "\n";
UNREACHABLE();
}
return solved;
};
auto make_diseq = [&]() {
if (delta >= 0)
delta++;
else
delta--;
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
VERIFY(argsv + coeff * (new_value - value(v)) != bound);
return true;
};
if (new_sign) {
switch (ineq.m_op) {
case ineq_kind::LE:
// args <= bound -> args > bound
SASSERT(argsv <= bound);
SASSERT(delta <= 0);
delta--;
--delta;
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
VERIFY(argsv + coeff * (new_value - value(v)) > bound);
return true;
case ineq_kind::LT:
// args < bound -> args >= bound
SASSERT(argsv <= ineq.m_bound);
SASSERT(delta <= 0);
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
VERIFY(argsv + coeff * (new_value - value(v)) >= bound);
return true;
case ineq_kind::EQ:
if (delta >= 0)
delta++;
else
delta--;
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
VERIFY(argsv + coeff * (new_value - value(v)) != bound);
return true;
return make_diseq();
case ineq_kind::NE:
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
solved = argsv + coeff * (new_value - value(v)) == bound;
if (!solved) verbose_stream() << "did not solve disequality " << ineq << " for " << v << "\n";
return solved;
return make_eq();
default:
UNREACHABLE();
break;
@ -263,18 +281,9 @@ namespace arith {
VERIFY(argsv + coeff * (new_value - value(v)) < bound);
return true;
case ineq_kind::NE:
if (delta >= 0)
delta++;
else
delta--;
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
VERIFY(argsv + coeff * (new_value - value(v)) != bound);
return true;
return make_diseq();
case ineq_kind::EQ:
new_value = value(v) + (abs(delta) + abs(coeff) - 1) / coeff;
solved = argsv + coeff * (new_value - value(v)) == bound;
if (!solved) verbose_stream() << "did not solve equality " << ineq << " for " << v << "\n";
return solved;
return make_eq();
default:
UNREACHABLE();
break;
@ -291,10 +300,10 @@ namespace arith {
int64_t new_value;
auto v = ineq.m_var_to_flip;
if (v == UINT_MAX) {
verbose_stream() << "no var to flip\n";
// verbose_stream() << "no var to flip\n";
return false;
}
if (!cm(sign, ineq, v, new_value)) {
if (!cm(!sign, ineq, v, new_value)) {
verbose_stream() << "no critical move for " << v << "\n";
return false;
}
@ -308,6 +317,7 @@ namespace arith {
// cached dts has to be updated when the score of literals are updated.
//
double sls::dscore(var_t v, int64_t new_value) const {
verbose_stream() << "dscore\n";
double score = 0;
#if 0
auto const& vi = m_vars[v];
@ -558,12 +568,12 @@ namespace arith {
auto* ineq = atom(bv);
if (!ineq)
return 0;
SASSERT(ineq->is_true() == sign);
SASSERT(ineq->is_true() != sign);
int64_t new_value;
for (auto const& [coeff, v] : ineq->m_args) {
double result = 0;
if (cm(sign, *ineq, v, coeff, new_value))
if (cm(!sign, *ineq, v, coeff, new_value))
result = dscore(v, new_value);
// just pick first positive, or pick a max?
if (result > 0) {
@ -576,7 +586,7 @@ namespace arith {
// switch to dscore mode
void sls::on_rescale() {
m_dscore_mode = true;
// m_dscore_mode = true;
}
void sls::on_save_model() {