feat(frontends/lean/inductive_cmd): don't force user to repeat argument declarations in every datatype in a mutually recursive datatype declaration

This commit is contained in:
Leonardo de Moura 2014-10-13 20:48:23 -07:00
parent 8aa60cbc1c
commit b7c1b348d1
4 changed files with 28 additions and 84 deletions

View file

@ -161,35 +161,18 @@ struct inductive_cmd_fn {
If this is the first declaration in a mutually recursive block, then this method If this is the first declaration in a mutually recursive block, then this method
stores the levels in m_explicit_levels, and set m_using_explicit_levels to true (iff they were provided). stores the levels in m_explicit_levels, and set m_using_explicit_levels to true (iff they were provided).
If this is not the first declaration, then this function just checks if the parsed parameters If this is not the first declaration, then this function just checks if the user
match the ones in the first declaration (stored in \c m_explicit_levels) is not providing explicit universe levels again.
*/ */
void parse_inductive_univ_params() { void parse_inductive_univ_params() {
buffer<name> curr_ls_buffer; buffer<name> curr_ls_buffer;
if (parse_univ_params(m_p, curr_ls_buffer)) { if (parse_univ_params(m_p, curr_ls_buffer)) {
if (m_first) { if (!m_first) {
m_using_explicit_levels = true;
m_explicit_levels.append(curr_ls_buffer);
} else if (!m_using_explicit_levels) {
throw_error("invalid mutually recursive declaration, " throw_error("invalid mutually recursive declaration, "
"explicit universe levels were not provided for previous inductive types in this declaration"); "explicit universe levels should only be provided to first inductive type in this declaration");
} else if (curr_ls_buffer.size() != m_explicit_levels.size()) {
throw_error("invalid mutually recursive declaration, "
"all inductive types must have the same number of universe paramaters");
} else {
for (unsigned i = 0; i < m_explicit_levels.size(); i++) {
if (curr_ls_buffer[i] != m_explicit_levels[i])
throw_error("invalid mutually recursive inductive declaration, "
"all inductive types must have the same universe paramaters");
}
}
} else {
if (m_first) {
m_using_explicit_levels = false;
} else if (m_using_explicit_levels) {
throw_error("invalid mutually recursive declaration, "
"explicit universe levels were provided for previous inductive types in this declaration");
} }
m_using_explicit_levels = true;
m_explicit_levels.append(curr_ls_buffer);
} }
} }
@ -200,8 +183,10 @@ struct inductive_cmd_fn {
m_pos = m_p.pos(); m_pos = m_p.pos();
if (m_p.curr_is_token(get_assign_tk())) { if (m_p.curr_is_token(get_assign_tk())) {
type = mk_sort(mk_level_placeholder()); type = mk_sort(mk_level_placeholder());
} else if (!m_p.curr_is_token(get_colon_tk())) { } else if (m_first && !m_p.curr_is_token(get_colon_tk())) {
lean_assert(m_params.empty());
m_p.parse_binders(ps); m_p.parse_binders(ps);
m_num_params = ps.size();
if (m_p.curr_is_token(get_colon_tk())) { if (m_p.curr_is_token(get_colon_tk())) {
m_p.next(); m_p.next();
type = m_p.parse_scoped_expr(ps); type = m_p.parse_scoped_expr(ps);
@ -210,17 +195,12 @@ struct inductive_cmd_fn {
} }
type = Pi(ps, type, m_p); type = Pi(ps, type, m_p);
} else { } else {
m_p.next(); m_p.check_token_next(get_colon_tk(), "invalid mutually recursive inductive declaration, "
"':' expected (remark: parameters should be provided only to first datatype)");
type = m_p.parse_expr(); type = m_p.parse_expr();
} }
// check if number of parameters if (!m_first)
if (m_first) { type = Pi(m_params, type, m_p);
m_num_params = ps.size();
} else if (m_num_params != ps.size()) {
// mutually recursive declaration checks
throw_error("invalid mutually recursive inductive declaration, "
"all inductive types must have the same number of arguments");
}
return type; return type;
} }
@ -240,49 +220,11 @@ struct inductive_cmd_fn {
return mk_local(m_p.mk_fresh_name(), binding_name(b), binding_domain(b), binding_info(b)); return mk_local(m_p.mk_fresh_name(), binding_name(b), binding_domain(b), binding_info(b));
} }
/** \brief Check if the parameters of \c d_type and \c first_d_type are equal. */
void check_params(expr d_type, expr first_d_type) {
for (unsigned i = 0; i < m_num_params; i++) {
if (!m_tc->is_def_eq(binding_domain(d_type), binding_domain(first_d_type)).first)
throw_error(sstream() << "invalid parameter #" << (i+1) << " in mutually recursive inductive declaration, "
<< "all inductive types must have equivalent parameters");
expr l = mk_local_for(d_type);
d_type = instantiate(binding_body(d_type), l);
first_d_type = instantiate(binding_body(first_d_type), l);
}
}
/** \brief Check if the level names in d_lvls and \c first_d_lvls are equal. */
void check_levels(level_param_names d_lvls, level_param_names first_d_lvls) {
while (!is_nil(d_lvls) && !is_nil(first_d_lvls)) {
if (head(d_lvls) != head(first_d_lvls))
throw_error(sstream() << "invalid mutually recursive inductive declaration, "
<< "all declarations must use the same universe parameter names, mismatch: '"
<< head(d_lvls) << "' and '" << head(first_d_lvls) << "' "
<< "(if the universe parameters were inferred, then provide them explicitly to fix the problem)");
d_lvls = tail(d_lvls);
first_d_lvls = tail(first_d_lvls);
}
if (!is_nil(d_lvls) || !is_nil(first_d_lvls))
throw_error("invalid mutually recursive inductive declaration, "
"all declarations must have the same number of arguments "
"(this is error is probably due to inferred implicit parameters, "
"provide all parameters explicitly to fix the problem");
}
/** \brief Set explicit datatype parameters as local constants in m_params */ /** \brief Set explicit datatype parameters as local constants in m_params */
void update_params(expr d_type) { void set_params(expr d_type) {
// Remark: if m_params is not empty, then this function will reuse their names. lean_assert(m_params.empty());
// Reason: reference to these names
lean_assert(m_num_params >= m_params.size());
buffer<name> param_names;
for (unsigned i = 0; i < m_num_params - m_params.size(); i++)
param_names.push_back(m_p.mk_fresh_name());
for (expr const & param : m_params)
param_names.push_back(mlocal_name(param)); // keep existing internal names
m_params.clear();
for (unsigned i = 0; i < m_num_params; i++) { for (unsigned i = 0; i < m_num_params; i++) {
expr l = mk_local(param_names[i], binding_name(d_type), binding_domain(d_type), binding_info(d_type)); expr l = mk_local(binding_name(d_type), binding_name(d_type), binding_domain(d_type), binding_info(d_type));
m_params.push_back(l); m_params.push_back(l);
d_type = instantiate(binding_body(d_type), l); d_type = instantiate(binding_body(d_type), l);
} }
@ -324,13 +266,17 @@ struct inductive_cmd_fn {
} }
void parse_inductive_decls(buffer<inductive_decl> & decls) { void parse_inductive_decls(buffer<inductive_decl> & decls) {
optional<expr> first_d_type;
while (true) { while (true) {
parser::local_scope l_scope(m_p); parser::local_scope l_scope(m_p);
pair<name, name> d_names = parse_inductive_decl_name(); pair<name, name> d_names = parse_inductive_decl_name();
name d_short_name = d_names.first; name d_short_name = d_names.first;
name d_name = d_names.second; name d_name = d_names.second;
parse_inductive_univ_params(); parse_inductive_univ_params();
if (!m_first) {
add_params_to_local_scope();
for (name const & lvl_name : m_explicit_levels)
m_p.add_local_level(lvl_name, mk_param_univ(lvl_name));
}
modifiers mods; modifiers mods;
mods.parse(m_p); mods.parse(m_p);
m_modifiers.insert(d_name, mods); m_modifiers.insert(d_name, mods);
@ -342,18 +288,16 @@ struct inductive_cmd_fn {
} }
if (m_first) { if (m_first) {
m_levels.append(m_explicit_levels); m_levels.append(m_explicit_levels);
update_params(d_type); set_params(d_type);
} else {
lean_assert(first_d_type);
check_params(d_type, *first_d_type);
} }
if (empty_type) { if (empty_type) {
decls.push_back(inductive_decl(d_name, d_type, list<intro_rule>())); decls.push_back(inductive_decl(d_name, d_type, list<intro_rule>()));
} else { } else {
if (m_first)
add_params_to_local_scope();
expr d_const = mk_constant(d_name, param_names_to_levels(to_list(m_explicit_levels.begin(), expr d_const = mk_constant(d_name, param_names_to_levels(to_list(m_explicit_levels.begin(),
m_explicit_levels.end()))); m_explicit_levels.end())));
m_p.add_local_expr(d_short_name, d_const); m_p.add_local_expr(d_short_name, d_const);
add_params_to_local_scope();
auto d_intro_rules = parse_intro_rules(d_name); auto d_intro_rules = parse_intro_rules(d_name);
decls.push_back(inductive_decl(d_name, d_type, d_intro_rules)); decls.push_back(inductive_decl(d_name, d_type, d_intro_rules));
} }
@ -362,7 +306,6 @@ struct inductive_cmd_fn {
} }
m_p.next(); m_p.next();
m_first = false; m_first = false;
first_d_type = d_type;
} }
} }
/** \brief Include in m_levels any local level referenced by decls. */ /** \brief Include in m_levels any local level referenced by decls. */

View file

@ -1,9 +1,10 @@
inductive tree (A : Type) : Type := inductive tree (A : Type) : Type :=
node : A → forest A → tree A node : A → forest A → tree A
with forest (A : Type) : Type := with forest : Type :=
nil : forest A, nil : forest A,
cons : tree A → forest A → forest A cons : tree A → forest A → forest A
check tree.{1} check tree.{1}
check forest.{1} check forest.{1}
check tree.rec.{1 1} check tree.rec.{1 1}

View file

@ -1,6 +1,6 @@
inductive tree.{u} (A : Type.{u}) : Type.{max u 1} := inductive tree.{u} (A : Type.{u}) : Type.{max u 1} :=
node : A → forest.{u} A → tree.{u} A node : A → forest.{u} A → tree.{u} A
with forest.{u} (A : Type.{u}) : Type.{max u 1} := with forest : Type.{max u 1} :=
nil : forest.{u} A, nil : forest.{u} A,
cons : tree.{u} A → forest.{u} A → forest.{u} A cons : tree.{u} A → forest.{u} A → forest.{u} A

View file

@ -14,7 +14,7 @@ constant num : Type.{1}
namespace Tree namespace Tree
inductive tree (A : Type) : Type := inductive tree (A : Type) : Type :=
node : A → forest A → tree A node : A → forest A → tree A
with forest (A : Type) : Type := with forest : Type :=
nil : forest A, nil : forest A,
cons : tree A → forest A → forest A cons : tree A → forest A → forest A
end Tree end Tree