Parse try as a keyword only in edition 2018 and up

This commit is contained in:
Lukas Wirth 2024-07-19 15:26:50 +02:00
parent 713c47f25b
commit d235d09bf9
8 changed files with 24 additions and 42 deletions

View file

@ -101,6 +101,8 @@ pub(super) fn atom_expr(
}
T![loop] => loop_expr(p, None),
T![while] => while_expr(p, None),
// test try_macro_fallback 2015
// fn foo() { try!(Ok(())); }
T![try] => try_block_expr(p, None),
T![match] => match_expr(p),
T![return] => return_expr(p),
@ -767,24 +769,6 @@ fn break_expr(p: &mut Parser<'_>, r: Restrictions) -> CompletedMarker {
fn try_block_expr(p: &mut Parser<'_>, m: Option<Marker>) -> CompletedMarker {
assert!(p.at(T![try]));
let m = m.unwrap_or_else(|| p.start());
// Special-case `try!` as macro.
// This is a hack until we do proper edition support
if p.nth_at(1, T![!]) {
// test try_macro_fallback
// fn foo() { try!(Ok(())); }
let macro_call = p.start();
let path = p.start();
let path_segment = p.start();
let name_ref = p.start();
p.bump_remap(IDENT);
name_ref.complete(p, NAME_REF);
path_segment.complete(p, PATH_SEGMENT);
path.complete(p, PATH);
let _block_like = items::macro_call_after_excl(p);
macro_call.complete(p, MACRO_CALL);
return m.complete(p, MACRO_EXPR);
}
p.bump(T![try]);
if p.at(T!['{']) {
stmt_list(p);

View file

@ -230,13 +230,8 @@ fn opt_item_without_modifiers(p: &mut Parser<'_>, m: Marker) -> Result<(), Marke
IDENT if p.at_contextual_kw(T![union]) && p.nth(1) == IDENT => adt::union(p, m),
T![macro] => macro_def(p, m),
// check if current token is "macro_rules" followed by "!" followed by an identifier or "try"
// try is keyword since the 2018 edition and the parser is not edition aware (yet!)
IDENT
if p.at_contextual_kw(T![macro_rules])
&& p.nth_at(1, BANG)
&& (p.nth_at(2, IDENT) || p.nth_at(2, T![try])) =>
{
// check if current token is "macro_rules" followed by "!" followed by an identifier
IDENT if p.at_contextual_kw(T![macro_rules]) && p.nth_at(1, BANG) && p.nth_at(2, IDENT) => {
macro_rules(p, m)
}
@ -334,23 +329,14 @@ pub(crate) fn extern_item_list(p: &mut Parser<'_>) {
m.complete(p, EXTERN_ITEM_LIST);
}
// test try_macro_rules 2015
// macro_rules! try { () => {} }
fn macro_rules(p: &mut Parser<'_>, m: Marker) {
assert!(p.at_contextual_kw(T![macro_rules]));
p.bump_remap(T![macro_rules]);
p.expect(T![!]);
// Special-case `macro_rules! try`.
// This is a hack until we do proper edition support
// test try_macro_rules
// macro_rules! try { () => {} }
if p.at(T![try]) {
let m = p.start();
p.bump_remap(IDENT);
m.complete(p, NAME);
} else {
name(p);
}
name(p);
match p.current() {
// test macro_rules_non_brace

View file

@ -559,11 +559,17 @@ mod ok {
fn try_expr() { run_and_expect_no_errors("test_data/parser/inline/ok/try_expr.rs"); }
#[test]
fn try_macro_fallback() {
run_and_expect_no_errors("test_data/parser/inline/ok/try_macro_fallback.rs");
run_and_expect_no_errors_with_edition(
"test_data/parser/inline/ok/try_macro_fallback.rs",
crate::Edition::Edition2015,
);
}
#[test]
fn try_macro_rules() {
run_and_expect_no_errors("test_data/parser/inline/ok/try_macro_rules.rs");
run_and_expect_no_errors_with_edition(
"test_data/parser/inline/ok/try_macro_rules.rs",
crate::Edition::Edition2015,
);
}
#[test]
fn tuple_attrs() { run_and_expect_no_errors("test_data/parser/inline/ok/tuple_attrs.rs"); }

View file

@ -1,5 +1,7 @@
SOURCE_FILE
FN
COMMENT "// 2015"
WHITESPACE "\n"
FN_KW "fn"
WHITESPACE " "
NAME

View file

@ -1 +1,2 @@
// 2015
fn foo() { try!(Ok(())); }

View file

@ -1,5 +1,7 @@
SOURCE_FILE
MACRO_RULES
COMMENT "// 2015"
WHITESPACE "\n"
MACRO_RULES_KW "macro_rules"
BANG "!"
WHITESPACE " "

View file

@ -1 +1,2 @@
// 2015
macro_rules! try { () => {} }