feat(kernel/expr): add O(1) predicates has_expr_metavar and has_univ_metavar
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
e366aadad0
commit
fc4df6a430
2 changed files with 33 additions and 20 deletions
|
@ -36,10 +36,11 @@ unsigned hash_levels(levels const & ls) {
|
||||||
|
|
||||||
MK_THREAD_LOCAL_GET(unsigned, get_hash_alloc_counter, 0)
|
MK_THREAD_LOCAL_GET(unsigned, get_hash_alloc_counter, 0)
|
||||||
|
|
||||||
expr_cell::expr_cell(expr_kind k, unsigned h, bool has_mv, bool has_local, bool has_param_univ):
|
expr_cell::expr_cell(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv, bool has_local, bool has_param_univ):
|
||||||
m_flags(0),
|
m_flags(0),
|
||||||
m_kind(static_cast<unsigned>(k)),
|
m_kind(static_cast<unsigned>(k)),
|
||||||
m_has_mv(has_mv),
|
m_has_expr_mv(has_expr_mv),
|
||||||
|
m_has_univ_mv(has_univ_mv),
|
||||||
m_has_local(has_local),
|
m_has_local(has_local),
|
||||||
m_has_param_univ(has_param_univ),
|
m_has_param_univ(has_param_univ),
|
||||||
m_hash(h),
|
m_hash(h),
|
||||||
|
@ -93,7 +94,7 @@ bool is_meta(expr const & e) {
|
||||||
|
|
||||||
// Expr variables
|
// Expr variables
|
||||||
expr_var::expr_var(unsigned idx):
|
expr_var::expr_var(unsigned idx):
|
||||||
expr_cell(expr_kind::Var, idx, false, false, false),
|
expr_cell(expr_kind::Var, idx, false, false, false, false),
|
||||||
m_vidx(idx) {
|
m_vidx(idx) {
|
||||||
if (idx == std::numeric_limits<unsigned>::max())
|
if (idx == std::numeric_limits<unsigned>::max())
|
||||||
throw exception("invalid free variable index, de Bruijn index is too big");
|
throw exception("invalid free variable index, de Bruijn index is too big");
|
||||||
|
@ -101,13 +102,14 @@ expr_var::expr_var(unsigned idx):
|
||||||
|
|
||||||
// Expr constants
|
// Expr constants
|
||||||
expr_const::expr_const(name const & n, levels const & ls):
|
expr_const::expr_const(name const & n, levels const & ls):
|
||||||
expr_cell(expr_kind::Constant, ::lean::hash(n.hash(), hash_levels(ls)), has_meta(ls), false, has_param(ls)),
|
expr_cell(expr_kind::Constant, ::lean::hash(n.hash(), hash_levels(ls)), false, has_meta(ls), false, has_param(ls)),
|
||||||
m_name(n),
|
m_name(n),
|
||||||
m_levels(ls) {}
|
m_levels(ls) {}
|
||||||
|
|
||||||
// Expr metavariables and local variables
|
// Expr metavariables and local variables
|
||||||
expr_mlocal::expr_mlocal(bool is_meta, name const & n, expr const & t):
|
expr_mlocal::expr_mlocal(bool is_meta, name const & n, expr const & t):
|
||||||
expr_cell(is_meta ? expr_kind::Meta : expr_kind::Local, n.hash(), is_meta || t.has_metavar(), !is_meta || t.has_local(), t.has_param_univ()),
|
expr_cell(is_meta ? expr_kind::Meta : expr_kind::Local, n.hash(), is_meta || t.has_expr_metavar(), t.has_univ_metavar(),
|
||||||
|
!is_meta || t.has_local(), t.has_param_univ()),
|
||||||
m_name(n),
|
m_name(n),
|
||||||
m_type(t) {}
|
m_type(t) {}
|
||||||
void expr_mlocal::dealloc(buffer<expr_cell*> & todelete) {
|
void expr_mlocal::dealloc(buffer<expr_cell*> & todelete) {
|
||||||
|
@ -125,15 +127,17 @@ void expr_local::dealloc(buffer<expr_cell*> & todelete) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Composite expressions
|
// Composite expressions
|
||||||
expr_composite::expr_composite(expr_kind k, unsigned h, bool has_mv, bool has_local, bool has_param_univ, unsigned d, unsigned fv_range):
|
expr_composite::expr_composite(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv,
|
||||||
expr_cell(k, h, has_mv, has_local, has_param_univ),
|
bool has_local, bool has_param_univ, unsigned d, unsigned fv_range):
|
||||||
|
expr_cell(k, h, has_expr_mv, has_univ_mv, has_local, has_param_univ),
|
||||||
m_depth(d),
|
m_depth(d),
|
||||||
m_free_var_range(fv_range) {}
|
m_free_var_range(fv_range) {}
|
||||||
|
|
||||||
// Expr applications
|
// Expr applications
|
||||||
expr_app::expr_app(expr const & fn, expr const & arg):
|
expr_app::expr_app(expr const & fn, expr const & arg):
|
||||||
expr_composite(expr_kind::App, ::lean::hash(fn.hash(), arg.hash()),
|
expr_composite(expr_kind::App, ::lean::hash(fn.hash(), arg.hash()),
|
||||||
fn.has_metavar() || arg.has_metavar(),
|
fn.has_expr_metavar() || arg.has_expr_metavar(),
|
||||||
|
fn.has_univ_metavar() || arg.has_univ_metavar(),
|
||||||
fn.has_local() || arg.has_local(),
|
fn.has_local() || arg.has_local(),
|
||||||
fn.has_param_univ() || arg.has_param_univ(),
|
fn.has_param_univ() || arg.has_param_univ(),
|
||||||
std::max(get_depth(fn), get_depth(arg)) + 1,
|
std::max(get_depth(fn), get_depth(arg)) + 1,
|
||||||
|
@ -160,7 +164,8 @@ bool operator==(binder_info const & i1, binder_info const & i2) {
|
||||||
// Expr binders (Lambda, Pi)
|
// Expr binders (Lambda, Pi)
|
||||||
expr_binding::expr_binding(expr_kind k, name const & n, expr const & t, expr const & b, binder_info const & i):
|
expr_binding::expr_binding(expr_kind k, name const & n, expr const & t, expr const & b, binder_info const & i):
|
||||||
expr_composite(k, ::lean::hash(t.hash(), b.hash()),
|
expr_composite(k, ::lean::hash(t.hash(), b.hash()),
|
||||||
t.has_metavar() || b.has_metavar(),
|
t.has_expr_metavar() || b.has_expr_metavar(),
|
||||||
|
t.has_univ_metavar() || b.has_univ_metavar(),
|
||||||
t.has_local() || b.has_local(),
|
t.has_local() || b.has_local(),
|
||||||
t.has_param_univ() || b.has_param_univ(),
|
t.has_param_univ() || b.has_param_univ(),
|
||||||
std::max(get_depth(t), get_depth(b)) + 1,
|
std::max(get_depth(t), get_depth(b)) + 1,
|
||||||
|
@ -178,7 +183,7 @@ void expr_binding::dealloc(buffer<expr_cell*> & todelete) {
|
||||||
|
|
||||||
// Expr Sort
|
// Expr Sort
|
||||||
expr_sort::expr_sort(level const & l):
|
expr_sort::expr_sort(level const & l):
|
||||||
expr_cell(expr_kind::Sort, ::lean::hash(l), has_meta(l), false, has_param(l)),
|
expr_cell(expr_kind::Sort, ::lean::hash(l), false, has_meta(l), false, has_param(l)),
|
||||||
m_level(l) {
|
m_level(l) {
|
||||||
}
|
}
|
||||||
expr_sort::~expr_sort() {}
|
expr_sort::~expr_sort() {}
|
||||||
|
@ -229,7 +234,8 @@ static unsigned get_free_var_range(unsigned num, expr const * args) {
|
||||||
expr_macro::expr_macro(macro_definition const & m, unsigned num, expr const * args):
|
expr_macro::expr_macro(macro_definition const & m, unsigned num, expr const * args):
|
||||||
expr_composite(expr_kind::Macro,
|
expr_composite(expr_kind::Macro,
|
||||||
lean::hash(num, [&](unsigned i) { return args[i].hash(); }, m.hash()),
|
lean::hash(num, [&](unsigned i) { return args[i].hash(); }, m.hash()),
|
||||||
std::any_of(args, args+num, [](expr const & e) { return e.has_metavar(); }),
|
std::any_of(args, args+num, [](expr const & e) { return e.has_expr_metavar(); }),
|
||||||
|
std::any_of(args, args+num, [](expr const & e) { return e.has_univ_metavar(); }),
|
||||||
std::any_of(args, args+num, [](expr const & e) { return e.has_local(); }),
|
std::any_of(args, args+num, [](expr const & e) { return e.has_local(); }),
|
||||||
std::any_of(args, args+num, [](expr const & e) { return e.has_param_univ(); }),
|
std::any_of(args, args+num, [](expr const & e) { return e.has_param_univ(); }),
|
||||||
max_depth(num, args) + 1,
|
max_depth(num, args) + 1,
|
||||||
|
|
|
@ -52,7 +52,8 @@ protected:
|
||||||
// Remark: we use atomic_uchar because these flags are computed lazily (i.e., after the expression is created)
|
// Remark: we use atomic_uchar because these flags are computed lazily (i.e., after the expression is created)
|
||||||
atomic_uchar m_flags;
|
atomic_uchar m_flags;
|
||||||
unsigned m_kind:8;
|
unsigned m_kind:8;
|
||||||
unsigned m_has_mv:1; // term contains metavariables
|
unsigned m_has_expr_mv:1; // term contains expression metavariables
|
||||||
|
unsigned m_has_univ_mv:1; // term contains universe metavariables
|
||||||
unsigned m_has_local:1; // term contains local constants
|
unsigned m_has_local:1; // term contains local constants
|
||||||
unsigned m_has_param_univ:1; // term constains parametric universe levels
|
unsigned m_has_param_univ:1; // term constains parametric universe levels
|
||||||
unsigned m_hash; // hash based on the structure of the expression (this is a good hash for structural equality)
|
unsigned m_hash; // hash based on the structure of the expression (this is a good hash for structural equality)
|
||||||
|
@ -67,11 +68,12 @@ protected:
|
||||||
|
|
||||||
static void dec_ref(expr & c, buffer<expr_cell*> & todelete);
|
static void dec_ref(expr & c, buffer<expr_cell*> & todelete);
|
||||||
public:
|
public:
|
||||||
expr_cell(expr_kind k, unsigned h, bool has_mv, bool has_local, bool has_param_univ);
|
expr_cell(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv, bool has_local, bool has_param_univ);
|
||||||
expr_kind kind() const { return static_cast<expr_kind>(m_kind); }
|
expr_kind kind() const { return static_cast<expr_kind>(m_kind); }
|
||||||
unsigned hash() const { return m_hash; }
|
unsigned hash() const { return m_hash; }
|
||||||
unsigned hash_alloc() const { return m_hash_alloc; }
|
unsigned hash_alloc() const { return m_hash_alloc; }
|
||||||
bool has_metavar() const { return m_has_mv; }
|
bool has_expr_metavar() const { return m_has_expr_mv; }
|
||||||
|
bool has_univ_metavar() const { return m_has_univ_mv; }
|
||||||
bool has_local() const { return m_has_local; }
|
bool has_local() const { return m_has_local; }
|
||||||
bool has_param_univ() const { return m_has_param_univ; }
|
bool has_param_univ() const { return m_has_param_univ; }
|
||||||
void set_tag(tag t);
|
void set_tag(tag t);
|
||||||
|
@ -113,7 +115,9 @@ public:
|
||||||
expr_kind kind() const { return m_ptr->kind(); }
|
expr_kind kind() const { return m_ptr->kind(); }
|
||||||
unsigned hash() const { return m_ptr ? m_ptr->hash() : 23; }
|
unsigned hash() const { return m_ptr ? m_ptr->hash() : 23; }
|
||||||
unsigned hash_alloc() const { return m_ptr ? m_ptr->hash_alloc() : 23; }
|
unsigned hash_alloc() const { return m_ptr ? m_ptr->hash_alloc() : 23; }
|
||||||
bool has_metavar() const { return m_ptr->has_metavar(); }
|
bool has_expr_metavar() const { return m_ptr->has_expr_metavar(); }
|
||||||
|
bool has_univ_metavar() const { return m_ptr->has_univ_metavar(); }
|
||||||
|
bool has_metavar() const { return has_expr_metavar() || has_univ_metavar(); }
|
||||||
bool has_local() const { return m_ptr->has_local(); }
|
bool has_local() const { return m_ptr->has_local(); }
|
||||||
bool has_param_univ() const { return m_ptr->has_param_univ(); }
|
bool has_param_univ() const { return m_ptr->has_param_univ(); }
|
||||||
|
|
||||||
|
@ -251,7 +255,8 @@ protected:
|
||||||
friend unsigned get_depth(expr const & e);
|
friend unsigned get_depth(expr const & e);
|
||||||
friend unsigned get_free_var_range(expr const & e);
|
friend unsigned get_free_var_range(expr const & e);
|
||||||
public:
|
public:
|
||||||
expr_composite(expr_kind k, unsigned h, bool has_mv, bool has_local, bool has_param_univ, unsigned d, unsigned fv_range);
|
expr_composite(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv, bool has_local,
|
||||||
|
bool has_param_univ, unsigned d, unsigned fv_range);
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \brief Applications */
|
/** \brief Applications */
|
||||||
|
@ -573,6 +578,8 @@ inline binder_info const & local_info(expr const & e) { return to_local(e)
|
||||||
|
|
||||||
inline bool is_constant(expr const & e, name const & n) { return is_constant(e) && const_name(e) == n; }
|
inline bool is_constant(expr const & e, name const & n) { return is_constant(e) && const_name(e) == n; }
|
||||||
inline bool has_metavar(expr const & e) { return e.has_metavar(); }
|
inline bool has_metavar(expr const & e) { return e.has_metavar(); }
|
||||||
|
inline bool has_expr_metavar(expr const & e) { return e.has_expr_metavar(); }
|
||||||
|
inline bool has_univ_metavar(expr const & e) { return e.has_univ_metavar(); }
|
||||||
inline bool has_local(expr const & e) { return e.has_local(); }
|
inline bool has_local(expr const & e) { return e.has_local(); }
|
||||||
inline bool has_param_univ(expr const & e) { return e.has_param_univ(); }
|
inline bool has_param_univ(expr const & e) { return e.has_param_univ(); }
|
||||||
unsigned get_depth(expr const & e);
|
unsigned get_depth(expr const & e);
|
||||||
|
|
Loading…
Reference in a new issue