/* Copyright (c) 2014 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura */ #pragma once #include #include #include "util/list.h" #include "kernel/metavar.h" #include "kernel/type_checker.h" #include "library/expr_lt.h" #include "library/unifier.h" #include "library/tactic/tactic.h" #include "library/local_context.h" #include "frontends/lean/elaborator_context.h" #include "frontends/lean/coercion_elaborator.h" #include "frontends/lean/util.h" #include "frontends/lean/calc_proof_elaborator.h" namespace lean { /** \brief Mapping from metavariable names to metavariable applications (?M ...) */ typedef name_map mvar2meta; /** \brief Helper class for implementing the \c elaborate functions. */ class elaborator : public coercion_info_manager { typedef name_map local_tactic_hints; typedef rb_map, expr_quick_cmp> cache; typedef std::vector> to_check_sorts; elaborator_context & m_ctx; name_generator m_ngen; type_checker_ptr m_tc[2]; // mapping from metavariable ?m to the (?m l_1 ... l_n) where [l_1 ... l_n] are the local constants // representing the context where ?m was created. local_context m_context; // current local context: a list of local constants local_context m_full_context; // superset of m_context, it also contains non-contextual locals. mvar2meta m_mvar2meta; // Set of metavariable that where created when m_relax_main_opaque flag is set to true. name_set m_relaxed_mvars; cache m_cache; // The following vector contains sorts that we should check // whether the computed universe is too specific or not. to_check_sorts m_to_check_sorts; // mapping from metavariable name ?m to tactic expression that should be used to solve it. // this mapping is populated by the 'by tactic-expr' expression. local_tactic_hints m_local_tactic_hints; // set of metavariables that we already reported unsolved/unassigned name_set m_displayed_errors; // if m_relax_main_opaque is true, then treat opaque definitions from the main module as transparent. bool m_relax_main_opaque; // if m_no_info is true, we do not collect information when true, // we set is to true whenever we find no_info annotation. bool m_no_info; // if m_in_equation_lhs is true, we are processing the left-hand-side of an equation // and inaccessible expressions are allowed bool m_in_equation_lhs; // if m_equation_lhs is not none, we are processing the right-hand-side of an equation // and decreasing expressions are allowed optional m_equation_lhs; // if m_equation_R is not none when elaborator is processing recursive equation using the well-founded relation R. optional m_equation_R; bool m_use_tactic_hints; info_manager m_pre_info_data; bool m_has_sorry; unifier_config m_unifier_config; // If m_nice_mvar_names is true, we append (when possible) a more informative name for a metavariable. // That is, whenever a metavariables comes from a binding, we add the binding name as a suffix bool m_nice_mvar_names; struct choice_expr_elaborator; environment const & env() const { return m_ctx.m_env; } io_state const & ios() const { return m_ctx.m_ios; } local_decls const & lls() const { return m_ctx.m_lls; } bool use_local_instances() const { return m_ctx.m_use_local_instances; } info_manager * infom() const { return m_ctx.m_info_manager; } pos_info_provider const * pip() const { return m_ctx.m_pos_provider; } bool check_unassigned() const { return m_ctx.m_check_unassigned; } expr mk_local(name const & n, expr const & t, binder_info const & bi); pair infer_type(expr const & e) { return m_tc[m_relax_main_opaque]->infer(e); } pair whnf(expr const & e) { return m_tc[m_relax_main_opaque]->whnf(e); } expr infer_type(expr const & e, constraint_seq & s) { return m_tc[m_relax_main_opaque]->infer(e, s); } expr whnf(expr const & e, constraint_seq & s) { return m_tc[m_relax_main_opaque]->whnf(e, s); } expr mk_app(expr const & f, expr const & a, tag g) { return ::lean::mk_app(f, a, g); } void register_meta(expr const & meta); optional mvar_to_meta(expr const & mvar); void save_type_data(expr const & e, expr const & r); void save_binder_type(expr const & e, expr const & r); void save_extra_type_data(expr const & e, expr const & r); void save_proof_state_info(proof_state const & ps, expr const & e); void save_identifier_info(expr const & f); void save_synth_data(expr const & e, expr const & r); void save_placeholder_info(expr const & e, expr const & r); virtual void save_coercion_info(expr const & e, expr const & c); virtual void erase_coercion_info(expr const & e); void copy_info_to_manager(substitution s); /** \brief If info manager is being used, then create a metavariable suffix based on binding_name(b) */ optional mk_mvar_suffix(expr const & b); expr mk_placeholder_meta(optional const & suffix, optional const & type, tag g, bool is_strict, bool inst_implicit, constraint_seq & cs); expr mk_placeholder_meta(optional const & type, tag g, bool is_strict, bool inst_implicit, constraint_seq & cs) { return mk_placeholder_meta(optional(), type, g, is_strict, inst_implicit, cs); } expr visit_expecting_type(expr const & e, constraint_seq & cs); expr visit_expecting_type_of(expr const & e, expr const & t, constraint_seq & cs); expr visit_choice(expr const & e, optional const & t, constraint_seq & cs); expr visit_by(expr const & e, optional const & t, constraint_seq & cs); expr visit_calc_proof(expr const & e, optional const & t, constraint_seq & cs); expr add_implict_args(expr e, constraint_seq & cs, bool relax); pair ensure_fun(expr f, constraint_seq & cs); bool has_coercions_from(expr const & a_type); bool has_coercions_to(expr d_type); expr apply_coercion(expr const & a, expr a_type, expr d_type); pair mk_delayed_coercion(expr const & a, expr const & a_type, expr const & expected_type, justification const & j); pair ensure_has_type(expr const & a, expr const & a_type, expr const & expected_type, justification const & j, bool relax); bool is_choice_app(expr const & e); expr visit_choice_app(expr const & e, constraint_seq & cs); expr visit_app(expr const & e, constraint_seq & cs); expr visit_placeholder(expr const & e, constraint_seq & cs); level replace_univ_placeholder(level const & l); expr visit_sort(expr const & e); expr visit_macro(expr const & e, constraint_seq & cs); expr visit_constant(expr const & e); expr ensure_type(expr const & e, constraint_seq & cs); expr instantiate_rev_locals(expr const & a, unsigned n, expr const * subst); expr visit_binding(expr e, expr_kind k, constraint_seq & cs); expr visit_pi(expr const & e, constraint_seq & cs); expr visit_lambda(expr const & e, constraint_seq & cs); expr visit_typed_expr(expr const & e, constraint_seq & cs); expr visit_let_value(expr const & e, constraint_seq & cs); bool is_sorry(expr const & e) const; expr visit_sorry(expr const & e); expr visit_core(expr const & e, constraint_seq & cs); pair visit(expr const & e); expr visit(expr const & e, constraint_seq & cs); unify_result_seq solve(constraint_seq const & cs); void display_unsolved_proof_state(expr const & mvar, proof_state const & ps, char const * msg, expr const & pos); void display_unsolved_proof_state(expr const & mvar, proof_state const & ps, char const * msg); void display_tactic_exception(tactic_exception const & ex, proof_state const & ps, expr const & pre_tac); optional get_pre_tactic_for(expr const & mvar); optional pre_tactic_to_tactic(expr const & pre_tac); bool try_using(substitution & subst, expr const & mvar, proof_state const & ps, expr const & pre_tac, tactic const & tac, bool show_failure); bool try_using_begin_end(substitution & subst, expr const & mvar, proof_state ps, expr const & pre_tac); void solve_unassigned_mvar(substitution & subst, expr mvar, name_set & visited); expr solve_unassigned_mvars(substitution & subst, expr e, name_set & visited); expr solve_unassigned_mvars(substitution & subst, expr const & e); void display_unassigned_mvars(expr const & e, substitution const & s); void check_sort_assignments(substitution const & s); expr apply(substitution & s, expr const & e, name_set & univ_params, buffer & new_params); std::tuple apply(substitution & s, expr const & e); pair elaborate_nested(list const & g, expr const & e, bool relax, bool use_tactic_hints, bool report_unassigned); expr const & get_equation_fn(expr const & eq) const; expr visit_equations(expr const & eqns, constraint_seq & cs); expr visit_equation(expr const & e, constraint_seq & cs); expr visit_inaccessible(expr const & e, constraint_seq & cs); expr visit_decreasing(expr const & e, constraint_seq & cs); constraint mk_equations_cnstr(expr const & m, expr const & eqns); bool is_structure(expr const & S); expr visit_structure_instance(expr const & e, constraint_seq & cs); public: elaborator(elaborator_context & ctx, name_generator const & ngen, bool nice_mvar_names = false); std::tuple operator()(list const & ctx, expr const & e, bool _ensure_type, bool relax_main_opaque); std::tuple operator()(expr const & t, expr const & v, name const & n, bool is_opaque); }; std::tuple elaborate(elaborator_context & env, list const & ctx, expr const & e, bool relax_main_opaque, bool ensure_type = false, bool nice_mvar_names = false); std::tuple elaborate(elaborator_context & env, name const & n, expr const & t, expr const & v, bool is_opaque); void initialize_elaborator(); void finalize_elaborator(); }