feat(library/definitional): add brec_on construction, closes #272
This commit is contained in:
parent
f948241bb9
commit
fca97d5bb2
12 changed files with 278 additions and 60 deletions
|
@ -36,6 +36,12 @@ mk :: (pr1 : A) (pr2 : B)
|
||||||
inductive and (a b : Prop) : Prop :=
|
inductive and (a b : Prop) : Prop :=
|
||||||
intro : a → b → and a b
|
intro : a → b → and a b
|
||||||
|
|
||||||
|
definition and.elim_left {a b : Prop} (H : and a b) : a :=
|
||||||
|
and.rec (λa b, a) H
|
||||||
|
|
||||||
|
definition and.elim_right {a b : Prop} (H : and a b) : b :=
|
||||||
|
and.rec (λa b, b) H
|
||||||
|
|
||||||
inductive sum (A B : Type) : Type :=
|
inductive sum (A B : Type) : Type :=
|
||||||
inl : A → sum A B,
|
inl : A → sum A B,
|
||||||
inr : B → sum A B
|
inr : B → sum A B
|
||||||
|
|
|
@ -235,12 +235,6 @@ namespace and
|
||||||
theorem elim (H₁ : a ∧ b) (H₂ : a → b → c) : c :=
|
theorem elim (H₁ : a ∧ b) (H₂ : a → b → c) : c :=
|
||||||
rec H₂ H₁
|
rec H₂ H₁
|
||||||
|
|
||||||
definition elim_left (H : a ∧ b) : a :=
|
|
||||||
rec (λa b, a) H
|
|
||||||
|
|
||||||
definition elim_right (H : a ∧ b) : b :=
|
|
||||||
rec (λa b, b) H
|
|
||||||
|
|
||||||
theorem swap (H : a ∧ b) : b ∧ a :=
|
theorem swap (H : a ∧ b) : b ∧ a :=
|
||||||
intro (elim_right H) (elim_left H)
|
intro (elim_right H) (elim_left H)
|
||||||
|
|
||||||
|
|
|
@ -726,26 +726,36 @@ struct inductive_cmd_fn {
|
||||||
for (inductive_decl const & d : decls) {
|
for (inductive_decl const & d : decls) {
|
||||||
name const & n = inductive_decl_name(d);
|
name const & n = inductive_decl_name(d);
|
||||||
pos_info pos = *m_decl_pos_map.find(n);
|
pos_info pos = *m_decl_pos_map.find(n);
|
||||||
env = mk_rec_on(env, inductive_decl_name(d));
|
env = mk_rec_on(env, n);
|
||||||
env = mk_induction_on(env, inductive_decl_name(d));
|
env = mk_induction_on(env, n);
|
||||||
save_def_info(name(n, "rec_on"), pos);
|
save_def_info(name(n, "rec_on"), pos);
|
||||||
save_def_info(name(n, "induction_on"), pos);
|
save_def_info(name(n, "induction_on"), pos);
|
||||||
if (has_unit) {
|
if (has_unit) {
|
||||||
env = mk_cases_on(env, inductive_decl_name(d));
|
env = mk_cases_on(env, n);
|
||||||
save_def_info(name(n, "cases_on"), pos);
|
save_def_info(name(n, "cases_on"), pos);
|
||||||
if (has_eq && has_heq) {
|
if (has_eq && has_heq) {
|
||||||
env = mk_no_confusion(env, inductive_decl_name(d));
|
env = mk_no_confusion(env, n);
|
||||||
save_if_defined(name{n, "no_confusion_type"}, pos);
|
save_if_defined(name{n, "no_confusion_type"}, pos);
|
||||||
save_if_defined(name(n, "no_confusion"), pos);
|
save_if_defined(name(n, "no_confusion"), pos);
|
||||||
}
|
}
|
||||||
if (has_prod) {
|
if (has_prod) {
|
||||||
env = mk_below(env, inductive_decl_name(d));
|
env = mk_below(env, n);
|
||||||
env = mk_ibelow(env, inductive_decl_name(d));
|
env = mk_ibelow(env, n);
|
||||||
save_if_defined(name{n, "below"}, pos);
|
save_if_defined(name{n, "below"}, pos);
|
||||||
save_if_defined(name(n, "ibelow"), pos);
|
save_if_defined(name(n, "ibelow"), pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for (inductive_decl const & d : decls) {
|
||||||
|
name const & n = inductive_decl_name(d);
|
||||||
|
pos_info pos = *m_decl_pos_map.find(n);
|
||||||
|
if (has_unit && has_prod) {
|
||||||
|
env = mk_brec_on(env, n);
|
||||||
|
env = mk_binduction_on(env, n);
|
||||||
|
save_if_defined(name{n, "brec_on"}, pos);
|
||||||
|
save_if_defined(name(n, "binduction_on"), pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
return env;
|
return env;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,15 +21,17 @@ static void throw_corrupted(name const & n) {
|
||||||
throw exception(sstream() << "error in 'brec_on' generation, '" << n << "' inductive datatype declaration is corrupted");
|
throw exception(sstream() << "error in 'brec_on' generation, '" << n << "' inductive datatype declaration is corrupted");
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool is_typeformer_app(buffer<name> const & typeformer_names, expr const & e) {
|
static optional<unsigned> is_typeformer_app(buffer<name> const & typeformer_names, expr const & e) {
|
||||||
expr const & fn = get_app_fn(e);
|
expr const & fn = get_app_fn(e);
|
||||||
if (!is_local(fn))
|
if (!is_local(fn))
|
||||||
return false;
|
return optional<unsigned>();
|
||||||
|
unsigned r = 0;
|
||||||
for (name const & n : typeformer_names) {
|
for (name const & n : typeformer_names) {
|
||||||
if (mlocal_name(fn) == n)
|
if (mlocal_name(fn) == n)
|
||||||
return true;
|
return optional<unsigned>(r);
|
||||||
|
r++;
|
||||||
}
|
}
|
||||||
return false;
|
return optional<unsigned>();
|
||||||
}
|
}
|
||||||
|
|
||||||
static environment mk_below(environment const & env, name const & n, bool ibelow) {
|
static environment mk_below(environment const & env, name const & n, bool ibelow) {
|
||||||
|
@ -48,49 +50,34 @@ static environment mk_below(environment const & env, name const & n, bool ibelow
|
||||||
unsigned ntypeformers = length(std::get<2>(decls));
|
unsigned ntypeformers = length(std::get<2>(decls));
|
||||||
level_param_names lps = rec_decl.get_univ_params();
|
level_param_names lps = rec_decl.get_univ_params();
|
||||||
bool is_reflexive = is_reflexive_datatype(tc, n);
|
bool is_reflexive = is_reflexive_datatype(tc, n);
|
||||||
level lvl = mk_param_univ(head(lps)); // universe we are eliminating to
|
level lvl = mk_param_univ(head(lps));
|
||||||
levels lvls = param_names_to_levels(tail(lps));
|
levels lvls = param_names_to_levels(tail(lps));
|
||||||
levels blvls; // universe level parameters of ibelow/below
|
level_param_names blvls; // universe level parameters of ibelow/below
|
||||||
level rlvl; // universe level of the resultant type
|
level rlvl; // universe level of the resultant type
|
||||||
name prod_name;
|
|
||||||
expr unit, outer_prod, inner_prod;
|
|
||||||
// The arguments of below (ibelow) are the ones in the recursor - minor premises.
|
// The arguments of below (ibelow) are the ones in the recursor - minor premises.
|
||||||
// The universe we map to is also different (l+1 for below) and (0 fo ibelow).
|
// The universe we map to is also different (l+1 for below of reflexive types) and (0 fo ibelow).
|
||||||
expr ref_type;
|
expr ref_type;
|
||||||
expr Type_result;
|
expr Type_result;
|
||||||
if (ibelow) {
|
if (ibelow) {
|
||||||
// we are eliminating to Prop
|
// we are eliminating to Prop
|
||||||
blvls = lvls;
|
blvls = tail(lps);
|
||||||
rlvl = mk_level_zero();
|
rlvl = mk_level_zero();
|
||||||
unit = mk_constant("true");
|
|
||||||
prod_name = name("and");
|
|
||||||
outer_prod = mk_constant(prod_name);
|
|
||||||
inner_prod = outer_prod;
|
|
||||||
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_level_zero());
|
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_level_zero());
|
||||||
Type_result = mk_sort(rlvl);
|
|
||||||
} else if (is_reflexive) {
|
} else if (is_reflexive) {
|
||||||
blvls = cons(lvl, lvls);
|
blvls = lps;
|
||||||
rlvl = get_datatype_level(ind_decl.get_type());
|
rlvl = get_datatype_level(ind_decl.get_type());
|
||||||
// if rlvl is of the form (max 1 l), then rlvl <- l
|
// if rlvl is of the form (max 1 l), then rlvl <- l
|
||||||
if (is_max(rlvl) && is_one(max_lhs(rlvl)))
|
if (is_max(rlvl) && is_one(max_lhs(rlvl)))
|
||||||
rlvl = max_rhs(rlvl);
|
rlvl = max_rhs(rlvl);
|
||||||
rlvl = mk_max(mk_succ(lvl), rlvl);
|
rlvl = mk_max(mk_succ(lvl), rlvl);
|
||||||
unit = mk_constant("unit", rlvl);
|
|
||||||
prod_name = name("prod");
|
|
||||||
outer_prod = mk_constant(prod_name, {rlvl, rlvl});
|
|
||||||
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_succ(lvl));
|
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_succ(lvl));
|
||||||
Type_result = mk_sort(rlvl);
|
|
||||||
} else {
|
} else {
|
||||||
// we can simplify the universe levels for non-reflexive datatypes
|
// we can simplify the universe levels for non-reflexive datatypes
|
||||||
blvls = cons(lvl, lvls);
|
blvls = lps;
|
||||||
rlvl = mk_max(mk_level_one(), lvl);
|
rlvl = mk_max(mk_level_one(), lvl);
|
||||||
unit = mk_constant("unit", rlvl);
|
|
||||||
prod_name = name("prod");
|
|
||||||
outer_prod = mk_constant(prod_name, {rlvl, rlvl});
|
|
||||||
inner_prod = mk_constant(prod_name, {lvl, rlvl});
|
|
||||||
ref_type = rec_decl.get_type();
|
ref_type = rec_decl.get_type();
|
||||||
Type_result = mk_sort(rlvl);
|
|
||||||
}
|
}
|
||||||
|
Type_result = mk_sort(rlvl);
|
||||||
buffer<expr> ref_args;
|
buffer<expr> ref_args;
|
||||||
to_telescope(ngen, ref_type, ref_args);
|
to_telescope(ngen, ref_type, ref_args);
|
||||||
if (ref_args.size() != nparams + ntypeformers + nminors + nindices + 1)
|
if (ref_args.size() != nparams + ntypeformers + nminors + nindices + 1)
|
||||||
|
@ -132,19 +119,15 @@ static environment mk_below(environment const & env, name const & n, bool ibelow
|
||||||
buffer<expr> minor_arg_args;
|
buffer<expr> minor_arg_args;
|
||||||
expr minor_arg_type = to_telescope(tc, mlocal_type(minor_arg), minor_arg_args);
|
expr minor_arg_type = to_telescope(tc, mlocal_type(minor_arg), minor_arg_args);
|
||||||
if (is_typeformer_app(typeformer_names, minor_arg_type)) {
|
if (is_typeformer_app(typeformer_names, minor_arg_type)) {
|
||||||
expr r = minor_arg;
|
expr fst = mlocal_type(minor_arg);
|
||||||
expr fst = mlocal_type(minor_arg);
|
|
||||||
expr snd = Pi(minor_arg_args, mk_app(r, minor_arg_args));
|
|
||||||
if (!ibelow && is_reflexive) {
|
|
||||||
// inner product is not constant
|
|
||||||
level fst_lvl = sort_level(tc.ensure_type(fst).first);
|
|
||||||
inner_prod = mk_constant(prod_name, {fst_lvl, rlvl});
|
|
||||||
}
|
|
||||||
prod_pairs.push_back(mk_app(inner_prod, fst, snd));
|
|
||||||
minor_arg = update_mlocal(minor_arg, Pi(minor_arg_args, Type_result));
|
minor_arg = update_mlocal(minor_arg, Pi(minor_arg_args, Type_result));
|
||||||
|
expr snd = Pi(minor_arg_args, mk_app(minor_arg, minor_arg_args));
|
||||||
|
prod_pairs.push_back(mk_prod(tc, fst, snd, ibelow));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expr new_arg = mk_bin_rop(outer_prod, unit, prod_pairs.size(), prod_pairs.data());
|
expr new_arg = foldr([&](expr const & a, expr const & b) { return mk_prod(tc, a, b, ibelow); },
|
||||||
|
[&]() { return mk_unit(rlvl, ibelow); },
|
||||||
|
prod_pairs.size(), prod_pairs.data());
|
||||||
rec = mk_app(rec, Fun(minor_args, new_arg));
|
rec = mk_app(rec, Fun(minor_args, new_arg));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -159,7 +142,7 @@ static environment mk_below(environment const & env, name const & n, bool ibelow
|
||||||
|
|
||||||
bool opaque = false;
|
bool opaque = false;
|
||||||
bool use_conv_opt = true;
|
bool use_conv_opt = true;
|
||||||
declaration new_d = mk_definition(env, below_name, rec_decl.get_univ_params(), below_type, below_value,
|
declaration new_d = mk_definition(env, below_name, blvls, below_type, below_value,
|
||||||
opaque, rec_decl.get_module_idx(), use_conv_opt);
|
opaque, rec_decl.get_module_idx(), use_conv_opt);
|
||||||
environment new_env = module::add(env, check(env, new_d));
|
environment new_env = module::add(env, check(env, new_d));
|
||||||
new_env = set_reducible(new_env, below_name, reducible_status::On);
|
new_env = set_reducible(new_env, below_name, reducible_status::On);
|
||||||
|
@ -173,4 +156,189 @@ environment mk_below(environment const & env, name const & n) {
|
||||||
environment mk_ibelow(environment const & env, name const & n) {
|
environment mk_ibelow(environment const & env, name const & n) {
|
||||||
return mk_below(env, n, true);
|
return mk_below(env, n, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static environment mk_brec_on(environment const & env, name const & n, bool ind) {
|
||||||
|
if (!is_recursive_datatype(env, n))
|
||||||
|
return env;
|
||||||
|
if (is_inductive_predicate(env, n))
|
||||||
|
return env;
|
||||||
|
inductive::inductive_decls decls = *inductive::is_inductive_decl(env, n);
|
||||||
|
type_checker tc(env);
|
||||||
|
name_generator ngen;
|
||||||
|
unsigned nparams = std::get<1>(decls);
|
||||||
|
declaration ind_decl = env.get(n);
|
||||||
|
declaration rec_decl = env.get(inductive::get_elim_name(n));
|
||||||
|
// declaration below_decl = env.get(name(n, ind ? "ibelow" : "below"));
|
||||||
|
unsigned nindices = *inductive::get_num_indices(env, n);
|
||||||
|
unsigned nminors = *inductive::get_num_minor_premises(env, n);
|
||||||
|
unsigned ntypeformers = length(std::get<2>(decls));
|
||||||
|
level_param_names lps = rec_decl.get_univ_params();
|
||||||
|
bool is_reflexive = is_reflexive_datatype(tc, n);
|
||||||
|
level lvl = mk_param_univ(head(lps));
|
||||||
|
levels lvls = param_names_to_levels(tail(lps));
|
||||||
|
level rlvl;
|
||||||
|
level_param_names blps;
|
||||||
|
levels blvls; // universe level parameters of brec_on/binduction_on
|
||||||
|
// The arguments of brec_on (binduction_on) are the ones in the recursor - minor premises.
|
||||||
|
// The universe we map to is also different (l+1 for below of reflexive types) and (0 fo ibelow).
|
||||||
|
expr ref_type;
|
||||||
|
if (ind) {
|
||||||
|
// we are eliminating to Prop
|
||||||
|
blps = tail(lps);
|
||||||
|
blvls = lvls;
|
||||||
|
rlvl = mk_level_zero();
|
||||||
|
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_level_zero());
|
||||||
|
} else if (is_reflexive) {
|
||||||
|
blps = lps;
|
||||||
|
blvls = cons(lvl, lvls);
|
||||||
|
rlvl = get_datatype_level(ind_decl.get_type());
|
||||||
|
// if rlvl is of the form (max 1 l), then rlvl <- l
|
||||||
|
if (is_max(rlvl) && is_one(max_lhs(rlvl)))
|
||||||
|
rlvl = max_rhs(rlvl);
|
||||||
|
rlvl = mk_max(mk_succ(lvl), rlvl);
|
||||||
|
// inner_prod, inner_prod_intro, pr1, pr2 do not use the same universe levels for
|
||||||
|
// reflective datatypes.
|
||||||
|
ref_type = instantiate_univ_param(rec_decl.get_type(), param_id(lvl), mk_succ(lvl));
|
||||||
|
} else {
|
||||||
|
// we can simplify the universe levels for non-reflexive datatypes
|
||||||
|
blps = lps;
|
||||||
|
blvls = cons(lvl, lvls);
|
||||||
|
rlvl = mk_max(mk_level_one(), lvl);
|
||||||
|
ref_type = rec_decl.get_type();
|
||||||
|
}
|
||||||
|
buffer<expr> ref_args;
|
||||||
|
to_telescope(ngen, ref_type, ref_args);
|
||||||
|
if (ref_args.size() != nparams + ntypeformers + nminors + nindices + 1)
|
||||||
|
throw_corrupted(n);
|
||||||
|
|
||||||
|
// args contains the brec_on/binduction_on arguments
|
||||||
|
buffer<expr> args;
|
||||||
|
buffer<name> typeformer_names;
|
||||||
|
// add parameters and typeformers
|
||||||
|
for (unsigned i = 0; i < nparams; i++)
|
||||||
|
args.push_back(ref_args[i]);
|
||||||
|
for (unsigned i = nparams; i < nparams + ntypeformers; i++) {
|
||||||
|
args.push_back(ref_args[i]);
|
||||||
|
typeformer_names.push_back(mlocal_name(ref_args[i]));
|
||||||
|
}
|
||||||
|
// add indices and major premise
|
||||||
|
for (unsigned i = nparams + ntypeformers + nminors; i < ref_args.size(); i++)
|
||||||
|
args.push_back(ref_args[i]);
|
||||||
|
// create below terms (one per datatype)
|
||||||
|
// (below.{lvls} params type-formers)
|
||||||
|
// Remark: it also creates the result type
|
||||||
|
buffer<expr> belows;
|
||||||
|
expr result_type;
|
||||||
|
unsigned k = 0;
|
||||||
|
for (auto const & decl : std::get<2>(decls)) {
|
||||||
|
name const & n1 = inductive::inductive_decl_name(decl);
|
||||||
|
if (n1 == n) {
|
||||||
|
result_type = ref_args[nparams + k];
|
||||||
|
for (unsigned i = nparams + ntypeformers + nminors; i < ref_args.size(); i++)
|
||||||
|
result_type = mk_app(result_type, ref_args[i]);
|
||||||
|
}
|
||||||
|
k++;
|
||||||
|
name bname = name(n1, ind ? "ibelow" : "below");
|
||||||
|
expr below = mk_constant(bname, blvls);
|
||||||
|
for (unsigned i = 0; i < nparams; i++)
|
||||||
|
below = mk_app(below, ref_args[i]);
|
||||||
|
for (unsigned i = nparams; i < nparams + ntypeformers; i++)
|
||||||
|
below = mk_app(below, ref_args[i]);
|
||||||
|
belows.push_back(below);
|
||||||
|
}
|
||||||
|
// create functionals (one for each type former)
|
||||||
|
// Pi idxs t, below idxs t -> C idxs t
|
||||||
|
buffer<expr> Fs;
|
||||||
|
name F_name("F");
|
||||||
|
for (unsigned i = nparams, j = 0; i < nparams + ntypeformers; i++, j++) {
|
||||||
|
expr const & C = ref_args[i];
|
||||||
|
buffer<expr> F_args;
|
||||||
|
to_telescope(ngen, mlocal_type(C), F_args);
|
||||||
|
expr F_result = mk_app(C, F_args);
|
||||||
|
expr F_below = mk_app(belows[j], F_args);
|
||||||
|
F_args.push_back(mk_local(ngen.next(), "f", F_below, binder_info()));
|
||||||
|
expr F_type = Pi(F_args, F_result);
|
||||||
|
expr F = mk_local(ngen.next(), F_name.append_after(j+1), F_type, binder_info());
|
||||||
|
Fs.push_back(F);
|
||||||
|
args.push_back(F);
|
||||||
|
}
|
||||||
|
|
||||||
|
// We define brec_on/binduction_on using the recursor for this type
|
||||||
|
levels rec_lvls = cons(rlvl, lvls);
|
||||||
|
expr rec = mk_constant(rec_decl.get_name(), rec_lvls);
|
||||||
|
// add parameters to rec
|
||||||
|
for (unsigned i = 0; i < nparams; i++)
|
||||||
|
rec = mk_app(rec, ref_args[i]);
|
||||||
|
// add type formers to rec
|
||||||
|
// Pi indices t, prod (C ... t) (below ... t)
|
||||||
|
for (unsigned i = nparams, j = 0; i < nparams + ntypeformers; i++, j++) {
|
||||||
|
expr const & C = ref_args[i];
|
||||||
|
buffer<expr> C_args;
|
||||||
|
to_telescope(ngen, mlocal_type(C), C_args);
|
||||||
|
expr C_t = mk_app(C, C_args);
|
||||||
|
expr below_t = mk_app(belows[j], C_args);
|
||||||
|
expr prod = mk_prod(tc, C_t, below_t, ind);
|
||||||
|
rec = mk_app(rec, Fun(C_args, prod));
|
||||||
|
}
|
||||||
|
// add minor premises to rec
|
||||||
|
for (unsigned i = nparams + ntypeformers, j = 0; i < nparams + ntypeformers + nminors; i++, j++) {
|
||||||
|
expr minor = ref_args[i];
|
||||||
|
expr minor_type = mlocal_type(minor);
|
||||||
|
buffer<expr> minor_args;
|
||||||
|
minor_type = to_telescope(ngen, minor_type, minor_args);
|
||||||
|
buffer<expr> pairs;
|
||||||
|
for (expr & minor_arg : minor_args) {
|
||||||
|
buffer<expr> minor_arg_args;
|
||||||
|
expr minor_arg_type = to_telescope(tc, mlocal_type(minor_arg), minor_arg_args);
|
||||||
|
if (auto k = is_typeformer_app(typeformer_names, minor_arg_type)) {
|
||||||
|
buffer<expr> C_args;
|
||||||
|
get_app_args(minor_arg_type, C_args);
|
||||||
|
expr new_minor_arg_type = mk_prod(tc, minor_arg_type, mk_app(belows[*k], C_args), ind);
|
||||||
|
minor_arg = update_mlocal(minor_arg, Pi(minor_arg_args, new_minor_arg_type));
|
||||||
|
if (minor_arg_args.empty()) {
|
||||||
|
pairs.push_back(minor_arg);
|
||||||
|
} else {
|
||||||
|
expr r = mk_app(minor_arg, minor_arg_args);
|
||||||
|
expr r_1 = Fun(minor_arg_args, mk_pr1(tc, r, ind));
|
||||||
|
expr r_2 = Fun(minor_arg_args, mk_pr2(tc, r, ind));
|
||||||
|
pairs.push_back(mk_pair(tc, r_1, r_2, ind));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expr b = foldr([&](expr const & a, expr const & b) { return mk_pair(tc, a, b, ind); },
|
||||||
|
[&]() { return mk_unit_mk(rlvl, ind); },
|
||||||
|
pairs.size(), pairs.data());
|
||||||
|
unsigned F_idx = *is_typeformer_app(typeformer_names, minor_type);
|
||||||
|
expr F = Fs[F_idx];
|
||||||
|
buffer<expr> F_args;
|
||||||
|
get_app_args(minor_type, F_args);
|
||||||
|
F_args.push_back(b);
|
||||||
|
expr new_arg = mk_pair(tc, mk_app(F, F_args), b, ind);
|
||||||
|
rec = mk_app(rec, Fun(minor_args, new_arg));
|
||||||
|
}
|
||||||
|
// add indices and major to rec
|
||||||
|
for (unsigned i = nparams + ntypeformers + nminors; i < ref_args.size(); i++)
|
||||||
|
rec = mk_app(rec, ref_args[i]);
|
||||||
|
|
||||||
|
|
||||||
|
name brec_on_name = name(n, ind ? "binduction_on" : "brec_on");
|
||||||
|
expr brec_on_type = Pi(args, result_type);
|
||||||
|
expr brec_on_value = Fun(args, mk_pr1(tc, rec, ind));
|
||||||
|
|
||||||
|
bool opaque = false;
|
||||||
|
bool use_conv_opt = true;
|
||||||
|
declaration new_d = mk_definition(env, brec_on_name, blps, brec_on_type, brec_on_value,
|
||||||
|
opaque, rec_decl.get_module_idx(), use_conv_opt);
|
||||||
|
environment new_env = module::add(env, check(env, new_d));
|
||||||
|
new_env = set_reducible(new_env, brec_on_name, reducible_status::On);
|
||||||
|
return add_protected(new_env, brec_on_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
environment mk_brec_on(environment const & env, name const & n) {
|
||||||
|
return mk_brec_on(env, n, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
environment mk_binduction_on(environment const & env, name const & n) {
|
||||||
|
return mk_brec_on(env, n, true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,8 +9,12 @@ Author: Leonardo de Moura
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
/** \brief Given an inductive datatype \c n in \c env, add
|
/** \brief Given an inductive datatype \c n in \c env, add
|
||||||
<tt>n.brec_on</tt> (aka below recursion on) to the environment.
|
<tt>n.below</tt> auxiliary construction for <tt>n.brec_on</t>
|
||||||
|
(aka below recursion on) to the environment.
|
||||||
*/
|
*/
|
||||||
environment mk_below(environment const & env, name const & n);
|
environment mk_below(environment const & env, name const & n);
|
||||||
environment mk_ibelow(environment const & env, name const & n);
|
environment mk_ibelow(environment const & env, name const & n);
|
||||||
|
|
||||||
|
environment mk_brec_on(environment const & env, name const & n);
|
||||||
|
environment mk_binduction_on(environment const & env, name const & n);
|
||||||
}
|
}
|
||||||
|
|
|
@ -132,12 +132,14 @@ expr to_telescope(type_checker & tc, expr type, buffer<expr> & telescope, option
|
||||||
}
|
}
|
||||||
|
|
||||||
static expr * g_true = nullptr;
|
static expr * g_true = nullptr;
|
||||||
|
static expr * g_true_intro = nullptr;
|
||||||
static expr * g_and = nullptr;
|
static expr * g_and = nullptr;
|
||||||
static expr * g_and_intro = nullptr;
|
static expr * g_and_intro = nullptr;
|
||||||
static expr * g_and_elim_left = nullptr;
|
static expr * g_and_elim_left = nullptr;
|
||||||
static expr * g_and_elim_right = nullptr;
|
static expr * g_and_elim_right = nullptr;
|
||||||
|
|
||||||
static name * g_unit = nullptr;
|
static name * g_unit_name = nullptr;
|
||||||
|
static name * g_unit_mk_name = nullptr;
|
||||||
static name * g_prod_name = nullptr;
|
static name * g_prod_name = nullptr;
|
||||||
static name * g_prod_mk_name = nullptr;
|
static name * g_prod_mk_name = nullptr;
|
||||||
static name * g_pr1_name = nullptr;
|
static name * g_pr1_name = nullptr;
|
||||||
|
@ -145,12 +147,14 @@ static name * g_pr2_name = nullptr;
|
||||||
|
|
||||||
void initialize_definitional_util() {
|
void initialize_definitional_util() {
|
||||||
g_true = new expr(mk_constant("true"));
|
g_true = new expr(mk_constant("true"));
|
||||||
|
g_true_intro = new expr(mk_constant(name({"true", "intro"})));
|
||||||
g_and = new expr(mk_constant("and"));
|
g_and = new expr(mk_constant("and"));
|
||||||
g_and_intro = new expr(mk_constant({"and", "intro"}));
|
g_and_intro = new expr(mk_constant(name({"and", "intro"})));
|
||||||
g_and_elim_left = new expr(mk_constant({"and", "elim_left"}));
|
g_and_elim_left = new expr(mk_constant(name({"and", "elim_left"})));
|
||||||
g_and_elim_right = new expr(mk_constant({"and", "elim_right"}));
|
g_and_elim_right = new expr(mk_constant(name({"and", "elim_right"})));
|
||||||
|
|
||||||
g_unit = new name("unit");
|
g_unit_name = new name("unit");
|
||||||
|
g_unit_mk_name = new name{"unit", "star"};
|
||||||
g_prod_name = new name("prod");
|
g_prod_name = new name("prod");
|
||||||
g_prod_mk_name = new name{"prod", "mk"};
|
g_prod_mk_name = new name{"prod", "mk"};
|
||||||
g_pr1_name = new name{"prod", "pr1"};
|
g_pr1_name = new name{"prod", "pr1"};
|
||||||
|
@ -159,11 +163,13 @@ void initialize_definitional_util() {
|
||||||
|
|
||||||
void finalize_definitional_util() {
|
void finalize_definitional_util() {
|
||||||
delete g_true;
|
delete g_true;
|
||||||
|
delete g_true_intro;
|
||||||
delete g_and;
|
delete g_and;
|
||||||
delete g_and_intro;
|
delete g_and_intro;
|
||||||
delete g_and_elim_left;
|
delete g_and_elim_left;
|
||||||
delete g_and_elim_right;
|
delete g_and_elim_right;
|
||||||
delete g_unit;
|
delete g_unit_name;
|
||||||
|
delete g_unit_mk_name;
|
||||||
delete g_prod_name;
|
delete g_prod_name;
|
||||||
delete g_prod_mk_name;
|
delete g_prod_mk_name;
|
||||||
delete g_pr1_name;
|
delete g_pr1_name;
|
||||||
|
@ -174,6 +180,10 @@ expr mk_true() {
|
||||||
return *g_true;
|
return *g_true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expr mk_true_intro() {
|
||||||
|
return *g_true_intro;
|
||||||
|
}
|
||||||
|
|
||||||
expr mk_and(expr const & a, expr const & b) {
|
expr mk_and(expr const & a, expr const & b) {
|
||||||
return mk_app(*g_and, a, b);
|
return mk_app(*g_and, a, b);
|
||||||
}
|
}
|
||||||
|
@ -193,7 +203,11 @@ expr mk_and_elim_right(type_checker & tc, expr const & H) {
|
||||||
}
|
}
|
||||||
|
|
||||||
expr mk_unit(level const & l) {
|
expr mk_unit(level const & l) {
|
||||||
return mk_constant(*g_unit, {l});
|
return mk_constant(*g_unit_name, {l});
|
||||||
|
}
|
||||||
|
|
||||||
|
expr mk_unit_mk(level const & l) {
|
||||||
|
return mk_constant(*g_unit_mk_name, {l});
|
||||||
}
|
}
|
||||||
|
|
||||||
expr mk_prod(type_checker & tc, expr const & A, expr const & B) {
|
expr mk_prod(type_checker & tc, expr const & A, expr const & B) {
|
||||||
|
@ -214,14 +228,14 @@ expr mk_pr1(type_checker & tc, expr const & p) {
|
||||||
expr AxB = tc.whnf(tc.infer(p).first).first;
|
expr AxB = tc.whnf(tc.infer(p).first).first;
|
||||||
expr const & A = app_arg(app_fn(AxB));
|
expr const & A = app_arg(app_fn(AxB));
|
||||||
expr const & B = app_arg(AxB);
|
expr const & B = app_arg(AxB);
|
||||||
return mk_app(mk_constant(*g_pr1_name, const_levels(AxB)), A, B, p);
|
return mk_app(mk_constant(*g_pr1_name, const_levels(get_app_fn(AxB))), A, B, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
expr mk_pr2(type_checker & tc, expr const & p) {
|
expr mk_pr2(type_checker & tc, expr const & p) {
|
||||||
expr AxB = tc.whnf(tc.infer(p).first).first;
|
expr AxB = tc.whnf(tc.infer(p).first).first;
|
||||||
expr const & A = app_arg(app_fn(AxB));
|
expr const & A = app_arg(app_fn(AxB));
|
||||||
expr const & B = app_arg(AxB);
|
expr const & B = app_arg(AxB);
|
||||||
return mk_app(mk_constant(*g_pr2_name, const_levels(AxB)), A, B, p);
|
return mk_app(mk_constant(*g_pr2_name, const_levels(get_app_fn(AxB))), A, B, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
expr mk_unit(level const & l, bool prop) {
|
expr mk_unit(level const & l, bool prop) {
|
||||||
|
@ -231,6 +245,13 @@ expr mk_unit(level const & l, bool prop) {
|
||||||
return mk_unit(l);
|
return mk_unit(l);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expr mk_unit_mk(level const & l, bool prop) {
|
||||||
|
if (prop)
|
||||||
|
return mk_true_intro();
|
||||||
|
else
|
||||||
|
return mk_unit_mk(l);
|
||||||
|
}
|
||||||
|
|
||||||
expr mk_prod(type_checker & tc, expr const & a, expr const & b, bool prop) {
|
expr mk_prod(type_checker & tc, expr const & a, expr const & b, bool prop) {
|
||||||
if (prop)
|
if (prop)
|
||||||
return mk_and(a, b);
|
return mk_and(a, b);
|
||||||
|
|
|
@ -54,18 +54,21 @@ level get_datatype_level(expr ind_type);
|
||||||
expr instantiate_univ_param (expr const & e, name const & p, level const & l);
|
expr instantiate_univ_param (expr const & e, name const & p, level const & l);
|
||||||
|
|
||||||
expr mk_true();
|
expr mk_true();
|
||||||
|
expr mk_true_intro();
|
||||||
expr mk_and(expr const & a, expr const & b);
|
expr mk_and(expr const & a, expr const & b);
|
||||||
expr mk_and_intro(type_checker & tc, expr const & Ha, expr const & Hb);
|
expr mk_and_intro(type_checker & tc, expr const & Ha, expr const & Hb);
|
||||||
expr mk_and_elim_left(type_checker & tc, expr const & H);
|
expr mk_and_elim_left(type_checker & tc, expr const & H);
|
||||||
expr mk_and_elim_right(type_checker & tc, expr const & H);
|
expr mk_and_elim_right(type_checker & tc, expr const & H);
|
||||||
|
|
||||||
expr mk_unit(level const & l);
|
expr mk_unit(level const & l);
|
||||||
|
expr mk_unit_mk(level const & l);
|
||||||
expr mk_prod(type_checker & tc, expr const & A, expr const & B);
|
expr mk_prod(type_checker & tc, expr const & A, expr const & B);
|
||||||
expr mk_pair(type_checker & tc, expr const & a, expr const & b);
|
expr mk_pair(type_checker & tc, expr const & a, expr const & b);
|
||||||
expr mk_pr1(type_checker & tc, expr const & p);
|
expr mk_pr1(type_checker & tc, expr const & p);
|
||||||
expr mk_pr2(type_checker & tc, expr const & p);
|
expr mk_pr2(type_checker & tc, expr const & p);
|
||||||
|
|
||||||
expr mk_unit(level const & l, bool prop);
|
expr mk_unit(level const & l, bool prop);
|
||||||
|
expr mk_unit_mk(level const & l, bool prop);
|
||||||
expr mk_prod(type_checker & tc, expr const & a, expr const & b, bool prop);
|
expr mk_prod(type_checker & tc, expr const & a, expr const & b, bool prop);
|
||||||
expr mk_pair(type_checker & tc, expr const & a, expr const & b, bool prop);
|
expr mk_pair(type_checker & tc, expr const & a, expr const & b, bool prop);
|
||||||
expr mk_pr1(type_checker & tc, expr const & p, bool prop);
|
expr mk_pr1(type_checker & tc, expr const & p, bool prop);
|
||||||
|
|
|
@ -9,6 +9,7 @@ pos_num.is_inhabited|inhabited pos_num
|
||||||
pos_num.is_one|pos_num → bool
|
pos_num.is_one|pos_num → bool
|
||||||
pos_num.inc|pos_num → pos_num
|
pos_num.inc|pos_num → pos_num
|
||||||
pos_num.ibelow|pos_num → Prop
|
pos_num.ibelow|pos_num → Prop
|
||||||
|
pos_num.binduction_on|Π (n : pos_num), (Π (n : pos_num), pos_num.ibelow n → ?C n) → ?C n
|
||||||
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
pos_num.succ|pos_num → pos_num
|
pos_num.succ|pos_num → pos_num
|
||||||
pos_num.bit1|pos_num → pos_num
|
pos_num.bit1|pos_num → pos_num
|
||||||
|
@ -22,6 +23,7 @@ pos_num.no_confusion_type|Type → pos_num → pos_num → Type
|
||||||
pos_num.num_bits|pos_num → pos_num
|
pos_num.num_bits|pos_num → pos_num
|
||||||
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
||||||
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
|
pos_num.brec_on|Π (n : pos_num), (Π (n : pos_num), pos_num.below n → ?C n) → ?C n
|
||||||
pos_num.add|pos_num → pos_num → pos_num
|
pos_num.add|pos_num → pos_num → pos_num
|
||||||
pos_num|Type
|
pos_num|Type
|
||||||
-- ENDFINDP
|
-- ENDFINDP
|
||||||
|
@ -36,6 +38,7 @@ pos_num.is_inhabited|inhabited pos_num
|
||||||
pos_num.is_one|pos_num → bool
|
pos_num.is_one|pos_num → bool
|
||||||
pos_num.inc|pos_num → pos_num
|
pos_num.inc|pos_num → pos_num
|
||||||
pos_num.ibelow|pos_num → Prop
|
pos_num.ibelow|pos_num → Prop
|
||||||
|
pos_num.binduction_on|Π (n : pos_num), (Π (n : pos_num), pos_num.ibelow n → ?C n) → ?C n
|
||||||
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
pos_num.succ|pos_num → pos_num
|
pos_num.succ|pos_num → pos_num
|
||||||
pos_num.bit1|pos_num → pos_num
|
pos_num.bit1|pos_num → pos_num
|
||||||
|
@ -48,6 +51,7 @@ pos_num.mul|pos_num → pos_num → pos_num
|
||||||
pos_num.no_confusion_type|Type → pos_num → pos_num → Type
|
pos_num.no_confusion_type|Type → pos_num → pos_num → Type
|
||||||
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
||||||
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
|
pos_num.brec_on|Π (n : pos_num), (Π (n : pos_num), pos_num.below n → ?C n) → ?C n
|
||||||
pos_num.add|pos_num → pos_num → pos_num
|
pos_num.add|pos_num → pos_num → pos_num
|
||||||
pos_num|Type
|
pos_num|Type
|
||||||
-- ENDFINDP
|
-- ENDFINDP
|
||||||
|
@ -58,6 +62,7 @@ pos_num.is_inhabited|inhabited pos_num
|
||||||
pos_num.is_one|pos_num → bool
|
pos_num.is_one|pos_num → bool
|
||||||
pos_num.inc|pos_num → pos_num
|
pos_num.inc|pos_num → pos_num
|
||||||
pos_num.ibelow|pos_num → Prop
|
pos_num.ibelow|pos_num → Prop
|
||||||
|
pos_num.binduction_on|Π (n : pos_num), (Π (n : pos_num), pos_num.ibelow n → ?C n) → ?C n
|
||||||
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.induction_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
pos_num.succ|pos_num → pos_num
|
pos_num.succ|pos_num → pos_num
|
||||||
pos_num.bit1|pos_num → pos_num
|
pos_num.bit1|pos_num → pos_num
|
||||||
|
@ -70,6 +75,7 @@ pos_num.mul|pos_num → pos_num → pos_num
|
||||||
pos_num.no_confusion_type|Type → pos_num → pos_num → Type
|
pos_num.no_confusion_type|Type → pos_num → pos_num → Type
|
||||||
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
pos_num.no_confusion|eq ?v1 ?v2 → pos_num.no_confusion_type ?P ?v1 ?v2
|
||||||
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
pos_num.rec_on|Π (n : pos_num), ?C pos_num.one → (Π (a : pos_num), ?C a → ?C (pos_num.bit1 a)) → (Π (a : pos_num), ?C a → ?C (pos_num.bit0 a)) → ?C n
|
||||||
|
pos_num.brec_on|Π (n : pos_num), (Π (n : pos_num), pos_num.below n → ?C n) → ?C n
|
||||||
pos_num.add|pos_num → pos_num → pos_num
|
pos_num.add|pos_num → pos_num → pos_num
|
||||||
pos_num|Type
|
pos_num|Type
|
||||||
-- ENDFINDP
|
-- ENDFINDP
|
||||||
|
|
|
@ -2,6 +2,7 @@ import data.nat.basic data.prod
|
||||||
open prod
|
open prod
|
||||||
|
|
||||||
namespace nat
|
namespace nat
|
||||||
|
namespace manual
|
||||||
definition brec_on {C : nat → Type} (n : nat) (F : Π (n : nat), @below C n → C n) : C n :=
|
definition brec_on {C : nat → Type} (n : nat) (F : Π (n : nat), @below C n → C n) : C n :=
|
||||||
have general : C n × @below C n, from
|
have general : C n × @below C n, from
|
||||||
rec_on n
|
rec_on n
|
||||||
|
@ -13,6 +14,7 @@ namespace nat
|
||||||
F (succ n₁) b,
|
F (succ n₁) b,
|
||||||
pair c b),
|
pair c b),
|
||||||
pr₁ general
|
pr₁ general
|
||||||
|
end manual
|
||||||
|
|
||||||
definition fib (n : nat) :=
|
definition fib (n : nat) :=
|
||||||
brec_on n (λ (n : nat),
|
brec_on n (λ (n : nat),
|
||||||
|
|
|
@ -49,7 +49,6 @@ definition pbelow.{l₁ l₂} {A : Type.{l₁}} {B : Type.{l₂}} (C : ftree A B
|
||||||
let p₂ : Prop := fc₂ ∧ fr₂ in
|
let p₂ : Prop := fc₂ ∧ fr₂ in
|
||||||
p₁ ∧ p₂)
|
p₁ ∧ p₂)
|
||||||
t
|
t
|
||||||
end manual
|
|
||||||
|
|
||||||
definition brec_on.{l l₁ l₂} {A : Type.{l₁}} {B : Type.{l₂}} {C : ftree A B → Type.{l+1}}
|
definition brec_on.{l l₁ l₂} {A : Type.{l₁}} {B : Type.{l₂}} {C : ftree A B → Type.{l+1}}
|
||||||
(t : ftree A B)
|
(t : ftree A B)
|
||||||
|
@ -110,4 +109,5 @@ have gen : C t ∧ @ibelow A B C t, from
|
||||||
and.intro c b)
|
and.intro c b)
|
||||||
t,
|
t,
|
||||||
and.elim_left gen
|
and.elim_left gen
|
||||||
|
end manual
|
||||||
end ftree
|
end ftree
|
||||||
|
|
|
@ -9,6 +9,7 @@ namespace vector
|
||||||
print definition no_confusion
|
print definition no_confusion
|
||||||
infixr `::` := vcons
|
infixr `::` := vcons
|
||||||
|
|
||||||
|
namespace play
|
||||||
section
|
section
|
||||||
universe variables l₁ l₂
|
universe variables l₁ l₂
|
||||||
variable {A : Type.{l₁}}
|
variable {A : Type.{l₁}}
|
||||||
|
@ -25,6 +26,7 @@ namespace vector
|
||||||
pair c b),
|
pair c b),
|
||||||
pr₁ general
|
pr₁ general
|
||||||
end
|
end
|
||||||
|
end play
|
||||||
|
|
||||||
print "====================="
|
print "====================="
|
||||||
definition append {A : Type} {n m : nat} (w : vector A m) (v : vector A n) : vector A (n + m) :=
|
definition append {A : Type} {n m : nat} (w : vector A m) (v : vector A n) : vector A (n + m) :=
|
||||||
|
|
|
@ -22,6 +22,7 @@ namespace vector
|
||||||
set_option pp.universes true
|
set_option pp.universes true
|
||||||
check @below
|
check @below
|
||||||
|
|
||||||
|
namespace manual
|
||||||
section
|
section
|
||||||
universe variables l₁ l₂
|
universe variables l₁ l₂
|
||||||
variable {A : Type.{l₁}}
|
variable {A : Type.{l₁}}
|
||||||
|
@ -38,6 +39,7 @@ namespace vector
|
||||||
pair c b),
|
pair c b),
|
||||||
pr₁ general
|
pr₁ general
|
||||||
end
|
end
|
||||||
|
end manual
|
||||||
|
|
||||||
-- check brec_on
|
-- check brec_on
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue