/* Copyright (c) 2018 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura Quotient types. */ #include "util/name_generator.h" #include "kernel/old_type_checker.h" #include "kernel/quot.h" #include "kernel/local_ctx.h" #include "kernel/inductive/inductive.h" namespace lean { name * quot_consts::g_quot = nullptr; name * quot_consts::g_quot_lift = nullptr; name * quot_consts::g_quot_ind = nullptr; name * quot_consts::g_quot_mk = nullptr; static environment add_constant(environment const & env, name const & n, std::initializer_list lvls, expr const & type) { auto cd = check(env, mk_axiom(n, level_param_names(lvls), type)); return env.add(cd); } static void check_eq_type(environment const & env) { optional decl = inductive::is_inductive_decl(env, "eq"); if (!decl) throw exception("failed to initialize quot module, environment does not have 'eq' type"); if (length(decl->m_level_params) != 1) throw exception("failed to initialize quot module, unexpected number of universe params at 'eq' type"); local_ctx lctx; name_generator g; level u = mk_univ_param(head(decl->m_level_params)); expr alpha = lctx.mk_local_decl(g, "α", mk_sort(u), mk_implicit_binder_info()); expr expected_eq_type = lctx.mk_pi(alpha, mk_arrow(alpha, mk_arrow(alpha, mk_Prop()))); if (decl->m_type != expected_eq_type) throw exception("failed to initialize quot module, 'eq' has an expected type"); if (length(decl->m_intro_rules) != 1) throw exception("failed to initialize quot module, unexpected number of constructors for 'eq' type"); expr a = lctx.mk_local_decl(g, "a", alpha); expr expected_eq_refl_type = lctx.mk_pi({alpha, a}, mk_app(mk_constant("eq", {u}), alpha, a, a)); if (local_type(head(decl->m_intro_rules)) != expected_eq_refl_type) { throw exception("failed to initialize quot module, unexpected type for 'eq' type constructor"); } } environment quot_declare(environment const & env) { check_eq_type(env); environment new_env = env; name u_name("u"); local_ctx lctx; name_generator g; level u = mk_univ_param(u_name); expr Sort_u = mk_sort(u); expr alpha = lctx.mk_local_decl(g, "α", Sort_u, mk_implicit_binder_info()); expr r = lctx.mk_local_decl(g, "r", mk_arrow(alpha, mk_arrow(alpha, mk_Prop()))); /* constant {u} quot {α : Sort u} (r : α → α → Prop) : Sort u */ new_env = add_constant(new_env, *quot_consts::g_quot, {u_name}, lctx.mk_pi({alpha, r}, Sort_u)); expr quot_r = mk_app(mk_constant(*quot_consts::g_quot, {u}), alpha, r); expr a = lctx.mk_local_decl(g, "a", alpha); /* constant {u} quot.mk {α : Sort u} (r : α → α → Prop) (a : α) : @quot.{u} α r */ new_env = add_constant(new_env, *quot_consts::g_quot_mk, {u_name}, lctx.mk_pi({alpha, r, a}, quot_r)); /* make r implicit */ lctx = local_ctx(); alpha = lctx.mk_local_decl(g, "α", Sort_u, mk_implicit_binder_info()); r = lctx.mk_local_decl(g, "r", mk_arrow(alpha, mk_arrow(alpha, mk_Prop())), mk_implicit_binder_info()); quot_r = mk_app(mk_constant(*quot_consts::g_quot, {u}), alpha, r); a = lctx.mk_local_decl(g, "a", alpha); name v_name("v"); level v = mk_univ_param(v_name); expr Sort_v = mk_sort(v); expr beta = lctx.mk_local_decl(g, "β", Sort_v, mk_implicit_binder_info()); expr f = lctx.mk_local_decl(g, "f", mk_arrow(alpha, beta)); expr b = lctx.mk_local_decl(g, "b", alpha); expr r_a_b = mk_app(r, a, b); /* f a = f b */ expr f_a_eq_f_b = mk_app(mk_constant("eq", {v}), beta, mk_app(f, a), mk_app(f, b)); /* (∀ a b : α, r a b → f a = f b) */ expr sanity = lctx.mk_pi({a, b}, mk_arrow(r_a_b, f_a_eq_f_b)); /* constant {u v} quot.lift {α : Sort u} {r : α → α → Prop} {β : Sort v} (f : α → β) : (∀ a b : α, r a b → f a = f b) → @quot.{u} α r → β */ new_env = add_constant(new_env, *quot_consts::g_quot_lift, {u_name, v_name}, lctx.mk_pi({alpha, r, beta, f}, mk_arrow(sanity, mk_arrow(quot_r, beta)))); /* {β : @quot.{u} α r → Prop} */ beta = lctx.mk_local_decl(g, "β", mk_arrow(quot_r, mk_Prop()), mk_implicit_binder_info()); expr quot_mk_a = mk_app(mk_constant(*quot_consts::g_quot_mk, {u}), alpha, r, a); expr all_quot = lctx.mk_pi(a, mk_app(beta, quot_mk_a)); expr q = lctx.mk_local_decl(g, "q", quot_r); expr beta_q = mk_app(beta, q); /* constant {u} quot.ind {α : Sort u} {r : α → α → Prop} {β : @quot.{u} α r → Prop} : (∀ a : α, β (@quot.mk.{u} α r a)) → ∀ q : @quot.{u} α r, β q */ new_env = add_constant(new_env, *quot_consts::g_quot_ind, {u_name}, lctx.mk_pi({alpha, r, beta}, mk_arrow(all_quot, lctx.mk_pi(q, beta_q)))); return new_env; } void initialize_quot() { quot_consts::g_quot = new name{"quot"}; quot_consts::g_quot_lift = new name{"quot", "lift"}; quot_consts::g_quot_ind = new name{"quot", "ind"}; quot_consts::g_quot_mk = new name{"quot", "mk"}; } void finalize_quot() { delete quot_consts::g_quot; delete quot_consts::g_quot_lift; delete quot_consts::g_quot_ind; delete quot_consts::g_quot_mk; } }