feat(frontends/lean): search for identifiers in the stack of namespaces; reject non-atomic names as local names
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
e6d4c01b88
commit
b43fb7448c
7 changed files with 66 additions and 36 deletions
|
@ -73,7 +73,7 @@ static expr parse_let(parser & p, pos_info const & pos) {
|
||||||
return parse_let_body(p, pos);
|
return parse_let_body(p, pos);
|
||||||
} else {
|
} else {
|
||||||
auto pos = p.pos();
|
auto pos = p.pos();
|
||||||
name id = p.check_id_next("invalid let declaration, identifier expected");
|
name id = p.check_atomic_id_next("invalid let declaration, identifier expected");
|
||||||
bool is_opaque = true;
|
bool is_opaque = true;
|
||||||
bool is_fact = false;
|
bool is_fact = false;
|
||||||
expr type, value;
|
expr type, value;
|
||||||
|
|
|
@ -284,6 +284,14 @@ name parser::check_id_next(char const * msg) {
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
name parser::check_atomic_id_next(char const * msg) {
|
||||||
|
auto p = pos();
|
||||||
|
name id = check_id_next(msg);
|
||||||
|
if (!id.is_atomic())
|
||||||
|
throw parser_error(msg, p);
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
expr parser::mk_app(expr fn, expr arg, pos_info const & p) {
|
expr parser::mk_app(expr fn, expr arg, pos_info const & p) {
|
||||||
return save_pos(::lean::mk_app(fn, arg), p);
|
return save_pos(::lean::mk_app(fn, arg), p);
|
||||||
}
|
}
|
||||||
|
@ -328,14 +336,6 @@ unsigned parser::get_local_index(name const & n) const {
|
||||||
return m_local_decls.find_idx(n);
|
return m_local_decls.find_idx(n);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Parse a sequence of identifiers <tt>ID*</tt>. Store the result in \c result. */
|
|
||||||
void parser::parse_names(buffer<std::pair<pos_info, name>> & result) {
|
|
||||||
while (curr_is_identifier()) {
|
|
||||||
result.emplace_back(pos(), get_name_val());
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static name g_period("."), g_colon(":"), g_lparen("("), g_rparen(")"), g_llevel_curly(".{");
|
static name g_period("."), g_colon(":"), g_lparen("("), g_rparen(")"), g_llevel_curly(".{");
|
||||||
static name g_lcurly("{"), g_rcurly("}"), g_ldcurly("⦃"), g_rdcurly("⦄"), g_lbracket("["), g_rbracket("]");
|
static name g_lcurly("{"), g_rcurly("}"), g_ldcurly("⦃"), g_rdcurly("⦄"), g_lbracket("["), g_rbracket("]");
|
||||||
static name g_bar("|"), g_comma(","), g_add("+"), g_max("max"), g_imax("imax"), g_cup("\u2294");
|
static name g_bar("|"), g_comma(","), g_add("+"), g_max("max"), g_imax("imax"), g_cup("\u2294");
|
||||||
|
@ -566,10 +566,7 @@ void parser::parse_close_binder_info(optional<binder_info> const & bi) {
|
||||||
/** \brief Parse <tt>ID ':' expr</tt>, where the expression represents the type of the identifier. */
|
/** \brief Parse <tt>ID ':' expr</tt>, where the expression represents the type of the identifier. */
|
||||||
expr parser::parse_binder_core(binder_info const & bi) {
|
expr parser::parse_binder_core(binder_info const & bi) {
|
||||||
auto p = pos();
|
auto p = pos();
|
||||||
if (!curr_is_identifier())
|
name id = check_atomic_id_next("invalid binder, atomic identifier expected");
|
||||||
throw parser_error("invalid binder, identifier expected", p);
|
|
||||||
name id = get_name_val();
|
|
||||||
next();
|
|
||||||
expr type;
|
expr type;
|
||||||
if (curr_is_token(g_colon)) {
|
if (curr_is_token(g_colon)) {
|
||||||
next();
|
next();
|
||||||
|
@ -597,7 +594,10 @@ expr parser::parse_binder() {
|
||||||
*/
|
*/
|
||||||
void parser::parse_binder_block(buffer<expr> & r, binder_info const & bi) {
|
void parser::parse_binder_block(buffer<expr> & r, binder_info const & bi) {
|
||||||
buffer<std::pair<pos_info, name>> names;
|
buffer<std::pair<pos_info, name>> names;
|
||||||
parse_names(names);
|
while (curr_is_identifier()) {
|
||||||
|
auto p = pos();
|
||||||
|
names.emplace_back(p, check_atomic_id_next("invalid binder, atomic identifier expected"));
|
||||||
|
}
|
||||||
if (names.empty())
|
if (names.empty())
|
||||||
throw parser_error("invalid binder, identifier expected", pos());
|
throw parser_error("invalid binder, identifier expected", pos());
|
||||||
optional<expr> type;
|
optional<expr> type;
|
||||||
|
@ -798,29 +798,42 @@ expr parser::id_to_expr(name const & id, pos_info const & p) {
|
||||||
next();
|
next();
|
||||||
ls = to_list(lvl_buffer.begin(), lvl_buffer.end());
|
ls = to_list(lvl_buffer.begin(), lvl_buffer.end());
|
||||||
}
|
}
|
||||||
// locals
|
if (id.is_atomic()) {
|
||||||
if (auto it1 = m_local_decls.find(id))
|
// locals
|
||||||
return copy_with_new_pos(propagate_levels(*it1, ls), p);
|
if (auto it1 = m_local_decls.find(id))
|
||||||
optional<expr> r;
|
return copy_with_new_pos(propagate_levels(*it1, ls), p);
|
||||||
// globals
|
optional<expr> r;
|
||||||
if (m_env.find(id))
|
// globals
|
||||||
r = save_pos(mk_constant(id, ls), p);
|
if (m_env.find(id))
|
||||||
// aliases
|
r = save_pos(mk_constant(id, ls), p);
|
||||||
auto as = get_alias_exprs(m_env, id);
|
// aliases
|
||||||
if (!is_nil(as)) {
|
auto as = get_alias_exprs(m_env, id);
|
||||||
buffer<expr> new_as;
|
if (!is_nil(as)) {
|
||||||
if (r)
|
buffer<expr> new_as;
|
||||||
new_as.push_back(*r);
|
if (r)
|
||||||
for (auto const & e : as) {
|
new_as.push_back(*r);
|
||||||
new_as.push_back(copy_with_new_pos(propagate_levels(e, ls), p));
|
for (auto const & e : as) {
|
||||||
|
new_as.push_back(copy_with_new_pos(propagate_levels(e, ls), p));
|
||||||
|
}
|
||||||
|
r = save_pos(mk_choice(new_as.size(), new_as.data()), p);
|
||||||
|
}
|
||||||
|
if (!r && m_no_undef_id_error)
|
||||||
|
r = save_pos(mk_constant(get_namespace(m_env) + id, ls), p);
|
||||||
|
if (!r)
|
||||||
|
throw parser_error(sstream() << "unknown identifier '" << id << "'", p);
|
||||||
|
return *r;
|
||||||
|
} else {
|
||||||
|
if (m_env.find(id)) {
|
||||||
|
return save_pos(mk_constant(id, ls), p);
|
||||||
|
} else {
|
||||||
|
for (name const & ns : get_namespaces(m_env)) {
|
||||||
|
auto new_id = ns + id;
|
||||||
|
if (m_env.find(new_id))
|
||||||
|
return save_pos(mk_constant(new_id, ls), p);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
r = save_pos(mk_choice(new_as.size(), new_as.data()), p);
|
|
||||||
}
|
|
||||||
if (!r && m_no_undef_id_error)
|
|
||||||
r = save_pos(mk_constant(get_namespace(m_env) + id, ls), p);
|
|
||||||
if (!r)
|
|
||||||
throw parser_error(sstream() << "unknown identifier '" << id << "'", p);
|
throw parser_error(sstream() << "unknown identifier '" << id << "'", p);
|
||||||
return *r;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
expr parser::parse_id() {
|
expr parser::parse_id() {
|
||||||
|
|
|
@ -183,6 +183,9 @@ public:
|
||||||
void check_token_next(name const & tk, char const * msg);
|
void check_token_next(name const & tk, char const * msg);
|
||||||
/** \brief Check if the current token is an identifier, if it is return it and move to next token, otherwise throw an exception. */
|
/** \brief Check if the current token is an identifier, if it is return it and move to next token, otherwise throw an exception. */
|
||||||
name check_id_next(char const * msg);
|
name check_id_next(char const * msg);
|
||||||
|
/** \brief Check if the current token is an atomic identifier, if it is return it and move to next token,
|
||||||
|
otherwise throw an exception. */
|
||||||
|
name check_atomic_id_next(char const * msg);
|
||||||
|
|
||||||
mpq const & get_num_val() const { return m_scanner.get_num_val(); }
|
mpq const & get_num_val() const { return m_scanner.get_num_val(); }
|
||||||
name const & get_name_val() const { return m_scanner.get_name_val(); }
|
name const & get_name_val() const { return m_scanner.get_name_val(); }
|
||||||
|
@ -193,7 +196,6 @@ public:
|
||||||
regular regular_stream() const { return regular(env(), ios()); }
|
regular regular_stream() const { return regular(env(), ios()); }
|
||||||
diagnostic diagnostic_stream() const { return diagnostic(env(), ios()); }
|
diagnostic diagnostic_stream() const { return diagnostic(env(), ios()); }
|
||||||
|
|
||||||
void parse_names(buffer<std::pair<pos_info, name>> & result);
|
|
||||||
unsigned get_small_nat();
|
unsigned get_small_nat();
|
||||||
unsigned parse_small_nat();
|
unsigned parse_small_nat();
|
||||||
double parse_double();
|
double parse_double();
|
||||||
|
|
|
@ -126,13 +126,20 @@ static environment update(environment const & env, aliases_ext const & ext) {
|
||||||
return env.update(g_ext.m_ext_id, std::make_shared<aliases_ext>(ext));
|
return env.update(g_ext.m_ext_id, std::make_shared<aliases_ext>(ext));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void check_atomic(name const & a) {
|
||||||
|
if (!a.is_atomic())
|
||||||
|
throw exception(sstream() << "invalid alias '" << a << "', aliases must be atomic names");
|
||||||
|
}
|
||||||
|
|
||||||
environment add_alias(environment const & env, name const & a, expr const & e) {
|
environment add_alias(environment const & env, name const & a, expr const & e) {
|
||||||
|
check_atomic(a);
|
||||||
aliases_ext ext = get_extension(env);
|
aliases_ext ext = get_extension(env);
|
||||||
ext.add_alias(a, e);
|
ext.add_alias(a, e);
|
||||||
return update(env, ext);
|
return update(env, ext);
|
||||||
}
|
}
|
||||||
|
|
||||||
environment add_decl_alias(environment const & env, name const & a, expr const & e) {
|
environment add_decl_alias(environment const & env, name const & a, expr const & e) {
|
||||||
|
check_atomic(a);
|
||||||
aliases_ext ext = get_extension(env);
|
aliases_ext ext = get_extension(env);
|
||||||
ext.add_decl_alias(a, e);
|
ext.add_decl_alias(a, e);
|
||||||
return update(env, ext);
|
return update(env, ext);
|
||||||
|
@ -154,6 +161,7 @@ static void check_no_shadow(environment const & env, name const & a) {
|
||||||
}
|
}
|
||||||
|
|
||||||
environment add_alias(environment const & env, name const & a, level const & l) {
|
environment add_alias(environment const & env, name const & a, level const & l) {
|
||||||
|
check_atomic(a);
|
||||||
check_no_shadow(env, a);
|
check_no_shadow(env, a);
|
||||||
aliases_ext ext = get_extension(env);
|
aliases_ext ext = get_extension(env);
|
||||||
ext.add_alias(a, l);
|
ext.add_alias(a, l);
|
||||||
|
|
|
@ -49,6 +49,10 @@ name const & get_namespace(environment const & env) {
|
||||||
return !is_nil(ext.m_namespaces) ? head(ext.m_namespaces) : name::anonymous();
|
return !is_nil(ext.m_namespaces) ? head(ext.m_namespaces) : name::anonymous();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
list<name> const & get_namespaces(environment const & env) {
|
||||||
|
return get_extension(env).m_namespaces;
|
||||||
|
}
|
||||||
|
|
||||||
bool in_section(environment const & env) {
|
bool in_section(environment const & env) {
|
||||||
scope_mng_ext const & ext = get_extension(env);
|
scope_mng_ext const & ext = get_extension(env);
|
||||||
return !is_nil(ext.m_in_section) && head(ext.m_in_section);
|
return !is_nil(ext.m_in_section) && head(ext.m_in_section);
|
||||||
|
|
|
@ -28,6 +28,7 @@ environment push_scope(environment const & env, io_state const & ios, name const
|
||||||
environment pop_scope(environment const & env);
|
environment pop_scope(environment const & env);
|
||||||
|
|
||||||
name const & get_namespace(environment const & env);
|
name const & get_namespace(environment const & env);
|
||||||
|
list<name> const & get_namespaces(environment const & env);
|
||||||
bool in_section(environment const & env);
|
bool in_section(environment const & env);
|
||||||
|
|
||||||
/** \brief Check if \c n may be a reference to a namespace, if it is return it.
|
/** \brief Check if \c n may be a reference to a namespace, if it is return it.
|
||||||
|
|
|
@ -77,6 +77,8 @@ namespace semigroup
|
||||||
end
|
end
|
||||||
|
|
||||||
namespace monoid
|
namespace monoid
|
||||||
|
check semigroup.mul
|
||||||
|
|
||||||
inductive monoid_struct (A : Type) : Type :=
|
inductive monoid_struct (A : Type) : Type :=
|
||||||
| mk_monoid_struct : Π (mul : A → A → A) (id : A), is_assoc mul → is_id mul id → monoid_struct A
|
| mk_monoid_struct : Π (mul : A → A → A) (id : A), is_assoc mul → is_id mul id → monoid_struct A
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue