/*++ Copyright (c) 2012 Microsoft Corporation Module Name: nlsat.cpp Abstract: nlsat procedure Author: Leonardo (leonardo) 2012-01-09 Notes: --*/ #include "nlsat/nlsat_assignment.h" #include "nlsat/nlsat_interval_set.h" #include "nlsat/nlsat_evaluator.h" #include "nlsat/nlsat_solver.h" #include "nlsat/levelwise.h" #include "util/util.h" #include "nlsat/nlsat_explain.h" #include "math/polynomial/polynomial_cache.h" #include "util/rlimit.h" #include #include nlsat::interval_set_ref tst_interval(nlsat::interval_set_ref const & s1, nlsat::interval_set_ref const & s2, unsigned expected_num_intervals, bool check_num_intervals = true) { nlsat::interval_set_manager & ism = s1.m(); nlsat::interval_set_ref r(ism); std::cout << "s1: " << s1 << "\n"; std::cout << "s2: " << s2 << "\n"; r = ism.mk_union(s1, s2); std::cout << "union(s1, s2): " << r << std::endl; ENSURE(!check_num_intervals || ism.num_intervals(r) == expected_num_intervals); ENSURE(ism.subset(s1, r)); ENSURE(ism.subset(s2, r)); if (ism.set_eq(s1, s2)) { ENSURE(ism.set_eq(s1, r)); ENSURE(ism.set_eq(s2, r)); } else { ENSURE(ism.subset(s1, s2) || !ism.subset(r, s2)); ENSURE(ism.subset(s2, s1) || !ism.subset(r, s1)); } nlsat::interval_set_ref r2(ism); r2 = ism.mk_union(s2, s1); ENSURE(ism.set_eq(r, r2)); anum zero; nlsat::interval_set_ref full(ism); nlsat::literal dummy(131, false); full = ism.mk(true, true, zero, true, true, zero, dummy, nullptr); ENSURE(ism.set_eq(r, full) == ism.is_full(r)); return r; } static void tst3() { enable_trace("nlsat_interval"); reslimit rl; unsynch_mpq_manager qm; anum_manager am(rl, qm); small_object_allocator allocator; nlsat::interval_set_manager ism(am, allocator); scoped_anum sqrt2(am), m_sqrt2(am), two(am), m_two(am), three(am), one(am), zero(am); am.set(two, 2); am.set(m_two, -2); am.set(one, 1); am.root(two, 2, sqrt2); am.set(m_sqrt2, sqrt2); am.neg(m_sqrt2); am.set(three, 3); nlsat::literal p1(1, false); nlsat::literal p2(2, false); nlsat::literal p3(3, false); nlsat::literal p4(4, false); nlsat::literal np2(2, true); nlsat::interval_set_ref s1(ism), s2(ism), s3(ism), s4(ism); s1 = ism.mk_empty(); std::cout << "s1: " << s1 << "\n"; s2 = ism.mk(true, true, zero, false, false, sqrt2, np2, nullptr); std::cout << "s2: " << s2 << "\n"; s3 = ism.mk(false, false, zero, false, false, two, p1, nullptr); std::cout << "s3: " << s3 << "\n"; s4 = ism.mk_union(s2, s3); std::cout << "s4: " << s4 << "\n"; // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, zero, false, false, two, p1, nullptr); s2 = ism.mk(false, false, zero, false, false, two, p2, nullptr); tst_interval(s1, s2, 1); // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, zero, false, false, two, p1, nullptr); s2 = ism.mk(false, false, m_sqrt2, false, false, one, p2, nullptr); s3 = ism.mk_union(s1, s2); tst_interval(s1, s2, 2); // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, m_sqrt2, false, false, one, p1, nullptr); s2 = ism.mk(false, false, zero, false, false, two, p2, nullptr); tst_interval(s1, s2, 2); // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, m_sqrt2, false, false, one, p1, nullptr); s2 = ism.mk(false, false, two, false, false, three, p2, nullptr); tst_interval(s1, s2, 2); // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, m_sqrt2, false, false, three, p1, nullptr); s2 = ism.mk(false, false, zero, false, false, two, p2, nullptr); tst_interval(s1, s2, 1); // Case // s1: [ ... ] // s2: [ ... ] [ ... ] s1 = ism.mk(false, false, m_two, false, false, two, p1, nullptr); s2 = ism.mk(false, false, m_sqrt2, false, false, zero, p2, nullptr); s3 = ism.mk(false, false, one, false, false, three, p2, nullptr); s2 = ism.mk_union(s2, s3); tst_interval(s1, s2, 2); // Case // s1: [ ... ] // s2: [ ... ] s1 = ism.mk(false, false, m_two, false, false, two, p1, nullptr); s2 = ism.mk(false, false, two, false, false, three, p2, nullptr); tst_interval(s1, s2, 2); s2 = ism.mk(true, false, two, false, false, three, p2, nullptr); tst_interval(s1, s2, 2); s2 = ism.mk(true, false, two, false, false, three, p1, nullptr); tst_interval(s1, s2, 1); s1 = ism.mk(false, false, m_two, true, false, two, p1, nullptr); tst_interval(s1, s2, 2); s1 = ism.mk(false, false, two, false, false, two, p1, nullptr); s2 = ism.mk(false, false, two, false, false, three, p2, nullptr); tst_interval(s1, s2, 1); // Case // s1: [ ... ] [ ... ] // s2: [ .. ] [ ... ] [ ... ] s1 = ism.mk(false, false, m_two, false, false, zero, p1, nullptr); s3 = ism.mk(false, false, one, false, false, three, p1, nullptr); s1 = ism.mk_union(s1, s3); s2 = ism.mk(true, true, zero, false, false, m_sqrt2, p2, nullptr); tst_interval(s1, s2, 3); s3 = ism.mk(false, false, one, false, false, sqrt2, p2, nullptr); s2 = ism.mk_union(s2, s3); s3 = ism.mk(false, false, two, true, true, zero, p2, nullptr); s2 = ism.mk_union(s2, s3); tst_interval(s1, s2, 4); // Case s1 = ism.mk(true, true, zero, false, false, one, p1, nullptr); s2 = ism.mk(true, false, one, true, true, zero, p2, nullptr); tst_interval(s1, s2, 2); s2 = ism.mk(true, false, one, false, false, two, p2, nullptr); s3 = ism.mk(false, false, two, true, true, zero, p1, nullptr); s2 = ism.mk_union(s2, s3); tst_interval(s1, s2, 3); } static nlsat::interval_set_ref mk_random(nlsat::interval_set_manager & ism, anum_manager & am, int range, int space, int tries, bool minus_inf, bool plus_inf, nlsat::literal lit) { static random_gen gen; ENSURE(range > 0); ENSURE(space > 0); nlsat::interval_set_ref r(ism), curr(ism); scoped_anum lower(am); scoped_anum upper(am); int prev = -range + (gen() % (space*4)); if (gen() % 3 == 0 && minus_inf) { int next = prev + (gen() % space); bool open = gen() % 2 == 0; am.set(upper, next); r = ism.mk(true, true, lower, open, false, upper, lit, nullptr); prev = next; } for (int i = 0; i < tries; ++i) { int l = prev + (gen() % space); int u = l + (gen() % space); bool lower_open = gen() % 2 == 0; bool upper_open = gen() % 2 == 0; if ((lower_open || upper_open) && l == u) u++; am.set(lower, l); am.set(upper, u); curr = ism.mk(lower_open, false, lower, upper_open, false, upper, lit, nullptr); r = ism.mk_union(r, curr); prev = u; } if (gen() % 3 == 0 && plus_inf) { int next = prev + (gen() % space); bool open = gen() % 2 == 0; am.set(lower, next); curr = ism.mk(open, false, lower, true, true, upper, lit, nullptr); r = ism.mk_union(r, curr); } return r; } static void check_subset_result(nlsat::interval_set_ref const & s1, nlsat::interval_set_ref const & s2, nlsat::interval_set_ref const & r, nlsat::literal l1, nlsat::literal l2) { nlsat::interval_set_manager ism(s1.m()); nlsat::interval_set_ref tmp(ism); unsigned num = ism.num_intervals(r); nlsat::literal_vector lits; ptr_vector clauses; ism.get_justifications(r, lits, clauses); ENSURE(lits.size() <= 2); for (unsigned i = 0; i < num; ++i) { tmp = ism.get_interval(r, i); ism.get_justifications(tmp, lits, clauses); ENSURE(lits.size() == 1); if (lits[0] == l1) { ENSURE(ism.subset(tmp, s1)); } else { ENSURE(lits[0] == l2); ENSURE(ism.subset(tmp, s2)); } } } static void tst4() { enable_trace("nlsat_interval"); reslimit rl; unsynch_mpq_manager qm; anum_manager am(rl, qm); small_object_allocator allocator; nlsat::interval_set_manager ism(am, allocator); nlsat::interval_set_ref s1(ism), s2(ism), r(ism); nlsat::literal l1(1, false); nlsat::literal l2(2, false); for (unsigned i = 0; i < 100; ++i) { s1 = mk_random(ism, am, 20, 3, 10, true, true, l1); s2 = mk_random(ism, am, 20, 3, 10, true, true, l2); r = tst_interval(s1, s2, 0, false); check_subset_result(s1, s2, r, l1, l2); } for (unsigned i = 0; i < 100; ++i) { s1 = mk_random(ism, am, 200, 100, 20, true, true, l1); s2 = mk_random(ism, am, 200, 100, 20, true, true, l2); r = tst_interval(s1, s2, 0, false); check_subset_result(s1, s2, r, l1, l2); } } static void tst5() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); small_object_allocator allocator; nlsat::interval_set_manager ism(am, allocator); nlsat::evaluator ev(s, as, pm, allocator); nlsat::var x0, x1; x0 = pm.mk_var(); x1 = pm.mk_var(); polynomial_ref p(pm); polynomial_ref _x0(pm), _x1(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); p = (_x0^2) + (_x1^2) - 2; nlsat::poly * _p[1] = { p.get() }; bool is_even[1] = { false }; nlsat::bool_var b = s.mk_ineq_atom(nlsat::atom::GT, 1, _p, is_even); nlsat::atom * a = s.bool_var2atom(b); ENSURE(a != nullptr); scoped_anum zero(am); am.set(zero, 0); as.set(0, zero); auto i = ev.infeasible_intervals(a, true, nullptr); std::cout << "1) " << i << "\n"; as.set(1, zero); auto i2 = ev.infeasible_intervals(a, true, nullptr); std::cout << "2) " << i2 << "\n"; } static void project(nlsat::solver& s, nlsat::explain& ex, nlsat::var x, unsigned num, nlsat::literal const* lits) { std::cout << "Project "; s.display(std::cout, num, lits); nlsat::scoped_literal_vector result(s); ex.project(x, num, lits, result); s.display(std::cout << "\n==>\n", result.size(), result.data()); std::cout << "\n"; } static void project_fa(nlsat::solver& s, nlsat::explain& ex, nlsat::var x, unsigned num, nlsat::literal const* lits) { std::cout << "Project "; nlsat::scoped_literal_vector result(s); ex.compute_conflict_explanation(num, lits, result); std::cout << "(or"; for (auto l : result) { s.display(std::cout << " ", l); } for (unsigned i = 0; i < num; ++i) { s.display(std::cout << " ", ~lits[i]); } std::cout << ")\n"; } static bool literal_holds(nlsat::solver& s, nlsat::evaluator& eval, nlsat::literal l) { if (l == nlsat::true_literal) return true; if (l == nlsat::false_literal) return false; nlsat::atom* a = s.bool_var2atom(l.var()); ENSURE(a != nullptr); return eval.eval(a, l.sign()); } static nlsat::literal mk_gt(nlsat::solver& s, nlsat::poly* p) { nlsat::poly * _p[1] = { p }; bool is_even[1] = { false }; return s.mk_ineq_literal(nlsat::atom::GT, 1, _p, is_even); } static nlsat::literal mk_lt(nlsat::solver& s, nlsat::poly* p) { nlsat::poly * _p[1] = { p }; bool is_even[1] = { false }; return s.mk_ineq_literal(nlsat::atom::LT, 1, _p, is_even); } static nlsat::literal mk_eq(nlsat::solver& s, nlsat::poly* p) { nlsat::poly * _p[1] = { p }; bool is_even[1] = { false }; return s.mk_ineq_literal(nlsat::atom::EQ, 1, _p, is_even); } static void set_assignment_value(nlsat::assignment& as, anum_manager& am, nlsat::var v, rational const& val) { scoped_anum tmp(am); am.set(tmp, val.to_mpq()); as.set(v, tmp); } static void tst_vandermond() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); nlsat::pmanager& pm = s.pm(); anum_manager & am = s.am(); nlsat::assignment as(am); scoped_anum zero(am), one(am), two(am), three(am); nlsat::explain& ex = s.get_explain(); nlsat::var x0 = s.mk_var(false); nlsat::var x1 = s.mk_var(false); nlsat::var x2 = s.mk_var(false); nlsat::var x3 = s.mk_var(false); am.set(one, 1); am.set(two, 2); as.set(x0, one); as.set(x1, two); as.set(x2, three); polynomial_ref _x0(pm), _x1(pm), _x2(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); polynomial_ref x0_sq(pm), x1_sq(pm), x2_sq(pm); x0_sq = _x0 * _x0; x1_sq = _x1 * _x1; x2_sq = _x2 * _x2; polynomial_ref vandermonde_flat(pm); vandermonde_flat = (_x1 * x2_sq) - (x1_sq * _x2) + (_x0 * x1_sq) - (x0_sq * _x1) + (x0_sq * _x2) - (_x0 * x2_sq); polynomial_ref vandermonde_factored(pm); vandermonde_factored = (_x1 - _x0) * (_x2 - _x0) * (_x2 - _x1); std::cout << "vandermonde_factored:" << vandermonde_factored << "\n"; polynomial_ref diff(pm); diff = vandermonde_flat - vandermonde_factored; ENSURE(pm.is_zero(diff.get())); pm.display(std::cout << "vandermonde(flat): ", vandermonde_flat); std::cout << "\n"; nlsat::scoped_literal_vector lits(s); lits.push_back(mk_gt(s, vandermonde_flat)); s.set_rvalues(as); project(s, ex, x2, lits.size(), lits.data()); as.set(x2, (one + two)/2); std::cout << am.eval_sign_at(vandermonde_flat, as) << "\n"; ;} static void tst6() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); nlsat::var x0, x1, x2, a, b, c, d; a = s.mk_var(false); b = s.mk_var(false); c = s.mk_var(false); d = s.mk_var(false); x0 = s.mk_var(false); x1 = s.mk_var(false); x2 = s.mk_var(false); polynomial_ref p1(pm), p2(pm), p3(pm), p4(pm), p5(pm); polynomial_ref _x0(pm), _x1(pm), _x2(pm); polynomial_ref _a(pm), _b(pm), _c(pm), _d(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); _a = pm.mk_polynomial(a); _b = pm.mk_polynomial(b); _c = pm.mk_polynomial(c); _d = pm.mk_polynomial(d); p1 = (_a*(_x0^2)) + _x2 + 2; p2 = (_b*_x1) - (2*_x2) - _x0 + 8; nlsat::scoped_literal_vector lits(s); lits.push_back(mk_gt(s, p1)); lits.push_back(mk_gt(s, p2)); lits.push_back(mk_gt(s, (_c*_x0) + _x2 + 1)); lits.push_back(mk_gt(s, (_d*_x0) - _x1 + 5*_x2)); scoped_anum zero(am), one(am), two(am); am.set(zero, 0); am.set(one, 1); am.set(two, 2); as.set(0, one); as.set(1, one); as.set(2, two); as.set(3, two); as.set(4, two); as.set(5, one); as.set(6, one); s.set_rvalues(as); project(s, ex, x0, 2, lits.data()); project(s, ex, x1, 3, lits.data()); project(s, ex, x2, 3, lits.data()); project(s, ex, x2, 2, lits.data()); project(s, ex, x2, 4, lits.data()); project(s, ex, x2, 3, lits.data()+1); } static void tst7() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); nlsat::pmanager & pm = s.pm(); nlsat::var x0, x1, x2, a, b, c, d; a = s.mk_var(false); b = s.mk_var(false); c = s.mk_var(false); d = s.mk_var(false); x0 = s.mk_var(false); x1 = s.mk_var(false); x2 = s.mk_var(false); polynomial_ref p1(pm), p2(pm), p3(pm), p4(pm), p5(pm); polynomial_ref _x0(pm), _x1(pm), _x2(pm); polynomial_ref _a(pm), _b(pm), _c(pm), _d(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); _a = pm.mk_polynomial(a); _b = pm.mk_polynomial(b); _c = pm.mk_polynomial(c); _d = pm.mk_polynomial(d); p1 = _x0 + _x1; p2 = _x2 - _x0; p3 = (-1*_x0) - _x1; nlsat::scoped_literal_vector lits(s); lits.push_back(mk_gt(s, p1)); lits.push_back(mk_gt(s, p2)); lits.push_back(mk_gt(s, p3)); nlsat::literal_vector litsv(lits.size(), lits.data()); lbool res = s.check(litsv); VERIFY(res == l_false); for (unsigned i = 0; i < litsv.size(); ++i) { s.display(std::cout, litsv[i]); std::cout << " "; } std::cout << "\n"; litsv.reset(); litsv.append(2, lits.data()); res = s.check(litsv); ENSURE(res == l_true); s.display(std::cout); s.am().display(std::cout, s.value(x0)); std::cout << "\n"; s.am().display(std::cout, s.value(x1)); std::cout << "\n"; s.am().display(std::cout, s.value(x2)); std::cout << "\n"; } static void tst8() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); nlsat::var x0, x1, x2, a, b, c, d; a = s.mk_var(false); b = s.mk_var(false); c = s.mk_var(false); d = s.mk_var(false); x0 = s.mk_var(false); x1 = s.mk_var(false); x2 = s.mk_var(false); polynomial_ref p1(pm), p2(pm), p3(pm), p4(pm), p5(pm); polynomial_ref _x0(pm), _x1(pm), _x2(pm); polynomial_ref _a(pm), _b(pm), _c(pm), _d(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); _a = pm.mk_polynomial(a); _b = pm.mk_polynomial(b); _c = pm.mk_polynomial(c); _d = pm.mk_polynomial(d); scoped_anum zero(am), one(am), two(am), six(am); am.set(zero, 0); am.set(one, 1); am.set(two, 2); am.set(six, 6); as.set(0, two); // a as.set(1, one); // b as.set(2, six); // c as.set(3, zero); // d as.set(4, zero); // x0 as.set(5, zero); // x1 as.set(6, two); // x2 s.set_rvalues(as); nlsat::scoped_literal_vector lits(s); lits.push_back(mk_eq(s, (_a*_x2*_x2) - (_b*_x2) - _c)); project(s, ex, x2, 1, lits.data()); } static void tst9() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); int num_lo = 4; int num_hi = 5; svector los, his; for (int i = 0; i < num_lo; ++i) { los.push_back(s.mk_var(false)); scoped_anum num(am); am.set(num, - i - 1); as.set(i, num); } for (int i = 0; i < num_hi; ++i) { his.push_back(s.mk_var(false)); scoped_anum num(am); am.set(num, i + 1); as.set(num_lo + i, num); } nlsat::var _z = s.mk_var(false); nlsat::var _x = s.mk_var(false); polynomial_ref x(pm), z(pm); x = pm.mk_polynomial(_x); scoped_anum val(am); am.set(val, 0); as.set(num_lo + num_hi, val); as.set(num_lo + num_hi + 1, val); s.set_rvalues(as); nlsat::scoped_literal_vector lits(s); for (int i = 0; i < num_lo; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(los[i]); lits.push_back(mk_gt(s, x - y)); } for (int i = 0; i < num_hi; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(his[i]); lits.push_back(mk_gt(s, y - x)); } z = pm.mk_polynomial(_z); lits.push_back(mk_eq(s, x - z)); #define TEST_ON_OFF() \ std::cout << "Off "; \ project(s, ex, _x, lits.size()-1, lits.data()); \ std::cout << "On "; \ project(s, ex, _x, lits.size()-1, lits.data()); \ std::cout << "Off "; \ project(s, ex, _x, lits.size(), lits.data()); \ std::cout << "On "; \ project(s, ex, _x, lits.size(), lits.data()) \ TEST_ON_OFF(); lits.reset(); polynomial_ref u(pm); u = pm.mk_polynomial(his[1]); for (int i = 0; i < num_lo; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(los[i]); lits.push_back(mk_gt(s, u*x - y)); } for (int i = 0; i < num_hi; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(his[i]); lits.push_back(mk_gt(s, y - u*x)); } z = pm.mk_polynomial(_z); lits.push_back(mk_eq(s, u*x - z)); TEST_ON_OFF(); lits.reset(); u = pm.mk_polynomial(los[1]); for (int i = 0; i < num_lo; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(los[i]); lits.push_back(mk_gt(s, u*x - y)); } for (int i = 0; i < num_hi; ++i) { polynomial_ref y(pm); y = pm.mk_polynomial(his[i]); lits.push_back(mk_gt(s, y - u*x)); } z = pm.mk_polynomial(_z); lits.push_back(mk_eq(s, x - z)); TEST_ON_OFF(); } #if 0 #endif static void test_root_literal(nlsat::solver& s, nlsat::explain& ex, nlsat::var x, nlsat::atom::kind k, unsigned i, nlsat::poly* p) { nlsat::scoped_literal_vector result(s); ex.test_root_literal(k, x, 1, p, result); nlsat::bool_var b = s.mk_root_atom(k, x, i, p); s.display(std::cout, nlsat::literal(b, false)); s.display(std::cout << " ==> ", result.size(), result.data()); std::cout << "\n"; } static bool satisfies_root(nlsat::solver& s, nlsat::atom::kind k, nlsat::poly* p) { nlsat::pmanager & pm = s.pm(); anum_manager & am = s.am(); nlsat::assignment as(am); s.get_rvalues(as); polynomial_ref pr(p, pm); switch (k) { case nlsat::atom::ROOT_EQ: return am.eval_sign_at(pr, as) == 0; case nlsat::atom::ROOT_LE: return am.eval_sign_at(pr, as) <= 0; case nlsat::atom::ROOT_LT: return am.eval_sign_at(pr, as) < 0; case nlsat::atom::ROOT_GE: return am.eval_sign_at(pr, as) >= 0; case nlsat::atom::ROOT_GT: return am.eval_sign_at(pr, as) > 0; default: UNREACHABLE(); return false; } } static void tst10() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); nlsat::var _a = s.mk_var(false); nlsat::var _b = s.mk_var(false); nlsat::var _c = s.mk_var(false); nlsat::var _x = s.mk_var(false); polynomial_ref x(pm), a(pm), b(pm), c(pm), p(pm); x = pm.mk_polynomial(_x); a = pm.mk_polynomial(_a); b = pm.mk_polynomial(_b); c = pm.mk_polynomial(_c); p = a*x*x + b*x + c; scoped_anum one(am), two(am), three(am), mone(am), mtwo(am), mthree(am), zero(am), one_a_half(am); am.set(zero, 0); am.set(one, 1); am.set(two, 2); am.set(three, 3); am.set(mone, -1); am.set(mtwo, -2); am.set(mthree, -3); rational oah(1,2); am.set(one_a_half, oah.to_mpq()); scoped_anum_vector nums(am); nums.push_back(one); nums.push_back(two); nums.push_back(one_a_half); nums.push_back(mone); nums.push_back(three); // a = 1, b = -3, c = 2: // has roots x = 2, x = 1: // 2^2 - 3*2 + 2 = 0 // 1 - 3 + 2 = 0 as.set(_a, one); as.set(_b, mthree); as.set(_c, two); for (unsigned i = 0; i < nums.size(); ++i) { as.set(_x, nums[i]); s.set_rvalues(as); std::cout << p << "\n"; as.display(std::cout); for (unsigned k = nlsat::atom::ROOT_EQ; k <= nlsat::atom::ROOT_GE; ++k) { if (satisfies_root(s, (nlsat::atom::kind) k, p)) { test_root_literal(s, ex, _x, (nlsat::atom::kind) k, 1, p); } } } as.set(_a, mone); as.set(_b, three); as.set(_c, mtwo); for (unsigned i = 0; i < nums.size(); ++i) { as.set(_x, nums[i]); s.set_rvalues(as); std::cout << p << "\n"; as.display(std::cout); for (unsigned k = nlsat::atom::ROOT_EQ; k <= nlsat::atom::ROOT_GE; ++k) { if (satisfies_root(s, (nlsat::atom::kind) k, p)) { test_root_literal(s, ex, _x, (nlsat::atom::kind) k, 1, p); } } } std::cout << "\n"; } void tst_nlsat_mv() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment assignment(am); nlsat::explain& ex = s.get_explain(); tst_vandermond(); return; // Regression: reproduce lemma 114 where main_operator adds spurious bounds. nlsat::var x0 = s.mk_var(false); nlsat::var x1 = s.mk_var(false); polynomial_ref _x0(pm), _x1(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); polynomial_ref x0_sq(pm), x0_cu(pm), x0_4(pm), x0_5(pm); x0_sq = _x0 * _x0; x0_cu = x0_sq * _x0; x0_4 = x0_cu * _x0; x0_5 = x0_4 * _x0; polynomial_ref x1_sq(pm), x1_cu(pm), x1_4(pm), x1_5(pm); x1_sq = _x1 * _x1; x1_cu = x1_sq * _x1; x1_4 = x1_cu * _x1; x1_5 = x1_4 * _x1; polynomial_ref root_arg(pm); root_arg = x1_5 + (_x0 * x1_4) - (18 * x1_4) - (2 * x0_sq * x1_cu) - (2 * x0_cu * x1_sq) + (36 * x0_sq * x1_sq) + (1296 * _x0 * x1_sq) + (864 * x1_sq) + (x0_4 * _x1) + (1296 * x0_sq * _x1) + (6048 * _x0 * _x1) + x0_5 - (18 * x0_4) + (864 * x0_sq); // should be (x1^5 + x0 x1^4 - 18 x1^4 - 2 x0^2 x1^3 - 2 x0^3 x1^2 + 36 x0^2 x1^2 + 1296 x0 x1^2 + 864 x1^2 + x0^4 x1 + 1296 x0^2 x1 + 6048 x0 x1 + x0^5 - 18 x0^4 + 864 x0^2) std::cout << "big poly:" << root_arg << std::endl; nlsat::literal x1_gt_0 = mk_gt(s, _x1); nlsat::bool_var root_gt = s.mk_root_atom(nlsat::atom::ROOT_GT, x1, 3, root_arg.get()); nlsat::literal x1_gt_root(root_gt, false); nlsat::scoped_literal_vector lits(s); lits.push_back(x1_gt_0); lits.push_back(~x1_gt_root); // !(x1 > root[3](root_arg)) scoped_anum one(am), one_dup(am); am.set(one, 1); assignment.set(x0, one); s.set_rvalues(assignment); nlsat::scoped_literal_vector result(s); ex.compute_conflict_explanation(lits.size(), lits.data(), result); std::cout << "main_operator root regression core:\n"; s.display(std::cout, lits.size(), lits.data()); s.display(std::cout << "\n==>\n", result.size(), result.data()); std::cout << "\n"; // Assign x1 only after the lemma is produced. am.set(one_dup, 1); assignment.set(x1, one_dup); s.set_rvalues(assignment); small_object_allocator allocator; nlsat::evaluator eval(s, assignment, pm, allocator); std::cout << "input literal values at x0 = 1, x1 = 1:\n"; for (nlsat::literal l : lits) { nlsat::atom* a = s.bool_var2atom(l.var()); if (!a) { std::cout << "conversion bug\n"; } bool value = a ? eval.eval(a, l.sign()) : false; s.display(std::cout << " ", l); std::cout << " -> " << (value ? "true" : "false") << "\n"; } std::cout << "new literal values at x0 = 1, x1 = 1:\n"; for (nlsat::literal l : result) { nlsat::atom* a = s.bool_var2atom(l.var()); bool value = a ? eval.eval(a, l.sign()) : false; if (!a) { std::cout << "conversion bug\n"; } s.display(std::cout << " ", l); std::cout << " -> " << (value ? "true" : "false") << "\n"; } std::cout << "\n"; } static void tst11() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); nlsat::var x, y, z; y = s.mk_var(false); z = s.mk_var(false); x = s.mk_var(false); polynomial_ref p1(pm), p2(pm), _x(pm), _y(pm), _z(pm); _x = pm.mk_polynomial(x); _y = pm.mk_polynomial(y); _z = pm.mk_polynomial(z); nlsat::scoped_literal_vector lits(s); scoped_anum zero(am), one(am), five(am); am.set(zero, 0); am.set(one, 1); am.set(five, 5); as.set(z, zero); as.set(y, five); as.set(x, five); s.set_rvalues(as); p1 = (_x - _y); p2 = ((_x*_x) - (_x*_y) - _z); lits.reset(); lits.push_back(mk_gt(s, p1)); lits.push_back(mk_eq(s, p2)); project_fa(s, ex, x, 2, lits.data()); // return; p1 = ((_x * _x) - (2 * _y * _x) - _z + (_y *_y)); p2 = _x + _y; as.set(_x, one); as.set(_y, zero); as.set(_z, one); lits.reset(); lits.push_back(mk_lt(s, p1)); lits.push_back(mk_eq(s, p2)); project_fa(s, ex, x, 2, lits.data()); return; as.set(z, zero); as.set(y, five); as.set(x, five); p1 = (_x - _y); p2 = ((_x*_x) - (_x*_y)); lits.reset(); lits.push_back(mk_gt(s, p1)); lits.push_back(mk_eq(s, p2)); project_fa(s, ex, x, 2, lits.data()); #if 0 !(x5^4 - 2 x3^2 x5^2 - 2 x1^2 x5^2 + 4 x0 x1 x5^2 - 2 x0^2 x5^2 + x3^4 - 2 x1^2 x3^2 + 4 x0 x1 x3^2 - 2 x0^2 x3^2 + x1^4 - 4 x0 x1^3 + 6 x0^2 x1^2 - 4 x0^3 x1 + x0^4 = 0) or !(x5 < 0) or !(x4 > root[1](x1 x4 - x0 x4 + x3)) or !(x3 + x1 - x0 > 0) or !(x1 - x0 < 0) or !(x7 > root[1](x1^2 x7 - 2 x0 x1 x7 + x0^2 x7 + x1 x3 - x0 x3)) or x7 - x4 = 0 or !(x1 x3 x7^2 - x0 x3 x7^2 - x5^2 x7 + x3^2 x7 + x1^2 x7 - 2 x0 x1 x7 + x0^2 x7 + x1 x3 - x0 x3 = 0) x0 := -1 x1 := -21.25 x2 := 0.0470588235? x3 := 2 x4 := -0.03125 x5 := -18.25 x6 := -0.5 x7 := 1 #endif } static void tst_polynomial_cache_mk_unique() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); nlsat::pmanager& pm = s.pm(); polynomial::cache cache(pm); nlsat::var x = s.mk_var(false); polynomial_ref x_poly(pm); x_poly = pm.mk_polynomial(x); polynomial_ref p(pm); p = 2 * x_poly; pm.display(std::cout, p); std::cout << "\n"; ENSURE(!cache.contains(p.get())); polynomial::polynomial* unique_p = cache.mk_unique(p.get()); ENSURE(unique_p != nullptr); ENSURE(cache.contains(unique_p)); pm.display(std::cout, unique_p); std::cout << "\n"; polynomial_ref p_again(pm); p_again = 2 * x_poly; ENSURE(cache.mk_unique(p_again.get()) == unique_p); polynomial_ref p_neg(pm); p_neg = -2 * x_poly; ENSURE(!cache.contains(p_neg.get())); polynomial::polynomial* unique_p_neg = cache.mk_unique(p_neg.get()); ENSURE(unique_p_neg != nullptr); ENSURE(unique_p_neg != unique_p); ENSURE(cache.contains(unique_p_neg)); polynomial_ref p_neg_again(pm); p_neg_again = -2 * x_poly; ENSURE(cache.mk_unique(p_neg_again.get()) == unique_p_neg); polynomial_ref ca(p, pm), cb(p_neg, pm); pm.primitive(ca, x, ca); pm.primitive(cb, x, cb); pm.display(std::cout << "ca:", ca) << "\n"; pm.display(std::cout << "cb:", cb) << "\n"; } static void tst_nullified_polynomial() { params_ref ps; reslimit rlim; nlsat::solver s(rlim, ps, false); nlsat::pmanager & pm = s.pm(); anum_manager & am = s.am(); polynomial::cache cache(pm); nlsat::var x0 = s.mk_var(false); nlsat::var x1 = s.mk_var(false); nlsat::var x2 = s.mk_var(false); nlsat::var x3 = s.mk_var(false); ENSURE(x0 < x1 && x1 < x2); polynomial_ref _x0(pm), _x1(pm), _x2(pm), _x3(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); _x3 = pm.mk_polynomial(x3); polynomial_ref p1(pm), p2(pm); p1 = _x2 * _x1; p2 = _x0; p1 = p1 + p2; p2 = _x3; polynomial_ref_vector polys(pm); polys.push_back(p1); polys.push_back(p2); nlsat::assignment as(am); scoped_anum zero(am); am.set(zero, 0); as.set(x0, zero); as.set(x1, zero); as.set(x2, zero); as.set(x3, zero); s.set_rvalues(as); unsigned max_x = 0; for (unsigned i = 0; i < polys.size(); ++i) { unsigned lvl = pm.max_var(polys.get(i)); if (lvl > max_x) max_x = lvl; } ENSURE(max_x == x3); nlsat::levelwise lws(s, polys, max_x, s.sample(), pm, am, cache); auto cell = lws.single_cell(); ENSURE(lws.failed()); } // Test case for unsound lemma lws2380 - comparing standard projection vs levelwise // The issue: x7 is unconstrained in levelwise output but affects the section polynomial static void tst_unsound_lws2380() { enable_trace("nlsat_explain"); auto run_test = [](bool use_lws) { std::cout << "=== tst_unsound_lws2380: " << (use_lws ? "Levelwise" : "Standard") << " projection (lws=" << use_lws << ") ===\n"; params_ref ps; ps.set_bool("lws", use_lws); reslimit rlim; nlsat::solver s(rlim, ps, false); anum_manager & am = s.am(); nlsat::pmanager & pm = s.pm(); nlsat::assignment as(am); nlsat::explain& ex = s.get_explain(); // Create 14 variables x0-x13 nlsat::var x0 = s.mk_var(false); nlsat::var x1 = s.mk_var(false); nlsat::var x2 = s.mk_var(false); nlsat::var x3 = s.mk_var(false); nlsat::var x4 = s.mk_var(false); nlsat::var x5 = s.mk_var(false); nlsat::var x6 = s.mk_var(false); nlsat::var x7 = s.mk_var(false); nlsat::var x8 = s.mk_var(false); nlsat::var x9 = s.mk_var(false); nlsat::var x10 = s.mk_var(false); nlsat::var x11 = s.mk_var(false); nlsat::var x12 = s.mk_var(false); nlsat::var x13 = s.mk_var(false); polynomial_ref _x0(pm), _x1(pm), _x2(pm), _x3(pm), _x4(pm), _x5(pm), _x6(pm); polynomial_ref _x7(pm), _x8(pm), _x9(pm), _x10(pm), _x11(pm), _x12(pm), _x13(pm); _x0 = pm.mk_polynomial(x0); _x1 = pm.mk_polynomial(x1); _x2 = pm.mk_polynomial(x2); _x3 = pm.mk_polynomial(x3); _x4 = pm.mk_polynomial(x4); _x5 = pm.mk_polynomial(x5); _x6 = pm.mk_polynomial(x6); _x7 = pm.mk_polynomial(x7); _x8 = pm.mk_polynomial(x8); _x9 = pm.mk_polynomial(x9); _x10 = pm.mk_polynomial(x10); _x11 = pm.mk_polynomial(x11); _x12 = pm.mk_polynomial(x12); _x13 = pm.mk_polynomial(x13); // p[0]: x13 polynomial_ref p0(pm); p0 = _x13; // p[1]: 170*x8*x13 + x10*x11*x12 - x11*x12 - x7*x8*x12 + x5*x10*x11 + 184*x1*x10*x11 // - x0*x10*x11 - x5*x11 - 184*x1*x11 + x0*x11 - x3*x8*x10 + x2*x8*x10 // - 2*x10 - 184*x1*x7*x8 - x2*x8 + 2 polynomial_ref p1(pm); p1 = (170 * _x8 * _x13) + (_x10 * _x11 * _x12) - (_x11 * _x12) - (_x7 * _x8 * _x12) + (_x5 * _x10 * _x11) + (184 * _x1 * _x10 * _x11) - (_x0 * _x10 * _x11) - (_x5 * _x11) - (184 * _x1 * _x11) + (_x0 * _x11) - (_x3 * _x8 * _x10) + (_x2 * _x8 * _x10) - (2 * _x10) - (184 * _x1 * _x7 * _x8) - (_x2 * _x8) + 2; std::cout << "p0: " << p0 << "\n"; std::cout << "p1: " << p1 << "\n"; // Set sample: x0=-1, x1=-1, x2=0, x3=-1, x5=0, x7=0, x8=2, x10=0, x11=0, x12=2 scoped_anum val(am); am.set(val, -1); as.set(x0, val); am.set(val, -1); as.set(x1, val); am.set(val, 0); as.set(x2, val); am.set(val, -1); as.set(x3, val); am.set(val, 0); as.set(x4, val); am.set(val, 0); as.set(x5, val); am.set(val, 0); as.set(x6, val); am.set(val, 0); as.set(x7, val); am.set(val, 2); as.set(x8, val); am.set(val, 0); as.set(x9, val); am.set(val, 0); as.set(x10, val); am.set(val, 0); as.set(x11, val); am.set(val, 2); as.set(x12, val); am.set(val, 1); as.set(x13, val); s.set_rvalues(as); // Create literals for the two polynomials nlsat::scoped_literal_vector lits(s); lits.push_back(mk_gt(s, p0.get())); // x13 > 0 lits.push_back(mk_gt(s, p1.get())); // p1 > 0 project_fa(s, ex, x13, lits.size(), lits.data()); std::cout << "\n"; }; run_test(false); // Standard projection run_test(true); // Levelwise projection } void tst_nlsat() { tst_unsound_lws2380(); std::cout << "------------------\n"; tst_polynomial_cache_mk_unique(); std::cout << "------------------\n"; tst_nullified_polynomial(); std::cout << "------------------\n"; tst11(); std::cout << "------------------\n"; tst10(); std::cout << "------------------\n"; tst9(); std::cout << "------------------\n"; tst8(); std::cout << "------------------\n"; tst7(); std::cout << "------------------\n"; tst6(); std::cout << "------------------\n"; tst5(); std::cout << "------------------\n"; tst4(); std::cout << "------------------\n"; tst3(); }