All Downloads are FREE. Search and download functionalities are using the official Maven repository.

z3-z3-4.13.0.src.test.bdd.cpp Maven / Gradle / Ivy

The newest version!
#include "math/dd/dd_bdd.h"
#include "math/dd/dd_fdd.h"
#include 

namespace dd {

class test_bdd {
public:

    static void test1() {
        bdd_manager m(20);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(1);
        bdd v2 = m.mk_var(2);
        bdd c1 = v0 && v1 && v2;
        bdd c2 = v2 && v0 && v1;
        std::cout << c1 << "\n";
        VERIFY(c1 == c2);
        std::cout << "cnf size: " << c1.cnf_size() << "\n";

        c1 = v0 || v1 || v2;
        c2 = v2 || v1 || v0;
        std::cout << c1 << "\n";
        VERIFY(c1 == c2);
        std::cout << "cnf size: " << c1.cnf_size() << "\n";
    }

    static void test2() {
        bdd_manager m(20);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(1);
        bdd v2 = m.mk_var(2);
        VERIFY(m.mk_ite(v0,v0,v1) == (v0 || v1));
        VERIFY(m.mk_ite(v0,v1,v1) == v1);
        VERIFY(m.mk_ite(v1,v0,v1) == (v0 && v1));
        VERIFY(m.mk_ite(v1,v0,v0) == v0);
        VERIFY(m.mk_ite(v0,!v0,v1) == (!v0 && v1));
        VERIFY(m.mk_ite(v0,v1,!v0) == (!v0 || v1));
        VERIFY((!(v0 && v1)) == (!v0 || !v1));
        VERIFY((!(v0 || v1)) == (!v0 && !v1));
    }

    static void test3() {
        bdd_manager m(20);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(1);
        bdd v2 = m.mk_var(2);
        bdd c1 = (v0 && v1) || v2;
        bdd c2 = m.mk_exists(0, c1);
        std::cout << c1 << "\n";
        std::cout << c2 << "\n";
        VERIFY(c2 == (v1 || v2));
        c2 = m.mk_exists(1, c1);
        VERIFY(c2 == (v0 || v2));
        c2 = m.mk_exists(2, c1);
        VERIFY(c2.is_true());
        VERIFY(m.mk_exists(3, c1) == c1);
        c1 = (v0 && v1) || (v1 && v2) || (!v0 && !v2);
        c2 = m.mk_exists(0, c1);
        VERIFY(c2 == (v1 || (v1 && v2) || !v2));
        c2 = m.mk_exists(1, c1);
        VERIFY(c2 == (v0 || v2 || (!v0 && !v2)));
        c2 = m.mk_exists(2, c1);
        VERIFY(c2 == ((v0 && v1) || v1 || !v0));
    }

    static void test4() {
        bdd_manager m(3);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(1);
        bdd v2 = m.mk_var(2);
        bdd c1 = (v0 && v2) || v1;
        std::cout << "before reorder:\n";
        std::cout << c1 << "\n";
        std::cout << c1.bdd_size() << "\n";
        m.gc();
        m.try_reorder();
        std::cout << "after reorder:\n";
        std::cout << c1 << "\n";
        std::cout << c1.bdd_size() << "\n";
    }

    static void test_xor() {
        bdd_manager m(4);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(0);
        VERIFY((m.mk_false() ^ v0) == v0);
        VERIFY((v0 ^ m.mk_false()) == v0);
        VERIFY((m.mk_true() ^ v0) == !v0);
        VERIFY((v0 ^ m.mk_true()) == !v0);
        VERIFY((v0 ^ v1) == ((v0 && !v1) || (!v0 && v1)));
    }

    static void test_bddv_ops_on_constants() {
        std::cout << "test_bddv_ops_on_constants\n";
        unsigned const num_bits = 3;
        rational const modulus = rational::power_of_two(num_bits);
        bdd_manager m(num_bits);

        VERIFY_EQ(m.to_val(m.mk_zero(num_bits)), rational(0));
        VERIFY_EQ(m.to_val(m.mk_ones(num_bits)), modulus - 1);

        for (unsigned n = 0; n < 8; ++n) {
            rational const nr(n);
            VERIFY_EQ(m.to_val(m.mk_num(nr, num_bits)), nr);
        }

        for (unsigned n = 0; n < 8; ++n) {
            for (unsigned k = 0; k < 8; ++k) {
                rational const nr(n);
                rational const kr(k);
                bddv const nv = m.mk_num(nr, num_bits);
                bddv const kv = m.mk_num(kr, num_bits);
                VERIFY_EQ((nv + kv).to_val(), (nr + kr) % modulus);
                VERIFY_EQ((nv - kv).to_val(), (nr - kr + modulus) % modulus);
                VERIFY_EQ((nv * kr).to_val(), (nr * kr) % modulus);
                VERIFY_EQ((nv * kv).to_val(), (nr * kr) % modulus);
                bdd eq = m.mk_eq(nv, kv);
                VERIFY(eq.is_const() && (eq.is_true() == (n == k)));
                eq = m.mk_eq(nv, kr);
                VERIFY(eq.is_const() && (eq.is_true() == (n == k)));

                VERIFY(m.mk_usub(nv).to_val() == (m.mk_zero(num_bits) - nv).to_val());

                bdd cmp = nv <= kv;
                VERIFY(cmp.is_const() && (cmp.is_true() == (nr <= kr)));
                cmp = nv >= kv;
                VERIFY(cmp.is_const() && (cmp.is_true() == (nr >= kr)));
                cmp = nv < kv;
                VERIFY(cmp.is_const() && (cmp.is_true() == (nr < kr)));
                cmp = nv > kv;
                VERIFY(cmp.is_const() && (cmp.is_true() == (nr > kr)));

                // signed versions
                rational const nrs = (nr < modulus / 2) ? nr : nr - modulus;
                rational const krs = (kr < modulus / 2) ? kr : kr - modulus;
                cmp = nv.sle(kv);
                VERIFY(cmp.is_const() && (cmp.is_true() == (nrs <= krs)));
                cmp = nv.sge(kv);
                VERIFY(cmp.is_const() && (cmp.is_true() == (nrs >= krs)));
                cmp = nv.slt(kv);
                VERIFY(cmp.is_const() && (cmp.is_true() == (nrs < krs)));
                cmp = nv.sgt(kv);
                VERIFY(cmp.is_const() && (cmp.is_true() == (nrs > krs)));

                bddv quotv = m.mk_zero(num_bits);
                bddv remv = m.mk_zero(num_bits);
                nv.quot_rem(kv, quotv, remv);
                if (k != 0) {
                    VERIFY_EQ(quotv.to_val(), rational(n / k));
                    VERIFY_EQ(remv.to_val(), rational(n % k));
                } else {
                    // std::cout << "n=" << n << "  k=" << k << "\n";
                    // std::cout << "  quot: " << quotv.to_val() << "\n";
                    // std::cout << "   rem: " << remv.to_val() << "\n";
                }
            }
        }
    }

    static void test_bddv_eqfind_small() {
        std::cout << "test_bddv_eqfind_small\n";
        bdd_manager m(4);
        fdd const x_dom(m, 1);
        bddv const x = x_dom.var();
        bdd x_is_one = (x == rational(1));
        std::cout << "x_is_one:\n" << x_is_one << "\n";
        rational r;
        auto res = x_dom.find(x_is_one, r);
        VERIFY_EQ(res, find_t::singleton);
        VERIFY_EQ(r, rational(1));
    }

    static void test_bddv_eqfind() {
        std::cout << "test_bddv_eqfind\n";
        bdd_manager m(4);

        unsigned_vector bits;
        bits.push_back(0);
        bits.push_back(1);
        bits.push_back(4);
        bits.push_back(27);

        fdd const x_dom(m, bits);
        bddv const x = x_dom.var();
        bddv const zero = m.mk_zero(x_dom.num_bits());
        bddv const one = m.mk_num(rational(1), x_dom.num_bits());
        bddv const five = m.mk_num(rational(5), x_dom.num_bits());

        VERIFY((one == rational(1)).is_true());
        VERIFY((five == rational(5)).is_true());
        VERIFY((five == rational(4)).is_false());
        VERIFY((five == five).is_true());
        VERIFY((five == one).is_false());

        // Test the three different mk_eq overloads
        {
            bdd const x_is_five = m.mk_eq(x, rational(5));
            rational r;
            auto res = x_dom.find(x_is_five, r);
            VERIFY_EQ(res, find_t::singleton);
            VERIFY_EQ(r, 5);
        }

        {
            bdd const x_is_five = m.mk_eq(bits, rational(5));
            rational r;
            auto res = x_dom.find(x_is_five, r);
            VERIFY_EQ(res, find_t::singleton);
            VERIFY_EQ(r, 5);
        }

        {
            bdd const x_is_five = m.mk_eq(x, five);
            rational r;
            auto res = x_dom.find(x_is_five, r);
            VERIFY_EQ(res, find_t::singleton);
            VERIFY_EQ(r, 5);
        }
    }

    static void test_bddv_addsub() {
        std::cout << "test_bddv_addsub\n";
        unsigned_vector bits;
        bits.push_back(0);
        bits.push_back(1);
        bits.push_back(2);
        bdd_manager m(bits.size());
        bddv const x = m.mk_var(bits);
        VERIFY_EQ(x - rational(3) == rational(2), x == rational(5));
        VERIFY_EQ(x + rational(3) == rational(5), x == rational(2));
        VERIFY_EQ(x - rational(3) == rational(6), x == rational(1));
        VERIFY_EQ(x + rational(3) == rational(2), x == rational(7));
    }

    static void test_bddv_mul() {
        std::cout << "test_bddv_mul\n";
        unsigned const num_bits = 4;
        bdd_manager m(num_bits);

        unsigned_vector bits;
        bits.push_back(0);
        bits.push_back(1);
        bits.push_back(4);
        bits.push_back(27);

        bddv const x = m.mk_var(bits);
        bddv const zero = m.mk_zero(num_bits);
        bddv const one = m.mk_num(rational(1), num_bits);
        bddv const five = m.mk_num(rational(5), num_bits);
        bddv const six = m.mk_num(rational(6), num_bits);

        // 5*x == 1 (mod 16)  =>  x == 13 (mod 16)
        bdd const five_inv = x * five == one;
        VERIFY_EQ(five_inv, x == rational(13));

        // 6*x == 1 (mod 16)  =>  no solution for x
        bdd const six_inv = x * six == one;
        VERIFY(six_inv.is_false());

        // 6*x == 0 (mod 16)  =>  x is either 0 or 8 (mod 16)
        bdd const b = six * x == zero;
        VERIFY_EQ(b, x == rational(0) || x == rational(8));
        VERIFY((b && (x != rational(0)) && (x != rational(8))).is_false());
        VERIFY_EQ(b && (x != rational(0)), x == rational(8));

        VERIFY_EQ((x * zero).bits(), (x * rational(0)).bits());
        VERIFY_EQ((x *  one).bits(), (x * rational(1)).bits());
        VERIFY_EQ((x * five).bits(), (x * rational(5)).bits());
        VERIFY_EQ((x *  six).bits(), (x * rational(6)).bits());
    }

    static void test_bddv_ule() {
        std::cout << "test_bddv_ule\n";
        unsigned const num_bits = 4;
        bdd_manager m(num_bits);

        unsigned_vector bits;
        bits.push_back(0);
        bits.push_back(1);
        bits.push_back(2);
        bits.push_back(3);

        bddv const x = m.mk_var(bits);
        bddv const three = m.mk_num(rational(3), num_bits);
        bddv const four = m.mk_num(rational(4), num_bits);
        bddv const five = m.mk_num(rational(5), num_bits);

        VERIFY_EQ(x >= four && x < five, x == four);
        VERIFY_EQ(four <= x && x < five, x == four);
        VERIFY_EQ(three < x && x < five, x == four);
        VERIFY_EQ(three < x && x <= four, x == four);
        VERIFY_EQ(three <= x && x < five, x == four || x == three);
    }

    static void test_fdd3() {
        std::cout << "test_fdd3\n";
        unsigned const w = 3;  // bit width
        bdd_manager m(w);

        fdd const x_dom(m, w);
        bddv const& x = x_dom.var();

        // Encodes the values x satisfying a*x + b == 0 (mod 2^w) as BDD.
        auto mk_affine = [] (unsigned a, bddv const& x, unsigned b) {
            return (rational(a)*x + rational(b) == rational(0));
        };

        vector num;
        for (unsigned n = 0; n < (1< num;
        for (unsigned n = 0; n < (1ul << x_dom.num_bits()); ++n) {
            num.push_back(x == rational(n));
            VERIFY(x_dom.contains(num[n], rational(n)));
            rational r;
            VERIFY_EQ(x_dom.find(num[n], r), find_t::singleton);
            VERIFY_EQ(r, n);
        }

        // need something extra to skew costs and trigger a reorder
        bdd atleast3 = (x >= rational(3));
        VERIFY(x_dom.contains(atleast3, rational(3)));

        auto const old_levels = m.m_var2level;
        std::cout << "old levels: " << old_levels << "\n";
        m.gc();
        m.try_reorder();
        std::cout << "new levels: " << m.m_var2level << "\n";
        VERIFY(old_levels != m.m_var2level);  // ensure that reorder actually did something.

        // Should still give the correct answer after reordering
        for (unsigned n = 0; n < (1ul << x_dom.num_bits()); ++n) {
            VERIFY(x_dom.contains(num[n], rational(n)));
            rational r;
            VERIFY_EQ(x_dom.find(num[n], r), find_t::singleton);
            VERIFY_EQ(r, n);
        }
    }

    static void test_fdd_twovars() {
        std::cout << "test_fdd_twovars\n";
        bdd_manager m(6);
        fdd const x_dom(m, 3, 0, 2);
        fdd const y_dom(m, 3, 1, 2);
        bddv const& x = x_dom.var();
        bddv const& y = y_dom.var();
        VERIFY_EQ(x - y <= rational(0), x == y);
    }

    static void test_fdd_find_hint() {
        std::cout << "test_fdd_find_hint\n";
        bdd_manager m(4);
        fdd const x_dom(m, 4);
        bddv const& x = x_dom.var();

        bdd s358 = x == rational(3) || x == rational(5) || x == rational(8);
        rational r;
        VERIFY_EQ(x_dom.find_hint(s358, rational(8), r), find_t::multiple);
        VERIFY_EQ(r, 8);
        VERIFY_EQ(x_dom.find_hint(s358, rational(5), r), find_t::multiple);
        VERIFY_EQ(r, 5);
        VERIFY_EQ(x_dom.find_hint(s358, rational(3), r), find_t::multiple);
        VERIFY_EQ(r, 3);
        VERIFY_EQ(x_dom.find_hint(s358, rational(7), r), find_t::multiple);
        VERIFY(r == 3 || r == 5 || r == 8);

        VERIFY_EQ(x_dom.find_hint(x == rational(5), rational(3), r), find_t::singleton);
        VERIFY_EQ(r, 5);
        VERIFY_EQ(x_dom.find_hint(x == rational(5), rational(5), r), find_t::singleton);
        VERIFY_EQ(r, 5);

        VERIFY_EQ(x_dom.find_hint(s358 && (x == rational(4)), rational(5), r), find_t::empty);
    }

    static void test_cofactor() {
	std::cout << "test_cofactor\n";
	bdd_manager m(20);
        bdd v0 = m.mk_var(0);
        bdd v1 = m.mk_var(1);
        bdd v2 = m.mk_var(2);
        bdd c1 = v0 && v1 && v2;
	VERIFY(c1.cofactor(v0) == (v1 && v2));
	VERIFY(c1.cofactor(v1) == (v0 && v2));
	VERIFY(c1.cofactor(v2) == (v0 && v1));
	VERIFY(c1.cofactor(!v1) == m.mk_false());
    }

    static void inc(bool_vector& x) {
        for (auto& b : x) {
            b = !b;
            if (b)
                break;
        }
    }

    static void dec(bool_vector& x) {
        for (auto& b : x) {
            b = !b;
            if (!b)
                break;
        }
    }

    static unsigned value(bool_vector const& x) {
        unsigned p = 1;
        unsigned v = 0;
        for (auto b : x) {
            v += p*b;
            p <<= 1;
        }
        return v;
    }

    static void test_sup() {
	std::cout << "test_sup\n";
	bdd_manager m(20);
        fdd const x_dom(m, 10);
        bddv const& x = x_dom.var();
        bdd c = (1 <= x && x <= 5) || (11 <= x && x <= 13) || (29 <= x && x <= 33);
        bool_vector lo(10, false);
        for (unsigned i = 0; i < 20; ++i) {
            unsigned v = value(lo);
            bool found = x_dom.sup(c, lo);
            std::cout << found << ": " << v << " - " << value(lo) << "\n";
            if (found) 
                std::cout << x_dom.sup(c, lo) << ": " << value(lo) << "\n";
            c = !c;
            inc(lo);
        }        
    }

    static void test_inf() {
	std::cout << "test_inf\n";
	bdd_manager m(20);
        fdd const x_dom(m, 10);
        bddv const& x = x_dom.var();
        bdd c = (1 <= x && x <= 5) || (11 <= x && x <= 13) || (29 <= x && x <= 33);
        bool_vector hi(10, true);
        for (unsigned i = 0; i < 10; ++i) {
            bool found = x_dom.inf(c, hi);
            std::cout << found << ": " << value(hi) << "\n";
            if (found) {
                std::cout << x_dom.inf(c, hi) << ": " << value(hi) << "\n";
                VERIFY(value(hi) == 0 || value(hi) == 1 || value(hi) == 5 || value(hi) == 6 ||
                       value(hi) == 10 || value(hi) == 11 || 
                       value(hi) == 13 || value(hi) == 14 || 
                       value(hi) == 28 || value(hi) == 29 || 
                       value(hi) == 33 || value(hi) == 34 || value(hi) == 1023);
            }
            c = !c;
            dec(hi);
        }        
    }



};

}

void tst_bdd() {
    dd::test_bdd::test1();
    dd::test_bdd::test2();
    dd::test_bdd::test3();
    dd::test_bdd::test4();
    dd::test_bdd::test_xor();
    dd::test_bdd::test_bddv_ops_on_constants();
    dd::test_bdd::test_bddv_eqfind_small();
    dd::test_bdd::test_bddv_eqfind();
    dd::test_bdd::test_bddv_addsub();
    dd::test_bdd::test_bddv_mul();
    dd::test_bdd::test_bddv_ule();
    dd::test_bdd::test_fdd3();
    dd::test_bdd::test_fdd4();
    dd::test_bdd::test_fdd_reorder();
    dd::test_bdd::test_fdd_twovars();
    dd::test_bdd::test_fdd_find_hint();
    dd::test_bdd::test_cofactor();
    dd::test_bdd::test_inf();
    dd::test_bdd::test_sup();
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy