refactor(library/unifier): process_flex_rigid method before adding yet another case split
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
8e222e6244
commit
1a8a2b0811
1 changed files with 56 additions and 18 deletions
|
@ -1327,22 +1327,17 @@ struct unifier_fn {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Process a flex rigid constraint */
|
/** \brief Append the auxiliary constraints \c aux to each alternative in \c alts */
|
||||||
bool process_flex_rigid(expr const & lhs, expr const & rhs, justification const & j, bool relax) {
|
void append_auxiliary_constraints(buffer<constraints> & alts, list<constraint> const & aux) {
|
||||||
lean_assert(is_meta(lhs));
|
if (is_nil(aux))
|
||||||
lean_assert(!is_meta(rhs));
|
return;
|
||||||
|
for (constraints & cs : alts)
|
||||||
|
cs = append(aux, cs);
|
||||||
|
}
|
||||||
|
|
||||||
|
void process_flex_rigid_core(expr const & lhs, expr const & rhs, justification const & j, bool relax, buffer<constraints> & alts) {
|
||||||
buffer<expr> margs;
|
buffer<expr> margs;
|
||||||
expr m = get_app_args(lhs, margs);
|
expr m = get_app_args(lhs, margs);
|
||||||
for (expr & marg : margs) {
|
|
||||||
// Make sure that if marg is reducible to a local constant, then it is replaced with it.
|
|
||||||
// We need that because of the optimization based on is_easy_flex_rigid_arg
|
|
||||||
if (!is_local(marg)) {
|
|
||||||
expr new_marg = m_tc[relax]->whnf(marg);
|
|
||||||
if (is_local(new_marg))
|
|
||||||
marg = new_marg;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
buffer<constraints> alts;
|
|
||||||
switch (rhs.kind()) {
|
switch (rhs.kind()) {
|
||||||
case expr_kind::Var: case expr_kind::Meta:
|
case expr_kind::Var: case expr_kind::Meta:
|
||||||
lean_unreachable(); // LCOV_EXCL_LINE
|
lean_unreachable(); // LCOV_EXCL_LINE
|
||||||
|
@ -1374,16 +1369,59 @@ struct unifier_fn {
|
||||||
else
|
else
|
||||||
mk_simple_nonlocal_projection(m, margs, i, rhs, j, alts, relax);
|
mk_simple_nonlocal_projection(m, margs, i, rhs, j, alts, relax);
|
||||||
}
|
}
|
||||||
} else if (is_constant(f)) {
|
} else {
|
||||||
|
lean_assert(is_constant(f));
|
||||||
mk_simple_projections(m, margs, rhs, j, alts, relax);
|
mk_simple_projections(m, margs, rhs, j, alts, relax);
|
||||||
mk_flex_rigid_app_cnstrs(m, margs, f, rhs, j, alts, relax);
|
mk_flex_rigid_app_cnstrs(m, margs, f, rhs, j, alts, relax);
|
||||||
} else {
|
}
|
||||||
|
break;
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** \brief When lhs is an application (f ...), make sure that if any argument that is reducible to a
|
||||||
|
local constant is replaced with a local constant.
|
||||||
|
|
||||||
|
\remark We store auxiliary constraints created in the reductions in \c aux. We return the new
|
||||||
|
"reduce" application.
|
||||||
|
|
||||||
|
\remark We need this step because of the optimization based on is_easy_flex_rigid_arg
|
||||||
|
*/
|
||||||
|
expr expose_local_args(expr const & lhs, bool relax, buffer<constraint> & aux) {
|
||||||
|
buffer<expr> margs;
|
||||||
|
expr m = get_app_args(lhs, margs);
|
||||||
|
bool modified = false;
|
||||||
|
for (expr & marg : margs) {
|
||||||
|
// Make sure that if marg is reducible to a local constant, then it is replaced with it.
|
||||||
|
// We need that because of the optimization based on is_easy_flex_rigid_arg
|
||||||
|
if (!is_local(marg)) {
|
||||||
|
expr new_marg = m_tc[relax]->whnf(marg, aux);
|
||||||
|
if (is_local(new_marg)) {
|
||||||
|
marg = new_marg;
|
||||||
|
modified = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return modified ? mk_app(m, margs) : lhs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** \brief Process a flex rigid constraint */
|
||||||
|
bool process_flex_rigid(expr lhs, expr const & rhs, justification const & j, bool relax) {
|
||||||
|
lean_assert(is_meta(lhs));
|
||||||
|
lean_assert(!is_meta(rhs));
|
||||||
|
if (is_app(rhs)) {
|
||||||
|
expr const & f = get_app_fn(rhs);
|
||||||
|
if (!is_local(f) && !is_constant(f)) {
|
||||||
expr new_rhs = m_tc[relax]->whnf(rhs);
|
expr new_rhs = m_tc[relax]->whnf(rhs);
|
||||||
lean_assert(new_rhs != rhs);
|
lean_assert(new_rhs != rhs);
|
||||||
return is_def_eq(lhs, new_rhs, j, relax);
|
return is_def_eq(lhs, new_rhs, j, relax);
|
||||||
}
|
}
|
||||||
break;
|
}
|
||||||
}}
|
|
||||||
|
buffer<constraint> aux;
|
||||||
|
lhs = expose_local_args(lhs, relax, aux);
|
||||||
|
buffer<constraints> alts;
|
||||||
|
process_flex_rigid_core(lhs, rhs, j, relax, alts);
|
||||||
|
append_auxiliary_constraints(alts, to_list(aux.begin(), aux.end()));
|
||||||
|
|
||||||
if (alts.empty()) {
|
if (alts.empty()) {
|
||||||
set_conflict(j);
|
set_conflict(j);
|
||||||
|
|
Loading…
Reference in a new issue