mirror of
https://github.com/Z3Prover/z3
synced 2026-06-13 20:35:39 +00:00
Fix unsound array equality rewrite for const-array store chains (#9572)
Z3 could return `sat` for an unsatisfiable QF_ABV formula equating two
store chains over distinct constant arrays. The rewrite path for array
equalities was missing a necessary base-value constraint in
finite-domain cases where stores cannot cover all indices.
- **Root cause**
- In `array_rewriter::mk_eq_core`, equality rewriting for nested stores
over const-array bases did not enforce equality of the underlying const
values when the index domain size exceeds the number of updated indices.
- **Rewriter fix**
- Added a sound rewrite branch for:
- `store* ((as const ...) v)` vs `store* ((as const ...) w)`
- When `|domain| > (#stores_lhs + #stores_rhs)`, rewrite now includes:
- select equalities for touched indices (existing behavior)
- **and** base-value equality `v = w` (new requirement)
- This prevents spurious models where only updated indices are
constrained.
- **Regression coverage**
- Added a focused regression in `src/test/mod_factor.cpp` that asserts
`unsat` for a minimized constant-array/store-chain BV case with
`(distinct x y)` and one store per side.
```cpp
(assert (distinct x y))
(assert (= (store A0 i0 e0) (store A1 i1 e1)))
(check-sat) ; expected: unsat
```
---------
Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com>
Co-authored-by: NikolajBjorner <3085284+NikolajBjorner@users.noreply.github.com>
This commit is contained in:
parent
af33dfaa7d
commit
34ba2962ef
2 changed files with 62 additions and 11 deletions
|
|
@ -4,6 +4,7 @@ Copyright (c) 2025 Microsoft Corporation
|
|||
|
||||
#include "api/z3.h"
|
||||
#include "util/util.h"
|
||||
#include <string>
|
||||
|
||||
// x mod 7 = 0 & (x*y) mod 7 != 0 should be unsat
|
||||
// Exercises: mod internalization path (is_mod with numeric divisor)
|
||||
|
|
@ -60,7 +61,31 @@ static void test_mod_factor_idiv_path() {
|
|||
Z3_del_context(ctx);
|
||||
}
|
||||
|
||||
static void test_const_array_store_chain_unsat() {
|
||||
Z3_config cfg = Z3_mk_config();
|
||||
Z3_context ctx = Z3_mk_context(cfg);
|
||||
const char* script = R"(
|
||||
(set-logic QF_ABV)
|
||||
(declare-const x (_ BitVec 8))
|
||||
(declare-const y (_ BitVec 8))
|
||||
(define-fun A0 () (Array (_ BitVec 2) (_ BitVec 8)) ((as const (Array (_ BitVec 2) (_ BitVec 8))) x))
|
||||
(define-fun A1 () (Array (_ BitVec 2) (_ BitVec 8)) ((as const (Array (_ BitVec 2) (_ BitVec 8))) y))
|
||||
(declare-const i0 (_ BitVec 2))
|
||||
(declare-const e0 (_ BitVec 8))
|
||||
(declare-const i1 (_ BitVec 2))
|
||||
(declare-const e1 (_ BitVec 8))
|
||||
(assert (distinct x y))
|
||||
(assert (= (store A0 i0 e0) (store A1 i1 e1)))
|
||||
(check-sat)
|
||||
)";
|
||||
std::string resp = Z3_eval_smtlib2_string(ctx, script);
|
||||
ENSURE(resp.find("unsat") != std::string::npos);
|
||||
Z3_del_config(cfg);
|
||||
Z3_del_context(ctx);
|
||||
}
|
||||
|
||||
void tst_mod_factor() {
|
||||
test_mod_factor_mod_path();
|
||||
test_mod_factor_idiv_path();
|
||||
test_const_array_store_chain_unsat();
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue