feat(library/type_inference): postpone "nontrivial" universe unification constraints
This commit is contained in:
parent
0bf069f016
commit
766fdd415a
2 changed files with 126 additions and 17 deletions
|
@ -215,14 +215,61 @@ expr type_inference::whnf(expr const & e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
static bool is_max_like(level const & l) {
|
||||||
|
return is_max(l) || is_imax(l);
|
||||||
|
}
|
||||||
|
|
||||||
|
lbool type_inference::quick_is_def_eq(level const & l1, level const & l2) {
|
||||||
|
if (is_equivalent(l1, l2)) {
|
||||||
|
return l_true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_uvar(l1)) {
|
||||||
|
if (auto v = get_assignment(l1)) {
|
||||||
|
return quick_is_def_eq(*v, l2);
|
||||||
|
} else {
|
||||||
|
update_assignment(l1, l2);
|
||||||
|
return l_true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_uvar(l2)) {
|
||||||
|
if (auto v = get_assignment(l2)) {
|
||||||
|
return quick_is_def_eq(l1, *v);
|
||||||
|
} else {
|
||||||
|
update_assignment(l2, l1);
|
||||||
|
return l_true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// postpone constraint if l1 or l2 is max, imax or meta.
|
||||||
|
if (is_max_like(l1) || is_max_like(l2) || is_meta(l1) || is_meta(l2))
|
||||||
|
return l_undef;
|
||||||
|
|
||||||
|
if (l1.kind() == l2.kind()) {
|
||||||
|
switch (l1.kind()) {
|
||||||
|
case level_kind::Succ:
|
||||||
|
return quick_is_def_eq(succ_of(l1), succ_of(l2));
|
||||||
|
case level_kind::Param: case level_kind::Global:
|
||||||
|
return l_false;
|
||||||
|
case level_kind::Max: case level_kind::IMax:
|
||||||
|
case level_kind::Zero: case level_kind::Meta:
|
||||||
|
lean_unreachable();
|
||||||
|
}
|
||||||
|
lean_unreachable();
|
||||||
|
} else {
|
||||||
|
return l_false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool type_inference::full_is_def_eq(level const & l1, level const & l2) {
|
||||||
if (is_equivalent(l1, l2)) {
|
if (is_equivalent(l1, l2)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_uvar(l1)) {
|
if (is_uvar(l1)) {
|
||||||
if (auto v = get_assignment(l1)) {
|
if (auto v = get_assignment(l1)) {
|
||||||
return is_def_eq(*v, l2);
|
return full_is_def_eq(*v, l2);
|
||||||
} else {
|
} else {
|
||||||
update_assignment(l1, l2);
|
update_assignment(l1, l2);
|
||||||
return true;
|
return true;
|
||||||
|
@ -231,7 +278,7 @@ bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
||||||
|
|
||||||
if (is_uvar(l2)) {
|
if (is_uvar(l2)) {
|
||||||
if (auto v = get_assignment(l2)) {
|
if (auto v = get_assignment(l2)) {
|
||||||
return is_def_eq(l1, *v);
|
return full_is_def_eq(l1, *v);
|
||||||
} else {
|
} else {
|
||||||
update_assignment(l2, l1);
|
update_assignment(l2, l1);
|
||||||
return true;
|
return true;
|
||||||
|
@ -245,7 +292,7 @@ bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (l1 != new_l1 || l2 != new_l2)
|
if (l1 != new_l1 || l2 != new_l2)
|
||||||
return is_def_eq(new_l1, new_l2);
|
return full_is_def_eq(new_l1, new_l2);
|
||||||
|
|
||||||
if (l1.kind() != l2.kind())
|
if (l1.kind() != l2.kind())
|
||||||
return false;
|
return false;
|
||||||
|
@ -253,14 +300,14 @@ bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
||||||
switch (l1.kind()) {
|
switch (l1.kind()) {
|
||||||
case level_kind::Max:
|
case level_kind::Max:
|
||||||
return
|
return
|
||||||
is_def_eq(max_lhs(l1), max_lhs(l2)) &&
|
full_is_def_eq(max_lhs(l1), max_lhs(l2)) &&
|
||||||
is_def_eq(max_rhs(l1), max_rhs(l2));
|
full_is_def_eq(max_rhs(l1), max_rhs(l2));
|
||||||
case level_kind::IMax:
|
case level_kind::IMax:
|
||||||
return
|
return
|
||||||
is_def_eq(imax_lhs(l1), imax_lhs(l2)) &&
|
full_is_def_eq(imax_lhs(l1), imax_lhs(l2)) &&
|
||||||
is_def_eq(imax_rhs(l1), imax_rhs(l2));
|
full_is_def_eq(imax_rhs(l1), imax_rhs(l2));
|
||||||
case level_kind::Succ:
|
case level_kind::Succ:
|
||||||
return is_def_eq(succ_of(l1), succ_of(l2));
|
return full_is_def_eq(succ_of(l1), succ_of(l2));
|
||||||
case level_kind::Param:
|
case level_kind::Param:
|
||||||
case level_kind::Global:
|
case level_kind::Global:
|
||||||
return false;
|
return false;
|
||||||
|
@ -271,6 +318,16 @@ bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
||||||
lean_unreachable();
|
lean_unreachable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool type_inference::is_def_eq(level const & l1, level const & l2) {
|
||||||
|
auto r = quick_is_def_eq(l1, l2);
|
||||||
|
if (r == l_undef) {
|
||||||
|
m_postponed.emplace_back(l1, l2);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return r == l_true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool type_inference::is_def_eq(levels const & ls1, levels const & ls2) {
|
bool type_inference::is_def_eq(levels const & ls1, levels const & ls2) {
|
||||||
if (is_nil(ls1) && is_nil(ls2)) {
|
if (is_nil(ls1) && is_nil(ls2)) {
|
||||||
return true;
|
return true;
|
||||||
|
@ -755,9 +812,54 @@ bool type_inference::is_def_eq_core(expr const & t, expr const & s) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool type_inference::process_postponed(unsigned old_sz) {
|
||||||
|
if (m_postponed.size() == old_sz)
|
||||||
|
return true; // no new universe constraints.
|
||||||
|
lean_assert(m_postponed.size() > old_sz);
|
||||||
|
buffer<pair<level, level>> b1, b2;
|
||||||
|
b1.append(m_postponed.size() - old_sz, m_postponed.data() + old_sz);
|
||||||
|
buffer<pair<level, level>> * curr, * next;
|
||||||
|
curr = &b1;
|
||||||
|
next = &b2;
|
||||||
|
while (true) {
|
||||||
|
for (auto p : *curr) {
|
||||||
|
auto r = quick_is_def_eq(p.first, p.second);
|
||||||
|
if (r == l_undef) {
|
||||||
|
next->push_back(p);
|
||||||
|
} else if (r == l_false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (next->empty()) {
|
||||||
|
return true; // all constraints have been processed
|
||||||
|
} else if (next->size() < curr->size()) {
|
||||||
|
// easy constraints have been processed in this iteration
|
||||||
|
curr->clear();
|
||||||
|
std::swap(next, curr);
|
||||||
|
lean_assert(next->empty());
|
||||||
|
} else {
|
||||||
|
// use full (and approximate) is_def_eq to process the first constraint
|
||||||
|
// in next.
|
||||||
|
auto p = (*next)[0];
|
||||||
|
if (!full_is_def_eq(p.first, p.second))
|
||||||
|
return false;
|
||||||
|
if (next->size() == 1)
|
||||||
|
return true; // the last constraint has been solved.
|
||||||
|
curr->clear();
|
||||||
|
curr->append(next->size() - 1, next->data() + 1);
|
||||||
|
next->clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool type_inference::is_def_eq(expr const & e1, expr const & e2) {
|
bool type_inference::is_def_eq(expr const & e1, expr const & e2) {
|
||||||
scope s(*this);
|
scope s(*this);
|
||||||
if (is_def_eq_core(e1, e2)) {
|
unsigned psz = m_postponed.size();
|
||||||
|
if (!is_def_eq_core(e1, e2)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (process_postponed(psz)) {
|
||||||
|
m_postponed.resize(psz);
|
||||||
s.commit();
|
s.commit();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,10 +22,12 @@ namespace lean {
|
||||||
class type_inference {
|
class type_inference {
|
||||||
struct ext_ctx;
|
struct ext_ctx;
|
||||||
friend struct ext_ctx;
|
friend struct ext_ctx;
|
||||||
environment m_env;
|
environment m_env;
|
||||||
name_generator m_ngen;
|
name_generator m_ngen;
|
||||||
std::unique_ptr<ext_ctx> m_ext_ctx;
|
std::unique_ptr<ext_ctx> m_ext_ctx;
|
||||||
name_map<projection_info> m_proj_info;
|
// postponed universe constraints
|
||||||
|
std::vector<pair<level, level>> m_postponed;
|
||||||
|
name_map<projection_info> m_proj_info;
|
||||||
|
|
||||||
bool is_opaque(declaration const & d) const;
|
bool is_opaque(declaration const & d) const;
|
||||||
optional<expr> expand_macro(expr const & m);
|
optional<expr> expand_macro(expr const & m);
|
||||||
|
@ -37,6 +39,8 @@ class type_inference {
|
||||||
optional<declaration> is_delta(expr const & e) const;
|
optional<declaration> is_delta(expr const & e) const;
|
||||||
expr whnf_core(expr e, unsigned h);
|
expr whnf_core(expr e, unsigned h);
|
||||||
|
|
||||||
|
lbool quick_is_def_eq(level const & l1, level const & l2);
|
||||||
|
bool full_is_def_eq(level const & l1, level const & l2);
|
||||||
bool is_def_eq(level const & l1, level const & l2);
|
bool is_def_eq(level const & l1, level const & l2);
|
||||||
bool is_def_eq(levels const & ls1, levels const & ls2);
|
bool is_def_eq(levels const & ls1, levels const & ls2);
|
||||||
|
|
||||||
|
@ -55,6 +59,8 @@ class type_inference {
|
||||||
reduction_status ext_reduction_step(expr & t_n, expr & s_n);
|
reduction_status ext_reduction_step(expr & t_n, expr & s_n);
|
||||||
lbool reduce_def_eq(expr & t_n, expr & s_n);
|
lbool reduce_def_eq(expr & t_n, expr & s_n);
|
||||||
|
|
||||||
|
bool process_postponed(unsigned old_sz);
|
||||||
|
|
||||||
expr infer_constant(expr const & e);
|
expr infer_constant(expr const & e);
|
||||||
expr infer_macro(expr const & e);
|
expr infer_macro(expr const & e);
|
||||||
expr infer_lambda(expr e);
|
expr infer_lambda(expr e);
|
||||||
|
@ -66,9 +72,10 @@ class type_inference {
|
||||||
struct scope {
|
struct scope {
|
||||||
type_inference & m_owner;
|
type_inference & m_owner;
|
||||||
bool m_keep;
|
bool m_keep;
|
||||||
scope(type_inference & o):m_owner(o), m_keep(false) { m_owner.push(); }
|
unsigned m_postponed_sz;
|
||||||
~scope() { if (!m_keep) m_owner.pop(); }
|
scope(type_inference & o):m_owner(o), m_keep(false), m_postponed_sz(o.m_postponed.size()) { m_owner.push(); }
|
||||||
void commit() { m_owner.commit(); m_keep = true; }
|
~scope() { m_owner.m_postponed.resize(m_postponed_sz); if (!m_keep) m_owner.pop(); }
|
||||||
|
void commit() { m_postponed_sz = m_owner.m_postponed.size(); m_owner.commit(); m_keep = true; }
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
Loading…
Reference in a new issue