3
0
Fork 0
mirror of https://github.com/Z3Prover/z3 synced 2025-11-09 23:52:02 +00:00

Add finite_set_rewriter implementation with basic rewrite rules

Co-authored-by: NikolajBjorner <3085284+NikolajBjorner@users.noreply.github.com>
This commit is contained in:
copilot-swe-agent[bot] 2025-10-14 08:21:46 +00:00
parent fe94b7d486
commit 3e495b1e80
6 changed files with 280 additions and 3 deletions

View file

@ -55,6 +55,7 @@ add_executable(test-z3
ext_numeral.cpp
f2n.cpp
finite_set.cpp
finite_set_rewriter.cpp
factor_rewriter.cpp
finder.cpp
fixed_bit_vector.cpp

View file

@ -0,0 +1,157 @@
/*++
Copyright (c) 2025 Microsoft Corporation
Module Name:
finite_set_rewriter.cpp
Abstract:
Test finite set rewriter
Author:
GitHub Copilot Agent 2025
--*/
#include "ast/ast.h"
#include "ast/finite_set_decl_plugin.h"
#include "ast/reg_decl_plugins.h"
#include "ast/arith_decl_plugin.h"
#include "ast/rewriter/finite_set_rewriter.h"
static void test_union_idempotent() {
ast_manager m;
reg_decl_plugins(m);
finite_set_util fsets(m);
finite_set_rewriter rw(m);
arith_util arith(m);
// Create a set
sort_ref int_sort(arith.mk_int(), m);
expr_ref zero(arith.mk_int(0), m);
expr_ref ten(arith.mk_int(10), m);
app_ref s1(fsets.mk_range(zero, ten), m);
// Test set.union(s1, s1) -> s1
expr* args[2] = { s1, s1 };
expr_ref result(m);
br_status st = rw.mk_union(2, args, result);
ENSURE(st == BR_DONE);
ENSURE(result == s1);
}
static void test_intersect_idempotent() {
ast_manager m;
reg_decl_plugins(m);
finite_set_util fsets(m);
finite_set_rewriter rw(m);
arith_util arith(m);
// Create a set
sort_ref int_sort(arith.mk_int(), m);
expr_ref zero(arith.mk_int(0), m);
expr_ref ten(arith.mk_int(10), m);
app_ref s1(fsets.mk_range(zero, ten), m);
// Test set.intersect(s1, s1) -> s1
expr* args[2] = { s1, s1 };
expr_ref result(m);
br_status st = rw.mk_intersect(2, args, result);
ENSURE(st == BR_DONE);
ENSURE(result == s1);
}
static void test_difference_same() {
ast_manager m;
reg_decl_plugins(m);
finite_set_util fsets(m);
finite_set_rewriter rw(m);
arith_util arith(m);
// Create a set
sort_ref int_sort(arith.mk_int(), m);
expr_ref zero(arith.mk_int(0), m);
expr_ref ten(arith.mk_int(10), m);
app_ref s1(fsets.mk_range(zero, ten), m);
// Test set.difference(s1, s1) -> empty
expr_ref result(m);
br_status st = rw.mk_difference(s1, s1, result);
ENSURE(st == BR_DONE);
ENSURE(fsets.is_empty(result));
}
static void test_subset_rewrite() {
ast_manager m;
reg_decl_plugins(m);
finite_set_util fsets(m);
finite_set_rewriter rw(m);
arith_util arith(m);
// Create two sets
sort_ref int_sort(arith.mk_int(), m);
expr_ref zero(arith.mk_int(0), m);
expr_ref ten(arith.mk_int(10), m);
expr_ref twenty(arith.mk_int(20), m);
app_ref s1(fsets.mk_range(zero, ten), m);
app_ref s2(fsets.mk_range(zero, twenty), m);
// Test set.subset(s1, s2) -> set.intersect(s1, s2) = s1
expr_ref result(m);
br_status st = rw.mk_subset(s1, s2, result);
ENSURE(st == BR_REWRITE3);
ENSURE(m.is_eq(result));
// Check that result is an equality
app* eq = to_app(result);
ENSURE(eq->get_num_args() == 2);
// The left side should be set.intersect(s1, s2)
expr* lhs = eq->get_arg(0);
ENSURE(fsets.is_intersect(lhs));
// The right side should be s1
expr* rhs = eq->get_arg(1);
ENSURE(rhs == s1);
}
static void test_mk_app_core() {
ast_manager m;
reg_decl_plugins(m);
finite_set_util fsets(m);
finite_set_rewriter rw(m);
arith_util arith(m);
// Create sets
sort_ref int_sort(arith.mk_int(), m);
expr_ref zero(arith.mk_int(0), m);
expr_ref ten(arith.mk_int(10), m);
app_ref s1(fsets.mk_range(zero, ten), m);
// Test union through mk_app_core
app_ref union_app(fsets.mk_union(s1, s1), m);
expr_ref result(m);
br_status st = rw.mk_app_core(union_app->get_decl(), union_app->get_num_args(), union_app->get_args(), result);
ENSURE(st == BR_DONE);
ENSURE(result == s1);
}
void tst_finite_set_rewriter() {
test_union_idempotent();
test_intersect_idempotent();
test_difference_same();
test_subset_rewrite();
test_mk_app_core();
}

View file

@ -283,4 +283,5 @@ int main(int argc, char ** argv) {
TST(sls_seq_plugin);
TST(ho_matcher);
TST(finite_set);
TST(finite_set_rewriter);
}