/* Copyright (c) 2013-2014 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura Soonho Kong */ #include #include #include #include #include #include "runtime/hash.h" #include "util/list_fn.h" #include "util/buffer.h" #include "kernel/expr.h" #include "kernel/expr_eq_fn.h" #include "kernel/expr_sets.h" #include "kernel/for_each_fn.h" #include "kernel/replace_fn.h" #include "kernel/abstract.h" #include "kernel/instantiate.h" namespace lean { /* Expression literal values */ literal::literal(char const * v): object_ref(mk_cnstr(static_cast(literal_kind::String), mk_string(v))) { } literal::literal(unsigned v): object_ref(mk_cnstr(static_cast(literal_kind::Nat), mk_nat_obj(v))) { } literal::literal(mpz const & v): object_ref(mk_cnstr(static_cast(literal_kind::Nat), mk_nat_obj(v))) { } literal::literal(nat const & v): object_ref(mk_cnstr(static_cast(literal_kind::Nat), v)) { } bool operator==(literal const & a, literal const & b) { if (a.kind() != b.kind()) return false; switch (a.kind()) { case literal_kind::String: return a.get_string() == b.get_string(); case literal_kind::Nat: return a.get_nat() == b.get_nat(); } lean_unreachable(); } bool operator<(literal const & a, literal const & b) { if (a.kind() != b.kind()) return static_cast(a.kind()) < static_cast(b.kind()); switch (a.kind()) { case literal_kind::String: return a.get_string() < b.get_string(); case literal_kind::Nat: return a.get_nat() < b.get_nat(); } lean_unreachable(); } bool is_atomic(expr const & e) { switch (e.kind()) { case expr_kind::Const: case expr_kind::Sort: case expr_kind::BVar: case expr_kind::Lit: case expr_kind::MVar: case expr_kind::FVar: return true; case expr_kind::App: case expr_kind::Lambda: case expr_kind::Pi: case expr_kind::Let: case expr_kind::MData: case expr_kind::Proj: case expr_kind::Local: return false; } lean_unreachable(); // LCOV_EXCL_LINE } extern "C" uint8 lean_expr_binder_info(object * e); binder_info binding_info(expr const & e) { return static_cast(lean_expr_binder_info(e.to_obj_arg())); } extern "C" object * lean_lit_type(obj_arg e); expr lit_type(literal const & lit) { return expr(lean_lit_type(lit.to_obj_arg())); } extern "C" usize lean_expr_hash(obj_arg e); unsigned hash(expr const & e) { return lean_expr_hash(e.to_obj_arg()); } extern "C" uint8 lean_expr_has_fvar(obj_arg e); bool has_fvar(expr const & e) { return lean_expr_has_fvar(e.to_obj_arg()); } extern "C" uint8 lean_expr_has_expr_mvar(obj_arg e); bool has_expr_mvar(expr const & e) { return lean_expr_has_expr_mvar(e.to_obj_arg()); } extern "C" uint8 lean_expr_has_level_mvar(obj_arg e); bool has_univ_mvar(expr const & e) { return lean_expr_has_level_mvar(e.to_obj_arg()); } extern "C" uint8 lean_expr_has_level_param(obj_arg e); bool has_univ_param(expr const & e) { return lean_expr_has_level_param(e.to_obj_arg()); } extern "C" unsigned lean_expr_loose_bvar_range(object * e); unsigned get_loose_bvar_range(expr const & e) { return lean_expr_loose_bvar_range(e.to_obj_arg()); } // ======================================= // Constructors static expr * g_dummy = nullptr; static expr const & get_dummy() { if (!g_dummy) { g_dummy = new expr(mk_constant("__expr_for_default_constructor__")); } return *g_dummy; } expr::expr():expr(get_dummy()) {} extern "C" object * lean_expr_mk_lit(obj_arg l); expr mk_lit(literal const & l) { return expr(lean_expr_mk_lit(l.to_obj_arg())); } extern "C" object * lean_expr_mk_mdata(obj_arg m, obj_arg e); expr mk_mdata(kvmap const & m, expr const & e) { return expr(lean_expr_mk_mdata(m.to_obj_arg(), e.to_obj_arg())); } extern "C" object * lean_expr_mk_proj(obj_arg s, obj_arg idx, obj_arg e); expr mk_proj(name const & s, nat const & idx, expr const & e) { return expr(lean_expr_mk_proj(s.to_obj_arg(), idx.to_obj_arg(), e.to_obj_arg())); } extern "C" object * lean_expr_mk_bvar(obj_arg idx); expr mk_bvar(nat const & idx) { return expr(lean_expr_mk_bvar(idx.to_obj_arg())); } extern "C" object * lean_expr_mk_fvar(obj_arg n); expr mk_fvar(name const & n) { return expr(lean_expr_mk_fvar(n.to_obj_arg())); } extern "C" object * lean_expr_mk_mvar(object * n); expr mk_mvar(name const & n) { return expr(lean_expr_mk_mvar(n.to_obj_arg())); } extern "C" object * lean_expr_mk_const(obj_arg n, obj_arg ls); expr mk_const(name const & n, levels const & ls) { return expr(lean_expr_mk_const(n.to_obj_arg(), ls.to_obj_arg())); } extern "C" object * lean_expr_mk_app(obj_arg f, obj_arg a); expr mk_app(expr const & f, expr const & a) { return expr(lean_expr_mk_app(f.to_obj_arg(), a.to_obj_arg())); } extern "C" object * lean_expr_mk_sort(obj_arg l); expr mk_sort(level const & l) { return expr(lean_expr_mk_sort(l.to_obj_arg())); } extern "C" object * lean_expr_mk_lambda(obj_arg n, obj_arg t, obj_arg e, uint8 bi); expr mk_lambda(name const & n, expr const & t, expr const & e, binder_info bi) { return expr(lean_expr_mk_lambda(n.to_obj_arg(), t.to_obj_arg(), e.to_obj_arg(), static_cast(bi))); } extern "C" object * lean_expr_mk_forall(obj_arg n, obj_arg t, obj_arg e, uint8 bi); expr mk_pi(name const & n, expr const & t, expr const & e, binder_info bi) { return expr(lean_expr_mk_forall(n.to_obj_arg(), t.to_obj_arg(), e.to_obj_arg(), static_cast(bi))); } static name * g_default_name = nullptr; expr mk_arrow(expr const & t, expr const & e) { return mk_pi(*g_default_name, t, e, mk_binder_info()); } extern "C" object * lean_expr_mk_let(object * n, object * t, object * v, object * b); expr mk_let(name const & n, expr const & t, expr const & v, expr const & b) { return expr(lean_expr_mk_let(n.to_obj_arg(), t.to_obj_arg(), v.to_obj_arg(), b.to_obj_arg())); } static expr * g_Prop = nullptr; static expr * g_Type0 = nullptr; expr mk_Prop() { return *g_Prop; } expr mk_Type() { return *g_Type0; } // ======================================= // Auxiliary constructors and accessors expr mk_app(expr const & f, unsigned num_args, expr const * args) { expr r = f; for (unsigned i = 0; i < num_args; i++) r = mk_app(r, args[i]); return r; } expr mk_app(unsigned num_args, expr const * args) { lean_assert(num_args >= 2); return mk_app(mk_app(args[0], args[1]), num_args - 2, args+2); } expr mk_app(expr const & f, list const & args) { buffer _args; to_buffer(args, _args); return mk_app(f, _args); } expr mk_rev_app(expr const & f, unsigned num_args, expr const * args) { expr r = f; unsigned i = num_args; while (i > 0) { --i; r = mk_app(r, args[i]); } return r; } expr mk_rev_app(unsigned num_args, expr const * args) { lean_assert(num_args >= 2); return mk_rev_app(mk_app(args[num_args-1], args[num_args-2]), num_args-2, args); } expr const & get_app_args(expr const & e, buffer & args) { unsigned sz = args.size(); expr const * it = &e; while (is_app(*it)) { args.push_back(app_arg(*it)); it = &(app_fn(*it)); } std::reverse(args.begin() + sz, args.end()); return *it; } expr const & get_app_args_at_most(expr const & e, unsigned num, buffer & args) { unsigned sz = args.size(); expr const * it = &e; unsigned i = 0; while (is_app(*it)) { if (i == num) break; args.push_back(app_arg(*it)); it = &(app_fn(*it)); i++; } std::reverse(args.begin() + sz, args.end()); return *it; } expr const & get_app_rev_args(expr const & e, buffer & args) { expr const * it = &e; while (is_app(*it)) { args.push_back(app_arg(*it)); it = &(app_fn(*it)); } return *it; } expr const & get_app_fn(expr const & e) { expr const * it = &e; while (is_app(*it)) { it = &(app_fn(*it)); } return *it; } unsigned get_app_num_args(expr const & e) { expr const * it = &e; unsigned n = 0; while (is_app(*it)) { it = &(app_fn(*it)); n++; } return n; } bool is_arrow(expr const & t) { if (!is_pi(t)) return false; if (has_loose_bvars(t)) { return !has_loose_bvar(binding_body(t), 0); } else { lean_assert(has_loose_bvars(binding_body(t)) == has_loose_bvar(binding_body(t), 0)); return !has_loose_bvars(binding_body(t)); } } bool is_default_var_name(name const & n) { return n == *g_default_name; } // ======================================= // Update expr update_mdata(expr const & e, expr const & t) { if (!is_eqp(mdata_expr(e), t)) return mk_mdata(mdata_data(e), t); else return e; } expr update_proj(expr const & e, expr const & t) { if (!is_eqp(proj_expr(e), t)) return mk_proj(proj_sname(e), proj_idx(e), t); else return e; } expr update_app(expr const & e, expr const & new_fn, expr const & new_arg) { if (!is_eqp(app_fn(e), new_fn) || !is_eqp(app_arg(e), new_arg)) return mk_app(new_fn, new_arg); else return e; } expr update_binding(expr const & e, expr const & new_domain, expr const & new_body) { if (!is_eqp(binding_domain(e), new_domain) || !is_eqp(binding_body(e), new_body)) return mk_binding(e.kind(), binding_name(e), new_domain, new_body, binding_info(e)); else return e; } expr update_binding(expr const & e, expr const & new_domain, expr const & new_body, binder_info bi) { if (!is_eqp(binding_domain(e), new_domain) || !is_eqp(binding_body(e), new_body) || bi != binding_info(e)) return mk_binding(e.kind(), binding_name(e), new_domain, new_body, bi); else return e; } expr update_sort(expr const & e, level const & new_level) { if (!is_eqp(sort_level(e), new_level)) return mk_sort(new_level); else return e; } expr update_const(expr const & e, levels const & new_levels) { if (!is_eqp(const_levels(e), new_levels)) return mk_const(const_name(e), new_levels); else return e; } expr update_let(expr const & e, expr const & new_type, expr const & new_value, expr const & new_body) { if (!is_eqp(let_type(e), new_type) || !is_eqp(let_value(e), new_value) || !is_eqp(let_body(e), new_body)) return mk_let(let_name(e), new_type, new_value, new_body); else return e; } extern "C" object * lean_expr_update_mdata(obj_arg e, obj_arg new_expr) { if (mdata_expr(TO_REF(expr, e)).raw() != new_expr) { object * r = lean_expr_mk_mdata(mdata_data(TO_REF(expr, e)).to_obj_arg(), new_expr); lean_dec_ref(e); return r; } else { lean_dec_ref(new_expr); return e; } } extern "C" object * lean_expr_update_const(obj_arg e, obj_arg new_levels) { if (const_levels(TO_REF(expr, e)).raw() != new_levels) { object * r = lean_expr_mk_const(const_name(TO_REF(expr, e)).to_obj_arg(), new_levels); lean_dec_ref(e); return r; } else { lean_dec(new_levels); return e; } } extern "C" object * lean_expr_update_sort(obj_arg e, obj_arg new_level) { if (sort_level(TO_REF(expr, e)).raw() != new_level) { object * r = lean_expr_mk_sort(new_level); lean_dec_ref(e); return r; } else { lean_dec(new_level); return e; } } extern "C" object * lean_expr_update_proj(obj_arg e, obj_arg new_expr) { if (proj_expr(TO_REF(expr, e)).raw() != new_expr) { object * r = lean_expr_mk_proj(proj_sname(TO_REF(expr, e)).to_obj_arg(), proj_idx(TO_REF(expr, e)).to_obj_arg(), new_expr); lean_dec_ref(e); return r; } else { lean_dec_ref(new_expr); return e; } } extern "C" object * lean_expr_update_app(obj_arg e, obj_arg new_fn, obj_arg new_arg) { if (app_fn(TO_REF(expr, e)).raw() != new_fn || app_arg(TO_REF(expr, e)).raw() != new_arg) { object * r = lean_expr_mk_app(new_fn, new_arg); lean_dec_ref(e); return r; } else { lean_dec_ref(new_fn); lean_dec_ref(new_arg); return e; } } extern "C" object * lean_expr_update_forall(obj_arg e, uint8 new_binfo, obj_arg new_domain, obj_arg new_body) { if (binding_domain(TO_REF(expr, e)).raw() != new_domain || binding_body(TO_REF(expr, e)).raw() != new_body || binding_info(TO_REF(expr, e)) != static_cast(new_binfo)) { object * r = lean_expr_mk_forall(binding_name(TO_REF(expr, e)).to_obj_arg(), new_domain, new_body, new_binfo); lean_dec_ref(e); return r; } else { lean_dec_ref(new_domain); lean_dec_ref(new_body); return e; } } extern "C" object * lean_expr_update_lambda(obj_arg e, uint8 new_binfo, obj_arg new_domain, obj_arg new_body) { if (binding_domain(TO_REF(expr, e)).raw() != new_domain || binding_body(TO_REF(expr, e)).raw() != new_body || binding_info(TO_REF(expr, e)) != static_cast(new_binfo)) { object * r = lean_expr_mk_lambda(binding_name(TO_REF(expr, e)).to_obj_arg(), new_domain, new_body, new_binfo); lean_dec_ref(e); return r; } else { lean_dec_ref(new_domain); lean_dec_ref(new_body); return e; } } extern "C" object * lean_expr_update_let(obj_arg e, obj_arg new_type, obj_arg new_val, obj_arg new_body) { if (let_type(TO_REF(expr, e)).raw() != new_type || let_value(TO_REF(expr, e)).raw() != new_val || let_body(TO_REF(expr, e)).raw() != new_body) { object * r = lean_expr_mk_let(let_name(TO_REF(expr, e)).to_obj_arg(), new_type, new_val, new_body); lean_dec_ref(e); return r; } else { lean_dec_ref(new_type); lean_dec_ref(new_val); lean_dec_ref(new_body); return e; } } // ======================================= // Loose bound variable management static bool has_loose_bvars_in_domain(expr const & b, unsigned vidx, bool strict) { if (is_pi(b)) { return (has_loose_bvar(binding_domain(b), vidx) && is_explicit(binding_info(b))) || has_loose_bvars_in_domain(binding_body(b), vidx+1, strict); } else if (!strict) { return has_loose_bvar(b, vidx); } else { return false; } } bool has_loose_bvar(expr const & e, unsigned i) { bool found = false; for_each(e, [&](expr const & e, unsigned offset) { if (found) return false; // already found unsigned n_i = i + offset; if (n_i < i) return false; // overflow, vidx can't be >= max unsigned if (n_i >= get_loose_bvar_range(e)) return false; // expression e does not contain bound variables with idx >= n_i if (is_var(e)) { nat const & vidx = bvar_idx(e); if (vidx == n_i) found = true; } return true; // continue search }); return found; } extern "C" uint8 lean_expr_has_loose_bvar(b_obj_arg e, b_obj_arg i) { if (!lean_is_scalar(i)) return false; return has_loose_bvar(TO_REF(expr, e), lean_unbox(i)); } expr lower_loose_bvars(expr const & e, unsigned s, unsigned d) { if (d == 0 || s >= get_loose_bvar_range(e)) return e; lean_assert(s >= d); return replace(e, [=](expr const & e, unsigned offset) -> optional { unsigned s1 = s + offset; if (s1 < s) return some_expr(e); // overflow, vidx can't be >= max unsigned if (s1 >= get_loose_bvar_range(e)) return some_expr(e); // expression e does not contain bound variables with idx >= s1 if (is_bvar(e) && bvar_idx(e) >= s1) { lean_assert(bvar_idx(e) >= offset + d); return some_expr(mk_bvar(bvar_idx(e) - nat(d))); } else { return none_expr(); } }); } expr lower_loose_bvars(expr const & e, unsigned d) { return lower_loose_bvars(e, d, d); } extern "C" object * lean_expr_lower_loose_bvars(b_obj_arg e, b_obj_arg s, b_obj_arg d) { if (!lean_is_scalar(s) || !lean_is_scalar(d) || lean_unbox(s) < lean_unbox(d)) { lean_inc(e); return e; } return lower_loose_bvars(TO_REF(expr, e), lean_unbox(s), lean_unbox(d)).steal(); } expr lift_loose_bvars(expr const & e, unsigned s, unsigned d) { if (d == 0 || s >= get_loose_bvar_range(e)) return e; return replace(e, [=](expr const & e, unsigned offset) -> optional { unsigned s1 = s + offset; if (s1 < s) return some_expr(e); // overflow, vidx can't be >= max unsigned if (s1 >= get_loose_bvar_range(e)) return some_expr(e); // expression e does not contain bound variables with idx >= s1 if (is_var(e) && bvar_idx(e) >= s + offset) { return some_expr(mk_bvar(bvar_idx(e) + nat(d))); } else { return none_expr(); } }); } expr lift_loose_bvars(expr const & e, unsigned d) { return lift_loose_bvars(e, 0, d); } extern "C" object * lean_expr_lift_loose_bvars(b_obj_arg e, b_obj_arg s, b_obj_arg d) { if (!lean_is_scalar(s) || !lean_is_scalar(d)) { lean_inc(e); return e; } return lift_loose_bvars(TO_REF(expr, e), lean_unbox(s), lean_unbox(d)).steal(); } // ======================================= // Implicit argument inference expr infer_implicit(expr const & t, unsigned num_params, bool strict) { if (num_params == 0) { return t; } else if (is_pi(t)) { expr new_body = infer_implicit(binding_body(t), num_params-1, strict); if (!is_explicit(binding_info(t))) { // argument is already marked as implicit return update_binding(t, binding_domain(t), new_body); } else if (has_loose_bvars_in_domain(new_body, 0, strict)) { return update_binding(t, binding_domain(t), new_body, mk_implicit_binder_info()); } else { return update_binding(t, binding_domain(t), new_body); } } else { return t; } } expr infer_implicit(expr const & t, bool strict) { return infer_implicit(t, std::numeric_limits::max(), strict); } // ======================================= // Initialization & Finalization void initialize_expr() { get_dummy(); g_default_name = new name("a"); g_Type0 = new expr(mk_sort(mk_level_one())); g_Prop = new expr(mk_sort(mk_level_zero())); /* TODO(Leo): add support for builtin constants in the kernel. Something similar to what we have in the library directory. */ } void finalize_expr() { delete g_Prop; delete g_Type0; delete g_dummy; delete g_default_name; } // ======================================= // Legacy binder_info local_info(expr const & e) { lean_assert(is_local(e)); return static_cast(lean_expr_binder_info(e.to_obj_arg())); } extern "C" object * lean_expr_mk_local(obj_arg n, obj_arg u, obj_arg t, uint8 bi); expr mk_local(name const & n, name const & pp_n, expr const & t, binder_info bi) { return expr(lean_expr_mk_local(n.to_obj_arg(), pp_n.to_obj_arg(), t.to_obj_arg(), static_cast(bi))); } expr update_local(expr const & e, expr const & new_type, binder_info bi) { if (is_eqp(local_type(e), new_type) && local_info(e) == bi) return e; else return mk_local(local_name(e), local_pp_name(e), new_type, bi); } expr update_local(expr const & e, binder_info bi) { return update_local(e, local_type(e), bi); } expr update_local(expr const & e, expr const & new_type) { if (is_eqp(local_type(e), new_type)) return e; else return mk_local(local_name(e), local_pp_name(e), new_type, local_info(e)); } optional has_expr_metavar_strict(expr const & e) { if (!has_expr_metavar(e)) return none_expr(); optional r; for_each(e, [&](expr const & e, unsigned) { if (r || !has_expr_metavar(e)) return false; if (is_metavar_app(e)) { r = e; return false; } if (is_local(e)) return false; // do not visit type return true; }); return r; } }