lean4-htt/src/library/expr_lt.cpp
Leonardo de Moura 85092412c7 refactor: remove Expr.FVar hack
@Kha @dselsam:
This hack was preventing us from making `Expr` a "real" Lean type.
This was bad for a few reasons:
- It was hard to extend/modify `Expr` in Lean since we would also have
to modify the C++ code that creates the `Expr` objects with the hidden
fields.
- `Expr.lam` and `Expr.forallE` were not following the Lean layout
standard where we sort fields by size. @Kha: recall we used that to
avoid a UB. The issue with `Expr.lam` and `Expr.forallE` is that they
have a "visible" field (`BinderInfo`), which is smaller than
hidden fields such as hash code.
- `Expr.fvar` had only one field at `Expr.lean,` but four behind the
scenes.

I added a new constructor `Local` that is only accessible from C++.
It is only used in legacy code we inherited from Lean2.
We will eventually delete it.

This refactoring was quite painful since many parts of the codebase
were mixing the new `Expr.fvar` with the old `Expr.local`.
I doubt I would be able to do it without the new staging framework
@Kha built.

BTW, some of the patches are horrible. I didn't care much since we
are going to deleted the super ugly files. That being said,
you should expect new weird bevaior due to `Expr.fvar` vs `Expr.local`.

Next step: use the new `ExprCachedData` to make all `Expr` hidden visibles
accessible from Lean.

checkpoint
2019-11-15 14:04:26 -08:00

203 lines
7.3 KiB
C++

/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#include "kernel/expr.h"
#include "library/expr_lt.h"
namespace lean {
bool is_lt(expr const & a, expr const & b, bool use_hash, local_context const * lctx) {
if (is_eqp(a, b)) return false;
if (a.kind() != b.kind()) return a.kind() < b.kind();
if (use_hash) {
if (hash(a) < hash(b)) return true;
if (hash(a) > hash(b)) return false;
}
if (a == b) return false;
switch (a.kind()) {
case expr_kind::Lit:
return lit_value(a) < lit_value(b);
case expr_kind::BVar:
return bvar_idx(a) < bvar_idx(b);
case expr_kind::MData:
if (mdata_expr(a) != mdata_expr(b))
return is_lt(mdata_expr(a), mdata_expr(b), use_hash, lctx);
else
return mdata_data(a) < mdata_data(b);
case expr_kind::Proj:
if (proj_expr(a) != proj_expr(b))
return is_lt(proj_expr(a), proj_expr(b), use_hash, lctx);
else if (proj_sname(a) != proj_sname(b))
return proj_sname(a) < proj_sname(b);
else
return proj_idx(a) < proj_idx(b);
case expr_kind::Const:
if (const_name(a) != const_name(b))
return const_name(a) < const_name(b);
else
return is_lt(const_levels(a), const_levels(b), use_hash);
case expr_kind::App:
if (app_fn(a) != app_fn(b))
return is_lt(app_fn(a), app_fn(b), use_hash, lctx);
else
return is_lt(app_arg(a), app_arg(b), use_hash, lctx);
case expr_kind::Lambda: case expr_kind::Pi:
if (binding_domain(a) != binding_domain(b))
return is_lt(binding_domain(a), binding_domain(b), use_hash, lctx);
else
return is_lt(binding_body(a), binding_body(b), use_hash, lctx);
case expr_kind::Let:
if (let_type(a) != let_type(b))
return is_lt(let_type(a), let_type(b), use_hash, lctx);
else if (let_value(a) != let_value(b))
return is_lt(let_value(a), let_value(b), use_hash, lctx);
else
return is_lt(let_body(a), let_body(b), use_hash, lctx);
case expr_kind::Sort:
return is_lt(sort_level(a), sort_level(b), use_hash);
case expr_kind::FVar:
if (lctx) {
if (auto d1 = lctx->find_local_decl(a))
if (auto d2 = lctx->find_local_decl(b))
return d1->get_idx() < d2->get_idx();
}
return fvar_name(a) < fvar_name(b);
case expr_kind::MVar:
return mvar_name(a) < mvar_name(b);
case expr_kind::Local:
return local_name(a) < local_name(b);
}
lean_unreachable(); // LCOV_EXCL_LINE
}
bool is_lt_no_level_params(level const & a, level const & b) {
if (is_eqp(a, b)) return false;
if (kind(a) != kind(b)) {
if (kind(a) == level_kind::Param || kind(b) == level_kind::Param)
return false;
return kind(a) < kind(b);
}
switch (kind(a)) {
case level_kind::Zero:
lean_unreachable(); // LCOV_EXCL_LINE
case level_kind::Param:
return false;
case level_kind::MVar:
return mvar_id(a) < mvar_id(b);
case level_kind::Max:
if (is_lt_no_level_params(max_lhs(a), max_lhs(b)))
return true;
else if (is_lt_no_level_params(max_lhs(b), max_lhs(a)))
return false;
else
return is_lt_no_level_params(max_rhs(a), max_rhs(b));
case level_kind::IMax:
if (is_lt_no_level_params(imax_lhs(a), imax_lhs(b)))
return true;
else if (is_lt_no_level_params(imax_lhs(b), imax_lhs(a)))
return false;
else
return is_lt_no_level_params(imax_rhs(a), imax_rhs(b));
case level_kind::Succ:
return is_lt_no_level_params(succ_of(a), succ_of(b));
}
lean_unreachable();
}
bool is_lt_no_level_params(levels const & as, levels const & bs) {
if (is_nil(as))
return !is_nil(bs);
else if (is_nil(bs))
return false;
else if (is_lt_no_level_params(car(as), car(bs)))
return true;
else if (is_lt_no_level_params(car(bs), car(as)))
return false;
else
return is_lt_no_level_params(cdr(as), cdr(bs));
}
bool is_lt_no_level_params(expr const & a, expr const & b) {
if (is_eqp(a, b)) return false;
if (a.kind() != b.kind()) return a.kind() < b.kind();
switch (a.kind()) {
case expr_kind::Lit:
return lit_value(a) < lit_value(b);
case expr_kind::BVar:
return bvar_idx(a) < bvar_idx(b);
case expr_kind::MData:
if (mdata_expr(a) != mdata_expr(b))
return is_lt_no_level_params(mdata_expr(a), mdata_expr(b));
else
return mdata_data(a) < mdata_data(b);
case expr_kind::Proj:
if (proj_expr(a) != proj_expr(b))
return is_lt_no_level_params(proj_expr(a), proj_expr(b));
else if (proj_sname(a) != proj_sname(b))
return proj_sname(a) < proj_sname(b);
else
return proj_idx(a) < proj_idx(b);
case expr_kind::Const:
if (const_name(a) != const_name(b))
return const_name(a) < const_name(b);
else
return is_lt_no_level_params(const_levels(a), const_levels(b));
case expr_kind::App:
if (is_lt_no_level_params(app_fn(a), app_fn(b)))
return true;
else if (is_lt_no_level_params(app_fn(b), app_fn(a)))
return false;
else
return is_lt_no_level_params(app_arg(a), app_arg(b));
case expr_kind::Lambda: case expr_kind::Pi:
if (is_lt_no_level_params(binding_domain(a), binding_domain(b)))
return true;
else if (is_lt_no_level_params(binding_domain(b), binding_domain(a)))
return false;
else
return is_lt_no_level_params(binding_body(a), binding_body(b));
case expr_kind::Let:
if (is_lt_no_level_params(let_type(a), let_type(b)))
return true;
else if (is_lt_no_level_params(let_type(b), let_type(a)))
return false;
else if (is_lt_no_level_params(let_value(a), let_value(b)))
return true;
else if (is_lt_no_level_params(let_value(b), let_value(a)))
return false;
else
return is_lt_no_level_params(let_body(a), let_body(b));
case expr_kind::Sort:
return is_lt_no_level_params(sort_level(a), sort_level(b));
case expr_kind::Local:
if (local_name(a) != local_name(b))
return local_name(a) < local_name(b);
else
return is_lt_no_level_params(local_type(a), local_type(b));
case expr_kind::FVar:
return fvar_name(a) < fvar_name(b);
case expr_kind::MVar:
return mvar_name(a) < mvar_name(b);
}
lean_unreachable();
}
int expr_cmp_no_level_params::operator()(expr const & e1, expr const & e2) const {
if (is_lt_no_level_params(e1, e2))
return -1;
else if (is_lt_no_level_params(e2, e1))
return 1;
else
return 0;
}
extern "C" uint8 lean_expr_quick_lt(b_obj_arg a, b_obj_arg b) {
return is_lt(expr(a, true), expr(b, true), true, nullptr);
}
extern "C" uint8 lean_expr_lt(b_obj_arg a, b_obj_arg b) {
return is_lt(expr(a, true), expr(b, true), false, nullptr);
}
}