feat(elaborator): solve more unification constraints, add more tests

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-10-21 16:45:14 -07:00
parent c3e87f106f
commit f4592da87f
4 changed files with 393 additions and 14 deletions

View file

@ -12,6 +12,7 @@ Author: Leonardo de Moura
#include "kernel/free_vars.h" #include "kernel/free_vars.h"
#include "kernel/normalizer.h" #include "kernel/normalizer.h"
#include "kernel/instantiate.h" #include "kernel/instantiate.h"
#include "kernel/replace.h"
#include "kernel/builtin.h" #include "kernel/builtin.h"
#include "library/type_inferer.h" #include "library/type_inferer.h"
#include "library/update_expr.h" #include "library/update_expr.h"
@ -165,6 +166,11 @@ class elaborator::imp {
m_state.m_queue.push_front(c); m_state.m_queue.push_front(c);
} }
/** \brief Add given constraint to the end of the current constraint queue */
void push_back(unification_constraint const & c) {
m_state.m_queue.push_back(c);
}
/** \brief Return true iff \c m is an assigned metavariable in the current state */ /** \brief Return true iff \c m is an assigned metavariable in the current state */
bool is_assigned(expr const & m) const { bool is_assigned(expr const & m) const {
lean_assert(is_metavar(m)); lean_assert(is_metavar(m));
@ -199,6 +205,14 @@ class elaborator::imp {
return ::lean::has_metavar(e, m, m_state.m_menv.get_substitutions()); return ::lean::has_metavar(e, m, m_state.m_menv.get_substitutions());
} }
/**
\brief Return true iff \c e contains an assigned metavariable in
the current state.
*/
bool has_assigned_metavar(expr const & e) const {
return ::lean::has_assigned_metavar(e, m_state.m_menv.get_substitutions());
}
/** /**
\brief Return a unassigned metavariable in the current state. \brief Return a unassigned metavariable in the current state.
Return the anonymous name if the state does not contain unassigned metavariables. Return the anonymous name if the state does not contain unassigned metavariables.
@ -291,12 +305,17 @@ class elaborator::imp {
/** /**
\brief Assign \c v to \c m with justification \c tr in the current state. \brief Assign \c v to \c m with justification \c tr in the current state.
*/ */
void assign(expr const & m, expr const & v, trace const & tr) { void assign(expr const & m, expr const & v, context const & ctx, trace const & tr) {
lean_assert(is_metavar(m)); lean_assert(is_metavar(m));
metavar_env & menv = m_state.m_menv; metavar_env & menv = m_state.m_menv;
m_state.m_menv.assign(m, v, tr); m_state.m_menv.assign(m, v, tr);
if (menv.has_type(m)) { if (menv.has_type(m)) {
buffer<unification_constraint> ucs;
expr tv = m_type_inferer(v, ctx, &menv, ucs);
for (auto c : ucs)
push_front(c);
trace new_trace(new typeof_mvar_trace(ctx, m, menv.get_type(m), tv, tr));
push_front(mk_eq_constraint(ctx, menv.get_type(m), tv, new_trace));
} }
} }
@ -342,7 +361,7 @@ class elaborator::imp {
m_conflict = trace(new unification_failure_trace(c)); m_conflict = trace(new unification_failure_trace(c));
return Failed; return Failed;
} else if (allow_assignment) { } else if (allow_assignment) {
assign(a, b, trace(new assignment_trace(c))); assign(a, b, get_context(c), trace(new assignment_trace(c)));
reset_quota(); reset_quota();
return Processed; return Processed;
} }
@ -350,7 +369,7 @@ class elaborator::imp {
local_entry const & me = head(metavar_lctx(a)); local_entry const & me = head(metavar_lctx(a));
if (me.is_lift() && !has_free_var(b, me.s(), me.s() + me.n())) { if (me.is_lift() && !has_free_var(b, me.s(), me.s() + me.n())) {
// Case 3 // Case 3
trace new_tr(new substitution_trace(c, get_mvar_trace(a))); trace new_tr(new normalize_trace(c));
expr new_a = pop_meta_context(a); expr new_a = pop_meta_context(a);
expr new_b = lower_free_vars(b, me.s() + me.n(), me.n()); expr new_b = lower_free_vars(b, me.s() + me.n(), me.n());
if (!is_lhs) if (!is_lhs)
@ -371,6 +390,48 @@ class elaborator::imp {
return Continue; return Continue;
} }
trace mk_subst_trace(unification_constraint const & c, buffer<trace> const & subst_traces) {
if (subst_traces.size() == 1) {
return trace(new substitution_trace(c, subst_traces[0]));
} else {
return trace(new multi_substitution_trace(c, subst_traces.size(), subst_traces.data()));
}
}
/**
\brief Return true iff \c a contains instantiated variables. If this is the case,
then constraint \c c is updated with a new \c a s.t. all metavariables of \c a
are instantiated.
\remark if \c is_lhs is true, then we are considering the left-hand-side of the constraint \c c.
*/
bool instantiate_metavars(bool is_lhs, expr const & a, unification_constraint const & c) {
lean_assert(is_eq(c) || is_convertible(c));
lean_assert(!is_eq(c) || !is_lhs || is_eqp(eq_lhs(c), a));
lean_assert(!is_eq(c) || is_lhs || is_eqp(eq_rhs(c), a));
lean_assert(!is_convertible(c) || !is_lhs || is_eqp(convertible_from(c), a));
lean_assert(!is_convertible(c) || is_lhs || is_eqp(convertible_to(c), a));
if (has_assigned_metavar(a)) {
metavar_env & menv = m_state.m_menv;
buffer<trace> traces;
auto f = [&](expr const & m, unsigned) -> expr {
if (is_metavar(m) && menv.is_assigned(m)) {
trace t = menv.get_trace(m);
if (t)
traces.push_back(t);
return menv.get_subst(m);
} else {
return m;
}
};
expr new_a = replace_fn<decltype(f)>(f)(a);
trace new_tr = mk_subst_trace(c, traces);
push_updated_constraint(c, is_lhs, new_a, new_tr);
return true;
} else {
return false;
}
}
/** \brief Unfold let-expression */ /** \brief Unfold let-expression */
void process_let(expr & a) { void process_let(expr & a) {
@ -512,6 +573,41 @@ class elaborator::imp {
} }
} }
/** \brief Return true iff the variable with id \c vidx has a body/definition in the context \c ctx. */
static bool has_body(context const & ctx, unsigned vidx) {
try {
context_entry const & e = lookup(ctx, vidx);
if (e.get_body())
return true;
} catch (exception&) {
}
return false;
}
/**
\brief Return true iff ctx |- a == b is a "simple" higher-order matching constraint. By simple, we mean
a constraint of the form
ctx |- (?m x) == c
The constraint is solved by assigning ?m to (fun (x : T), c).
*/
bool process_simple_ho_match(context const & ctx, expr const & a, expr const & b, bool is_lhs, unification_constraint const & c) {
// Solve constraint of the form
// ctx |- (?m x) == c
// using imitation
if (is_eq(c) && is_meta_app(a) && is_var(arg(a, 1)) && !has_body(ctx, var_idx(arg(a, 1))) && closed(b)) {
expr m = arg(a, 0);
expr t = lookup(ctx, var_idx(arg(a, 1))).get_domain();
trace new_trace(new destruct_trace(c));
expr s = mk_lambda(g_x_name, t, b);
if (!is_lhs)
swap(m, s);
push_front(mk_eq_constraint(ctx, m, s, new_trace));
return true;
} else {
return false;
}
}
bool process_eq_convertible(context const & ctx, expr const & a, expr const & b, unification_constraint const & c) { bool process_eq_convertible(context const & ctx, expr const & a, expr const & b, unification_constraint const & c) {
bool eq = is_eq(c); bool eq = is_eq(c);
if (a == b) { if (a == b) {
@ -532,10 +628,19 @@ class elaborator::imp {
r = process_metavar(c, b, a, false, !is_type(a) && !is_meta(a) && a != Bool); r = process_metavar(c, b, a, false, !is_type(a) && !is_meta(a) && a != Bool);
if (r != Continue) { return r == Processed; } if (r != Continue) { return r == Processed; }
if (process_simple_ho_match(ctx, a, b, true, c) ||
process_simple_ho_match(ctx, b, a, false, c))
return true;
if (a.kind() == b.kind()) { if (a.kind() == b.kind()) {
switch (a.kind()) { switch (a.kind()) {
case expr_kind::Constant: case expr_kind::Var: case expr_kind::Value: case expr_kind::Constant: case expr_kind::Var: case expr_kind::Value:
return a == b; if (a == b) {
return true;
} else {
m_conflict = trace(new unification_failure_trace(c));
return false;
}
case expr_kind::Type: case expr_kind::Type:
if ((!eq && m_env.is_ge(ty_level(b), ty_level(a))) || (eq && a == b)) { if ((!eq && m_env.is_ge(ty_level(b), ty_level(a))) || (eq && a == b)) {
return true; return true;
@ -586,10 +691,18 @@ class elaborator::imp {
} }
} }
if (!is_meta(a) && !is_meta(b) && a.kind() != b.kind()) if (!is_meta(a) && !is_meta(b) && a.kind() != b.kind()) {
m_conflict = trace(new unification_failure_trace(c));
return false; return false;
}
if (instantiate_metavars(true, a, c) ||
instantiate_metavars(false, b, c)) {
return true;
}
std::cout << "Postponed: "; display(std::cout, c); std::cout << "Postponed: "; display(std::cout, c);
push_back(c);
return true; return true;
} }
@ -600,15 +713,24 @@ class elaborator::imp {
return true; return true;
} }
bool process_choice(unification_constraint const &) { bool process_choice(unification_constraint const & c) {
// TODO(Leo) std::unique_ptr<case_split> new_cs(new choice_case_split(c, m_state));
return true; bool r = new_cs->next(*this);
lean_assert(r);
m_case_splits.push_back(std::move(new_cs));
return r;
} }
void resolve_conflict() { void resolve_conflict() {
lean_assert(m_conflict); lean_assert(m_conflict);
std::cout << "Resolve conflict, num case_splits: " << m_case_splits.size() << "\n";
formatter fmt = mk_simple_formatter();
std::cout << m_conflict.pp(fmt, options(), nullptr, true) << "\n";
while (!m_case_splits.empty()) { while (!m_case_splits.empty()) {
std::unique_ptr<case_split> & d = m_case_splits.back(); std::unique_ptr<case_split> & d = m_case_splits.back();
std::cout << "Assumption " << d->m_curr_assumption.pp(fmt, options(), nullptr, true) << "\n";
if (depends_on(m_conflict, d->m_curr_assumption)) { if (depends_on(m_conflict, d->m_curr_assumption)) {
d->m_failed_traces.push_back(m_conflict); d->m_failed_traces.push_back(m_conflict);
if (d->next(*this)) { if (d->next(*this)) {
@ -630,7 +752,6 @@ class elaborator::imp {
s.m_curr_assumption = mk_assumption(); s.m_curr_assumption = mk_assumption();
m_state = s.m_prev_state; m_state = s.m_prev_state;
push_front(mk_eq_constraint(get_context(choice), choice_mvar(choice), choice_ith(choice, idx), s.m_curr_assumption)); push_front(mk_eq_constraint(get_context(choice), choice_mvar(choice), choice_ith(choice, idx), s.m_curr_assumption));
s.m_idx++;
return true; return true;
} else { } else {
m_conflict = trace(new unification_failure_by_cases_trace(choice, s.m_failed_traces.size(), s.m_failed_traces.data())); m_conflict = trace(new unification_failure_by_cases_trace(choice, s.m_failed_traces.size(), s.m_failed_traces.data()));
@ -710,7 +831,7 @@ public:
while (true) { while (true) {
check_interrupted(m_interrupted); check_interrupted(m_interrupted);
cnstr_queue & q = m_state.m_queue; cnstr_queue & q = m_state.m_queue;
if (q.empty()) { if (q.empty() || m_quota < -static_cast<int>(q.size())) {
name m = find_unassigned_metavar(); name m = find_unassigned_metavar();
std::cout << "Queue is empty\n"; display(std::cout); std::cout << "Queue is empty\n"; display(std::cout);
if (m) { if (m) {
@ -722,7 +843,7 @@ public:
} }
} else { } else {
unification_constraint c = q.front(); unification_constraint c = q.front();
std::cout << "Processing "; display(std::cout, c); std::cout << "Processing, quota: " << m_quota << " "; display(std::cout, c);
q.pop_front(); q.pop_front();
if (!process(c)) { if (!process(c)) {
resolve_conflict(); resolve_conflict();

View file

@ -48,9 +48,8 @@ expr const & propagation_trace::get_main_expr() const {
} }
format propagation_trace::pp_header(formatter const & fmt, options const & opts) const { format propagation_trace::pp_header(formatter const & fmt, options const & opts) const {
format r; format r;
unsigned indent = get_pp_indent(opts);
r += format(get_prop_name()); r += format(get_prop_name());
r += nest(indent, compose(line(), get_constraint().pp(fmt, opts, nullptr, false))); r += compose(line(), get_constraint().pp(fmt, opts, nullptr, false));
return r; return r;
} }
@ -93,6 +92,36 @@ void multi_substitution_trace::get_children(buffer<trace_cell*> & r) const {
append(r, m_assignment_traces); append(r, m_assignment_traces);
} }
// -------------------------
// typeof metavar trace
// -------------------------
typeof_mvar_trace::typeof_mvar_trace(context const & ctx, expr const & m, expr const & tm, expr const & t, trace const & tr):
m_context(ctx),
m_mvar(m),
m_typeof_mvar(tm),
m_type(t),
m_trace(tr) {
}
typeof_mvar_trace::~typeof_mvar_trace() {
}
format typeof_mvar_trace::pp_header(formatter const & fmt, options const & opts) const {
format r;
unsigned indent = get_pp_indent(opts);
r += format("Propagate type,");
{
format body;
body += fmt(m_context, m_mvar, false, opts);
body += space();
body += colon();
body += nest(indent, compose(line(), fmt(m_context, m_typeof_mvar, false, opts)));
r += nest(indent, compose(line(), body));
}
return group(r);
}
void typeof_mvar_trace::get_children(buffer<trace_cell*> & r) const {
push_back(r, m_trace);
}
// ------------------------- // -------------------------
// Synthesis trace objects // Synthesis trace objects
// ------------------------- // -------------------------

View file

@ -130,6 +130,23 @@ public:
virtual void get_children(buffer<trace_cell*> & r) const; virtual void get_children(buffer<trace_cell*> & r) const;
}; };
/**
\brief Trace object used to justify a <tt>typeof(m) == t</tt> constraint generated when
we assign a metavariable \c m.
*/
class typeof_mvar_trace : public trace_cell {
context m_context;
expr m_mvar;
expr m_typeof_mvar;
expr m_type;
trace m_trace;
public:
typeof_mvar_trace(context const & ctx, expr const & m, expr const & mt, expr const & t, trace const & tr);
virtual ~typeof_mvar_trace();
virtual format pp_header(formatter const &, options const &) const;
virtual void get_children(buffer<trace_cell*> & r) const;
};
/** /**
\brief Base class for synthesis_failure_trace and synthesized_assignment_trace \brief Base class for synthesis_failure_trace and synthesized_assignment_trace
*/ */

View file

@ -53,8 +53,220 @@ static void tst1() {
substitution s = elb.next(); substitution s = elb.next();
} }
static void tst2() {
/*
Solve elaboration problem for
g : Pi (A : Type), A -> A
a : Int
Axiom H : g _ a <= 0
The following elaboration problem is created
?m1 (g ?m2 (?m3 a)) (?m4 a)
?m1 in { Nat::Le, Int::Le, Real::Le }
?m3 in { Id, int2real }
?m4 in { Id, nat2int, nat2real }
*/
environment env;
import_all(env);
metavar_env menv;
buffer<unification_constraint> ucs;
type_checker checker(env);
expr A = Const("A");
expr g = Const("g");
env.add_var("g", Pi({A, Type()}, A >> A));
expr a = Const("a");
env.add_var("a", Int);
expr m1 = menv.mk_metavar();
expr m2 = menv.mk_metavar();
expr m3 = menv.mk_metavar();
expr m4 = menv.mk_metavar();
expr int_id = Fun({a, Int}, a);
expr nat_id = Fun({a, Nat}, a);
expr F = m1(g(m2, m3(a)), m4(nVal(0)));
std::cout << F << "\n";
std::cout << checker.infer_type(F, context(), &menv, ucs) << "\n";
ucs.push_back(mk_choice_constraint(context(), m1, { mk_nat_le_fn(), mk_int_le_fn(), mk_real_le_fn() }, trace()));
ucs.push_back(mk_choice_constraint(context(), m3, { int_id, mk_int_to_real_fn() }, trace()));
ucs.push_back(mk_choice_constraint(context(), m4, { nat_id, mk_nat_to_int_fn(), mk_nat_to_real_fn() }, trace()));
elaborator elb(env, menv, ucs.size(), ucs.data());
substitution s = elb.next();
}
static void tst3() {
/*
Solve elaboration problem for
a : Int
(fun x, (f x) > 10) a
The following elaboration problem is created
(fun x : ?m1, ?m2 (f ?m3 x) (?m4 10)) (?m5 a)
?m2 in { Nat::Le, Int::Le, Real::Le }
?m4 in { Id, nat2int, nat2real }
?m5 in { Id, int2real }
*/
environment env;
import_all(env);
metavar_env menv;
buffer<unification_constraint> ucs;
type_checker checker(env);
expr A = Const("A");
expr f = Const("f");
env.add_var("f", Pi({A, Type()}, A >> A));
expr a = Const("a");
env.add_var("a", Int);
expr m1 = menv.mk_metavar();
expr m2 = menv.mk_metavar();
expr m3 = menv.mk_metavar();
expr m4 = menv.mk_metavar();
expr m5 = menv.mk_metavar();
expr int_id = Fun({a, Int}, a);
expr nat_id = Fun({a, Nat}, a);
expr x = Const("x");
expr F = Fun({x, m1}, m2(f(m3, x), m4(nVal(10))))(m5(a));
std::cout << F << "\n";
std::cout << checker.infer_type(F, context(), &menv, ucs) << "\n";
ucs.push_back(mk_choice_constraint(context(), m2, { mk_nat_le_fn(), mk_int_le_fn(), mk_real_le_fn() }, trace()));
ucs.push_back(mk_choice_constraint(context(), m4, { nat_id, mk_nat_to_int_fn(), mk_nat_to_real_fn() }, trace()));
ucs.push_back(mk_choice_constraint(context(), m5, { int_id, mk_int_to_real_fn() }, trace()));
elaborator elb(env, menv, ucs.size(), ucs.data());
substitution s = elb.next();
}
static void tst4() {
/*
Variable f {A : Type} (a : A) : A
Variable a : Int
Variable b : Real
(fun x y, (f x) > (f y)) a b
The following elaboration problem is created
(fun (x : ?m1) (y : ?m2), ?m3 (f ?m4 x) (f ?m5 y)) (?m6 a) b
?m3 in { Nat::Le, Int::Le, Real::Le }
?m6 in { Id, int2real }
*/
environment env;
import_all(env);
metavar_env menv;
buffer<unification_constraint> ucs;
type_checker checker(env);
expr A = Const("A");
expr f = Const("f");
env.add_var("f", Pi({A, Type()}, A >> A));
expr a = Const("a");
expr b = Const("b");
env.add_var("a", Int);
env.add_var("b", Real);
expr m1 = menv.mk_metavar();
expr m2 = menv.mk_metavar();
expr m3 = menv.mk_metavar();
expr m4 = menv.mk_metavar();
expr m5 = menv.mk_metavar();
expr m6 = menv.mk_metavar();
expr x = Const("x");
expr y = Const("y");
expr int_id = Fun({a, Int}, a);
expr F = Fun({{x, m1}, {y, m2}}, m3(f(m4, x), f(m5, y)))(m6(a), b);
std::cout << F << "\n";
std::cout << checker.infer_type(F, context(), &menv, ucs) << "\n";
ucs.push_back(mk_choice_constraint(context(), m3, { mk_nat_le_fn(), mk_int_le_fn(), mk_real_le_fn() }, trace()));
ucs.push_back(mk_choice_constraint(context(), m6, { int_id, mk_int_to_real_fn() }, trace()));
elaborator elb(env, menv, ucs.size(), ucs.data());
substitution s = elb.next();
}
static void tst5() {
/*
Variable f {A : Type} (a b : A) : Bool
Variable a : Int
Variable b : Real
(fun x y, f x y) a b
The following elaboration problem is created
(fun (x : ?m1) (y : ?m2), (f ?m3 x y)) (?m4 a) b
?m4 in { Id, int2real }
*/
environment env;
import_all(env);
metavar_env menv;
buffer<unification_constraint> ucs;
type_checker checker(env);
expr A = Const("A");
expr f = Const("f");
env.add_var("f", Pi({A, Type()}, A >> (A >> A)));
expr a = Const("a");
expr b = Const("b");
env.add_var("a", Int);
env.add_var("b", Real);
expr m1 = menv.mk_metavar();
expr m2 = menv.mk_metavar();
expr m3 = menv.mk_metavar();
expr m4 = menv.mk_metavar();
expr x = Const("x");
expr y = Const("y");
expr int_id = Fun({a, Int}, a);
expr F = Fun({{x, m1}, {y, m2}}, f(m3, x, y))(m4(a), b);
std::cout << F << "\n";
std::cout << checker.infer_type(F, context(), &menv, ucs) << "\n";
ucs.push_back(mk_choice_constraint(context(), m4, { int_id, mk_int_to_real_fn() }, trace()));
elaborator elb(env, menv, ucs.size(), ucs.data());
substitution s = elb.next();
}
static void tst6() {
/*
Subst : Π (A : Type U) (a b : A) (P : A Bool), (P a) (a = b) (P b)
f : Int -> Int -> Int
a : Int
b : Int
H1 : (f a (f a b)) == a
H2 : a = b
Theorem T : (f a (f b b)) == a := Subst _ _ _ _ H1 H2
*/
environment env;
import_all(env);
metavar_env menv;
buffer<unification_constraint> ucs;
type_checker checker(env);
expr f = Const("f");
expr a = Const("a");
expr b = Const("b");
expr H1 = Const("H1");
expr H2 = Const("H2");
expr m1 = menv.mk_metavar();
expr m2 = menv.mk_metavar();
expr m3 = menv.mk_metavar();
expr m4 = menv.mk_metavar();
env.add_var("f", Int >> (Int >> Int));
env.add_var("a", Int);
env.add_var("b", Int);
env.add_axiom("H1", Eq(f(a, f(a, b)), a));
env.add_axiom("H2", Eq(a, b));
expr V = Subst(m1, m2, m3, m4, H1, H2);
expr expected = Eq(f(a, f(b, b)), a);
expr given = checker.infer_type(V, context(), &menv, ucs);
ucs.push_back(mk_eq_constraint(context(), expected, given, trace()));
elaborator elb(env, menv, ucs.size(), ucs.data());
substitution s = elb.next();
}
int main() { int main() {
tst1(); tst1();
tst2();
tst3();
tst4();
tst5();
tst6();
return has_violations() ? 1 : 0; return has_violations() ? 1 : 0;
} }