perf(kernel/metavar): avoid destructive update in occurs method
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
4286f5dd36
commit
adddf06e44
2 changed files with 21 additions and 8 deletions
|
@ -186,15 +186,20 @@ expr substitution::instantiate_metavars_wo_jst(expr const & e) {
|
||||||
return instantiate_metavars_fn(*this, false)(e);
|
return instantiate_metavars_fn(*this, false)(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool substitution::occurs_expr(name const & m, expr const & e) {
|
|
||||||
if (!has_expr_metavar(e))
|
bool substitution::occurs_expr_core(name const & m, expr const & e, name_set & visited) const {
|
||||||
return false;
|
|
||||||
expr new_e = instantiate(e);
|
|
||||||
bool found = false;
|
bool found = false;
|
||||||
for_each(new_e, [&](expr const & e, unsigned) {
|
for_each(e, [&](expr const & e, unsigned) {
|
||||||
if (found || !has_expr_metavar(e)) return false;
|
if (found || !has_expr_metavar(e)) return false;
|
||||||
if (is_metavar(e)) {
|
if (is_metavar(e)) {
|
||||||
if (mlocal_name(e) == m)
|
name const & n = mlocal_name(e);
|
||||||
|
if (n == m)
|
||||||
|
found = true;
|
||||||
|
auto s = get_expr(e);
|
||||||
|
if (!s || visited.contains(n))
|
||||||
|
return false; // do not visit type
|
||||||
|
visited.insert(n);
|
||||||
|
if (s && occurs_expr_core(m, *s, visited))
|
||||||
found = true;
|
found = true;
|
||||||
return false; // do not visit type
|
return false; // do not visit type
|
||||||
}
|
}
|
||||||
|
@ -203,4 +208,11 @@ bool substitution::occurs_expr(name const & m, expr const & e) {
|
||||||
});
|
});
|
||||||
return found;
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool substitution::occurs_expr(name const & m, expr const & e) const {
|
||||||
|
if (!has_expr_metavar(e))
|
||||||
|
return false;
|
||||||
|
name_set visited;
|
||||||
|
return occurs_expr_core(m, e, visited);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ class substitution {
|
||||||
friend class instantiate_metavars_fn;
|
friend class instantiate_metavars_fn;
|
||||||
std::pair<level, justification> instantiate_metavars(level const & l, bool use_jst);
|
std::pair<level, justification> instantiate_metavars(level const & l, bool use_jst);
|
||||||
expr instantiate_metavars_wo_jst(expr const & e);
|
expr instantiate_metavars_wo_jst(expr const & e);
|
||||||
|
bool occurs_expr_core(name const & m, expr const & e, name_set & visited) const;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
substitution();
|
substitution();
|
||||||
|
@ -78,7 +79,7 @@ public:
|
||||||
expr instantiate(expr const & e) { return instantiate_metavars_wo_jst(e); }
|
expr instantiate(expr const & e) { return instantiate_metavars_wo_jst(e); }
|
||||||
|
|
||||||
/** \brief Return true iff the metavariable \c m occurrs (directly or indirectly) in \c e. */
|
/** \brief Return true iff the metavariable \c m occurrs (directly or indirectly) in \c e. */
|
||||||
bool occurs_expr(name const & m, expr const & e);
|
bool occurs_expr(name const & m, expr const & e) const;
|
||||||
bool occurs(expr const & m, expr const & e) { lean_assert(is_metavar(m)); return occurs_expr(mlocal_name(m), e); }
|
bool occurs(expr const & m, expr const & e) const { lean_assert(is_metavar(m)); return occurs_expr(mlocal_name(m), e); }
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue