feat(library/unifier): handle 'first-order' flex-flex constraints

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2014-07-11 04:51:28 +01:00
parent 024299f56b
commit 1230e942aa
2 changed files with 29 additions and 3 deletions

View file

@ -1122,9 +1122,27 @@ struct unifier_fn {
return process_flex_rigid(cnstr_rhs_expr(c), cnstr_lhs_expr(c), c.get_justification()); return process_flex_rigid(cnstr_rhs_expr(c), cnstr_lhs_expr(c), c.get_justification());
} }
bool process_flex_flex(constraint const &) { bool process_flex_flex(constraint const & c) {
// We just ignore flex-flex constraints. expr const & lhs = cnstr_lhs_expr(c);
// This kind of constraint does not occur very often. expr const & rhs = cnstr_rhs_expr(c);
// We ignore almost all flex-flex constraints.
// We just handle flex_flex "first-order" case
// ?M_1 l_1 ... l_k =?= ?M_2 l_1 ... l_k
if (!is_simple_meta(lhs) || !is_simple_meta(rhs))
return true;
buffer<expr> lhs_args, rhs_args;
expr ml = get_app_args(lhs, lhs_args);
expr mr = get_app_args(rhs, rhs_args);
if (ml == mr || lhs_args.size() != rhs_args.size())
return true;
lean_assert(!m_subst.is_assigned(ml));
lean_assert(!m_subst.is_assigned(mr));
unsigned i = 0;
for (; i < lhs_args.size(); i++)
if (lhs_args[i] != rhs_args[i])
break;
if (i == lhs_args.size())
return assign(ml, mr, c.get_justification());
return true; return true;
} }

View file

@ -0,0 +1,8 @@
import standard
using num tactic
variable p : num → num → num → Bool
axiom H1 : ∃ x y z, p x y z
axiom H2 : ∀ {x y z : num}, p x y z → p x x x
theorem tst : ∃ x, p x x x
:= obtains a b c H [fact], from H1,
by (apply exists_intro; apply H2; eassumption)