Skip to content

Commit a0648ea

Browse files
committed
Auto merge of #88835 - FabianWolff:issue-88770, r=petrochenkov
Fix error recovery in format macro parsing Fixes #88770. Basically, the assumption in the following comment is incorrect: https://github.com/rust-lang/rust/blob/b69fe57261086e70aea9d5b58819a1794bf7c121/compiler/rustc_builtin_macros/src/format.rs#L167-L172 This is only true in the first iteration of the loop, when [`p.clear_expected_tokens()`](https://github.com/rust-lang/rust/blob/b69fe57261086e70aea9d5b58819a1794bf7c121/compiler/rustc_builtin_macros/src/format.rs#L164) is called. In subsequent iterations, `p.expected_tokens` won't be empty, so `p.expect()` won't actually call `unexpected_try_recover()`: https://github.com/rust-lang/rust/blob/b69fe57261086e70aea9d5b58819a1794bf7c121/compiler/rustc_parse/src/parser/mod.rs#L487-L498 Instead, it will call `expect_one_of()`, which _can_ recover and return `Ok()`. This PR handles this case to fix the ICE in #88770.
2 parents 91d8da1 + a8421ca commit a0648ea

File tree

4 files changed

+94
-18
lines changed

4 files changed

+94
-18
lines changed

compiler/rustc_builtin_macros/src/format.rs

+15-16
Original file line numberDiff line numberDiff line change
@@ -164,23 +164,22 @@ fn parse_args<'a>(
164164
p.clear_expected_tokens();
165165
}
166166

167-
// `Parser::expect` tries to recover using the
168-
// `Parser::unexpected_try_recover` function. This function is able
169-
// to recover if the expected token is a closing delimiter.
170-
//
171-
// As `,` is not a closing delimiter, it will always return an `Err`
172-
// variant.
173-
let mut err = p.expect(&token::Comma).unwrap_err();
174-
175-
match token::TokenKind::Comma.similar_tokens() {
176-
Some(tks) if tks.contains(&p.token.kind) => {
177-
// If a similar token is found, then it may be a typo. We
178-
// consider it as a comma, and continue parsing.
179-
err.emit();
180-
p.bump();
167+
match p.expect(&token::Comma) {
168+
Err(mut err) => {
169+
match token::TokenKind::Comma.similar_tokens() {
170+
Some(tks) if tks.contains(&p.token.kind) => {
171+
// If a similar token is found, then it may be a typo. We
172+
// consider it as a comma, and continue parsing.
173+
err.emit();
174+
p.bump();
175+
}
176+
// Otherwise stop the parsing and return the error.
177+
_ => return Err(err),
178+
}
179+
}
180+
Ok(recovered) => {
181+
assert!(recovered);
181182
}
182-
// Otherwise stop the parsing and return the error.
183-
_ => return Err(err),
184183
}
185184
}
186185
first = false;

compiler/rustc_parse/src/parser/diagnostics.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -277,7 +277,7 @@ impl<'a> Parser<'a> {
277277
self.struct_span_err(sp, &msg)
278278
.span_suggestion_short(sp, "change this to `;`", ";".to_string(), appl)
279279
.emit();
280-
return Ok(false);
280+
return Ok(true);
281281
} else if self.look_ahead(0, |t| {
282282
t == &token::CloseDelim(token::Brace)
283283
|| (
@@ -295,7 +295,7 @@ impl<'a> Parser<'a> {
295295
.span_label(self.token.span, "unexpected token")
296296
.span_suggestion_short(sp, "add `;` here", ";".to_string(), appl)
297297
.emit();
298-
return Ok(false);
298+
return Ok(true);
299299
}
300300
}
301301

src/test/ui/parser/issue-88770.rs

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
// Regression test for the ICE described in #88770.
2+
3+
// error-pattern:this file contains an unclosed delimiter
4+
// error-pattern:expected one of
5+
// error-pattern:missing `in` in `for` loop
6+
// error-pattern:expected `;`, found `e`
7+
8+
fn m(){print!("",(c for&g
9+
u
10+
e
11+
e

src/test/ui/parser/issue-88770.stderr

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
error: this file contains an unclosed delimiter
2+
--> $DIR/issue-88770.rs:11:3
3+
|
4+
LL | fn m(){print!("",(c for&g
5+
| - - - unclosed delimiter
6+
| | |
7+
| | unclosed delimiter
8+
| unclosed delimiter
9+
...
10+
LL | e
11+
| ^
12+
13+
error: this file contains an unclosed delimiter
14+
--> $DIR/issue-88770.rs:11:3
15+
|
16+
LL | fn m(){print!("",(c for&g
17+
| - - - unclosed delimiter
18+
| | |
19+
| | unclosed delimiter
20+
| unclosed delimiter
21+
...
22+
LL | e
23+
| ^
24+
25+
error: this file contains an unclosed delimiter
26+
--> $DIR/issue-88770.rs:11:3
27+
|
28+
LL | fn m(){print!("",(c for&g
29+
| - - - unclosed delimiter
30+
| | |
31+
| | unclosed delimiter
32+
| unclosed delimiter
33+
...
34+
LL | e
35+
| ^
36+
37+
error: missing `in` in `for` loop
38+
--> $DIR/issue-88770.rs:8:26
39+
|
40+
LL | fn m(){print!("",(c for&g
41+
| __________________________^
42+
LL | | u
43+
| |_ help: try adding `in` here
44+
45+
error: expected one of `!`, `)`, `,`, `.`, `::`, `?`, `{`, or an operator, found keyword `for`
46+
--> $DIR/issue-88770.rs:8:21
47+
|
48+
LL | fn m(){print!("",(c for&g
49+
| ^^^ expected one of 8 possible tokens
50+
51+
error: expected `;`, found `e`
52+
--> $DIR/issue-88770.rs:10:2
53+
|
54+
LL | e
55+
| ^ help: add `;` here
56+
LL | e
57+
| - unexpected token
58+
59+
error: expected one of `!`, `,`, `.`, `::`, `?`, `{`, or an operator, found `)`
60+
--> $DIR/issue-88770.rs:11:3
61+
|
62+
LL | e
63+
| ^ expected one of 7 possible tokens
64+
65+
error: aborting due to 7 previous errors
66+

0 commit comments

Comments
 (0)