fix(frontends/lean/elaborator): bug when reporting error position

This commit is contained in:
Leonardo de Moura 2014-10-10 17:31:12 -07:00
parent d204d9c025
commit 5e6ff3eef3
3 changed files with 61 additions and 16 deletions

View file

@ -377,21 +377,29 @@ pair<expr, expr> elaborator::ensure_fun(expr f, constraint_seq & cs) {
} }
bool elaborator::has_coercions_from(expr const & a_type) { bool elaborator::has_coercions_from(expr const & a_type) {
expr const & a_cls = get_app_fn(whnf(a_type).first); try {
return is_constant(a_cls) && ::lean::has_coercions_from(env(), const_name(a_cls)); expr const & a_cls = get_app_fn(whnf(a_type).first);
return is_constant(a_cls) && ::lean::has_coercions_from(env(), const_name(a_cls));
} catch (exception&) {
return false;
}
} }
bool elaborator::has_coercions_to(expr d_type) { bool elaborator::has_coercions_to(expr d_type) {
d_type = whnf(d_type).first; try {
expr const & fn = get_app_fn(d_type); d_type = whnf(d_type).first;
if (is_constant(fn)) expr const & fn = get_app_fn(d_type);
return ::lean::has_coercions_to(env(), const_name(fn)); if (is_constant(fn))
else if (is_pi(d_type)) return ::lean::has_coercions_to(env(), const_name(fn));
return ::lean::has_coercions_to_fun(env()); else if (is_pi(d_type))
else if (is_sort(d_type)) return ::lean::has_coercions_to_fun(env());
return ::lean::has_coercions_to_sort(env()); else if (is_sort(d_type))
else return ::lean::has_coercions_to_sort(env());
else
return false;
} catch (exception&) {
return false; return false;
}
} }
expr elaborator::apply_coercion(expr const & a, expr a_type, expr d_type) { expr elaborator::apply_coercion(expr const & a, expr a_type, expr d_type) {
@ -410,9 +418,12 @@ expr elaborator::apply_coercion(expr const & a, expr a_type, expr d_type) {
for (expr const & coe : coes) { for (expr const & coe : coes) {
expr r = mk_app(coe, a, a.get_tag()); expr r = mk_app(coe, a, a.get_tag());
expr r_type = infer_type(r).first; expr r_type = infer_type(r).first;
if (m_tc[m_relax_main_opaque]->is_def_eq(r_type, d_type).first) { try {
save_coercion_info(a, r); if (m_tc[m_relax_main_opaque]->is_def_eq(r_type, d_type).first) {
return r; save_coercion_info(a, r);
return r;
}
} catch (exception&) {
} }
} }
erase_coercion_info(a); erase_coercion_info(a);
@ -444,7 +455,12 @@ pair<expr, constraint_seq> elaborator::ensure_has_type(
} else if (is_meta(a_type) && has_coercions_to(expected_type)) { } else if (is_meta(a_type) && has_coercions_to(expected_type)) {
return mk_delayed_coercion(a, a_type, expected_type, j); return mk_delayed_coercion(a, a_type, expected_type, j);
} else { } else {
auto dcs = m_tc[relax]->is_def_eq(a_type, expected_type, j); pair<bool, constraint_seq> dcs;
try {
dcs = m_tc[relax]->is_def_eq(a_type, expected_type, j);
} catch (exception&) {
dcs.first = false;
}
if (dcs.first) { if (dcs.first) {
return to_ecs(a, dcs.second); return to_ecs(a, dcs.second);
} else { } else {
@ -453,7 +469,11 @@ pair<expr, constraint_seq> elaborator::ensure_has_type(
bool coercion_worked = false; bool coercion_worked = false;
if (!is_eqp(a, new_a)) { if (!is_eqp(a, new_a)) {
expr new_a_type = infer_type(new_a, cs); expr new_a_type = infer_type(new_a, cs);
coercion_worked = m_tc[relax]->is_def_eq(new_a_type, expected_type, j, cs); try {
coercion_worked = m_tc[relax]->is_def_eq(new_a_type, expected_type, j, cs);
} catch (exception&) {
coercion_worked = false;
}
} }
if (coercion_worked) { if (coercion_worked) {
return to_ecs(new_a, cs); return to_ecs(new_a, cs);

View file

@ -0,0 +1,9 @@
inductive category (ob : Type) : Type :=
mk : Π (hom : ob → ob → Type)
(comp : Π⦃a b c : ob⦄, hom b c → hom a b → hom a c),
category ob
inductive Category : Type := mk : Π (ob : Type), category ob → Category
definition MK (a b c : Category) : _ :=
Category.mk a (category.mk b c)

View file

@ -0,0 +1,16 @@
error_pos_bug.lean:9:0: error: type error in placeholder assigned to
λ (a : Category) (b : Category) (c : Category),
a
placeholder has type
Category
but is expected to have type
Type
the assignment was attempted when trying to solve
type mismatch at application
Category.mk a (category.mk b c)
term
category.mk b c
has type
category a
but is expected to have type
category a