feat(frontends/lean/notation_cmd): allow local notation to override reserved notation

closes #712
This commit is contained in:
Leonardo de Moura 2015-07-07 17:30:46 -07:00
parent 4b1b3e277f
commit a27b20cd9c
3 changed files with 111 additions and 37 deletions

View file

@ -145,7 +145,9 @@ static auto parse_mixfix_notation(parser & p, mixfix_kind k, bool overload, nota
} }
if (p.curr_is_token(get_colon_tk())) { if (p.curr_is_token(get_colon_tk())) {
if (reserved_pt) // Remark: we do not throw an exception, if it is local notation.
// We allow local notation to override reserved one.
if (!g_allow_local && reserved_pt)
throw parser_error("invalid notation declaration, invalid ':' occurrence " throw parser_error("invalid notation declaration, invalid ':' occurrence "
"(declaration matches reserved notation)", p.pos()); "(declaration matches reserved notation)", p.pos());
p.next(); p.next();
@ -182,7 +184,7 @@ static auto parse_mixfix_notation(parser & p, mixfix_kind k, bool overload, nota
"solution: use the 'precedence' command", p.pos()); "solution: use the 'precedence' command", p.pos());
} }
if (reserved_action) { if (!g_allow_local && reserved_action) {
switch (k) { switch (k) {
case mixfix_kind::infixl: case mixfix_kind::infixl:
if (reserved_action->kind() != notation::action_kind::Expr || reserved_action->rbp() != *prec) if (reserved_action->kind() != notation::action_kind::Expr || reserved_action->rbp() != *prec)
@ -439,6 +441,35 @@ static unsigned parse_binders_rbp(parser & p) {
} }
} }
static transition parse_transition(parser & p, optional<parse_table> const & pt, name const & tk,
buffer<expr> & locals, buffer<token_entry> & new_tokens, notation_entry_group grp) {
if (p.curr_is_token_or_id(get_binder_tk())) {
p.next();
unsigned rbp = parse_binders_rbp(p);
return transition(tk, mk_binder_action(rbp));
} else if (p.curr_is_token_or_id(get_binders_tk())) {
p.next();
unsigned rbp = parse_binders_rbp(p);
return transition(tk, mk_binders_action(rbp));
} else if (p.curr_is_identifier()) {
unsigned default_prec = get_default_prec(pt, tk);
name n = p.get_name_val();
p.next();
action a = parse_action(p, tk, default_prec, locals, new_tokens, grp);
expr local_type = mk_Prop(); // type used in notation local declarations, it is irrelevant
expr l = mk_local(n, local_type);
p.add_local(l);
locals.push_back(l);
return transition(tk, a);
} else if (p.curr_is_quoted_symbol() || p.curr_is_keyword() ||
p.curr_is_token(get_assign_tk()) || p.curr_is_command() || p.curr_is_eof()) {
return transition(tk, mk_skip_action());
} else {
throw parser_error("invalid notation declaration, quoted-symbol, identifier, "
"'binder', 'binders' expected", p.pos());
}
}
static notation_entry parse_notation_core(parser & p, bool overload, notation_entry_group grp, buffer<token_entry> & new_tokens, bool parse_only, static notation_entry parse_notation_core(parser & p, bool overload, notation_entry_group grp, buffer<token_entry> & new_tokens, bool parse_only,
unsigned priority) { unsigned priority) {
buffer<expr> locals; buffer<expr> locals;
@ -477,6 +508,10 @@ static notation_entry parse_notation_core(parser & p, bool overload, notation_en
switch (a.kind()) { switch (a.kind()) {
case notation::action_kind::Skip: case notation::action_kind::Skip:
if (!p.curr_is_quoted_symbol() && !p.curr_is_keyword() && !p.curr_is_token(get_assign_tk())) { if (!p.curr_is_quoted_symbol() && !p.curr_is_keyword() && !p.curr_is_token(get_assign_tk())) {
if (g_allow_local && !p.curr_is_token_or_id(get_binders_tk())) {
ts.push_back(parse_transition(p, pt, tk, locals, new_tokens, grp));
break;
}
p.check_token_or_id_next(get_binders_tk(), p.check_token_or_id_next(get_binders_tk(),
"invalid notation declaration, quoted-symbol, keyword or `:=` expected " "invalid notation declaration, quoted-symbol, keyword or `:=` expected "
"(declaration prefix matches reserved notation)"); "(declaration prefix matches reserved notation)");
@ -484,12 +519,20 @@ static notation_entry parse_notation_core(parser & p, bool overload, notation_en
ts.push_back(transition(tk, a)); ts.push_back(transition(tk, a));
break; break;
case notation::action_kind::Binder: case notation::action_kind::Binder:
p.check_token_or_id_next(get_binders_tk(), if (g_allow_local && !p.curr_is_token_or_id(get_binder_tk())) {
ts.push_back(parse_transition(p, pt, tk, locals, new_tokens, grp));
break;
}
p.check_token_or_id_next(get_binder_tk(),
"invalid notation declaration, 'binder' expected " "invalid notation declaration, 'binder' expected "
"(declaration prefix matches reserved notation)"); "(declaration prefix matches reserved notation)");
ts.push_back(transition(tk, a)); ts.push_back(transition(tk, a));
break; break;
case notation::action_kind::Binders: case notation::action_kind::Binders:
if (g_allow_local && !p.curr_is_token_or_id(get_binders_tk())) {
ts.push_back(parse_transition(p, pt, tk, locals, new_tokens, grp));
break;
}
p.check_token_or_id_next(get_binders_tk(), p.check_token_or_id_next(get_binders_tk(),
"invalid notation declaration, 'binders' expected " "invalid notation declaration, 'binders' expected "
"(declaration prefix matches reserved notation)"); "(declaration prefix matches reserved notation)");
@ -497,45 +540,38 @@ static notation_entry parse_notation_core(parser & p, bool overload, notation_en
break; break;
case notation::action_kind::Expr: case notation::action_kind::Exprs: case notation::action_kind::ScopedExpr: case notation::action_kind::Expr: case notation::action_kind::Exprs: case notation::action_kind::ScopedExpr:
case notation::action_kind::Ext: case notation::action_kind::LuaExt: { case notation::action_kind::Ext: case notation::action_kind::LuaExt: {
if (g_allow_local && !p.curr_is_identifier()) {
ts.push_back(parse_transition(p, pt, tk, locals, new_tokens, grp));
break;
}
name n = p.check_id_next("invalid notation declaration, identifier expected " name n = p.check_id_next("invalid notation declaration, identifier expected "
"(declaration prefix matches reserved notation)"); "(declaration prefix matches reserved notation)");
expr local_type = mk_Prop(); // type used in notation local declarations, it is irrelevant if (p.curr_is_token(get_colon_tk())) {
expr l = mk_local(n, local_type); if (g_allow_local) {
p.add_local(l);
locals.push_back(l);
ts.push_back(transition(tk, a));
break;
}}
if (p.curr_is_token(get_colon_tk()))
throw parser_error("invalid notation declaration, invalid ':' occurrence "
"(declaration prefix matches reserved notation)", p.pos());
} else {
reserved_pt = optional<parse_table>();
if (p.curr_is_token_or_id(get_binder_tk())) {
p.next();
unsigned rbp = parse_binders_rbp(p);
ts.push_back(transition(tk, mk_binder_action(rbp)));
} else if (p.curr_is_token_or_id(get_binders_tk())) {
p.next();
unsigned rbp = parse_binders_rbp(p);
ts.push_back(transition(tk, mk_binders_action(rbp)));
} else if (p.curr_is_identifier()) {
unsigned default_prec = get_default_prec(pt, tk); unsigned default_prec = get_default_prec(pt, tk);
name n = p.get_name_val();
p.next();
action a = parse_action(p, tk, default_prec, locals, new_tokens, grp); action a = parse_action(p, tk, default_prec, locals, new_tokens, grp);
expr local_type = mk_Prop(); // type used in notation local declarations, it is irrelevant expr local_type = mk_Prop(); // type used in notation local declarations, it is irrelevant
expr l = mk_local(n, local_type); expr l = mk_local(n, local_type);
p.add_local(l); p.add_local(l);
locals.push_back(l); locals.push_back(l);
ts.push_back(transition(tk, a)); ts.push_back(transition(tk, a));
} else if (p.curr_is_quoted_symbol() || p.curr_is_keyword() || break;
p.curr_is_token(get_assign_tk()) || p.curr_is_command() || p.curr_is_eof()) {
ts.push_back(transition(tk, mk_skip_action()));
} else { } else {
throw parser_error("invalid notation declaration, quoted-symbol, identifier, " throw parser_error("invalid notation declaration, invalid ':' occurrence "
"'binder', 'binders' expected", p.pos()); "(declaration prefix matches reserved notation)", p.pos());
} }
} else {
expr local_type = mk_Prop(); // type used in notation local declarations, it is irrelevant
expr l = mk_local(n, local_type);
p.add_local(l);
locals.push_back(l);
ts.push_back(transition(tk, a));
break;
}
}}
} else {
reserved_pt = optional<parse_table>();
ts.push_back(parse_transition(p, pt, tk, locals, new_tokens, grp));
} }
pt = find_match(pt, ts.back()); pt = find_match(pt, ts.back());
} }

30
tests/lean/712.lean Normal file
View file

@ -0,0 +1,30 @@
reserve infix `~~~`:50
reserve notation `[` a `][` b:10 `]`
section
local infix `~~~` := eq
print notation ~~~
local infix `~~~`:50 := eq
print notation ~~~
local infix `~~~`:100 := eq
infix `~~~`:100 := eq -- FAIL
print notation ~~~
local notation `[` a `][`:10 b:20 `]` := a = b
print notation ][
end
notation `[` a `][`:10 b:20 `]` := a = b -- FAIL
notation `[` a `][` b `]` := a = b
infix `~~~` := eq
print notation ~~~
print notation ][

View file

@ -0,0 +1,8 @@
_ `~~~`:50 _:50 := eq #1 #0
_ `~~~`:50 _:50 := eq #1 #0
712.lean:15:11: error: invalid notation declaration, invalid ':' occurrence (declaration matches reserved notation)
_ `~~~`:100 _:100 := eq #1 #0
`[`:1024 _:1 `][`:10 _:20 `]`:0 := eq #1 #0
712.lean:24:24: error: invalid notation declaration, invalid ':' occurrence (declaration prefix matches reserved notation)
_ `~~~`:50 _:50 := eq #1 #0
`[`:1024 _:1 `][`:1 _:10 `]`:0 := eq #1 #0