/* Copyright (c) 2014 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura */ #include #include "util/sstream.h" #include "kernel/abstract.h" #include "kernel/instantiate.h" #include "library/scoped_ext.h" #include "library/locals.h" #include "library/explicit.h" #include "frontends/lean/parser.h" namespace lean { void check_atomic(name const & n) { if (!n.is_atomic()) throw exception(sstream() << "invalid declaration name '" << n << "', identifier must be atomic"); } void check_in_section(parser const & p) { if (!in_section(p.env())) throw exception(sstream() << "invalid command, it must be used in a section"); } static name g_root("_root_"); bool is_root_namespace(name const & n) { return n == g_root; } name remove_root_prefix(name const & n) { return n.replace_prefix(g_root, name()); } // Sort local_names by order of occurrence in the section, and copy the associated parameters to section_ps void sort_section_params(expr_struct_set const & locals, parser const & p, buffer & section_ps) { for (expr const & l : locals) section_ps.push_back(l); std::sort(section_ps.begin(), section_ps.end(), [&](expr const & p1, expr const & p2) { return p.get_local_index(mlocal_name(p1)) < p.get_local_index(mlocal_name(p2)); }); } // Return the levels in \c ls that are defined in the section levels collect_section_levels(level_param_names const & ls, parser & p) { buffer section_ls_buffer; for (name const & l : ls) { if (p.get_local_level_index(l)) section_ls_buffer.push_back(mk_param_univ(l)); else break; } return to_list(section_ls_buffer.begin(), section_ls_buffer.end()); } // Collect local (section) constants occurring in type and value, sort them, and store in section_ps void collect_section_locals(expr const & type, expr const & value, parser const & p, buffer & section_ps) { expr_struct_set ls; collect_locals(type, ls); collect_locals(value, ls); sort_section_params(ls, p, section_ps); } list locals_to_context(expr const & e, parser const & p) { expr_struct_set ls; collect_locals(e, ls); buffer locals; sort_section_params(ls, p, locals); std::reverse(locals.begin(), locals.end()); return to_list(locals.begin(), locals.end()); } expr Fun(buffer const & locals, expr const & e, parser & p) { return p.rec_save_pos(Fun(locals, e), p.pos_of(e)); } expr Pi(buffer const & locals, expr const & e, parser & p) { return p.rec_save_pos(Pi(locals, e), p.pos_of(e)); } template static expr mk_binding_as_is(unsigned num, expr const * locals, expr const & b) { expr r = abstract_locals(b, num, locals); unsigned i = num; while (i > 0) { --i; expr const & l = locals[i]; expr t = abstract_locals(mlocal_type(l), i, locals); if (is_lambda) r = mk_lambda(local_pp_name(l), mk_as_is(t), r, local_info(l)); else r = mk_pi(local_pp_name(l), mk_as_is(t), r, local_info(l)); } return r; } expr Fun_as_is(buffer const & locals, expr const & e, parser & p) { return p.rec_save_pos(mk_binding_as_is(locals.size(), locals.data(), e), p.pos_of(e)); } expr Pi_as_is(buffer const & locals, expr const & e, parser & p) { return p.rec_save_pos(mk_binding_as_is(locals.size(), locals.data(), e), p.pos_of(e)); } }