3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-10-26 17:29:21 +00:00

refactor lws

Signed-off-by: Lev Nachmanson <levnach@hotmail.com>
This commit is contained in:
Lev Nachmanson 2025-08-15 20:14:30 -07:00
parent 599c426045
commit 8950b3d21d

View file

@ -246,38 +246,42 @@ namespace nlsat {
} }
} }
// Step 1b: bucketize roots by equality of values // Find an existing bucket that has the same algebraic value; returns nullptr if none found
void bucketize_roots(std::vector<root_item_t> const& roots, bucket_t* find_bucket_by_value(anum const& v,
std::vector<std::unique_ptr<bucket_t>>& buckets) { std::vector<std::unique_ptr<bucket_t>>& buckets) {
for (auto const& r : roots) { for (auto& b : buckets)
bool placed = false; if (m_am.compare(v, b->val) == 0)
for (auto& b : buckets) { return b.get();
if (m_am.compare(r.val, b->val) == 0) { return nullptr;
b->members.push_back(r.ire);
placed = true;
break;
}
}
if (!placed) {
auto nb = std::make_unique<bucket_t>(m_am);
nb->members.push_back(r.ire);
m_am.set(nb->val, r.val);
buckets.push_back(std::move(nb));
}
}
} }
// Step 2: finalize representation (I and omega buckets) from collected buckets // Append a root to a given bucket (does not change the bucket's value)
void finalize_representation_from_buckets(unsigned i, void add_root_to_bucket(bucket_t& bucket, root_item_t const& r) {
std::vector<std::unique_ptr<bucket_t>>& buckets, bucket.members.push_back(r.ire);
symbolic_interval& I, }
std::vector<std::vector<indexed_root_expr>>& omega_buckets) {
var y = i;
// default: whole line sector (-inf, +inf)
I.section = false;
I.l = nullptr; I.u = nullptr; I.l_index = 0; I.u_index = 0;
// Sort buckets by numeric value and form omega_buckets // Step 1b: bucketize roots by equality of values
void add_root_to_buckets(root_item_t const& r,
std::vector<std::unique_ptr<bucket_t>>& buckets) {
if (auto* b = find_bucket_by_value(r.val, buckets)) {
add_root_to_bucket(*b, r);
return;
}
auto nb = std::make_unique<bucket_t>(m_am);
m_am.set(nb->val, r.val);
add_root_to_bucket(*nb, r);
buckets.push_back(std::move(nb));
}
void bucketize_roots(std::vector<root_item_t> const& roots,
std::vector<std::unique_ptr<bucket_t>>& buckets) {
for (auto const& r : roots)
add_root_to_buckets(r, buckets);
}
// Step 2a: sort buckets and form omega_buckets
void build_omega_buckets(std::vector<std::unique_ptr<bucket_t>>& buckets,
std::vector<std::vector<indexed_root_expr>>& omega_buckets) {
std::sort(buckets.begin(), buckets.end(), [&](std::unique_ptr<bucket_t> const& a, std::unique_ptr<bucket_t> const& b){ std::sort(buckets.begin(), buckets.end(), [&](std::unique_ptr<bucket_t> const& a, std::unique_ptr<bucket_t> const& b){
return m_am.lt(a->val, b->val); return m_am.lt(a->val, b->val);
}); });
@ -285,35 +289,43 @@ namespace nlsat {
omega_buckets.reserve(buckets.size()); omega_buckets.reserve(buckets.size());
for (auto& b : buckets) for (auto& b : buckets)
omega_buckets.push_back(b->members); omega_buckets.push_back(b->members);
}
// Helper: set I as a section if sample matches a bucket value; returns true if set
bool initialize_section_from_bucket(unsigned i,
std::vector<std::unique_ptr<bucket_t>>& buckets,
symbolic_interval& I) {
var y = i;
if (!sample().is_assigned(y)) if (!sample().is_assigned(y))
return; return false;
anum const& y_val = sample().value(y); anum const& y_val = sample().value(y);
// Check for section first
for (auto const& b : buckets) { for (auto const& b : buckets) {
if (m_am.compare(y_val, b->val) == 0) { if (m_am.compare(y_val, b->val) == 0) {
I.section = true; I.section = true;
// pick a representative indexed root expression from the bucket
auto const& ire = b->members.front(); auto const& ire = b->members.front();
I.l = ire.p; I.l = ire.p;
I.l_index = ire.i; I.l_index = ire.i;
I.u = nullptr; I.u_index = 0; I.u = nullptr; I.u_index = 0;
return; return true;
} }
} }
// Otherwise compute nearest lower/upper buckets return false;
}
// Helper: set sector bounds from neighboring buckets; assumes buckets sorted; no-op if sample unassigned
void set_sector_bounds_from_buckets(unsigned i,
std::vector<std::unique_ptr<bucket_t>>& buckets,
symbolic_interval& I) {
var y = i;
if (!sample().is_assigned(y))
return;
anum const& y_val = sample().value(y);
bucket_t* lower_b = nullptr; bucket_t* lower_b = nullptr;
bucket_t* upper_b = nullptr; bucket_t* upper_b = nullptr;
for (auto& b : buckets) { for (auto& b : buckets) {
int cmp = m_am.compare(y_val, b->val); int cmp = m_am.compare(y_val, b->val);
if (cmp > 0) { if (cmp > 0) lower_b = b.get();
lower_b = b.get(); else if (cmp < 0) { upper_b = b.get(); break; }
}
else if (cmp < 0) {
upper_b = b.get();
break;
}
} }
if (lower_b) { if (lower_b) {
auto const& ire = lower_b->members.front(); auto const& ire = lower_b->members.front();
@ -327,40 +339,54 @@ namespace nlsat {
} }
} }
result_struct construct_interval(unsigned i) { // Step 2b: compute interval I from (sorted) buckets and current sample
result_struct ret; void compute_interval_from_buckets(unsigned i,
std::vector<std::unique_ptr<bucket_t>>& buckets,
symbolic_interval& I) {
// default: whole line sector (-inf, +inf)
I.section = false;
I.l = nullptr; I.u = nullptr; I.l_index = 0; I.u_index = 0;
if (initialize_section_from_bucket(i, buckets, I))
return;
set_sector_bounds_from_buckets(i, buckets, I);
}
// Part A of construct_interval: apply pre-conditions (line 8-11 scaffolding)
bool apply_property_rules(unsigned i) {
std::vector<property> to_refine = greatest_to_refine(i); std::vector<property> to_refine = greatest_to_refine(i);
for (const auto& p: to_refine) { for (const auto& p: to_refine)
apply_pre(p); apply_pre(p);
} return !m_fail;
if (m_fail) { }
ret.fail = true;
return ret;
}
/*
P_{nonnull} := {p | sgn_inv(p) Q |i s.t. p(s[i1], xi ) ̸= 0}
6 E:= irExpr(P_{nonnull} , s[i1])
7 choose representation (I, E, ) of with respect to s
8 if i > 1 then
9 foreach q Q |i where q is the greatest element with respect to (from Definition 4.5) and q ̸= holds(I) do
10 Q := apply_pre(i, Q ,q,(s, I, ))
11 if Q == FAIL return FAIL
*/
// Part B of construct_interval: build (I, E, ≼) representation for level i
void build_representation(unsigned i, result_struct& ret) {
std::vector<const poly*> p_non_null; std::vector<const poly*> p_non_null;
for (const auto & pr: m_Q) { for (const auto & pr: m_Q) {
if (pr.prop == prop_enum::sgn_inv_irreducible && m_pm.max_var(pr.p) == i && poly_is_not_nullified_at_sample_at_level(pr.p, i)) if (pr.prop == prop_enum::sgn_inv_irreducible && m_pm.max_var(pr.p) == i && poly_is_not_nullified_at_sample_at_level(pr.p, i))
p_non_null.push_back(pr.p); p_non_null.push_back(pr.p);
} }
// Line 7: choose representation (I, E, ≼) with respect to s.
std::vector<std::unique_ptr<bucket_t>> buckets; std::vector<std::unique_ptr<bucket_t>> buckets;
std::vector<root_item_t> roots; std::vector<root_item_t> roots;
collect_E_and_roots(p_non_null, i, ret.E, roots); collect_E_and_roots(p_non_null, i, ret.E, roots);
bucketize_roots(roots, buckets); bucketize_roots(roots, buckets);
finalize_representation_from_buckets(i, buckets, ret.I, ret.omega_buckets); build_omega_buckets(buckets, ret.omega_buckets);
compute_interval_from_buckets(i, buckets, ret.I);
}
result_struct construct_interval(unsigned i) {
result_struct ret;
ret.fail = false;
if (!apply_property_rules(i)) {
ret.fail = true;
return ret;
}
build_representation(i, ret);
// Keep Q unchanged for now until apply_pre is implemented
ret.Q = m_Q;
return ret; return ret;
} }
// overload exists in explain; keep local poly*-based API only for now // overload exists in explain; keep local poly*-based API only for now