feat(frontends/lean/elaborator): add support for coercions in let declarations

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-10-24 17:01:06 -07:00
parent 872b698bc3
commit d1a2a4ea7e

View file

@ -48,23 +48,21 @@ expr const & get_choice(expr const & e, unsigned i) {
class coercion_justification_cell : public justification_cell { class coercion_justification_cell : public justification_cell {
context m_ctx; context m_ctx;
expr m_app; expr m_src;
expr m_arg;
public: public:
coercion_justification_cell(context const & c, expr const & app, expr const & arg):m_ctx(c), m_app(app), m_arg(arg) {} coercion_justification_cell(context const & c, expr const & src):m_ctx(c), m_src(src) {}
virtual ~coercion_justification_cell() {} virtual ~coercion_justification_cell() {}
virtual format pp_header(formatter const & fmt, options const & opts) const { virtual format pp_header(formatter const & fmt, options const & opts) const {
unsigned indent = get_pp_indent(opts); unsigned indent = get_pp_indent(opts);
format expr_fmt = fmt(m_ctx, m_arg, false, opts); format expr_fmt = fmt(m_ctx, m_src, false, opts);
format r; format r;
r += format("Coercion for"); r += format("Coercion for");
r += nest(indent, compose(line(), expr_fmt)); r += nest(indent, compose(line(), expr_fmt));
return r; return r;
} }
virtual void get_children(buffer<justification_cell*> &) const {} virtual void get_children(buffer<justification_cell*> &) const {}
virtual expr const & get_main_expr() const { return m_arg; } virtual expr const & get_main_expr() const { return m_src; }
context const & get_context() const { return m_ctx; } context const & get_context() const { return m_ctx; }
expr const & get_app() const { return m_app; }
}; };
class overload_justification_cell : public justification_cell { class overload_justification_cell : public justification_cell {
@ -88,8 +86,8 @@ public:
}; };
inline justification mk_coercion_justification(context const & ctx, expr const & app, expr const & arg) { inline justification mk_coercion_justification(context const & ctx, expr const & e) {
return justification(new coercion_justification_cell(ctx, app, arg)); return justification(new coercion_justification_cell(ctx, e));
} }
inline justification mk_overload_justification(context const & ctx, expr const & app) { inline justification mk_overload_justification(context const & ctx, expr const & app) {
@ -163,7 +161,7 @@ class frontend_elaborator::imp {
} }
expr add_coercion_mvar_app(list<expr_pair> const & l, expr const & a, expr const & a_t, expr add_coercion_mvar_app(list<expr_pair> const & l, expr const & a, expr const & a_t,
context const & ctx, expr const & original_app, expr const & original_a) { context const & ctx, expr const & original_a) {
buffer<expr> choices; buffer<expr> choices;
expr mvar = m_ref.m_menv.mk_metavar(ctx); expr mvar = m_ref.m_menv.mk_metavar(ctx);
for (auto p : l) { for (auto p : l) {
@ -172,7 +170,7 @@ class frontend_elaborator::imp {
choices.push_back(mk_lambda(g_x_name, a_t, mk_var(0))); // add indentity function choices.push_back(mk_lambda(g_x_name, a_t, mk_var(0))); // add indentity function
std::reverse(choices.begin(), choices.end()); std::reverse(choices.begin(), choices.end());
m_ref.m_ucs.push_back(mk_choice_constraint(ctx, mvar, choices.size(), choices.data(), m_ref.m_ucs.push_back(mk_choice_constraint(ctx, mvar, choices.size(), choices.data(),
mk_coercion_justification(ctx, original_app, original_a))); mk_coercion_justification(ctx, original_a)));
return mk_app(mvar, a); return mk_app(mvar, a);
} }
@ -187,9 +185,13 @@ class frontend_elaborator::imp {
/** /**
\brief Try to solve overload at preprocessing time. \brief Try to solve overload at preprocessing time.
*/ */
#if 0
bool choose(buffer<expr> const & f_choices, buffer<expr> const & f_choice_types, bool choose(buffer<expr> const & f_choices, buffer<expr> const & f_choice_types,
buffer<expr> & args, buffer<expr> & arg_types, buffer<expr> & args, buffer<expr> & arg_types,
context const & ctx, expr const & src) { context const & ctx, expr const & src) {
#else
bool choose(buffer<expr> const &, buffer<expr> const &, buffer<expr> &, buffer<expr> &, context const &, expr const &) {
#endif
// TODO(Leo) // TODO(Leo)
return false; return false;
} }
@ -235,7 +237,7 @@ class frontend_elaborator::imp {
if (arg_types[i]) { if (arg_types[i]) {
list<expr_pair> coercions = m_ref.m_frontend.get_coercions(arg_types[i]); list<expr_pair> coercions = m_ref.m_frontend.get_coercions(arg_types[i]);
if (coercions) if (coercions)
args[i] = add_coercion_mvar_app(coercions, args[i], arg_types[i], ctx, e, arg(e, i)); args[i] = add_coercion_mvar_app(coercions, args[i], arg_types[i], ctx, arg(e, i));
} }
} }
} }
@ -254,7 +256,7 @@ class frontend_elaborator::imp {
list<expr_pair> coercions = m_ref.m_frontend.get_coercions(new_a_t); list<expr_pair> coercions = m_ref.m_frontend.get_coercions(new_a_t);
if (coercions) { if (coercions) {
if (!new_f_t) { if (!new_f_t) {
new_a = add_coercion_mvar_app(coercions, new_a, new_a_t, ctx, e, a); new_a = add_coercion_mvar_app(coercions, new_a, new_a_t, ctx, a);
} else { } else {
expr expected = abst_domain(new_f_t); expr expected = abst_domain(new_f_t);
if (expected != new_a_t) { if (expected != new_a_t) {
@ -262,7 +264,7 @@ class frontend_elaborator::imp {
if (c) if (c)
new_a = mk_app(c, new_a); // apply coercion new_a = mk_app(c, new_a); // apply coercion
else else
new_a = add_coercion_mvar_app(coercions, new_a, new_a_t, ctx, e, a); new_a = add_coercion_mvar_app(coercions, new_a, new_a_t, ctx, a);
} }
} }
} }
@ -275,6 +277,29 @@ class frontend_elaborator::imp {
} }
} }
virtual expr visit_let(expr const & e, context const & ctx) {
lean_assert(is_let(e));
return update_let(e, [&](expr const & t, expr const & v, expr const & b) {
expr new_t = t ? visit(t, ctx) : expr();
expr new_v = visit(v, ctx);
if (new_t) {
expr new_v_t = get_type(new_v, ctx);
if (new_v_t && new_t != new_v_t) {
list<expr_pair> coercions = m_ref.m_frontend.get_coercions(new_v_t);
if (coercions) {
new_v = add_coercion_mvar_app(coercions, new_v, new_v_t, ctx, v);
}
}
}
expr new_b;
{
cache::mk_scope sc(m_cache);
new_b = visit(b, extend(ctx, let_name(e), new_t, new_v));
}
return std::make_tuple(new_t, new_v, new_b);
});
}
virtual expr visit(expr const & e, context const & ctx) { virtual expr visit(expr const & e, context const & ctx) {
check_interrupted(m_ref.m_interrupted); check_interrupted(m_ref.m_interrupted);
expr r = replace_visitor::visit(e, ctx); expr r = replace_visitor::visit(e, ctx);