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

add tests for evaluation

This commit is contained in:
Nikolaj Bjorner 2024-02-16 09:58:24 +07:00
parent 1cf008dd0a
commit ddf2d28350
7 changed files with 484 additions and 88 deletions

View file

@ -89,12 +89,14 @@ namespace bv {
sls_valuation* sls_eval::alloc_valuation(unsigned bit_width) {
auto* r = alloc(sls_valuation, bit_width);
while (m_tmp.size() < r->nw) {
while (m_tmp.size() < 2 * r->nw) {
m_tmp.push_back(0);
m_tmp2.push_back(0);
m_tmp2.push_back(0);
m_tmp2.push_back(0);
m_tmp3.push_back(0);
m_tmp4.push_back(0);
m_zero.push_back(0);
m_one.push_back(0);
m_minus_one.push_back(~0);
m_one[0] = 1;
}
return r;
@ -195,9 +197,11 @@ namespace bv {
auto& b = wval0(e->get_arg(1));
unsigned c;
a.set(m_zero, a.bw - 1, true);
mpn.add(a.bits.data(), a.nw, m_zero.data(), a.nw, m_tmp.data(), a.nw, &c);
mpn.add(b.bits.data(), a.nw, m_zero.data(), a.nw, m_tmp2.data(), a.nw, &c);
mpn.add(a.bits.data(), a.nw, m_zero.data(), a.nw, m_tmp.data(), a.nw + 1, &c);
mpn.add(b.bits.data(), a.nw, m_zero.data(), a.nw, m_tmp2.data(), a.nw + 1, &c);
a.set(m_zero, a.bw - 1, false);
a.clear_overflow_bits(m_tmp);
a.clear_overflow_bits(m_tmp2);
return f(mpn.compare(m_tmp.data(), a.nw, m_tmp2.data(), b.nw));
};
@ -209,8 +213,7 @@ namespace bv {
for (unsigned i = a.nw; i < 2 * a.nw; ++i)
if (m_tmp2[i] != 0)
return true;
return !a.has_overflow(m_tmp);
return true;
return a.has_overflow(m_tmp2);
};
switch (e->get_decl_kind()) {
@ -246,7 +249,7 @@ namespace bv {
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
digit_t c = 0;
mpn.add(a.bits.data(), a.nw, b.bits.data(), b.nw, m_tmp.data(), a.nw, &c);
mpn.add(a.bits.data(), a.nw, b.bits.data(), b.nw, m_tmp.data(), a.nw + 1, &c);
return c != 0 || a.has_overflow(m_tmp);
}
case OP_BNEG_OVFL:
@ -295,6 +298,63 @@ namespace bv {
val.set(wval0(e).bits);
return;
}
auto set_sdiv = [&]() {
// d = udiv(abs(x), abs(y))
// y = 0, x > 0 -> -1
// y = 0, x <= 0 -> -1
// x = 0, y != 0 -> 0
// x > 0, y < 0 -> -d
// x < 0, y > 0 -> -d
// x > 0, y > 0 -> d
// x < 0, y < 0 -> d
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
if (a.is_zero() && b.is_zero())
val.set(m_minus_one);
else if (a.is_zero())
val.set(m_zero);
else if (b.is_zero())
val.set(m_minus_one);
else if (!a.sign() && b.is_zero())
val.set(m_one);
else {
bool sign_a = a.sign();
bool sign_b = b.sign();
digit_t c;
if (sign_a)
mpn.sub(m_zero.data(), a.nw, a.bits.data(), a.nw, m_tmp.data(), &c);
else
a.get(m_tmp);
val.clear_overflow_bits(m_tmp);
if (sign_b)
mpn.sub(m_zero.data(), a.nw, b.bits.data(), a.nw, m_tmp2.data(), &c);
else
b.get(m_tmp2);
val.clear_overflow_bits(m_tmp2);
mpn.div(m_tmp.data(), a.nw, m_tmp2.data(), a.nw, m_tmp3.data(), m_tmp4.data());
if (sign_a == sign_b)
val.set(m_tmp3);
else
mpn.sub(m_zero.data(), a.nw, m_tmp3.data(), a.nw, m_tmp.data(), &c),
val.set(m_tmp);
}
};
auto mk_rotate_left = [&](unsigned n) {
auto& a = wval0(e->get_arg(0));
if (n == 0 || a.bw == 1)
val.set(a.bits);
else {
for (unsigned i = a.bw - n; i < a.bw; ++i)
val.set(val.bits, i + n - a.bw, a.get(a.bits, i));
for (unsigned i = 0; i < a.bw - n; ++i)
val.set(val.bits, i + a.bw - n, a.get(a.bits, i));
}
};
SASSERT(e->get_family_id() == bv.get_fid());
switch (e->get_decl_kind()) {
case OP_BV_NUM: {
@ -307,7 +367,7 @@ namespace bv {
SASSERT(e->get_num_args() == 2);
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
for (unsigned i = 0; i < a.bw; ++i)
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i] & b.bits[i];
break;
}
@ -315,7 +375,7 @@ namespace bv {
SASSERT(e->get_num_args() == 2);
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
for (unsigned i = 0; i < a.bw; ++i)
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i] | b.bits[i];
break;
}
@ -323,7 +383,7 @@ namespace bv {
SASSERT(e->get_num_args() == 2);
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
for (unsigned i = 0; i < a.bw; ++i)
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i] ^ b.bits[i];
break;
}
@ -331,7 +391,7 @@ namespace bv {
SASSERT(e->get_num_args() == 2);
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
for (unsigned i = 0; i < a.bw; ++i)
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = ~(a.bits[i] & b.bits[i]);
break;
}
@ -340,7 +400,15 @@ namespace bv {
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
digit_t c;
mpn.add(a.bits.data(), a.nw, b.bits.data(), b.nw, val.bits.data(), val.nw, &c);
mpn.add(a.bits.data(), a.nw, b.bits.data(), b.nw, val.bits.data(), val.nw + 1, &c);
break;
}
case OP_BSUB: {
SASSERT(e->get_num_args() == 2);
auto const& a = wval0(e->get_arg(0));
auto const& b = wval0(e->get_arg(1));
digit_t c;
mpn.sub(a.bits.data(), a.nw, b.bits.data(), b.nw, val.bits.data(), &c);
break;
}
case OP_BMUL: {
@ -393,17 +461,13 @@ namespace bv {
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
auto sh = b.to_nat(b.bits, b.bw);
if (sh == 0) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i];
}
else if (sh >= b.bw) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = 0;
}
if (sh == 0)
val.set(a.bits);
else if (sh >= b.bw)
val.set_zero();
else {
for (unsigned i = 0; i < a.bw; ++i)
val.set(val.bits, i, i >= sh && b.get(b.bits, i - sh));
val.set(val.bits, i, i >= sh && a.get(a.bits, i - sh));
}
break;
}
@ -411,17 +475,13 @@ namespace bv {
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
auto sh = b.to_nat(b.bits, b.bw);
if (sh == 0) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i];
}
else if (sh >= b.bw) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = 0;
}
if (sh == 0)
val.set(a.bits);
else if (sh >= b.bw)
val.set_zero();
else {
for (unsigned i = 0; i < a.bw; ++i)
val.set(val.bits, i, i + sh < a.bw && b.get(b.bits, i + sh));
val.set(val.bits, i, i + sh < a.bw && a.get(a.bits, i + sh));
}
break;
}
@ -429,33 +489,31 @@ namespace bv {
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
auto sh = b.to_nat(b.bits, b.bw);
auto sign = a.get(a.bits, a.bw - 1);
if (sh == 0) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = a.bits[i];
}
auto sign = a.sign();
if (sh == 0)
val.set(a.bits);
else if (sh >= b.bw) {
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = sign ? ~0 : 0;
}
else {
for (unsigned i = 0; i < a.bw; ++i)
val.set(val.bits, i, i + sh < a.bw && b.get(b.bits, i + sh));
for (unsigned i = 0; i < a.bw; ++i)
val.set(val.bits, i, i + sh < a.bw && a.get(a.bits, i + sh));
if (sign)
val.set_range(val.bits, 0, a.bw - sh, true);
val.set_range(val.bits, a.bw - sh, a.bw, true);
}
break;
}
case OP_SIGN_EXT: {
auto& a = wval0(e->get_arg(0));
a.set(val.bits);
bool sign = a.get(a.bits, a.bw - 1);
val.set(a.bits);
bool sign = a.sign();
val.set_range(val.bits, a.bw, val.bw, sign);
break;
}
case OP_ZERO_EXT: {
auto& a = wval0(e->get_arg(0));
a.set(val.bits);
val.set(a.bits);
val.set_range(val.bits, a.bw, val.bw, false);
break;
}
@ -505,7 +563,7 @@ namespace bv {
mpn.sub(b.bits.data(), a.nw, m_tmp2.data(), a.nw, m_tmp.data(), &c),
val.set(m_tmp);
else if (b.sign())
mpn.add(b.bits.data(), a.nw, m_tmp2.data(), a.nw, m_tmp.data(), a.nw, &c),
mpn.add(b.bits.data(), a.nw, m_tmp2.data(), a.nw, m_tmp.data(), a.nw + 1, &c),
val.set(m_tmp);
else
val.set(m_tmp2);
@ -518,11 +576,8 @@ namespace bv {
// x div 0 = -1
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
if (b.is_zero()) {
val.set(m_zero);
for (unsigned i = 0; i < a.nw; ++i)
val.bits[i] = ~val.bits[i];
}
if (b.is_zero())
val.set(m_minus_one);
else {
mpn.div(a.bits.data(), a.nw,
b.bits.data(), b.nw,
@ -535,17 +590,57 @@ namespace bv {
case OP_BSDIV:
case OP_BSDIV_I:
case OP_BSDIV0:
// d = udiv(abs(x), abs(y))
// y = 0, x > 0 -> 1
// y = 0, x <= 0 -> -1
// x = 0, y != 0 -> 0
// x > 0, y < 0 -> -d
// x < 0, y > 0 -> -d
// x > 0, y > 0 -> d
// x < 0, y < 0 -> d
case OP_BSDIV0: {
set_sdiv();
break;
}
case OP_BSREM:
case OP_BSREM0:
case OP_BSREM_I: {
// y = 0 -> x
// else x - sdiv(x, y) * y
//
auto& a = wval0(e->get_arg(0));
auto& b = wval0(e->get_arg(1));
if (b.is_zero())
val.set(a.bits);
else {
digit_t c;
set_sdiv();
mpn.mul(val.bits.data(), a.nw, b.bits.data(), a.nw, m_tmp.data());
mpn.sub(a.bits.data(), a.nw, m_tmp.data(), a.nw, m_tmp2.data(), &c);
val.set(m_tmp2);
}
break;
}
case OP_ROTATE_LEFT: {
unsigned n = e->get_parameter(0).get_int() % val.bw;
mk_rotate_left(n);
break;
}
case OP_ROTATE_RIGHT: {
unsigned n = e->get_parameter(0).get_int() % val.bw;
mk_rotate_left(val.bw - n);
break;
}
case OP_EXT_ROTATE_LEFT: {
auto& b = wval0(e->get_arg(1));
rational n;
b.get_value(b.bits, n);
n = mod(n, rational(val.bw));
SASSERT(n.is_unsigned());
mk_rotate_left(n.get_unsigned());
break;
}
case OP_EXT_ROTATE_RIGHT: {
auto& b = wval0(e->get_arg(1));
rational n;
b.get_value(b.bits, n);
n = mod(n, rational(val.bw));
SASSERT(n.is_unsigned());
mk_rotate_left(val.bw - n.get_unsigned());
break;
}
case OP_BREDAND:
case OP_BREDOR:
case OP_BXNOR:
@ -693,22 +788,37 @@ namespace bv {
VERIFY(bv.is_bit2bool(e, arg, idx));
return try_repair_bit2bool(wval0(e, 0), idx);
}
case OP_BSDIV:
case OP_BSDIV_I:
case OP_BSDIV0:
return try_repair_sdiv(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_BUDIV:
case OP_BUDIV_I:
case OP_BUDIV0:
return try_repair_udiv(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_BUREM:
case OP_BUREM_I:
case OP_BUREM0:
return try_repair_urem(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_BSREM:
case OP_BSREM_I:
case OP_BSREM0:
return try_repair_srem(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_BSMOD:
case OP_BSMOD_I:
case OP_BSMOD0:
return try_repair_smod(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_ROTATE_LEFT:
return try_repair_rotate_left(wval0(e), wval0(e, 0), e->get_parameter(0).get_int());
case OP_ROTATE_RIGHT:
return try_repair_rotate_left(wval0(e), wval0(e, 0), wval0(e).bw - e->get_parameter(0).get_int());
case OP_EXT_ROTATE_LEFT:
return try_repair_rotate_left(wval0(e), wval0(e, 0), wval0(e, 1), i);
case OP_EXT_ROTATE_RIGHT:
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:
case OP_BNEG_OVFL:
case OP_BSADD_OVFL:
@ -883,7 +993,7 @@ namespace bv {
return a.try_set(b.bits);
else {
digit_t c;
mpn.add(b.bits.data(), a.nw, m_one.data(), a.nw, &c, a.nw, m_tmp.data());
mpn.add(b.bits.data(), a.nw, m_one.data(), a.nw, &c, a.nw + 1, m_tmp.data());
return a.try_set(m_tmp);
}
}
@ -918,9 +1028,8 @@ namespace bv {
unsigned sh = b.to_nat(b.bits, b.bw);
if (sh == 0)
return a.try_set(e.bits);
else if (sh >= b.bw) {
return false;
}
else if (sh >= b.bw)
return false;
else {
//
// e = a << sh
@ -936,29 +1045,91 @@ namespace bv {
}
}
else {
// NB. blind sub-range of possible values for b
SASSERT(i == 1);
unsigned sh = m_rand(a.bw + 1);
b.set(m_tmp, sh);
return b.try_set(m_tmp);
}
return false;
}
bool sls_eval::try_repair_ashr(bvval const& e, bvval & a, bvval& b, unsigned i) {
if (i == 0) {
unsigned sh = b.to_nat(b.bits, b.bw);
if (sh == 0)
return a.try_set(e.bits);
else if (sh >= b.bw)
return false;
else {
// e = a >> sh
// a[bw-1:sh] = e[bw-sh-1:0]
// a[sh-1:0] = a[sh-1:0]
// ignore sign
for (unsigned i = 0; i < a.bw - sh; ++i)
a.set(m_tmp, i + sh, e.get(e.bits, i));
for (unsigned i = 0; i < sh; ++i)
a.set(m_tmp, i, a.get(a.bits, i));
return a.try_set(m_tmp);
}
}
else {
// NB. blind sub-range of possible values for b
SASSERT(i == 1);
unsigned sh = m_rand(a.bw + 1);
b.set(m_tmp, sh);
return b.try_set(m_tmp);
}
return false;
}
bool sls_eval::try_repair_lshr(bvval const& e, bvval& a, bvval& b, unsigned i) {
if (i == 0) {
return try_repair_ashr(e, a, b, i);
}
bool sls_eval::try_repair_sdiv(bvval const& e, bvval& a, bvval& b, unsigned i) {
return false;
}
bool sls_eval::try_repair_udiv(bvval const& e, bvval& a, bvval& b, unsigned i) {
return false;
}
bool sls_eval::try_repair_smod(bvval const& e, bvval& a, bvval& b, unsigned i) {
return false;
}
bool sls_eval::try_repair_urem(bvval const& e, bvval& a, bvval& b, unsigned i) {
return false;
}
bool sls_eval::try_repair_srem(bvval const& e, bvval& a, bvval& b, unsigned i) {
return false;
}
bool sls_eval::try_repair_rotate_left(bvval const& e, bvval& a, unsigned n) {
// a := rotate_right(e, n)
n = (n % a.bw) - n;
for (unsigned i = a.bw - n; i < a.bw; ++i)
a.set(m_tmp, i + n - a.bw, e.get(e.bits, i));
for (unsigned i = 0; i < a.bw - n; ++i)
a.set(m_tmp, i + a.bw - n, e.get(e.bits, i));
return a.try_set(m_tmp);
}
bool sls_eval::try_repair_rotate_left(bvval const& e, bvval& a, bvval& b, unsigned i) {
if (i == 0) {
rational n;
b.get_value(b.bits, n);
n = mod(n, rational(b.bw));
return try_repair_rotate_left(e, a, n.get_unsigned());
}
else {
}
return false;
SASSERT(i == 1);
unsigned sh = m_rand(b.bw);
b.set(m_tmp, sh);
return b.try_set(m_tmp);
}
}
void sls_eval::repair_up(expr* e) {

View file

@ -27,10 +27,11 @@ namespace bv {
class sls_eval {
friend class sls_fixed;
friend class sls_test;
ast_manager& m;
bv_util bv;
sls_fixed m_fix;
mutable mpn_manager mpn;
mutable mpn_manager mpn;
ptr_vector<expr> m_todo;
random_gen m_rand;
@ -38,7 +39,7 @@ namespace bv {
bool_vector m_eval; // expr-id -> boolean valuation
bool_vector m_fixed; // expr-id -> is Boolean fixed
mutable svector<digit_t> m_tmp, m_tmp2, m_zero, m_one;
mutable svector<digit_t> m_tmp, m_tmp2, m_tmp3, m_tmp4, m_zero, m_one, m_minus_one;
using bvval = sls_valuation;
@ -82,6 +83,14 @@ namespace bv {
bool try_repair_ashr(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_lshr(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_bit2bool(bvval& a, unsigned idx);
bool try_repair_sdiv(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_udiv(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_smod(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_urem(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_srem(bvval const& e, bvval& a, bvval& b, unsigned i);
bool try_repair_rotate_left(bvval const& e, bvval& a, unsigned n);
bool try_repair_rotate_left(bvval const& e, bvval& a, bvval& b, unsigned i);
sls_valuation& wval0(app* e, unsigned i) { return wval0(e->get_arg(i)); }
@ -123,7 +132,6 @@ namespace bv {
m_eval[e->get_id()] = b;
}
/*
* Try to invert value of child to repair value assignment of parent.
*/

View file

@ -25,10 +25,10 @@ namespace bv {
sls_valuation::sls_valuation(unsigned bw): bw(bw) {
nw = (bw + sizeof(digit_t) * 8 - 1) / (8 * sizeof(digit_t));
unsigned num_bytes = nw * sizeof(digit_t);
lo.reserve(nw);
hi.reserve(nw);
bits.reserve(nw);
fixed.reserve(nw);
lo.reserve(nw + 1);
hi.reserve(nw + 1);
bits.reserve(nw + 1);
fixed.reserve(nw + 1);
// have lo, hi bits, fixed point to memory allocated within this of size num_bytes each allocated
for (unsigned i = 0; i < nw; ++i)
lo[i] = 0, hi[i] = 0, bits[i] = 0, fixed[i] = 0;
@ -84,6 +84,11 @@ namespace bv {
}
}
void sls_valuation::get(svector<digit_t>& dst) const {
for (unsigned i = 0; i < nw; ++i)
dst[i] = bits[i];
}
void sls_valuation::set1(svector<digit_t>& bits) {
for (unsigned i = 0; i < bw; ++i)
set(bits, i, true);

View file

@ -40,8 +40,10 @@ namespace bv {
void set_value(svector<digit_t>& bits, rational const& r);
void get_value(svector<digit_t> const& bits, rational& r) const;
void get(svector<digit_t>& dst) const;
void add_range(rational lo, rational hi);
void set1(svector<digit_t>& bits);
void clear_overflow_bits(svector<digit_t>& bits) const;
bool can_set(svector<digit_t> const& bits) const;
@ -90,6 +92,11 @@ namespace bv {
clear_overflow_bits(bits);
}
void set_zero() {
for (unsigned i = 0; i < nw; ++i)
bits[i] = 0;
}
void set_fixed(svector<digit_t> const& src) {
for (unsigned i = nw; i-- > 0; )

View file

@ -108,6 +108,7 @@ add_executable(test-z3
simple_parser.cpp
simplex.cpp
simplifier.cpp
sls_test.cpp
small_object_allocator.cpp
smt2print_parse.cpp
smt_context.cpp

View file

@ -267,4 +267,5 @@ int main(int argc, char ** argv) {
TST(distribution);
TST(euf_bv_plugin);
TST(euf_arith_plugin);
TST(sls_test);
}

203
src/test/sls_test.cpp Normal file
View file

@ -0,0 +1,203 @@
#include "ast/sls/bv_sls_eval.h"
#include "ast/rewriter/th_rewriter.h"
#include "ast/reg_decl_plugins.h"
#include "ast/ast_pp.h"
namespace bv {
class sls_test {
ast_manager& m;
bv_util bv;
public:
sls_test(ast_manager& m):
m(m),
bv(m)
{}
void check_eval(expr* e) {
std::function<bool(expr*, unsigned)> value = [](expr*, unsigned) {
return false;
};
expr_ref_vector es(m);
bv_util bv(m);
es.push_back(e);
sls_eval ev(m);
ev.init_eval(es, value);
th_rewriter rw(m);
expr_ref r(e, m);
rw(r);
if (bv.is_bv(e)) {
auto const& val = ev.wval0(e);
rational n1, n2;
val.get_value(val.bits, n1);
VERIFY(bv.is_numeral(r, n2));
if (n1 != n2) {
verbose_stream() << mk_pp(e, m) << " computed value " << val << "\n";
verbose_stream() << "should be " << n2 << "\n";
}
SASSERT(n1 == n2);
VERIFY(n1 == n2);
}
else if (m.is_bool(e)) {
auto val1 = ev.bval0(e);
auto val2 = m.is_true(r) ? true : false;
if (val1 != val2) {
verbose_stream() << mk_pp(e, m) << " computed value " << val1 << " at odds with definition\n";
}
SASSERT(val1 == val2);
VERIFY(val1 == val2);
}
}
void check(expr* a, expr* b) {
expr_ref e(m);
auto& validator = *this;
e = bv.mk_bv_add(a, b);
validator.check_eval(e);
e = bv.mk_bv_mul(a, b);
validator.check_eval(e);
e = bv.mk_bv_sub(a, b);
validator.check_eval(e);
e = bv.mk_bv_udiv(a, b);
validator.check_eval(e);
e = bv.mk_bv_sdiv(a, b);
validator.check_eval(e);
e = bv.mk_bv_srem(a, b);
validator.check_eval(e);
e = bv.mk_bv_urem(a, b);
validator.check_eval(e);
e = bv.mk_bv_smod(a, b);
validator.check_eval(e);
e = bv.mk_bv_shl(a, b);
validator.check_eval(e);
e = bv.mk_bv_ashr(a, b);
validator.check_eval(e);
e = bv.mk_bv_lshr(a, b);
validator.check_eval(e);
e = bv.mk_bv_and(a, b);
validator.check_eval(e);
e = bv.mk_bv_or(a, b);
validator.check_eval(e);
e = bv.mk_bv_xor(a, b);
validator.check_eval(e);
e = bv.mk_bv_neg(a);
validator.check_eval(e);
e = bv.mk_bv_not(a);
validator.check_eval(e);
e = bv.mk_bvumul_ovfl(a, b);
validator.check_eval(e);
e = bv.mk_bvumul_no_ovfl(a, b);
validator.check_eval(e);
e = bv.mk_zero_extend(3, a);
validator.check_eval(e);
e = bv.mk_sign_extend(3, a);
validator.check_eval(e);
e = bv.mk_ule(a, b);
validator.check_eval(e);
e = bv.mk_sle(a, b);
validator.check_eval(e);
e = bv.mk_concat(a, b);
validator.check_eval(e);
e = bv.mk_extract(6, 3, a);
validator.check_eval(e);
e = bv.mk_bvuadd_ovfl(a, b);
validator.check_eval(e);
#if 0
e = bv.mk_bvsadd_ovfl(a, b);
validator.check_eval(e);
e = bv.mk_bvneg_ovfl(a);
validator.check_eval(e);
e = bv.mk_bvsmul_no_ovfl(a, b);
validator.check_eval(e);
e = bv.mk_bvsmul_no_udfl(a, b);
validator.check_eval(e);
e = bv.mk_bvsmul_ovfl(a, b);
validator.check_eval(e);
e = bv.mk_bvsdiv_ovfl(a, b);
validator.check_eval(e);
#endif
#if 0
e = bv.mk_rotate_left(a, b);
validator.check_eval(e);
e = bv.mk_rotate_right(a, b);
validator.check_eval(e);
e = bv.mk_rotate_left_ext(a, b);
validator.check_eval(e);
e = bv.mk_rotate_right_ext(a, b);
validator.check_eval(e);
#endif
}
};
}
static void test_eval1() {
ast_manager m;
reg_decl_plugins(m);
bv_util bv(m);
expr_ref e(m);
bv::sls_test validator(m);
unsigned k = 0;
for (unsigned i = 0; i < 256; ++i) {
expr_ref a(bv.mk_numeral(rational(i), 8), m);
for (unsigned j = 0; j < 256; ++j) {
expr_ref b(bv.mk_numeral(rational(j), 8), m);
++k;
if (k % 1000 == 0)
verbose_stream() << "tests " << k << "\n";
validator.check(a, b);
}
}
}
void tst_sls_test() {
test_eval1();
}