/*++ Copyright (c) 2024 Microsoft Corporation Module Name: bv_sls_fixed.cpp Author: Nikolaj Bjorner (nbjorner) 2024-02-07 --*/ #include "ast/ast_pp.h" #include "ast/ast_ll_pp.h" #include "ast/sls/bv_sls_fixed.h" #include "ast/sls/bv_sls_eval.h" namespace bv { sls_fixed::sls_fixed(sls_eval& ev): ev(ev), m(ev.m), bv(ev.bv) {} void sls_fixed::init(expr_ref_vector const& es) { // init_ranges(es); ev.sort_assertions(es); for (expr* e : ev.m_todo) { if (!is_app(e)) continue; app* a = to_app(e); ev.m_fixed.setx(a->get_id(), is_fixed1(a), false); if (a->get_family_id() == basic_family_id) init_fixed_basic(a); else if (a->get_family_id() == bv.get_family_id()) init_fixed_bv(a); else ; } ev.m_todo.reset(); } void sls_fixed::init_ranges(expr_ref_vector const& es) { for (expr* e : es) { bool sign = m.is_not(e, e); if (is_app(e)) init_range(to_app(e), sign); } } // s <=s t <=> s + K <= t + K, K = 2^{bw-1} void sls_fixed::init_range(app* e, bool sign) { expr* s, * t, * x, * y; rational a, b; unsigned idx; auto N = [&](expr* s) { auto b = bv.get_bv_size(s); return b > 0 ? rational::power_of_two(b - 1) : rational(0); }; if (bv.is_ule(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(x, a, y, b, sign); } else if (bv.is_ult(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(y, b, x, a, !sign); } else if (bv.is_uge(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(y, b, x, a, sign); } else if (bv.is_ugt(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(x, a, y, b, !sign); } else if (bv.is_sle(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(x, a + N(s), y, b + N(s), sign); } else if (bv.is_slt(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(y, b + N(s), x, a + N(s), !sign); } else if (bv.is_sge(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(y, b + N(s), x, a + N(s), sign); } else if (bv.is_sgt(e, s, t)) { get_offset(s, x, a); get_offset(t, y, b); init_range(x, a + N(s), y, b + N(s), !sign); } else if (!sign && m.is_eq(e, s, t)) { if (bv.is_numeral(s, a)) // t - a <= 0 init_range(t, -a, nullptr, rational(0), false); else if (bv.is_numeral(t, a)) init_range(s, -a, nullptr, rational(0), false); } else if (bv.is_bit2bool(e, s, idx)) { auto& val = wval0(s); val.set_bit(idx, !sign); val.set(val.fixed, idx, true); val.init_fixed(); } } // // x + a <= b <=> x in [-a, b - a + 1[ b != -1 // a <= x + b <=> x in [a - b, -b[ a != 0 // x + a <= x + b <=> x in [-a, -b[ a != b // // x + a < b <=> ! (b <= x + a) <=> x not in [-b, a - b + 1[ <=> x in [a - b + 1, -b [ b != 0 // a < x + b <=> ! (x + b <= a) <=> x not in [-a, b - a [ <=> x in [b - a, -a [ a != -1 // x + a < x + b <=> ! (x + b <= x + a) <=> x in [-a, -b [ a != b // void sls_fixed::init_range(expr* x, rational const& a, expr* y, rational const& b, bool sign) { if (!x && !y) return; if (!x) { // a <= y + b if (a == 0) return; auto& v = wval0(y); if (!sign) v.add_range(a - b, -b); else v.add_range(-b, a - b); } else if (!y) { if (mod(b + 1, rational::power_of_two(bv.get_bv_size(x))) == 0) return; auto& v = wval0(x); if (!sign) v.add_range(-a, b - a + 1); else v.add_range(b - a + 1, -a); } else if (x == y) { if (a == b) return; auto& v = wval0(x); if (!sign) v.add_range(-a, -b); else v.add_range(-b, -a); } } void sls_fixed::get_offset(expr* e, expr*& x, rational& offset) { expr* s, * t; x = e; offset = 0; if (bv.is_bv_add(e, s, t)) { if (bv.is_numeral(s, offset)) x = t; else if (bv.is_numeral(t, offset)) x = s; } else if (bv.is_numeral(e, offset)) x = nullptr; } sls_valuation& sls_fixed::wval0(expr* e) { return ev.wval0(e); } void sls_fixed::init_fixed_basic(app* e) { if (bv.is_bv(e) && m.is_ite(e)) { auto& val = wval0(e); auto& val_th = wval0(e->get_arg(1)); auto& val_el = wval0(e->get_arg(2)); for (unsigned i = 0; i < val.nw; ++i) val.fixed[i] = val_el.fixed[i] & val_th.fixed[i] & ~(val_el.bits(i) ^ val_th.bits(i)); val.init_fixed(); } } void sls_fixed::init_fixed_bv(app* e) { if (bv.is_bv(e)) set_fixed_bw(e); } bool sls_fixed::is_fixed1(app* e) const { if (is_uninterp(e)) return false; if (e->get_family_id() == basic_family_id) return is_fixed1_basic(e); return all_of(*e, [&](expr* arg) { return ev.is_fixed0(arg); }); } bool sls_fixed::is_fixed1_basic(app* e) const { switch (e->get_decl_kind()) { case OP_TRUE: case OP_FALSE: return true; case OP_AND: return any_of(*e, [&](expr* arg) { return ev.is_fixed0(arg) && !ev.bval0(e); }); case OP_OR: return any_of(*e, [&](expr* arg) { return ev.is_fixed0(arg) && ev.bval0(e); }); default: return all_of(*e, [&](expr* arg) { return ev.is_fixed0(arg); }); } } void sls_fixed::set_fixed_bw(app* e) { SASSERT(bv.is_bv(e)); SASSERT(e->get_family_id() == bv.get_fid()); auto& v = ev.wval0(e); if (all_of(*e, [&](expr* arg) { return ev.is_fixed0(arg); })) { for (unsigned i = 0; i < v.bw; ++i) v.set(v.fixed, i, true); ev.m_fixed.setx(e->get_id(), true, false); return; } switch (e->get_decl_kind()) { case OP_BAND: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); // (a.fixed & b.fixed) | (a.fixed & ~a.bits) | (b.fixed & ~b.bits) for (unsigned i = 0; i < a.nw; ++i) v.fixed[i] = (a.fixed[i] & b.fixed[i]) | (a.fixed[i] & ~a.bits(i)) | (b.fixed[i] & ~b.bits(i)); break; } case OP_BOR: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); // (a.fixed & b.fixed) | (a.fixed & a.bits) | (b.fixed & b.bits) for (unsigned i = 0; i < a.nw; ++i) v.fixed[i] = (a.fixed[i] & b.fixed[i]) | (a.fixed[i] & a.bits(i)) | (b.fixed[i] & b.bits(i)); break; } case OP_BXOR: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); for (unsigned i = 0; i < a.nw; ++i) v.fixed[i] = a.fixed[i] & b.fixed[i]; break; } case OP_BNOT: { auto& a = wval0(e->get_arg(0)); for (unsigned i = 0; i < a.nw; ++i) v.fixed[i] = a.fixed[i]; break; } case OP_BADD: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); bool pfixed = true; for (unsigned i = 0; i < v.bw; ++i) { if (pfixed && a.get(a.fixed, i) && b.get(b.fixed, i)) v.set(v.fixed, i, true); else if (!pfixed && a.get(a.fixed, i) && b.get(b.fixed, i) && !a.get_bit(i) && !b.get_bit(i)) { pfixed = true; v.set(v.fixed, i, false); } else { pfixed = false; v.set(v.fixed, i, false); } } rational r, rlo, rhi; if (bv.is_numeral(e->get_arg(0), r) && !b.eq(b.lo, b.hi)) { b.get_value(b.lo, rlo); b.get_value(b.hi, rhi); v.add_range(r + rlo, r + rhi); } if (bv.is_numeral(e->get_arg(1), r) && !a.eq(a.lo, a.hi)) { a.get_value(a.lo, rlo); a.get_value(a.hi, rhi); v.add_range(r + rlo, r + rhi); } break; } case OP_BMUL: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); unsigned j = 0, k = 0, zj = 0, zk = 0, hzj = 0, hzk = 0; // i'th bit depends on bits j + k = i // if the first j, resp k bits are 0, the bits j + k are 0 for (; j < v.bw; ++j) if (!a.get(a.fixed, j)) break; for (; k < v.bw; ++k) if (!b.get(b.fixed, k)) break; for (; zj < v.bw; ++zj) if (!a.get(a.fixed, zj) || a.get_bit(zj)) break; for (; zk < v.bw; ++zk) if (!b.get(b.fixed, zk) || b.get_bit(zk)) break; for (; hzj < v.bw; ++hzj) if (!a.get(a.fixed, v.bw - hzj - 1) || a.get_bit(v.bw - hzj - 1)) break; for (; hzk < v.bw; ++hzk) if (!b.get(b.fixed, v.bw - hzk - 1) || b.get_bit(v.bw - hzk - 1)) break; if (j > 0 && k > 0) { for (unsigned i = 0; i < std::min(k, j); ++i) { SASSERT(!v.get_bit(i)); v.set(v.fixed, i, true); } } // lower zj + jk bits are 0 if (zk > 0 || zj > 0) { for (unsigned i = 0; i < zk + zj; ++i) { SASSERT(!v.get_bit(i)); v.set(v.fixed, i, true); } } // upper bits are 0, if enough high order bits of a, b are 0. // TODO - buggy if (false && hzj < v.bw && hzk < v.bw && hzj + hzk > v.bw) { hzj = v.bw - hzj; hzk = v.bw - hzk; for (unsigned i = hzj + hzk - 1; i < v.bw; ++i) { SASSERT(!v.get_bit(i)); v.set(v.fixed, i, true); } } break; } case OP_CONCAT: { auto& a = wval0(e->get_arg(0)); auto& b = wval0(e->get_arg(1)); for (unsigned i = 0; i < b.bw; ++i) v.set(v.fixed, i, b.get(b.fixed, i)); for (unsigned i = 0; i < a.bw; ++i) v.set(v.fixed, i + b.bw, a.get(a.fixed, i)); break; } case OP_EXTRACT: { expr* child; unsigned lo, hi; VERIFY(bv.is_extract(e, lo, hi, child)); auto& a = wval0(child); for (unsigned i = lo; i <= hi; ++i) v.set(v.fixed, i - lo, a.get(a.fixed, i)); break; } case OP_BNEG: { auto& a = wval0(e->get_arg(0)); bool pfixed = true; for (unsigned i = 0; i < v.bw; ++i) { if (pfixed && a.get(a.fixed, i)) v.set(v.fixed, i, true); else { pfixed = false; v.set(v.fixed, i, false); } } break; } case OP_BSHL: { // determine range of b. // if b = 0, then inherit fixed from a // if b >= v.bw then make e fixed to 0 // if 0 < b < v.bw is known, then inherit shift of fixed values of a // if 0 < b < v.bw but not known, then inherit run lengths of equal bits of a // that are fixed. break; } case OP_BASHR: case OP_BLSHR: case OP_INT2BV: case OP_BCOMP: case OP_BNAND: case OP_BREDAND: case OP_BREDOR: case OP_BSDIV: case OP_BSDIV_I: case OP_BSDIV0: case OP_BUDIV: case OP_BUDIV_I: case OP_BUDIV0: case OP_BUREM: case OP_BUREM_I: case OP_BUREM0: case OP_BSMOD: case OP_BSMOD_I: case OP_BSMOD0: case OP_BXNOR: // NOT_IMPLEMENTED_YET(); break; case OP_BV_NUM: case OP_BIT0: case OP_BIT1: case OP_BV2INT: case OP_BNEG_OVFL: case OP_BSADD_OVFL: case OP_BUADD_OVFL: case OP_BSDIV_OVFL: case OP_BSMUL_NO_OVFL: case OP_BSMUL_NO_UDFL: case OP_BSMUL_OVFL: case OP_BUMUL_NO_OVFL: case OP_BUMUL_OVFL: case OP_BIT2BOOL: case OP_ULEQ: case OP_UGEQ: case OP_UGT: case OP_ULT: case OP_SLEQ: case OP_SGEQ: case OP_SGT: case OP_SLT: UNREACHABLE(); break; } v.init_fixed(); } }