mirror of
https://github.com/Z3Prover/z3
synced 2025-11-23 22:21:28 +00:00
* Introduce X-macro-based trace tag definition - Created trace_tags.def to centralize TRACE tag definitions - Each tag includes a symbolic name and description - Set up enum class TraceTag for type-safe usage in TRACE macros * Add script to generate Markdown documentation from trace_tags.def - Python script parses trace_tags.def and outputs trace_tags.md * Refactor TRACE_NEW to prepend TraceTag and pass enum to is_trace_enabled * trace: improve trace tag handling system with hierarchical tagging - Introduce hierarchical tag-class structure: enabling a tag class activates all child tags - Unify TRACE, STRACE, SCTRACE, and CTRACE under enum TraceTag - Implement initial version of trace_tag.def using X(tag, tag_class, description) (class names and descriptions to be refined in a future update) * trace: replace all string-based TRACE tags with enum TraceTag - Migrated all TRACE, STRACE, SCTRACE, and CTRACE macros to use enum TraceTag values instead of raw string literals * trace : add cstring header * trace : Add Markdown documentation generation from trace_tags.def via mk_api_doc.py * trace : rename macro parameter 'class' to 'tag_class' and remove Unicode comment in trace_tags.h. * trace : Add TODO comment for future implementation of tag_class activation * trace : Disable code related to tag_class until implementation is ready (#7663).
143 lines
3.8 KiB
C++
143 lines
3.8 KiB
C++
/*++
|
|
Copyright (c) 2012 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
unit_subsumption_tactic.cpp
|
|
|
|
Abstract:
|
|
|
|
Simplify goal using subsumption based on unit propagation.
|
|
|
|
Author:
|
|
|
|
Nikolaj Bjorner (nbjorner) 2012-9-6
|
|
|
|
--*/
|
|
|
|
#include "ast/ast_util.h"
|
|
#include "smt/tactic/unit_subsumption_tactic.h"
|
|
#include "smt/smt_context.h"
|
|
|
|
struct unit_subsumption_tactic : public tactic {
|
|
ast_manager& m;
|
|
params_ref m_params;
|
|
smt_params m_fparams;
|
|
smt::context m_context;
|
|
expr_ref_vector m_clauses;
|
|
unsigned m_clause_count;
|
|
bit_vector m_is_deleted;
|
|
unsigned_vector m_deleted;
|
|
|
|
unit_subsumption_tactic(
|
|
ast_manager& m,
|
|
params_ref const& p):
|
|
m(m),
|
|
m_params(p),
|
|
m_context(m, m_fparams, p),
|
|
m_clauses(m) {
|
|
}
|
|
|
|
char const* name() const override { return "unit_subsumption"; }
|
|
|
|
void cleanup() override {}
|
|
|
|
void operator()(/* in */ goal_ref const & in,
|
|
/* out */ goal_ref_buffer & result) override {
|
|
tactic_report report("unit-subsume-simplify", *in);
|
|
fail_if_proof_generation("unit-subsume-simplify", in);
|
|
reduce_core(in, result);
|
|
}
|
|
|
|
void updt_params(params_ref const& p) override {
|
|
m_params.append(p);
|
|
// m_context.updt_params(p); does not exist.
|
|
}
|
|
|
|
tactic* translate(ast_manager& m) override {
|
|
return alloc(unit_subsumption_tactic, m, m_params);
|
|
}
|
|
|
|
void checkpoint() {
|
|
tactic::checkpoint(m);
|
|
}
|
|
|
|
void reduce_core(goal_ref const& g, goal_ref_buffer& result) {
|
|
init(g);
|
|
m_context.push();
|
|
assert_clauses(g);
|
|
m_context.push(); // internalize assertions.
|
|
prune_clauses();
|
|
goal_ref r(g);
|
|
insert_result(r);
|
|
r->elim_true();
|
|
result.push_back(r.get());
|
|
m_context.pop(2);
|
|
TRACE(unit_subsumption_tactic, g->display(tout); r->display(tout););
|
|
}
|
|
|
|
void assert_clauses(goal_ref const& g) {
|
|
for (unsigned i = 0; i < g->size(); ++i) {
|
|
expr_ref fml(m.mk_iff(new_clause(), g->form(i)), m);
|
|
m_context.assert_expr(fml);
|
|
}
|
|
}
|
|
|
|
void prune_clauses() {
|
|
for (unsigned i = 0; i < m_clause_count; ++i) {
|
|
prune_clause(i);
|
|
}
|
|
}
|
|
|
|
void prune_clause(unsigned i) {
|
|
m_context.push();
|
|
for (unsigned j = 0; j < m_clause_count; ++j) {
|
|
if (i == j) {
|
|
expr_ref fml(mk_not(m, m_clauses.get(j)), m);
|
|
m_context.assert_expr(fml);
|
|
}
|
|
else if (!m_is_deleted.get(j)) {
|
|
m_context.assert_expr(m_clauses.get(j));
|
|
}
|
|
}
|
|
m_context.push(); // force propagation
|
|
bool is_unsat = m_context.inconsistent();
|
|
m_context.pop(2);
|
|
if (is_unsat) {
|
|
TRACE(unit_subsumption_tactic, tout << "Removing clause " << i << "\n";);
|
|
m_is_deleted.set(i, true);
|
|
m_deleted.push_back(i);
|
|
}
|
|
}
|
|
|
|
void insert_result(goal_ref& result) {
|
|
for (auto d : m_deleted) result->update(d, m.mk_true());
|
|
}
|
|
|
|
void init(goal_ref const& g) {
|
|
m_clause_count = 0;
|
|
m_is_deleted.reset();
|
|
m_is_deleted.resize(g->size());
|
|
m_deleted.reset();
|
|
|
|
}
|
|
|
|
expr* new_bool(unsigned& count, expr_ref_vector& v, char const* name) {
|
|
SASSERT(count <= v.size());
|
|
if (count == v.size()) {
|
|
v.push_back(m.mk_fresh_const(name, m.mk_bool_sort()));
|
|
}
|
|
return v[count++].get();
|
|
}
|
|
|
|
expr* new_clause() {
|
|
return new_bool(m_clause_count, m_clauses, "#clause");
|
|
}
|
|
|
|
|
|
};
|
|
|
|
tactic * mk_unit_subsumption_tactic(ast_manager & m, params_ref const & p) {
|
|
return alloc(unit_subsumption_tactic, m, p);
|
|
}
|
|
|