Skip to content

Commit a13fbaa

Browse files
committed
Refactor maybe_recover_trailing_expr
1 parent e95c640 commit a13fbaa

File tree

1 file changed

+24
-21
lines changed
  • compiler/rustc_parse/src/parser

1 file changed

+24
-21
lines changed

compiler/rustc_parse/src/parser/pat.rs

+24-21
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@ use rustc_session::errors::ExprParenthesesNeeded;
2424
use rustc_span::source_map::{respan, Spanned};
2525
use rustc_span::symbol::{kw, sym, Ident};
2626
use rustc_span::{ErrorGuaranteed, Span};
27-
use std::borrow::Cow;
2827
use thin_vec::{thin_vec, ThinVec};
2928

3029
#[derive(PartialEq, Copy, Clone)]
@@ -337,11 +336,19 @@ impl<'a> Parser<'a> {
337336
}
338337
}
339338

340-
/// Ensures that the last parsed pattern is not followed by a method call or an operator.
339+
/// Ensures that the last parsed pattern (or pattern range bound) is not followed by a method call or an operator.
340+
///
341+
/// `is_end_bound` indicates whether the last parsed thing was the end bound of a range pattern (see [`parse_pat_range_end`](Self::parse_pat_range_end))
342+
/// in order to say "expected a pattern range bound" instead of "expected a pattern";
343+
/// ```text
344+
/// 0..=1 + 2
345+
/// ^^^^^
346+
/// ```
347+
/// Only the end bound is spanned, and this function have no idea if there were a `..=` before `pat_span`, hence the parameter.
341348
#[must_use = "the pattern must be discarded as `PatKind::Err` if this function returns Some"]
342-
fn maybe_recover_trailing_expr<'b>(
343-
&'b mut self,
344-
pat: Cow<'b, P<Expr>>,
349+
fn maybe_recover_trailing_expr(
350+
&mut self,
351+
pat_span: Span,
345352
is_end_bound: bool,
346353
) -> Option<ErrorGuaranteed> {
347354
if self.prev_token.is_keyword(kw::Underscore) || !self.may_recover() {
@@ -353,7 +360,7 @@ impl<'a> Parser<'a> {
353360
let has_trailing_method = self.check_noexpect(&token::Dot)
354361
&& self.look_ahead(1, |tok| {
355362
tok.ident()
356-
.and_then(|(ident, _)| ident.name.to_string().chars().next())
363+
.and_then(|(ident, _)| ident.name.as_str().chars().next())
357364
.is_some_and(char::is_lowercase)
358365
})
359366
&& self.look_ahead(2, |tok| tok.kind == token::OpenDelim(Delimiter::Parenthesis));
@@ -374,15 +381,16 @@ impl<'a> Parser<'a> {
374381
}
375382

376383
// Let's try to parse an expression to emit a better diagnostic.
377-
let pat = pat.into_owned();
378-
let pat_span = pat.span;
379-
380384
let mut snapshot = self.create_snapshot_for_diagnostic();
381385
snapshot.restrictions.insert(Restrictions::IS_PAT);
382386

383387
// Parse `?`, `.f`, `(arg0, arg1, ...)` or `[expr]` until they've all been eaten.
384388
if let Ok(expr) = snapshot
385-
.parse_expr_dot_or_call_with(pat, pat_span, AttrVec::new())
389+
.parse_expr_dot_or_call_with(
390+
self.mk_expr_err(pat_span), // equivalent to transforming the parsed pattern into an `Expr`
391+
pat_span,
392+
AttrVec::new(),
393+
)
386394
.map_err(|err| err.cancel())
387395
{
388396
let non_assoc_span = expr.span;
@@ -396,19 +404,17 @@ impl<'a> Parser<'a> {
396404
self.restore_snapshot(snapshot);
397405
self.restrictions.remove(Restrictions::IS_PAT);
398406

399-
let span = expr.span;
400-
401407
let is_bound = is_end_bound
402408
// is_start_bound: either `..` or `)..`
403409
|| self.token.is_range_separator()
404410
|| self.token.kind == token::CloseDelim(Delimiter::Parenthesis)
405411
&& self.look_ahead(1, Token::is_range_separator);
406412

407413
// Check that `parse_expr_assoc_with` didn't eat a rhs.
408-
let is_method_call = has_trailing_method && non_assoc_span == span;
414+
let is_method_call = has_trailing_method && non_assoc_span == expr.span;
409415

410416
return Some(self.dcx().emit_err(UnexpectedExpressionInPattern {
411-
span,
417+
span: expr.span,
412418
is_bound,
413419
is_method_call,
414420
}));
@@ -524,8 +530,7 @@ impl<'a> Parser<'a> {
524530
} else if self.check(&token::OpenDelim(Delimiter::Parenthesis)) {
525531
self.parse_pat_tuple_struct(qself, path)?
526532
} else {
527-
let pat = self.mk_expr(span, ExprKind::Path(qself.clone(), path.clone()));
528-
match self.maybe_recover_trailing_expr(Cow::Owned(pat), false) {
533+
match self.maybe_recover_trailing_expr(span, false) {
529534
Some(guar) => PatKind::Err(guar),
530535
None => PatKind::Path(qself, path),
531536
}
@@ -558,8 +563,7 @@ impl<'a> Parser<'a> {
558563
// Try to parse everything else as literal with optional minus
559564
match self.parse_literal_maybe_minus() {
560565
Ok(begin) => {
561-
let begin = match self.maybe_recover_trailing_expr(Cow::Borrowed(&begin), false)
562-
{
566+
let begin = match self.maybe_recover_trailing_expr(begin.span, false) {
563567
Some(_) => self.mk_expr_err(begin.span),
564568
None => begin,
565569
};
@@ -963,7 +967,7 @@ impl<'a> Parser<'a> {
963967
self.parse_literal_maybe_minus()
964968
}?;
965969

966-
let recovered = self.maybe_recover_trailing_expr(Cow::Borrowed(&bound), true);
970+
let recovered = self.maybe_recover_trailing_expr(bound.span, true);
967971

968972
// recover trailing `)`
969973
if let Some(open_paren) = open_paren {
@@ -1047,9 +1051,8 @@ impl<'a> Parser<'a> {
10471051
// Check for method calls after the `ident`,
10481052
// but not `ident @ subpat` as `subpat` was already checked and `ident` continues with `@`.
10491053

1050-
let expr = self.mk_expr(ident.span, ExprKind::Path(None, Path::from_ident(ident)));
10511054
let pat = if sub.is_none()
1052-
&& let Some(guar) = self.maybe_recover_trailing_expr(Cow::Owned(expr), false)
1055+
&& let Some(guar) = self.maybe_recover_trailing_expr(ident.span, false)
10531056
{
10541057
PatKind::Err(guar)
10551058
} else {

0 commit comments

Comments
 (0)