refactor(library/blast): we should reuse uref/mref/href in idxs

Motivation: avoid nasty bugs when caching results
This commit is contained in:
Leonardo de Moura 2015-11-09 12:41:52 -08:00
parent d952a1d299
commit 9bdf3a0d33
7 changed files with 52 additions and 20 deletions

View file

@ -191,7 +191,7 @@ class blastenv {
} }
virtual level mk_uvar() { virtual level mk_uvar() {
return m_benv.m_curr_state.mk_uref(); return mk_fresh_uref();
} }
virtual expr mk_mvar(expr const & type) { virtual expr mk_mvar(expr const & type) {
@ -238,7 +238,7 @@ class blastenv {
if (auto r = m_uvar2uref.find(meta_id(l))) { if (auto r = m_uvar2uref.find(meta_id(l))) {
return *r; return *r;
} else { } else {
level uref = m_state.mk_uref(); level uref = mk_fresh_uref();
m_uvar2uref.insert(meta_id(l), uref); m_uvar2uref.insert(meta_id(l), uref);
return uref; return uref;
} }
@ -547,6 +547,7 @@ public:
m_fun_info_manager(*m_tmp_ctx), m_fun_info_manager(*m_tmp_ctx),
m_congr_lemma_manager(m_app_builder, m_fun_info_manager), m_congr_lemma_manager(m_app_builder, m_fun_info_manager),
m_tctx(*this) { m_tctx(*this) {
init_uref_mref_href_idxs();
set_options(m_ios.get_options()); set_options(m_ios.get_options());
} }

View file

@ -80,13 +80,11 @@ double branch::compute_weight(unsigned hidx, expr const & /* type */) {
return 1.0 / (static_cast<double>(hidx) + 1.0); return 1.0 / (static_cast<double>(hidx) + 1.0);
} }
expr branch::add_hypothesis(name const & n, expr const & type, expr const & value) { expr branch::add_hypothesis(unsigned new_hidx, name const & n, expr const & type, expr const & value) {
hypothesis new_h; hypothesis new_h;
new_h.m_name = n; new_h.m_name = n;
new_h.m_type = type; new_h.m_type = type;
new_h.m_value = value; new_h.m_value = value;
unsigned new_hidx = m_next;
m_next++;
add_deps(new_h, new_hidx); add_deps(new_h, new_hidx);
m_context.insert(new_hidx, new_h); m_context.insert(new_hidx, new_h);
if (new_h.is_assumption()) if (new_h.is_assumption())
@ -98,8 +96,13 @@ expr branch::add_hypothesis(name const & n, expr const & type, expr const & valu
static name * g_prefix = nullptr; static name * g_prefix = nullptr;
expr branch::add_hypothesis(name const & n, expr const & type, expr const & value) {
return add_hypothesis(mk_href_idx(), n, type, value);
}
expr branch::add_hypothesis(expr const & type, expr const & value) { expr branch::add_hypothesis(expr const & type, expr const & value) {
return add_hypothesis(name(*g_prefix, m_next), type, value); unsigned hidx = mk_href_idx();
return add_hypothesis(hidx, name(*g_prefix, hidx), type, value);
} }
void branch::update_indices(unsigned /* hidx */) { void branch::update_indices(unsigned /* hidx */) {

View file

@ -21,7 +21,6 @@ class branch {
typedef hypothesis_idx_map<hypothesis_idx_set> forward_deps; typedef hypothesis_idx_map<hypothesis_idx_set> forward_deps;
typedef rb_map<double, unsigned, double_cmp> todo_queue; typedef rb_map<double, unsigned, double_cmp> todo_queue;
friend class state; friend class state;
unsigned m_next{0};
context m_context; context m_context;
// We break the set of hypotheses in m_context in 3 sets that are not necessarily disjoint: // We break the set of hypotheses in m_context in 3 sets that are not necessarily disjoint:
// - assumption // - assumption
@ -59,6 +58,8 @@ class branch {
We will update indices and data-structures (e.g., congruence closure). */ We will update indices and data-structures (e.g., congruence closure). */
void update_indices(unsigned hidx); void update_indices(unsigned hidx);
expr add_hypothesis(unsigned new_hidx, name const & n, expr const & type, expr const & value);
public: public:
branch() {} branch() {}

View file

@ -65,6 +65,34 @@ bool has_mref(expr const & e) {
return lean::has_expr_metavar(e); return lean::has_expr_metavar(e);
} }
LEAN_THREAD_VALUE(unsigned, m_next_uref_idx, 0);
LEAN_THREAD_VALUE(unsigned, m_next_mref_idx, 0);
LEAN_THREAD_VALUE(unsigned, m_next_href_idx, 0);
unsigned mk_uref_idx() {
unsigned r = m_next_uref_idx;
m_next_uref_idx++;
return r;
}
unsigned mk_mref_idx() {
unsigned r = m_next_mref_idx;
m_next_mref_idx++;
return r;
}
unsigned mk_href_idx() {
unsigned r = m_next_href_idx;
m_next_href_idx++;
return r;
}
void init_uref_mref_href_idxs() {
m_next_uref_idx = 0;
m_next_mref_idx = 0;
m_next_href_idx = 0;
}
void initialize_expr() { void initialize_expr() {
g_prefix = new name(name::mk_internal_unique_name()); g_prefix = new name(name::mk_internal_unique_name());
g_dummy_type = new expr(mk_constant(*g_prefix)); g_dummy_type = new expr(mk_constant(*g_prefix));

View file

@ -31,6 +31,16 @@ bool has_mref(expr const & e);
inline bool is_local_non_href(expr const & e) { return is_local(e) && !is_href(e); } inline bool is_local_non_href(expr const & e) { return is_local(e) && !is_href(e); }
/** \brief Return the a fresh index for uref/mref/href.
\remark It is implemented using thread local storage. */
unsigned mk_uref_idx();
unsigned mk_mref_idx();
unsigned mk_href_idx();
/** \brief Reset thread local counters */
void init_uref_mref_href_idxs();
inline level mk_fresh_uref() { return mk_uref(mk_uref_idx()); }
void initialize_expr(); void initialize_expr();
void finalize_expr(); void finalize_expr();
} }

View file

@ -24,7 +24,7 @@ bool metavar_decl::restrict_context_using(metavar_decl const & other) {
return !to_erase.empty(); return !to_erase.empty();
} }
state::state():m_next_uref_index(0), m_next_mref_index(0) {} state::state() {}
/** \brief Mark that hypothesis h with index hidx is fixed by the meta-variable midx. /** \brief Mark that hypothesis h with index hidx is fixed by the meta-variable midx.
That is, `h` occurs in the type of `midx`. */ That is, `h` occurs in the type of `midx`. */
@ -42,14 +42,8 @@ void state::add_fixed_by(unsigned hidx, unsigned midx) {
} }
} }
level state::mk_uref() {
unsigned idx = m_next_mref_index;
m_next_mref_index++;
return blast::mk_uref(idx);
}
expr state::mk_metavar(hypothesis_idx_set const & c, expr const & type) { expr state::mk_metavar(hypothesis_idx_set const & c, expr const & type) {
unsigned midx = m_next_mref_index; unsigned midx = mk_mref_idx();
for_each(type, [&](expr const & e, unsigned) { for_each(type, [&](expr const & e, unsigned) {
if (!has_href(e)) if (!has_href(e))
return false; return false;
@ -60,7 +54,6 @@ expr state::mk_metavar(hypothesis_idx_set const & c, expr const & type) {
} }
return true; // continue search return true; // continue search
}); });
m_next_mref_index++;
m_metavar_decls.insert(midx, metavar_decl(c, type)); m_metavar_decls.insert(midx, metavar_decl(c, type));
return blast::mk_mref(midx); return blast::mk_mref(midx);
} }

View file

@ -68,9 +68,7 @@ class state {
typedef metavar_idx_map<level> uassignment; typedef metavar_idx_map<level> uassignment;
typedef hypothesis_idx_map<metavar_idx_set> fixed_by; typedef hypothesis_idx_map<metavar_idx_set> fixed_by;
typedef list<proof_step> proof_steps; typedef list<proof_step> proof_steps;
unsigned m_next_uref_index; // index of the next universe metavariable
uassignment m_uassignment; uassignment m_uassignment;
unsigned m_next_mref_index; // index of the next metavariable
metavar_decls m_metavar_decls; metavar_decls m_metavar_decls;
eassignment m_eassignment; eassignment m_eassignment;
branch m_main; branch m_main;
@ -99,8 +97,6 @@ class state {
public: public:
state(); state();
level mk_uref();
bool is_uref_assigned(level const & l) const { bool is_uref_assigned(level const & l) const {
lean_assert(is_uref(l)); lean_assert(is_uref(l));
return m_uassignment.contains(uref_index(l)); return m_uassignment.contains(uref_index(l));