/* 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 "util/list_fn.h" #include "util/hash.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" #ifndef LEAN_INITIAL_EXPR_CACHE_CAPACITY #define LEAN_INITIAL_EXPR_CACHE_CAPACITY 1024*16 #endif 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))) { } 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(); } static inline unsigned hash(literal const & a) { switch (a.kind()) { case literal_kind::Nat: return a.get_nat().hash(); case literal_kind::String: return hash_str(a.get_string().num_bytes(), a.get_string().data(), 17); } lean_unreachable(); } static unsigned hash(levels const & ls) { unsigned r = 23; for (auto const & l : ls) r = hash(hash(l), r); return r; } // ======================================= // Safe arithmetic static unsigned safe_add(unsigned w1, unsigned w2) { unsigned r = w1 + w2; if (r < w1) r = std::numeric_limits::max(); // overflow return r; } static unsigned safe_inc(unsigned w) { if (w < std::numeric_limits::max()) return w+1; else return w; } static unsigned safe_dec(unsigned k) { return k == 0 ? 0 : k - 1; } // ======================================= // Scalar data offset calculation and setters/getters inline constexpr unsigned num_obj_fields(expr_kind k) { return k == expr_kind::App ? 2 : k == expr_kind::Const ? 2 : k == expr_kind::FVar ? 3 : // TODO(Leo): it should be 1 after we remove support for legacy code k == expr_kind::Lambda ? 3 : k == expr_kind::Pi ? 3 : k == expr_kind::BVar ? 1 : k == expr_kind::Let ? 4 : k == expr_kind::MVar ? 2 : k == expr_kind::Sort ? 1 : k == expr_kind::Lit ? 1 : k == expr_kind::MData ? 2 : k == expr_kind::Proj ? 2 : /* k == expr_kind::Quote */ 1; } /* Expression scalar data offset. */ inline constexpr unsigned scalar_offset(expr_kind k) { return num_obj_fields(k) * sizeof(object*); } inline constexpr unsigned binder_info_offset(expr_kind k) { // Only for: k == expr_kind::Pi || k == expr_kind::Lambda || k == expr_kind::FVar return scalar_offset(k); } /* Legacy support */ inline constexpr unsigned reflected_offset(expr_kind k) { // Only for: k == expr_kind::Quote return scalar_offset(k); } inline constexpr unsigned hash_offset(expr_kind k) { return k == expr_kind::FVar ? scalar_offset(k) + sizeof(unsigned char) : // for binder_info, TODO(Leo): delete after we remove support for legacy code k == expr_kind::Lambda ? scalar_offset(k) + sizeof(unsigned char) : // for binder_info k == expr_kind::Pi ? scalar_offset(k) + sizeof(unsigned char) : // for binder_info scalar_offset(k); } inline constexpr size_t flags_offset(expr_kind k) { return hash_offset(k) + sizeof(unsigned); } inline constexpr size_t weight_offset(expr_kind k) { return flags_offset(k) + sizeof(unsigned char); } inline constexpr size_t depth_offset(expr_kind k) { return weight_offset(k) + sizeof(unsigned); } inline constexpr size_t loose_bvar_range_offset(expr_kind k) { return depth_offset(k) + sizeof(unsigned); } /* Size for scalar value area for non recursive expression. */ inline constexpr size_t expr_scalar_size(expr_kind k) { return flags_offset(k) + sizeof(unsigned char); } /* Size for scalar value area for recursive expression. */ inline constexpr size_t rec_expr_scalar_size(expr_kind k) { return loose_bvar_range_offset(k) + sizeof(unsigned); } /* Set expr cached hash code and flags. All expressions contain them. We provide the kind `k` to allow the compiler to compute offsets at compilation time. */ template void set_scalar(expr const & e, unsigned hash, bool has_expr_mvar, bool has_univ_mvar, bool has_fvar, bool has_univ_param) { lean_assert(e.kind() == k); unsigned char d = (has_expr_mvar ? 1 : 0) + (has_univ_mvar ? 2 : 0) + (has_fvar ? 4 : 0) + (has_univ_param ? 8 : 0); cnstr_set_scalar(e.raw(), hash_offset(k), hash); cnstr_set_scalar(e.raw(), flags_offset(k), d); } /* Set expr cached weight, depth and loose bvar range. We only store this information in recursive expr constructors. We provide the kind `k` to allow the compiler to compute offsets at compilation time. */ template void set_rec_scalar(expr const & e, unsigned weight, unsigned depth, unsigned loose_bvar_range) { lean_assert(e.kind() == k); cnstr_set_scalar(e.raw(), weight_offset(k), weight); cnstr_set_scalar(e.raw(), depth_offset(k), depth); cnstr_set_scalar(e.raw(), loose_bvar_range_offset(k), loose_bvar_range); } template void set_binder_info(expr const & e, binder_info bi) { lean_assert(e.kind() == k); cnstr_set_scalar(e.raw(), binder_info_offset(k), static_cast(bi)); } /* Legacy support */ unsigned hash(expr const & e) { return cnstr_get_scalar(e.raw(), hash_offset(e.kind())); } static inline unsigned char get_flags(expr const & e) { return cnstr_get_scalar(e.raw(), flags_offset(e.kind())); } bool has_expr_mvar(expr const & e) { return (get_flags(e) & 1) != 0; } bool has_univ_mvar(expr const & e) { return (get_flags(e) & 2) != 0; } bool has_fvar(expr const & e) { return (get_flags(e) & 4) != 0; } bool has_univ_param(expr const & e) { return (get_flags(e) & 8) != 0; } template unsigned get_weight_core(expr const & e) { return cnstr_get_scalar(e.raw(), weight_offset(k)); } unsigned get_weight(expr const & e) { switch (e.kind()) { case expr_kind::BVar: case expr_kind::Const: case expr_kind::Sort: case expr_kind::MVar: case expr_kind::FVar: case expr_kind::Lit: return 1; case expr_kind::Lambda: return get_weight_core(e); case expr_kind::Pi: return get_weight_core(e); case expr_kind::App: return get_weight_core(e); case expr_kind::Let: return get_weight_core(e); case expr_kind::MData: return get_weight_core(e); case expr_kind::Proj: return get_weight_core(e); } lean_unreachable(); // LCOV_EXCL_LINE } template unsigned get_depth_core(expr const & e) { return cnstr_get_scalar(e.raw(), depth_offset(k)); } unsigned get_depth(expr const & e) { switch (e.kind()) { case expr_kind::BVar: case expr_kind::Const: case expr_kind::Sort: case expr_kind::MVar: case expr_kind::FVar: case expr_kind::Lit: return 1; case expr_kind::Lambda: return get_depth_core(e); case expr_kind::Pi: return get_depth_core(e); case expr_kind::App: return get_depth_core(e); case expr_kind::Let: return get_depth_core(e); case expr_kind::MData: return get_depth_core(e); case expr_kind::Proj: return get_depth_core(e); } lean_unreachable(); // LCOV_EXCL_LINE } template unsigned get_loose_bvar_range_core(expr const & e) { return cnstr_get_scalar(e.raw(), loose_bvar_range_offset(k)); } unsigned get_loose_bvar_range(expr const & e) { switch (e.kind()) { case expr_kind::Const: case expr_kind::Sort: case expr_kind::Lit: return 0; case expr_kind::BVar: { nat const & idx = bvar_idx(e); return idx.is_small() ? safe_inc(idx.get_small_value()) : std::numeric_limits::max(); } case expr_kind::MVar: return get_loose_bvar_range_core(e); case expr_kind::FVar: return get_loose_bvar_range_core(e); case expr_kind::Lambda: return get_loose_bvar_range_core(e); case expr_kind::Pi: return get_loose_bvar_range_core(e); case expr_kind::App: return get_loose_bvar_range_core(e); case expr_kind::Let: return get_loose_bvar_range_core(e); case expr_kind::MData: return get_loose_bvar_range_core(e); case expr_kind::Proj: return get_loose_bvar_range_core(e); } lean_unreachable(); // LCOV_EXCL_LINE } 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: return true; case expr_kind::App: case expr_kind::MVar: case expr_kind::FVar: case expr_kind::Lambda: case expr_kind::Pi: case expr_kind::Let: case expr_kind::MData: case expr_kind::Proj: return false; } lean_unreachable(); // LCOV_EXCL_LINE } binder_info binding_info(expr const & e) { lean_assert(is_lambda(e) || is_pi(e)); // Remark: lambda and Pi have the same memory layout return static_cast(cnstr_get_scalar(e.raw(), binder_info_offset(expr_kind::Lambda))); } /* Legacy */ binder_info local_info(expr const & e) { lean_assert(is_local(e)); return static_cast(cnstr_get_scalar(e.raw(), binder_info_offset(expr_kind::FVar))); } static expr * g_nat_type = nullptr; static expr * g_string_type = nullptr; expr const & lit_type(expr const & e) { switch (lit_value(e).kind()) { case literal_kind::String: return *g_string_type; case literal_kind::Nat: return *g_nat_type; } lean_unreachable(); } // ======================================= // Constructors static expr * g_dummy = nullptr; expr::expr():expr(*g_dummy) {} expr mk_lit(literal const & l) { expr r(mk_cnstr(static_cast(expr_kind::Lit), l, expr_scalar_size(expr_kind::Lit))); set_scalar(r, hash(l), false, false, false, false); return r; } expr mk_mdata(kvmap const & m, expr const & e) { expr r(mk_cnstr(static_cast(expr_kind::MData), m, e, rec_expr_scalar_size(expr_kind::MData))); unsigned w = safe_inc(get_weight(e)); unsigned d = get_depth(e) + 1; unsigned h = hash(hash(e), hash(w, d)); set_scalar(r, h, has_expr_mvar(e), has_univ_mvar(e), has_fvar(e), has_univ_param(e)); set_rec_scalar(r, w, d, get_loose_bvar_range(e)); return r; } expr mk_proj(nat const & idx, expr const & e) { expr r(mk_cnstr(static_cast(expr_kind::Proj), idx, e, rec_expr_scalar_size(expr_kind::Proj))); unsigned w = safe_inc(get_weight(e)); unsigned d = get_depth(e) + 1; unsigned h = hash(hash(e), hash(idx.hash(), w)); set_scalar(r, h, has_expr_mvar(e), has_univ_mvar(e), has_fvar(e), has_univ_param(e)); set_rec_scalar(r, w, d, get_loose_bvar_range(e)); return r; } expr mk_bvar(nat const & idx) { expr r(mk_cnstr(static_cast(expr_kind::BVar), idx, expr_scalar_size(expr_kind::BVar))); set_scalar(r, idx.hash(), false, false, false, false); return r; } /* Legacy */ expr mk_local(name const & n, name const & pp_n, expr const & t, binder_info bi) { expr r(mk_cnstr(static_cast(expr_kind::FVar), n, pp_n, t, rec_expr_scalar_size(expr_kind::FVar))); set_binder_info(r, bi); set_scalar(r, n.hash(), has_expr_mvar(t), has_univ_mvar(t), true, has_univ_param(t)); set_rec_scalar(r, 1, 1, get_loose_bvar_range(t)); return r; } expr mk_fvar(name const & n) { return mk_local(n, n, expr(), mk_binder_info()); } expr mk_const(name const & n, levels const & ls) { expr r(mk_cnstr(static_cast(expr_kind::Const), n, ls, expr_scalar_size(expr_kind::Const))); set_scalar(r, hash(n.hash(), hash(ls)), false, has_mvar(ls), false, has_param(ls)); return r; } expr mk_app(expr const & f, expr const & a) { expr r(mk_cnstr(static_cast(expr_kind::App), f, a, rec_expr_scalar_size(expr_kind::App))); unsigned w = safe_inc(safe_add(get_weight(f), get_weight(a))); unsigned d = std::max(get_depth(f), get_depth(a)) + 1; unsigned h = hash(hash(hash(f), hash(a)), hash(d, w)); set_scalar(r, h, has_expr_mvar(f) || has_expr_mvar(a), has_univ_mvar(f) || has_univ_mvar(a), has_fvar(f) || has_fvar(a), has_univ_param(f) || has_univ_param(a)); set_rec_scalar(r, w, d, std::max(get_loose_bvar_range(f), get_loose_bvar_range(a))); return r; } expr mk_sort(level const & l) { expr r(mk_cnstr(static_cast(expr_kind::Sort), l, expr_scalar_size(expr_kind::Sort))); set_scalar(r, hash(l), false, has_mvar(l), false, has_param(l)); return r; } template expr mk_binding(name const & n, expr const & t, expr const & e, binder_info bi) { lean_assert(k == expr_kind::Pi || k == expr_kind::Lambda); expr r(mk_cnstr(static_cast(k), n, t, e, rec_expr_scalar_size(k))); unsigned w = safe_inc(safe_add(get_weight(t), get_weight(e))); unsigned d = std::max(get_depth(t), get_depth(e)) + 1; unsigned h = hash(hash(d, w), hash(hash(t), hash(e))); unsigned lbvr = std::max(get_loose_bvar_range(t), safe_dec(get_loose_bvar_range(e))); set_binder_info(r, bi); set_scalar(r, h, has_expr_mvar(t) || has_expr_mvar(e), has_univ_mvar(t) || has_univ_mvar(e), has_fvar(t) || has_fvar(e), has_univ_param(t) || has_univ_param(e)); set_rec_scalar(r, w, d, lbvr); return r; } expr mk_lambda(name const & n, expr const & t, expr const & e, binder_info bi) { return mk_binding(n, t, e, bi); } expr mk_pi(name const & n, expr const & t, expr const & e, binder_info bi) { return mk_binding(n, t, e, 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()); } expr mk_let(name const & n, expr const & t, expr const & v, expr const & b) { expr r(mk_cnstr(static_cast(expr_kind::Let), n, t, v, b, rec_expr_scalar_size(expr_kind::Let))); unsigned w = safe_inc(safe_add(safe_add(get_weight(t), get_weight(v)), get_weight(b))); unsigned d = std::max(get_depth(t), std::max(get_depth(v), get_depth(b))) + 1; unsigned h = hash(hash(w, d), hash(hash(hash(t), hash(v)), hash(b))); unsigned lbvr = std::max(get_loose_bvar_range(t), std::max(get_loose_bvar_range(v), safe_dec(get_loose_bvar_range(b)))); set_scalar(r, h, has_expr_mvar(t) || has_expr_mvar(v) || has_expr_mvar(b), has_univ_mvar(t) || has_univ_mvar(v) || has_univ_mvar(b), has_fvar(t) || has_fvar(v) || has_fvar(b), has_univ_param(t) || has_univ_param(v) || has_univ_param(b)); set_rec_scalar(r, w, d, lbvr); return r; } expr mk_mvar(name const & n, expr const & t) { expr r(mk_cnstr(static_cast(expr_kind::MVar), n, t, rec_expr_scalar_size(expr_kind::MVar))); set_scalar(r, n.hash(), true, has_univ_mvar(t), has_fvar(t), has_univ_param(t)); set_rec_scalar(r, 1, 1, get_loose_bvar_range(t)); return r; } 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; } /* Legacy */ 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; } // ======================================= // 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_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_mvar(expr const & e, expr const & new_type) { if (is_eqp(mvar_type(e), new_type)) return e; else return mk_mvar(mvar_name(e), new_type); } 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; } /* Legacy */ 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); } /* Legacy */ expr update_local(expr const & e, binder_info bi) { return update_local(e, local_type(e), bi); } /* Legacy */ 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)); } // ======================================= // 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; } 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); } 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); } // ======================================= // 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() { g_dummy = new expr(mk_constant("__expr_for_default_constructor__")); 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. */ g_nat_type = new expr(mk_constant("nat")); g_string_type = new expr(mk_constant("string")); } void finalize_expr() { delete g_Prop; delete g_Type0; delete g_dummy; delete g_default_name; delete g_nat_type; delete g_string_type; } #if 0 // Expr metavariables and local variables expr_mlocal::expr_mlocal(bool is_meta, name const & n, name const & pp_n, expr const & t): expr_composite(is_meta ? expr_kind::MVar : expr_kind::FVar, n.hash(), is_meta || t.has_expr_metavar(), t.has_univ_metavar(), !is_meta || t.has_fvar(), t.has_param_univ(), 1, get_loose_bvar_range(t)), m_name(n), m_pp_name(pp_n), m_type(t) {} void expr_mlocal::dealloc(buffer & todelete) { dec_ref(m_type, todelete); delete this; } #endif }