mirror of
https://github.com/Z3Prover/z3
synced 2026-06-19 23:26:30 +00:00
Spacer can crash on small HORN/ADT benchmarks when model construction
reaches datatype enodes without a fully populated constructor state. The
failure manifested as a null/invalid-path dereference inside datatype
model value generation.
- **Root cause area: datatype model extraction path**
- Hardened `theory_datatype::mk_value` to handle incomplete theory state
safely instead of assuming constructor metadata is always present.
- Added guarded fallback to a factory-provided datatype value when:
- `th_var` is missing,
- union-find lookup is invalid,
- var data/constructor is unavailable.
- **Behavioral change**
- Missing constructor state now degrades to a safe model value
(`expr_wrapper_proc`) instead of crashing during model generation.
- **Regression coverage**
- Added a focused API regression in `src/test/api_datalog.cpp` using a
Spacer + ADT HORN script (with reproducing seed) to ensure the code path
executes without parser/runtime failure.
```cpp
// theory_datatype::mk_value fallback shape
if (v == null_theory_var || invalid_var_data || d->m_constructor == nullptr) {
app* val = to_app(m_factory->get_some_value(n->get_sort()));
return alloc(expr_wrapper_proc, val);
}
```
---------
Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com>
Co-authored-by: NikolajBjorner <3085284+NikolajBjorner@users.noreply.github.com>
94 lines
2.7 KiB
C++
94 lines
2.7 KiB
C++
/*++
|
|
Copyright (c) 2025 Daily Test Coverage Improver
|
|
|
|
Module Name:
|
|
|
|
api_datalog.cpp
|
|
|
|
Abstract:
|
|
|
|
Test API datalog/fixedpoint functions
|
|
|
|
Author:
|
|
|
|
Daily Test Coverage Improver 2025-09-17
|
|
|
|
Notes:
|
|
|
|
--*/
|
|
#include "api/z3.h"
|
|
#include "util/trace.h"
|
|
#include "util/debug.h"
|
|
|
|
void tst_api_datalog() {
|
|
Z3_config cfg = Z3_mk_config();
|
|
Z3_context ctx = Z3_mk_context(cfg);
|
|
Z3_del_config(cfg);
|
|
|
|
// Test 1: Z3_mk_finite_domain_sort and size functions
|
|
{
|
|
Z3_symbol name = Z3_mk_string_symbol(ctx, "Domain");
|
|
Z3_sort finite_sort = Z3_mk_finite_domain_sort(ctx, name, 5);
|
|
ENSURE(finite_sort != nullptr);
|
|
|
|
uint64_t size;
|
|
bool success = Z3_get_finite_domain_sort_size(ctx, finite_sort, &size);
|
|
ENSURE(success);
|
|
ENSURE(size == 5);
|
|
|
|
// Test with non-finite domain sort (should fail)
|
|
Z3_sort int_sort = Z3_mk_int_sort(ctx);
|
|
uint64_t wrong_size;
|
|
bool wrong_success = Z3_get_finite_domain_sort_size(ctx, int_sort, &wrong_size);
|
|
ENSURE(!wrong_success);
|
|
}
|
|
|
|
// Test 2: Z3_mk_fixedpoint basic operations
|
|
{
|
|
Z3_fixedpoint fp = Z3_mk_fixedpoint(ctx);
|
|
ENSURE(fp != nullptr);
|
|
|
|
Z3_fixedpoint_inc_ref(ctx, fp);
|
|
// Test string conversion (empty fixedpoint)
|
|
Z3_string fp_str = Z3_fixedpoint_to_string(ctx, fp, 0, nullptr);
|
|
ENSURE(fp_str != nullptr);
|
|
|
|
// Test statistics
|
|
Z3_stats stats = Z3_fixedpoint_get_statistics(ctx, fp);
|
|
ENSURE(stats != nullptr);
|
|
|
|
// Test reason unknown
|
|
Z3_string reason = Z3_fixedpoint_get_reason_unknown(ctx, fp);
|
|
(void)reason; // May be null
|
|
|
|
Z3_fixedpoint_dec_ref(ctx, fp);
|
|
}
|
|
|
|
// Regression test for Spacer model construction on ADT CHCs
|
|
{
|
|
char const* chc =
|
|
"(set-logic HORN)\n"
|
|
"(set-option :fp.engine spacer)\n"
|
|
"(set-option :fp.spacer.random_seed 51)\n"
|
|
"(set-option :timeout 2000)\n"
|
|
"(declare-datatypes ((L 0)) (((cons (hd Int) (tl L)) (nil))))\n"
|
|
"(declare-fun reva (L L L) Bool)\n"
|
|
"(assert (forall ((a L)) (reva nil a a)))\n"
|
|
"(assert (forall ((x L) (acc L) (r L) (h Int))\n"
|
|
" (=> (reva x (cons h acc) r)\n"
|
|
" (reva (cons h x) acc r))))\n"
|
|
"(assert (forall ((B L) (C L) (D L) (E L) (F L))\n"
|
|
" (=> (and (reva B C D)\n"
|
|
" (reva D nil E)\n"
|
|
" (reva C B F)\n"
|
|
" (not (= E F)))\n"
|
|
" false)))\n"
|
|
"(check-sat)\n";
|
|
|
|
Z3_string response = Z3_eval_smtlib2_string(ctx, chc);
|
|
ENSURE(response != nullptr);
|
|
ENSURE(Z3_get_error_code(ctx) == Z3_OK);
|
|
}
|
|
|
|
Z3_del_context(ctx);
|
|
}
|