rustc_parse/parser/
pat.rs

1use std::ops::Bound;
2
3use rustc_ast::mut_visit::{self, MutVisitor};
4use rustc_ast::token::NtPatKind::*;
5use rustc_ast::token::{self, IdentIsRaw, MetaVarKind, Token};
6use rustc_ast::util::parser::ExprPrecedence;
7use rustc_ast::visit::{self, Visitor};
8use rustc_ast::{
9    self as ast, Arm, AttrVec, BindingMode, ByRef, Expr, ExprKind, LocalKind, MacCall, Mutability,
10    Pat, PatField, PatFieldsRest, PatKind, Path, QSelf, RangeEnd, RangeSyntax, Stmt, StmtKind,
11};
12use rustc_ast_pretty::pprust;
13use rustc_errors::{Applicability, Diag, DiagArgValue, PResult, StashKey};
14use rustc_session::errors::ExprParenthesesNeeded;
15use rustc_span::source_map::{Spanned, respan};
16use rustc_span::{BytePos, ErrorGuaranteed, Ident, Span, kw, sym};
17use thin_vec::{ThinVec, thin_vec};
18
19use super::{ForceCollect, Parser, PathStyle, Restrictions, Trailing, UsePreAttrPos};
20use crate::errors::{
21    self, AmbiguousRangePattern, AtDotDotInStructPattern, AtInStructPattern,
22    DotDotDotForRemainingFields, DotDotDotRangeToPatternNotAllowed, DotDotDotRestPattern,
23    EnumPatternInsteadOfIdentifier, ExpectedBindingLeftOfAt, ExpectedCommaAfterPatternField,
24    GenericArgsInPatRequireTurbofishSyntax, InclusiveRangeExtraEquals, InclusiveRangeMatchArrow,
25    InclusiveRangeNoEnd, InvalidMutInPattern, ParenRangeSuggestion, PatternOnWrongSideOfAt,
26    RemoveLet, RepeatedMutInPattern, SwitchRefBoxOrder, TopLevelOrPatternNotAllowed,
27    TopLevelOrPatternNotAllowedSugg, TrailingVertNotAllowed, UnexpectedExpressionInPattern,
28    UnexpectedExpressionInPatternSugg, UnexpectedLifetimeInPattern, UnexpectedParenInRangePat,
29    UnexpectedParenInRangePatSugg, UnexpectedVertVertBeforeFunctionParam,
30    UnexpectedVertVertInPattern, WrapInParens,
31};
32use crate::parser::expr::{DestructuredFloat, could_be_unclosed_char_literal};
33use crate::{exp, maybe_recover_from_interpolated_ty_qpath};
34
35#[derive(PartialEq, Copy, Clone)]
36pub enum Expected {
37    ParameterName,
38    ArgumentName,
39    Identifier,
40    BindingPattern,
41}
42
43impl Expected {
44    // FIXME(#100717): migrate users of this to proper localization
45    fn to_string_or_fallback(expected: Option<Expected>) -> &'static str {
46        match expected {
47            Some(Expected::ParameterName) => "parameter name",
48            Some(Expected::ArgumentName) => "argument name",
49            Some(Expected::Identifier) => "identifier",
50            Some(Expected::BindingPattern) => "binding pattern",
51            None => "pattern",
52        }
53    }
54}
55
56const WHILE_PARSING_OR_MSG: &str = "while parsing this or-pattern starting here";
57
58/// Whether or not to recover a `,` when parsing or-patterns.
59#[derive(PartialEq, Copy, Clone)]
60pub enum RecoverComma {
61    Yes,
62    No,
63}
64
65/// Whether or not to recover a `:` when parsing patterns that were meant to be paths.
66#[derive(PartialEq, Copy, Clone)]
67pub enum RecoverColon {
68    Yes,
69    No,
70}
71
72/// Whether or not to recover a `a, b` when parsing patterns as `(a, b)` or that *and* `a | b`.
73#[derive(PartialEq, Copy, Clone)]
74pub enum CommaRecoveryMode {
75    LikelyTuple,
76    EitherTupleOrPipe,
77}
78
79/// The result of `eat_or_separator`. We want to distinguish which case we are in to avoid
80/// emitting duplicate diagnostics.
81#[derive(Debug, Clone, Copy)]
82enum EatOrResult {
83    /// We recovered from a trailing vert.
84    TrailingVert,
85    /// We ate an `|` (or `||` and recovered).
86    AteOr,
87    /// We did not eat anything (i.e. the current token is not `|` or `||`).
88    None,
89}
90
91/// The syntax location of a given pattern. Used for diagnostics.
92#[derive(Clone, Copy)]
93pub enum PatternLocation {
94    LetBinding,
95    FunctionParameter,
96}
97
98impl<'a> Parser<'a> {
99    /// Parses a pattern.
100    ///
101    /// Corresponds to `Pattern` in RFC 3637 and admits guard patterns at the top level.
102    /// Used when parsing patterns in all cases where neither `PatternNoTopGuard` nor
103    /// `PatternNoTopAlt` (see below) are used.
104    pub fn parse_pat_allow_top_guard(
105        &mut self,
106        expected: Option<Expected>,
107        rc: RecoverComma,
108        ra: RecoverColon,
109        rt: CommaRecoveryMode,
110    ) -> PResult<'a, Box<Pat>> {
111        let pat = self.parse_pat_no_top_guard(expected, rc, ra, rt)?;
112
113        if self.eat_keyword(exp!(If)) {
114            let cond = self.parse_expr()?;
115            // Feature-gate guard patterns
116            self.psess.gated_spans.gate(sym::guard_patterns, cond.span);
117            let span = pat.span.to(cond.span);
118            Ok(self.mk_pat(span, PatKind::Guard(pat, cond)))
119        } else {
120            Ok(pat)
121        }
122    }
123
124    /// Parses a pattern.
125    ///
126    /// Corresponds to `PatternNoTopAlt` in RFC 3637 and does not admit or-patterns
127    /// or guard patterns at the top level. Used when parsing the parameters of lambda
128    /// expressions, functions, function pointers, and `pat_param` macro fragments.
129    pub fn parse_pat_no_top_alt(
130        &mut self,
131        expected: Option<Expected>,
132        syntax_loc: Option<PatternLocation>,
133    ) -> PResult<'a, Box<Pat>> {
134        self.parse_pat_with_range_pat(true, expected, syntax_loc)
135    }
136
137    /// Parses a pattern.
138    ///
139    /// Corresponds to `PatternNoTopGuard` in RFC 3637 and allows or-patterns, but not
140    /// guard patterns, at the top level. Used for parsing patterns in `pat` fragments (until
141    /// the next edition) and `let`, `if let`, and `while let` expressions.
142    ///
143    /// Note that after the FCP in <https://github.com/rust-lang/rust/issues/81415>,
144    /// a leading vert is allowed in nested or-patterns, too. This allows us to
145    /// simplify the grammar somewhat.
146    pub fn parse_pat_no_top_guard(
147        &mut self,
148        expected: Option<Expected>,
149        rc: RecoverComma,
150        ra: RecoverColon,
151        rt: CommaRecoveryMode,
152    ) -> PResult<'a, Box<Pat>> {
153        self.parse_pat_no_top_guard_inner(expected, rc, ra, rt, None).map(|(pat, _)| pat)
154    }
155
156    /// Returns the pattern and a bool indicating whether we recovered from a trailing vert (true =
157    /// recovered).
158    fn parse_pat_no_top_guard_inner(
159        &mut self,
160        expected: Option<Expected>,
161        rc: RecoverComma,
162        ra: RecoverColon,
163        rt: CommaRecoveryMode,
164        syntax_loc: Option<PatternLocation>,
165    ) -> PResult<'a, (Box<Pat>, bool)> {
166        // Keep track of whether we recovered from a trailing vert so that we can avoid duplicated
167        // suggestions (which bothers rustfix).
168        //
169        // Allow a '|' before the pats (RFCs 1925, 2530, and 2535).
170        let (leading_vert_span, mut trailing_vert) = match self.eat_or_separator(None) {
171            EatOrResult::AteOr => (Some(self.prev_token.span), false),
172            EatOrResult::TrailingVert => (None, true),
173            EatOrResult::None => (None, false),
174        };
175
176        // Parse the first pattern (`p_0`).
177        let mut first_pat = match self.parse_pat_no_top_alt(expected, syntax_loc) {
178            Ok(pat) => pat,
179            Err(err)
180                if self.token.is_reserved_ident()
181                    && !self.token.is_keyword(kw::In)
182                    && !self.token.is_keyword(kw::If) =>
183            {
184                err.emit();
185                self.bump();
186                self.mk_pat(self.token.span, PatKind::Wild)
187            }
188            Err(err) => return Err(err),
189        };
190        if rc == RecoverComma::Yes && !first_pat.could_be_never_pattern() {
191            self.maybe_recover_unexpected_comma(first_pat.span, rt)?;
192        }
193
194        // If the next token is not a `|`,
195        // this is not an or-pattern and we should exit here.
196        if !self.check(exp!(Or)) && self.token != token::OrOr {
197            // If we parsed a leading `|` which should be gated,
198            // then we should really gate the leading `|`.
199            // This complicated procedure is done purely for diagnostics UX.
200
201            // Check if the user wrote `foo:bar` instead of `foo::bar`.
202            if ra == RecoverColon::Yes {
203                first_pat = self.maybe_recover_colon_colon_in_pat_typo(first_pat, expected);
204            }
205
206            if let Some(leading_vert_span) = leading_vert_span {
207                // If there was a leading vert, treat this as an or-pattern. This improves
208                // diagnostics.
209                let span = leading_vert_span.to(self.prev_token.span);
210                return Ok((self.mk_pat(span, PatKind::Or(thin_vec![first_pat])), trailing_vert));
211            }
212
213            return Ok((first_pat, trailing_vert));
214        }
215
216        // Parse the patterns `p_1 | ... | p_n` where `n > 0`.
217        let lo = leading_vert_span.unwrap_or(first_pat.span);
218        let mut pats = thin_vec![first_pat];
219        loop {
220            match self.eat_or_separator(Some(lo)) {
221                EatOrResult::AteOr => {}
222                EatOrResult::None => break,
223                EatOrResult::TrailingVert => {
224                    trailing_vert = true;
225                    break;
226                }
227            }
228            let pat = self.parse_pat_no_top_alt(expected, syntax_loc).map_err(|mut err| {
229                err.span_label(lo, WHILE_PARSING_OR_MSG);
230                err
231            })?;
232            if rc == RecoverComma::Yes && !pat.could_be_never_pattern() {
233                self.maybe_recover_unexpected_comma(pat.span, rt)?;
234            }
235            pats.push(pat);
236        }
237        let or_pattern_span = lo.to(self.prev_token.span);
238
239        Ok((self.mk_pat(or_pattern_span, PatKind::Or(pats)), trailing_vert))
240    }
241
242    /// Parse a pattern and (maybe) a `Colon` in positions where a pattern may be followed by a
243    /// type annotation (e.g. for `let` bindings or `fn` params).
244    ///
245    /// Generally, this corresponds to `pat_no_top_alt` followed by an optional `Colon`. It will
246    /// eat the `Colon` token if one is present.
247    ///
248    /// The return value represents the parsed pattern and `true` if a `Colon` was parsed (`false`
249    /// otherwise).
250    pub(super) fn parse_pat_before_ty(
251        &mut self,
252        expected: Option<Expected>,
253        rc: RecoverComma,
254        syntax_loc: PatternLocation,
255    ) -> PResult<'a, (Box<Pat>, bool)> {
256        // We use `parse_pat_allow_top_alt` regardless of whether we actually want top-level
257        // or-patterns so that we can detect when a user tries to use it. This allows us to print a
258        // better error message.
259        let (pat, trailing_vert) = self.parse_pat_no_top_guard_inner(
260            expected,
261            rc,
262            RecoverColon::No,
263            CommaRecoveryMode::LikelyTuple,
264            Some(syntax_loc),
265        )?;
266        let colon = self.eat(exp!(Colon));
267
268        if let PatKind::Or(pats) = &pat.kind {
269            let span = pat.span;
270            let sub = if pats.len() == 1 {
271                Some(TopLevelOrPatternNotAllowedSugg::RemoveLeadingVert {
272                    span: span.with_hi(span.lo() + BytePos(1)),
273                })
274            } else {
275                Some(TopLevelOrPatternNotAllowedSugg::WrapInParens {
276                    span,
277                    suggestion: WrapInParens { lo: span.shrink_to_lo(), hi: span.shrink_to_hi() },
278                })
279            };
280
281            let err = self.dcx().create_err(match syntax_loc {
282                PatternLocation::LetBinding => {
283                    TopLevelOrPatternNotAllowed::LetBinding { span, sub }
284                }
285                PatternLocation::FunctionParameter => {
286                    TopLevelOrPatternNotAllowed::FunctionParameter { span, sub }
287                }
288            });
289            if trailing_vert {
290                err.delay_as_bug();
291            } else {
292                err.emit();
293            }
294        }
295
296        Ok((pat, colon))
297    }
298
299    /// Parse the pattern for a function or function pointer parameter, followed by a colon.
300    ///
301    /// The return value represents the parsed pattern and `true` if a `Colon` was parsed (`false`
302    /// otherwise).
303    pub(super) fn parse_fn_param_pat_colon(&mut self) -> PResult<'a, (Box<Pat>, bool)> {
304        // In order to get good UX, we first recover in the case of a leading vert for an illegal
305        // top-level or-pat. Normally, this means recovering both `|` and `||`, but in this case,
306        // a leading `||` probably doesn't indicate an or-pattern attempt, so we handle that
307        // separately.
308        if let token::OrOr = self.token.kind {
309            self.dcx().emit_err(UnexpectedVertVertBeforeFunctionParam { span: self.token.span });
310            self.bump();
311        }
312
313        self.parse_pat_before_ty(
314            Some(Expected::ParameterName),
315            RecoverComma::No,
316            PatternLocation::FunctionParameter,
317        )
318    }
319
320    /// Eat the or-pattern `|` separator.
321    /// If instead a `||` token is encountered, recover and pretend we parsed `|`.
322    fn eat_or_separator(&mut self, lo: Option<Span>) -> EatOrResult {
323        if self.recover_trailing_vert(lo) {
324            EatOrResult::TrailingVert
325        } else if self.token.kind == token::OrOr {
326            // Found `||`; Recover and pretend we parsed `|`.
327            self.dcx().emit_err(UnexpectedVertVertInPattern { span: self.token.span, start: lo });
328            self.bump();
329            EatOrResult::AteOr
330        } else if self.eat(exp!(Or)) {
331            EatOrResult::AteOr
332        } else {
333            EatOrResult::None
334        }
335    }
336
337    /// Recover if `|` or `||` is the current token and we have one of the
338    /// tokens `=>`, `if`, `=`, `:`, `;`, `,`, `]`, `)`, or `}` ahead of us.
339    ///
340    /// These tokens all indicate that we reached the end of the or-pattern
341    /// list and can now reliably say that the `|` was an illegal trailing vert.
342    /// Note that there are more tokens such as `@` for which we know that the `|`
343    /// is an illegal parse. However, the user's intent is less clear in that case.
344    fn recover_trailing_vert(&mut self, lo: Option<Span>) -> bool {
345        let is_end_ahead = self.look_ahead(1, |token| {
346            matches!(
347                &token.uninterpolate().kind,
348                token::FatArrow // e.g. `a | => 0,`.
349                | token::Ident(kw::If, token::IdentIsRaw::No) // e.g. `a | if expr`.
350                | token::Eq // e.g. `let a | = 0`.
351                | token::Semi // e.g. `let a |;`.
352                | token::Colon // e.g. `let a | :`.
353                | token::Comma // e.g. `let (a |,)`.
354                | token::CloseBracket // e.g. `let [a | ]`.
355                | token::CloseParen // e.g. `let (a | )`.
356                | token::CloseBrace // e.g. `let A { f: a | }`.
357            )
358        });
359        match (is_end_ahead, &self.token.kind) {
360            (true, token::Or | token::OrOr) => {
361                // A `|` or possibly `||` token shouldn't be here. Ban it.
362                self.dcx().emit_err(TrailingVertNotAllowed {
363                    span: self.token.span,
364                    start: lo,
365                    token: self.token,
366                    note_double_vert: self.token.kind == token::OrOr,
367                });
368                self.bump();
369                true
370            }
371            _ => false,
372        }
373    }
374
375    /// Ensures that the last parsed pattern (or pattern range bound) is not followed by an expression.
376    ///
377    /// `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))
378    /// in order to say "expected a pattern range bound" instead of "expected a pattern";
379    /// ```text
380    /// 0..=1 + 2
381    ///     ^^^^^
382    /// ```
383    /// Only the end bound is spanned in this case, and this function has no idea if there was a `..=` before `pat_span`, hence the parameter.
384    ///
385    /// This function returns `Some` if a trailing expression was recovered, and said expression's span.
386    #[must_use = "the pattern must be discarded as `PatKind::Err` if this function returns Some"]
387    fn maybe_recover_trailing_expr(
388        &mut self,
389        pat_span: Span,
390        is_end_bound: bool,
391    ) -> Option<(ErrorGuaranteed, Span)> {
392        if self.prev_token.is_keyword(kw::Underscore) || !self.may_recover() {
393            // Don't recover anything after an `_` or if recovery is disabled.
394            return None;
395        }
396
397        // Returns `true` iff `token` is an unsuffixed integer.
398        let is_one_tuple_index = |_: &Self, token: &Token| -> bool {
399            use token::{Lit, LitKind};
400
401            matches!(
402                token.kind,
403                token::Literal(Lit { kind: LitKind::Integer, symbol: _, suffix: None })
404            )
405        };
406
407        // Returns `true` iff `token` is an unsuffixed `x.y` float.
408        let is_two_tuple_indexes = |this: &Self, token: &Token| -> bool {
409            use token::{Lit, LitKind};
410
411            if let token::Literal(Lit { kind: LitKind::Float, symbol, suffix: None }) = token.kind
412                && let DestructuredFloat::MiddleDot(..) = this.break_up_float(symbol, token.span)
413            {
414                true
415            } else {
416                false
417            }
418        };
419
420        // Check for `.hello` or `.0`.
421        let has_dot_expr = self.check_noexpect(&token::Dot) // `.`
422            && self.look_ahead(1, |tok| {
423                tok.is_ident() // `hello`
424                || is_one_tuple_index(&self, &tok) // `0`
425                || is_two_tuple_indexes(&self, &tok) // `0.0`
426            });
427
428        // Check for operators.
429        // `|` is excluded as it is used in pattern alternatives and lambdas,
430        // `?` is included for error propagation,
431        // `[` is included for indexing operations,
432        // `[]` is excluded as `a[]` isn't an expression and should be recovered as `a, []` (cf. `tests/ui/parser/pat-lt-bracket-7.rs`),
433        // `as` is included for type casts
434        let has_trailing_operator = matches!(
435                self.token.kind,
436                token::Plus | token::Minus | token::Star | token::Slash | token::Percent
437                | token::Caret | token::And | token::Shl | token::Shr // excludes `Or`
438            )
439            || self.token == token::Question
440            || (self.token == token::OpenBracket
441                && self.look_ahead(1, |t| *t != token::CloseBracket)) // excludes `[]`
442            || self.token.is_keyword(kw::As);
443
444        if !has_dot_expr && !has_trailing_operator {
445            // Nothing to recover here.
446            return None;
447        }
448
449        // Let's try to parse an expression to emit a better diagnostic.
450        let mut snapshot = self.create_snapshot_for_diagnostic();
451        snapshot.restrictions.insert(Restrictions::IS_PAT);
452
453        // Parse `?`, `.f`, `(arg0, arg1, ...)` or `[expr]` until they've all been eaten.
454        let Ok(expr) = snapshot
455            .parse_expr_dot_or_call_with(
456                AttrVec::new(),
457                self.mk_expr(pat_span, ExprKind::Dummy), // equivalent to transforming the parsed pattern into an `Expr`
458                pat_span,
459            )
460            .map_err(|err| err.cancel())
461        else {
462            // We got a trailing method/operator, but that wasn't an expression.
463            return None;
464        };
465
466        // Parse an associative expression such as `+ expr`, `% expr`, ...
467        // Assignments, ranges and `|` are disabled by [`Restrictions::IS_PAT`].
468        let Ok((expr, _)) = snapshot
469            .parse_expr_assoc_rest_with(Bound::Unbounded, false, expr)
470            .map_err(|err| err.cancel())
471        else {
472            // We got a trailing method/operator, but that wasn't an expression.
473            return None;
474        };
475
476        // We got a valid expression.
477        self.restore_snapshot(snapshot);
478        self.restrictions.remove(Restrictions::IS_PAT);
479
480        let is_bound = is_end_bound
481            // is_start_bound: either `..` or `)..`
482            || self.token.is_range_separator()
483            || self.token == token::CloseParen
484                && self.look_ahead(1, Token::is_range_separator);
485
486        let span = expr.span;
487
488        Some((
489            self.dcx()
490                .create_err(UnexpectedExpressionInPattern {
491                    span,
492                    is_bound,
493                    expr_precedence: expr.precedence(),
494                })
495                .stash(span, StashKey::ExprInPat)
496                .unwrap(),
497            span,
498        ))
499    }
500
501    /// Called by [`Parser::parse_stmt_without_recovery`], used to add statement-aware subdiagnostics to the errors stashed
502    /// by [`Parser::maybe_recover_trailing_expr`].
503    pub(super) fn maybe_augment_stashed_expr_in_pats_with_suggestions(&mut self, stmt: &Stmt) {
504        if self.dcx().has_errors().is_none() {
505            // No need to walk the statement if there's no stashed errors.
506            return;
507        }
508
509        struct PatVisitor<'a> {
510            /// `self`
511            parser: &'a Parser<'a>,
512            /// The freshly-parsed statement.
513            stmt: &'a Stmt,
514            /// The current match arm (for arm guard suggestions).
515            arm: Option<&'a Arm>,
516            /// The current struct field (for variable name suggestions).
517            field: Option<&'a PatField>,
518        }
519
520        impl<'a> PatVisitor<'a> {
521            /// Looks for stashed [`StashKey::ExprInPat`] errors in `stash_span`, and emit them with suggestions.
522            /// `stash_span` is contained in `expr_span`, the latter being larger in borrow patterns;
523            /// ```txt
524            /// &mut x.y
525            /// -----^^^ `stash_span`
526            /// |
527            /// `expr_span`
528            /// ```
529            /// `is_range_bound` is used to exclude arm guard suggestions in range pattern bounds.
530            fn maybe_add_suggestions_then_emit(
531                &self,
532                stash_span: Span,
533                expr_span: Span,
534                is_range_bound: bool,
535            ) {
536                self.parser.dcx().try_steal_modify_and_emit_err(
537                    stash_span,
538                    StashKey::ExprInPat,
539                    |err| {
540                        // Includes pre-pats (e.g. `&mut <err>`) in the diagnostic.
541                        err.span.replace(stash_span, expr_span);
542
543                        let sm = self.parser.psess.source_map();
544                        let stmt = self.stmt;
545                        let line_lo = sm.span_extend_to_line(stmt.span).shrink_to_lo();
546                        let indentation = sm.indentation_before(stmt.span).unwrap_or_default();
547                        let Ok(expr) = self.parser.span_to_snippet(expr_span) else {
548                            // FIXME: some suggestions don't actually need the snippet; see PR #123877's unresolved conversations.
549                            return;
550                        };
551
552                        if let StmtKind::Let(local) = &stmt.kind {
553                            match &local.kind {
554                                LocalKind::Decl | LocalKind::Init(_) => {
555                                    // It's kinda hard to guess what the user intended, so don't make suggestions.
556                                    return;
557                                }
558
559                                LocalKind::InitElse(_, _) => {}
560                            }
561                        }
562
563                        // help: use an arm guard `if val == expr`
564                        // FIXME(guard_patterns): suggest this regardless of a match arm.
565                        if let Some(arm) = &self.arm
566                            && !is_range_bound
567                        {
568                            let (ident, ident_span) = match self.field {
569                                Some(field) => {
570                                    (field.ident.to_string(), field.ident.span.to(expr_span))
571                                }
572                                None => ("val".to_owned(), expr_span),
573                            };
574
575                            // Are parentheses required around `expr`?
576                            // HACK: a neater way would be preferable.
577                            let expr = match &err.args["expr_precedence"] {
578                                DiagArgValue::Number(expr_precedence) => {
579                                    if *expr_precedence <= ExprPrecedence::Compare as i32 {
580                                        format!("({expr})")
581                                    } else {
582                                        format!("{expr}")
583                                    }
584                                }
585                                _ => unreachable!(),
586                            };
587
588                            match &arm.guard {
589                                None => {
590                                    err.subdiagnostic(
591                                        UnexpectedExpressionInPatternSugg::CreateGuard {
592                                            ident_span,
593                                            pat_hi: arm.pat.span.shrink_to_hi(),
594                                            ident,
595                                            expr,
596                                        },
597                                    );
598                                }
599                                Some(guard) => {
600                                    // Are parentheses required around the old guard?
601                                    let wrap_guard = guard.precedence() <= ExprPrecedence::LAnd;
602
603                                    err.subdiagnostic(
604                                        UnexpectedExpressionInPatternSugg::UpdateGuard {
605                                            ident_span,
606                                            guard_lo: if wrap_guard {
607                                                Some(guard.span.shrink_to_lo())
608                                            } else {
609                                                None
610                                            },
611                                            guard_hi: guard.span.shrink_to_hi(),
612                                            guard_hi_paren: if wrap_guard { ")" } else { "" },
613                                            ident,
614                                            expr,
615                                        },
616                                    );
617                                }
618                            }
619                        }
620
621                        // help: extract the expr into a `const VAL: _ = expr`
622                        let ident = match self.field {
623                            Some(field) => field.ident.as_str().to_uppercase(),
624                            None => "VAL".to_owned(),
625                        };
626                        err.subdiagnostic(UnexpectedExpressionInPatternSugg::Const {
627                            stmt_lo: line_lo,
628                            ident_span: expr_span,
629                            expr,
630                            ident,
631                            indentation,
632                        });
633                    },
634                );
635            }
636        }
637
638        impl<'a> Visitor<'a> for PatVisitor<'a> {
639            fn visit_arm(&mut self, a: &'a Arm) -> Self::Result {
640                self.arm = Some(a);
641                visit::walk_arm(self, a);
642                self.arm = None;
643            }
644
645            fn visit_pat_field(&mut self, fp: &'a PatField) -> Self::Result {
646                self.field = Some(fp);
647                visit::walk_pat_field(self, fp);
648                self.field = None;
649            }
650
651            fn visit_pat(&mut self, p: &'a Pat) -> Self::Result {
652                match &p.kind {
653                    // Base expression
654                    PatKind::Err(_) | PatKind::Expr(_) => {
655                        self.maybe_add_suggestions_then_emit(p.span, p.span, false)
656                    }
657
658                    // Sub-patterns
659                    // FIXME: this doesn't work with recursive subpats (`&mut &mut <err>`)
660                    PatKind::Box(subpat) | PatKind::Ref(subpat, _)
661                        if matches!(subpat.kind, PatKind::Err(_) | PatKind::Expr(_)) =>
662                    {
663                        self.maybe_add_suggestions_then_emit(subpat.span, p.span, false)
664                    }
665
666                    // Sub-expressions
667                    PatKind::Range(start, end, _) => {
668                        if let Some(start) = start {
669                            self.maybe_add_suggestions_then_emit(start.span, start.span, true);
670                        }
671
672                        if let Some(end) = end {
673                            self.maybe_add_suggestions_then_emit(end.span, end.span, true);
674                        }
675                    }
676
677                    // Walk continuation
678                    _ => visit::walk_pat(self, p),
679                }
680            }
681        }
682
683        // Starts the visit.
684        PatVisitor { parser: self, stmt, arm: None, field: None }.visit_stmt(stmt);
685    }
686
687    fn eat_metavar_pat(&mut self) -> Option<Box<Pat>> {
688        // Must try both kinds of pattern nonterminals.
689        if let Some(pat) = self.eat_metavar_seq_with_matcher(
690            |mv_kind| matches!(mv_kind, MetaVarKind::Pat(PatParam { .. })),
691            |this| this.parse_pat_no_top_alt(None, None),
692        ) {
693            Some(pat)
694        } else if let Some(pat) = self.eat_metavar_seq(MetaVarKind::Pat(PatWithOr), |this| {
695            this.parse_pat_no_top_guard(
696                None,
697                RecoverComma::No,
698                RecoverColon::No,
699                CommaRecoveryMode::EitherTupleOrPipe,
700            )
701        }) {
702            Some(pat)
703        } else {
704            None
705        }
706    }
707
708    /// Parses a pattern, with a setting whether modern range patterns (e.g., `a..=b`, `a..b` are
709    /// allowed).
710    fn parse_pat_with_range_pat(
711        &mut self,
712        allow_range_pat: bool,
713        expected: Option<Expected>,
714        syntax_loc: Option<PatternLocation>,
715    ) -> PResult<'a, Box<Pat>> {
716        maybe_recover_from_interpolated_ty_qpath!(self, true);
717
718        if let Some(pat) = self.eat_metavar_pat() {
719            return Ok(pat);
720        }
721
722        let mut lo = self.token.span;
723
724        if self.token.is_keyword(kw::Let)
725            && self.look_ahead(1, |tok| {
726                tok.can_begin_pattern(token::NtPatKind::PatParam { inferred: false })
727            })
728        {
729            self.bump();
730            // Trim extra space after the `let`
731            let span = lo.with_hi(self.token.span.lo());
732            self.dcx().emit_err(RemoveLet { span: lo, suggestion: span });
733            lo = self.token.span;
734        }
735
736        let pat = if self.check(exp!(And)) || self.token == token::AndAnd {
737            self.parse_pat_deref(expected)?
738        } else if self.check(exp!(OpenParen)) {
739            self.parse_pat_tuple_or_parens()?
740        } else if self.check(exp!(OpenBracket)) {
741            // Parse `[pat, pat,...]` as a slice pattern.
742            let (pats, _) =
743                self.parse_delim_comma_seq(exp!(OpenBracket), exp!(CloseBracket), |p| {
744                    p.parse_pat_allow_top_guard(
745                        None,
746                        RecoverComma::No,
747                        RecoverColon::No,
748                        CommaRecoveryMode::EitherTupleOrPipe,
749                    )
750                })?;
751            PatKind::Slice(pats)
752        } else if self.check(exp!(DotDot)) && !self.is_pat_range_end_start(1) {
753            // A rest pattern `..`.
754            self.bump(); // `..`
755            PatKind::Rest
756        } else if self.check(exp!(DotDotDot)) && !self.is_pat_range_end_start(1) {
757            self.recover_dotdotdot_rest_pat(lo)
758        } else if let Some(form) = self.parse_range_end() {
759            self.parse_pat_range_to(form)? // `..=X`, `...X`, or `..X`.
760        } else if self.eat(exp!(Bang)) {
761            // Parse `!`
762            self.psess.gated_spans.gate(sym::never_patterns, self.prev_token.span);
763            PatKind::Never
764        } else if self.eat_keyword(exp!(Underscore)) {
765            // Parse `_`
766            PatKind::Wild
767        } else if self.eat_keyword(exp!(Mut)) {
768            self.parse_pat_ident_mut()?
769        } else if self.eat_keyword(exp!(Ref)) {
770            if self.check_keyword(exp!(Box)) {
771                // Suggest `box ref`.
772                let span = self.prev_token.span.to(self.token.span);
773                self.bump();
774                self.dcx().emit_err(SwitchRefBoxOrder { span });
775            }
776            // Parse ref ident @ pat / ref mut ident @ pat
777            let mutbl = self.parse_mutability();
778            self.parse_pat_ident(BindingMode(ByRef::Yes(mutbl), Mutability::Not), syntax_loc)?
779        } else if self.eat_keyword(exp!(Box)) {
780            self.parse_pat_box()?
781        } else if self.check_inline_const(0) {
782            // Parse `const pat`
783            let const_expr = self.parse_const_block(lo.to(self.token.span), true)?;
784
785            if let Some(re) = self.parse_range_end() {
786                self.parse_pat_range_begin_with(const_expr, re)?
787            } else {
788                PatKind::Expr(const_expr)
789            }
790        } else if self.is_builtin() {
791            self.parse_pat_builtin()?
792        }
793        // Don't eagerly error on semantically invalid tokens when matching
794        // declarative macros, as the input to those doesn't have to be
795        // semantically valid. For attribute/derive proc macros this is not the
796        // case, so doing the recovery for them is fine.
797        else if self.can_be_ident_pat()
798            || (self.is_lit_bad_ident().is_some() && self.may_recover())
799        {
800            // Parse `ident @ pat`
801            // This can give false positives and parse nullary enums,
802            // they are dealt with later in resolve.
803            self.parse_pat_ident(BindingMode::NONE, syntax_loc)?
804        } else if self.is_start_of_pat_with_path() {
805            // Parse pattern starting with a path
806            let (qself, path) = if self.eat_lt() {
807                // Parse a qualified path
808                let (qself, path) = self.parse_qpath(PathStyle::Pat)?;
809                (Some(qself), path)
810            } else {
811                // Parse an unqualified path
812                (None, self.parse_path(PathStyle::Pat)?)
813            };
814            let span = lo.to(self.prev_token.span);
815
816            if qself.is_none() && self.check(exp!(Bang)) {
817                self.parse_pat_mac_invoc(path)?
818            } else if let Some(form) = self.parse_range_end() {
819                let begin = self.mk_expr(span, ExprKind::Path(qself, path));
820                self.parse_pat_range_begin_with(begin, form)?
821            } else if self.check(exp!(OpenBrace)) {
822                self.parse_pat_struct(qself, path)?
823            } else if self.check(exp!(OpenParen)) {
824                self.parse_pat_tuple_struct(qself, path)?
825            } else {
826                match self.maybe_recover_trailing_expr(span, false) {
827                    Some((guar, _)) => PatKind::Err(guar),
828                    None => PatKind::Path(qself, path),
829                }
830            }
831        } else if let Some((lt, IdentIsRaw::No)) = self.token.lifetime()
832            // In pattern position, we're totally fine with using "next token isn't colon"
833            // as a heuristic. We could probably just always try to recover if it's a lifetime,
834            // because we never have `'a: label {}` in a pattern position anyways, but it does
835            // keep us from suggesting something like `let 'a: Ty = ..` => `let 'a': Ty = ..`
836            && could_be_unclosed_char_literal(lt)
837            && !self.look_ahead(1, |token| token.kind == token::Colon)
838        {
839            // Recover a `'a` as a `'a'` literal
840            let lt = self.expect_lifetime();
841            let (lit, _) =
842                self.recover_unclosed_char(lt.ident, Parser::mk_token_lit_char, |self_| {
843                    let expected = Expected::to_string_or_fallback(expected);
844                    let msg = format!(
845                        "expected {}, found {}",
846                        expected,
847                        super::token_descr(&self_.token)
848                    );
849
850                    self_
851                        .dcx()
852                        .struct_span_err(self_.token.span, msg)
853                        .with_span_label(self_.token.span, format!("expected {expected}"))
854                });
855            PatKind::Expr(self.mk_expr(lo, ExprKind::Lit(lit)))
856        } else {
857            // Try to parse everything else as literal with optional minus
858            match self.parse_literal_maybe_minus() {
859                Ok(begin) => {
860                    let begin = self
861                        .maybe_recover_trailing_expr(begin.span, false)
862                        .map(|(guar, sp)| self.mk_expr_err(sp, guar))
863                        .unwrap_or(begin);
864
865                    match self.parse_range_end() {
866                        Some(form) => self.parse_pat_range_begin_with(begin, form)?,
867                        None => PatKind::Expr(begin),
868                    }
869                }
870                Err(err) => return self.fatal_unexpected_non_pat(err, expected),
871            }
872        };
873
874        let pat = self.mk_pat(lo.to(self.prev_token.span), pat);
875        let pat = self.maybe_recover_from_bad_qpath(pat)?;
876        let pat = self.recover_intersection_pat(pat)?;
877
878        if !allow_range_pat {
879            self.ban_pat_range_if_ambiguous(&pat)
880        }
881
882        Ok(pat)
883    }
884
885    /// Recover from a typoed `...` pattern that was encountered
886    /// Ref: Issue #70388
887    fn recover_dotdotdot_rest_pat(&mut self, lo: Span) -> PatKind {
888        // A typoed rest pattern `...`.
889        self.bump(); // `...`
890
891        // The user probably mistook `...` for a rest pattern `..`.
892        self.dcx().emit_err(DotDotDotRestPattern {
893            span: lo,
894            suggestion: lo.with_lo(lo.hi() - BytePos(1)),
895        });
896        PatKind::Rest
897    }
898
899    /// Try to recover the more general form `intersect ::= $pat_lhs @ $pat_rhs`.
900    ///
901    /// Allowed binding patterns generated by `binding ::= ref? mut? $ident @ $pat_rhs`
902    /// should already have been parsed by now at this point,
903    /// if the next token is `@` then we can try to parse the more general form.
904    ///
905    /// Consult `parse_pat_ident` for the `binding` grammar.
906    ///
907    /// The notion of intersection patterns are found in
908    /// e.g. [F#][and] where they are called AND-patterns.
909    ///
910    /// [and]: https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/pattern-matching
911    fn recover_intersection_pat(&mut self, lhs: Box<Pat>) -> PResult<'a, Box<Pat>> {
912        if self.token != token::At {
913            // Next token is not `@` so it's not going to be an intersection pattern.
914            return Ok(lhs);
915        }
916
917        // At this point we attempt to parse `@ $pat_rhs` and emit an error.
918        self.bump(); // `@`
919        let mut rhs = self.parse_pat_no_top_alt(None, None)?;
920        let whole_span = lhs.span.to(rhs.span);
921
922        if let PatKind::Ident(_, _, sub @ None) = &mut rhs.kind {
923            // The user inverted the order, so help them fix that.
924            let lhs_span = lhs.span;
925            // Move the LHS into the RHS as a subpattern.
926            // The RHS is now the full pattern.
927            *sub = Some(lhs);
928
929            self.dcx().emit_err(PatternOnWrongSideOfAt {
930                whole_span,
931                whole_pat: pprust::pat_to_string(&rhs),
932                pattern: lhs_span,
933                binding: rhs.span,
934            });
935        } else {
936            // The special case above doesn't apply so we may have e.g. `A(x) @ B(y)`.
937            rhs.kind = PatKind::Wild;
938            self.dcx().emit_err(ExpectedBindingLeftOfAt {
939                whole_span,
940                lhs: lhs.span,
941                rhs: rhs.span,
942            });
943        }
944
945        rhs.span = whole_span;
946        Ok(rhs)
947    }
948
949    /// Ban a range pattern if it has an ambiguous interpretation.
950    fn ban_pat_range_if_ambiguous(&self, pat: &Pat) {
951        match pat.kind {
952            PatKind::Range(
953                ..,
954                Spanned { node: RangeEnd::Included(RangeSyntax::DotDotDot), .. },
955            ) => return,
956            PatKind::Range(..) => {}
957            _ => return,
958        }
959
960        self.dcx().emit_err(AmbiguousRangePattern {
961            span: pat.span,
962            suggestion: ParenRangeSuggestion {
963                lo: pat.span.shrink_to_lo(),
964                hi: pat.span.shrink_to_hi(),
965            },
966        });
967    }
968
969    /// Parse `&pat` / `&mut pat`.
970    fn parse_pat_deref(&mut self, expected: Option<Expected>) -> PResult<'a, PatKind> {
971        self.expect_and()?;
972        if let Some((lifetime, _)) = self.token.lifetime() {
973            self.bump(); // `'a`
974
975            self.dcx().emit_err(UnexpectedLifetimeInPattern {
976                span: self.prev_token.span,
977                symbol: lifetime.name,
978                suggestion: self.prev_token.span.until(self.token.span),
979            });
980        }
981
982        let mutbl = self.parse_mutability();
983        let subpat = self.parse_pat_with_range_pat(false, expected, None)?;
984        Ok(PatKind::Ref(subpat, mutbl))
985    }
986
987    /// Parse a tuple or parenthesis pattern.
988    fn parse_pat_tuple_or_parens(&mut self) -> PResult<'a, PatKind> {
989        let open_paren = self.token.span;
990
991        let (fields, trailing_comma) = self.parse_paren_comma_seq(|p| {
992            p.parse_pat_allow_top_guard(
993                None,
994                RecoverComma::No,
995                RecoverColon::No,
996                CommaRecoveryMode::LikelyTuple,
997            )
998        })?;
999
1000        // Here, `(pat,)` is a tuple pattern.
1001        // For backward compatibility, `(..)` is a tuple pattern as well.
1002        let paren_pattern =
1003            fields.len() == 1 && !(matches!(trailing_comma, Trailing::Yes) || fields[0].is_rest());
1004
1005        let pat = if paren_pattern {
1006            let pat = fields.into_iter().next().unwrap();
1007            let close_paren = self.prev_token.span;
1008
1009            match &pat.kind {
1010                // recover ranges with parentheses around the `(start)..`
1011                PatKind::Expr(begin)
1012                    if self.may_recover()
1013                        && let Some(form) = self.parse_range_end() =>
1014                {
1015                    self.dcx().emit_err(UnexpectedParenInRangePat {
1016                        span: vec![open_paren, close_paren],
1017                        sugg: UnexpectedParenInRangePatSugg {
1018                            start_span: open_paren,
1019                            end_span: close_paren,
1020                        },
1021                    });
1022
1023                    self.parse_pat_range_begin_with(begin.clone(), form)?
1024                }
1025                // recover ranges with parentheses around the `(start)..`
1026                PatKind::Err(guar)
1027                    if self.may_recover()
1028                        && let Some(form) = self.parse_range_end() =>
1029                {
1030                    self.dcx().emit_err(UnexpectedParenInRangePat {
1031                        span: vec![open_paren, close_paren],
1032                        sugg: UnexpectedParenInRangePatSugg {
1033                            start_span: open_paren,
1034                            end_span: close_paren,
1035                        },
1036                    });
1037
1038                    self.parse_pat_range_begin_with(self.mk_expr_err(pat.span, *guar), form)?
1039                }
1040
1041                // (pat) with optional parentheses
1042                _ => PatKind::Paren(pat),
1043            }
1044        } else {
1045            PatKind::Tuple(fields)
1046        };
1047
1048        Ok(match self.maybe_recover_trailing_expr(open_paren.to(self.prev_token.span), false) {
1049            None => pat,
1050            Some((guar, _)) => PatKind::Err(guar),
1051        })
1052    }
1053
1054    /// Parse a mutable binding with the `mut` token already eaten.
1055    fn parse_pat_ident_mut(&mut self) -> PResult<'a, PatKind> {
1056        let mut_span = self.prev_token.span;
1057
1058        self.recover_additional_muts();
1059
1060        let byref = self.parse_byref();
1061
1062        self.recover_additional_muts();
1063
1064        // Make sure we don't allow e.g. `let mut $p;` where `$p:pat`.
1065        if let Some(MetaVarKind::Pat(_)) = self.token.is_metavar_seq() {
1066            self.expected_ident_found_err().emit();
1067        }
1068
1069        // Parse the pattern we hope to be an identifier.
1070        let mut pat = self.parse_pat_no_top_alt(Some(Expected::Identifier), None)?;
1071
1072        // If we don't have `mut $ident (@ pat)?`, error.
1073        if let PatKind::Ident(BindingMode(br @ ByRef::No, m @ Mutability::Not), ..) = &mut pat.kind
1074        {
1075            // Don't recurse into the subpattern.
1076            // `mut` on the outer binding doesn't affect the inner bindings.
1077            *br = byref;
1078            *m = Mutability::Mut;
1079        } else {
1080            // Add `mut` to any binding in the parsed pattern.
1081            let changed_any_binding = Self::make_all_value_bindings_mutable(&mut pat);
1082            self.ban_mut_general_pat(mut_span, &pat, changed_any_binding);
1083        }
1084
1085        if matches!(pat.kind, PatKind::Ident(BindingMode(ByRef::Yes(_), Mutability::Mut), ..)) {
1086            self.psess.gated_spans.gate(sym::mut_ref, pat.span);
1087        }
1088        Ok(pat.kind)
1089    }
1090
1091    /// Turn all by-value immutable bindings in a pattern into mutable bindings.
1092    /// Returns `true` if any change was made.
1093    fn make_all_value_bindings_mutable(pat: &mut Box<Pat>) -> bool {
1094        struct AddMut(bool);
1095        impl MutVisitor for AddMut {
1096            fn visit_pat(&mut self, pat: &mut Pat) {
1097                if let PatKind::Ident(BindingMode(ByRef::No, m @ Mutability::Not), ..) =
1098                    &mut pat.kind
1099                {
1100                    self.0 = true;
1101                    *m = Mutability::Mut;
1102                }
1103                mut_visit::walk_pat(self, pat);
1104            }
1105        }
1106
1107        let mut add_mut = AddMut(false);
1108        add_mut.visit_pat(pat);
1109        add_mut.0
1110    }
1111
1112    /// Error on `mut $pat` where `$pat` is not an ident.
1113    fn ban_mut_general_pat(&self, lo: Span, pat: &Pat, changed_any_binding: bool) {
1114        self.dcx().emit_err(if changed_any_binding {
1115            InvalidMutInPattern::NestedIdent {
1116                span: lo.to(pat.span),
1117                pat: pprust::pat_to_string(pat),
1118            }
1119        } else {
1120            InvalidMutInPattern::NonIdent { span: lo.until(pat.span) }
1121        });
1122    }
1123
1124    /// Eat any extraneous `mut`s and error + recover if we ate any.
1125    fn recover_additional_muts(&mut self) {
1126        let lo = self.token.span;
1127        while self.eat_keyword(exp!(Mut)) {}
1128        if lo == self.token.span {
1129            return;
1130        }
1131
1132        let span = lo.to(self.prev_token.span);
1133        let suggestion = span.with_hi(self.token.span.lo());
1134        self.dcx().emit_err(RepeatedMutInPattern { span, suggestion });
1135    }
1136
1137    /// Parse macro invocation
1138    fn parse_pat_mac_invoc(&mut self, path: Path) -> PResult<'a, PatKind> {
1139        self.bump();
1140        let args = self.parse_delim_args()?;
1141        let mac = Box::new(MacCall { path, args });
1142        Ok(PatKind::MacCall(mac))
1143    }
1144
1145    fn fatal_unexpected_non_pat(
1146        &mut self,
1147        err: Diag<'a>,
1148        expected: Option<Expected>,
1149    ) -> PResult<'a, Box<Pat>> {
1150        err.cancel();
1151
1152        let expected = Expected::to_string_or_fallback(expected);
1153        let msg = format!("expected {}, found {}", expected, super::token_descr(&self.token));
1154
1155        let mut err = self.dcx().struct_span_err(self.token.span, msg);
1156        err.span_label(self.token.span, format!("expected {expected}"));
1157
1158        let sp = self.psess.source_map().start_point(self.token.span);
1159        if let Some(sp) = self.psess.ambiguous_block_expr_parse.borrow().get(&sp) {
1160            err.subdiagnostic(ExprParenthesesNeeded::surrounding(*sp));
1161        }
1162
1163        Err(err)
1164    }
1165
1166    /// Parses the range pattern end form `".." | "..." | "..=" ;`.
1167    fn parse_range_end(&mut self) -> Option<Spanned<RangeEnd>> {
1168        let re = if self.eat(exp!(DotDotDot)) {
1169            RangeEnd::Included(RangeSyntax::DotDotDot)
1170        } else if self.eat(exp!(DotDotEq)) {
1171            RangeEnd::Included(RangeSyntax::DotDotEq)
1172        } else if self.eat(exp!(DotDot)) {
1173            RangeEnd::Excluded
1174        } else {
1175            return None;
1176        };
1177        Some(respan(self.prev_token.span, re))
1178    }
1179
1180    /// Parse a range pattern `$begin $form $end?` where `$form = ".." | "..." | "..=" ;`.
1181    /// `$begin $form` has already been parsed.
1182    fn parse_pat_range_begin_with(
1183        &mut self,
1184        begin: Box<Expr>,
1185        re: Spanned<RangeEnd>,
1186    ) -> PResult<'a, PatKind> {
1187        let end = if self.is_pat_range_end_start(0) {
1188            // Parsing e.g. `X..=Y`.
1189            Some(self.parse_pat_range_end()?)
1190        } else {
1191            // Parsing e.g. `X..`.
1192            if let RangeEnd::Included(_) = re.node {
1193                // FIXME(Centril): Consider semantic errors instead in `ast_validation`.
1194                self.inclusive_range_with_incorrect_end();
1195            }
1196            None
1197        };
1198        Ok(PatKind::Range(Some(begin), end, re))
1199    }
1200
1201    pub(super) fn inclusive_range_with_incorrect_end(&mut self) -> ErrorGuaranteed {
1202        let tok = &self.token;
1203        let span = self.prev_token.span;
1204        // If the user typed "..==" instead of "..=", we want to give them
1205        // a specific error message telling them to use "..=".
1206        // If they typed "..=>", suggest they use ".. =>".
1207        // Otherwise, we assume that they meant to type a half open exclusive
1208        // range and give them an error telling them to do that instead.
1209        let no_space = tok.span.lo() == span.hi();
1210        match tok.kind {
1211            token::Eq if no_space => {
1212                let span_with_eq = span.to(tok.span);
1213
1214                // Ensure the user doesn't receive unhelpful unexpected token errors
1215                self.bump();
1216                if self.is_pat_range_end_start(0) {
1217                    let _ = self.parse_pat_range_end().map_err(|e| e.cancel());
1218                }
1219
1220                self.dcx().emit_err(InclusiveRangeExtraEquals { span: span_with_eq })
1221            }
1222            token::Gt if no_space => {
1223                let after_pat = span.with_hi(span.hi() - BytePos(1)).shrink_to_hi();
1224                self.dcx().emit_err(InclusiveRangeMatchArrow { span, arrow: tok.span, after_pat })
1225            }
1226            _ => self.dcx().emit_err(InclusiveRangeNoEnd {
1227                span,
1228                suggestion: span.with_lo(span.hi() - BytePos(1)),
1229            }),
1230        }
1231    }
1232
1233    /// Parse a range-to pattern, `..X` or `..=X` where `X` remains to be parsed.
1234    ///
1235    /// The form `...X` is prohibited to reduce confusion with the potential
1236    /// expression syntax `...expr` for splatting in expressions.
1237    fn parse_pat_range_to(&mut self, mut re: Spanned<RangeEnd>) -> PResult<'a, PatKind> {
1238        let end = self.parse_pat_range_end()?;
1239        if let RangeEnd::Included(syn @ RangeSyntax::DotDotDot) = &mut re.node {
1240            *syn = RangeSyntax::DotDotEq;
1241            self.dcx().emit_err(DotDotDotRangeToPatternNotAllowed { span: re.span });
1242        }
1243        Ok(PatKind::Range(None, Some(end), re))
1244    }
1245
1246    /// Is the token `dist` away from the current suitable as the start of a range patterns end?
1247    fn is_pat_range_end_start(&self, dist: usize) -> bool {
1248        self.check_inline_const(dist)
1249            || self.look_ahead(dist, |t| {
1250                t.is_path_start() // e.g. `MY_CONST`;
1251                || *t == token::Dot // e.g. `.5` for recovery;
1252                || matches!(t.kind, token::Literal(..) | token::Minus)
1253                || t.is_bool_lit()
1254                || t.is_metavar_expr()
1255                || t.is_lifetime() // recover `'a` instead of `'a'`
1256                || (self.may_recover() // recover leading `(`
1257                    && *t == token::OpenParen
1258                    && self.look_ahead(dist + 1, |t| *t != token::OpenParen)
1259                    && self.is_pat_range_end_start(dist + 1))
1260            })
1261    }
1262
1263    /// Parse a range pattern end bound
1264    fn parse_pat_range_end(&mut self) -> PResult<'a, Box<Expr>> {
1265        // recover leading `(`
1266        let open_paren = (self.may_recover() && self.eat_noexpect(&token::OpenParen))
1267            .then_some(self.prev_token.span);
1268
1269        let bound = if self.check_inline_const(0) {
1270            self.parse_const_block(self.token.span, true)
1271        } else if self.check_path() {
1272            let lo = self.token.span;
1273            let (qself, path) = if self.eat_lt() {
1274                // Parse a qualified path
1275                let (qself, path) = self.parse_qpath(PathStyle::Pat)?;
1276                (Some(qself), path)
1277            } else {
1278                // Parse an unqualified path
1279                (None, self.parse_path(PathStyle::Pat)?)
1280            };
1281            let hi = self.prev_token.span;
1282            Ok(self.mk_expr(lo.to(hi), ExprKind::Path(qself, path)))
1283        } else {
1284            self.parse_literal_maybe_minus()
1285        }?;
1286
1287        let recovered = self.maybe_recover_trailing_expr(bound.span, true);
1288
1289        // recover trailing `)`
1290        if let Some(open_paren) = open_paren {
1291            self.expect(exp!(CloseParen))?;
1292
1293            self.dcx().emit_err(UnexpectedParenInRangePat {
1294                span: vec![open_paren, self.prev_token.span],
1295                sugg: UnexpectedParenInRangePatSugg {
1296                    start_span: open_paren,
1297                    end_span: self.prev_token.span,
1298                },
1299            });
1300        }
1301
1302        Ok(match recovered {
1303            Some((guar, sp)) => self.mk_expr_err(sp, guar),
1304            None => bound,
1305        })
1306    }
1307
1308    /// Is this the start of a pattern beginning with a path?
1309    fn is_start_of_pat_with_path(&mut self) -> bool {
1310        self.check_path()
1311        // Just for recovery (see `can_be_ident`).
1312        || self.token.is_ident() && !self.token.is_bool_lit() && !self.token.is_keyword(kw::In)
1313    }
1314
1315    /// Would `parse_pat_ident` be appropriate here?
1316    fn can_be_ident_pat(&mut self) -> bool {
1317        self.check_ident()
1318        && !self.token.is_bool_lit() // Avoid `true` or `false` as a binding as it is a literal.
1319        && !self.token.is_path_segment_keyword() // Avoid e.g. `Self` as it is a path.
1320        // Avoid `in`. Due to recovery in the list parser this messes with `for ( $pat in $expr )`.
1321        && !self.token.is_keyword(kw::In)
1322        // Try to do something more complex?
1323        && self.look_ahead(1, |t| !matches!(t.kind, token::OpenParen // A tuple struct pattern.
1324            | token::OpenBrace // A struct pattern.
1325            | token::DotDotDot | token::DotDotEq | token::DotDot // A range pattern.
1326            | token::PathSep // A tuple / struct variant pattern.
1327            | token::Bang)) // A macro expanding to a pattern.
1328    }
1329
1330    /// Parses `ident` or `ident @ pat`.
1331    /// Used by the copy foo and ref foo patterns to give a good
1332    /// error message when parsing mistakes like `ref foo(a, b)`.
1333    fn parse_pat_ident(
1334        &mut self,
1335        binding_annotation: BindingMode,
1336        syntax_loc: Option<PatternLocation>,
1337    ) -> PResult<'a, PatKind> {
1338        let ident = self.parse_ident_common(false)?;
1339
1340        if self.may_recover()
1341            && !matches!(syntax_loc, Some(PatternLocation::FunctionParameter))
1342            && self.check_noexpect(&token::Lt)
1343            && self.look_ahead(1, |t| t.can_begin_type())
1344        {
1345            return Err(self.dcx().create_err(GenericArgsInPatRequireTurbofishSyntax {
1346                span: self.token.span,
1347                suggest_turbofish: self.token.span.shrink_to_lo(),
1348            }));
1349        }
1350
1351        let sub = if self.eat(exp!(At)) {
1352            Some(self.parse_pat_no_top_alt(Some(Expected::BindingPattern), None)?)
1353        } else {
1354            None
1355        };
1356
1357        // Just to be friendly, if they write something like `ref Some(i)`,
1358        // we end up here with `(` as the current token.
1359        // This shortly leads to a parse error. Note that if there is no explicit
1360        // binding mode then we do not end up here, because the lookahead
1361        // will direct us over to `parse_enum_variant()`.
1362        if self.token == token::OpenParen {
1363            return Err(self
1364                .dcx()
1365                .create_err(EnumPatternInsteadOfIdentifier { span: self.prev_token.span }));
1366        }
1367
1368        // Check for method calls after the `ident`,
1369        // but not `ident @ subpat` as `subpat` was already checked and `ident` continues with `@`.
1370
1371        let pat = if sub.is_none()
1372            && let Some((guar, _)) = self.maybe_recover_trailing_expr(ident.span, false)
1373        {
1374            PatKind::Err(guar)
1375        } else {
1376            PatKind::Ident(binding_annotation, ident, sub)
1377        };
1378        Ok(pat)
1379    }
1380
1381    /// Parse a struct ("record") pattern (e.g. `Foo { ... }` or `Foo::Bar { ... }`).
1382    fn parse_pat_struct(&mut self, qself: Option<Box<QSelf>>, path: Path) -> PResult<'a, PatKind> {
1383        if qself.is_some() {
1384            // Feature gate the use of qualified paths in patterns
1385            self.psess.gated_spans.gate(sym::more_qualified_paths, path.span);
1386        }
1387        self.bump();
1388        let (fields, etc) = self.parse_pat_fields().unwrap_or_else(|mut e| {
1389            e.span_label(path.span, "while parsing the fields for this pattern");
1390            let guar = e.emit();
1391            self.recover_stmt();
1392            // When recovering, pretend we had `Foo { .. }`, to avoid cascading errors.
1393            (ThinVec::new(), PatFieldsRest::Recovered(guar))
1394        });
1395        self.bump();
1396        Ok(PatKind::Struct(qself, path, fields, etc))
1397    }
1398
1399    /// Parse tuple struct or tuple variant pattern (e.g. `Foo(...)` or `Foo::Bar(...)`).
1400    fn parse_pat_tuple_struct(
1401        &mut self,
1402        qself: Option<Box<QSelf>>,
1403        path: Path,
1404    ) -> PResult<'a, PatKind> {
1405        let (fields, _) = self.parse_paren_comma_seq(|p| {
1406            p.parse_pat_allow_top_guard(
1407                None,
1408                RecoverComma::No,
1409                RecoverColon::No,
1410                CommaRecoveryMode::EitherTupleOrPipe,
1411            )
1412        })?;
1413        if qself.is_some() {
1414            self.psess.gated_spans.gate(sym::more_qualified_paths, path.span);
1415        }
1416        Ok(PatKind::TupleStruct(qself, path, fields))
1417    }
1418
1419    /// Are we sure this could not possibly be the start of a pattern?
1420    ///
1421    /// Currently, this only accounts for tokens that can follow identifiers
1422    /// in patterns, but this can be extended as necessary.
1423    fn isnt_pattern_start(&self) -> bool {
1424        [
1425            token::Eq,
1426            token::Colon,
1427            token::Comma,
1428            token::Semi,
1429            token::At,
1430            token::OpenBrace,
1431            token::CloseBrace,
1432            token::CloseParen,
1433        ]
1434        .contains(&self.token.kind)
1435    }
1436
1437    fn parse_pat_builtin(&mut self) -> PResult<'a, PatKind> {
1438        self.parse_builtin(|self_, _lo, ident| {
1439            Ok(match ident.name {
1440                // builtin#deref(PAT)
1441                sym::deref => Some(ast::PatKind::Deref(self_.parse_pat_allow_top_guard(
1442                    None,
1443                    RecoverComma::Yes,
1444                    RecoverColon::Yes,
1445                    CommaRecoveryMode::LikelyTuple,
1446                )?)),
1447                _ => None,
1448            })
1449        })
1450    }
1451
1452    /// Parses `box pat`
1453    fn parse_pat_box(&mut self) -> PResult<'a, PatKind> {
1454        let box_span = self.prev_token.span;
1455
1456        if self.isnt_pattern_start() {
1457            let descr = super::token_descr(&self.token);
1458            self.dcx().emit_err(errors::BoxNotPat {
1459                span: self.token.span,
1460                kw: box_span,
1461                lo: box_span.shrink_to_lo(),
1462                descr,
1463            });
1464
1465            // We cannot use `parse_pat_ident()` since it will complain `box`
1466            // is not an identifier.
1467            let sub = if self.eat(exp!(At)) {
1468                Some(self.parse_pat_no_top_alt(Some(Expected::BindingPattern), None)?)
1469            } else {
1470                None
1471            };
1472
1473            Ok(PatKind::Ident(BindingMode::NONE, Ident::new(kw::Box, box_span), sub))
1474        } else {
1475            let pat = self.parse_pat_with_range_pat(false, None, None)?;
1476            self.psess.gated_spans.gate(sym::box_patterns, box_span.to(self.prev_token.span));
1477            Ok(PatKind::Box(pat))
1478        }
1479    }
1480
1481    /// Parses the fields of a struct-like pattern.
1482    fn parse_pat_fields(&mut self) -> PResult<'a, (ThinVec<PatField>, PatFieldsRest)> {
1483        let mut fields: ThinVec<PatField> = ThinVec::new();
1484        let mut etc = PatFieldsRest::None;
1485        let mut ate_comma = true;
1486        let mut delayed_err: Option<Diag<'a>> = None;
1487        let mut first_etc_and_maybe_comma_span = None;
1488        let mut last_non_comma_dotdot_span = None;
1489
1490        while self.token != token::CloseBrace {
1491            // check that a comma comes after every field
1492            if !ate_comma {
1493                let err = if self.token == token::At {
1494                    let prev_field = fields
1495                        .last()
1496                        .expect("Unreachable on first iteration, not empty otherwise")
1497                        .ident;
1498                    self.report_misplaced_at_in_struct_pat(prev_field)
1499                } else {
1500                    let mut err = self
1501                        .dcx()
1502                        .create_err(ExpectedCommaAfterPatternField { span: self.token.span });
1503                    self.recover_misplaced_pattern_modifiers(&fields, &mut err);
1504                    err
1505                };
1506                if let Some(delayed) = delayed_err {
1507                    delayed.emit();
1508                }
1509                return Err(err);
1510            }
1511            ate_comma = false;
1512
1513            if self.check(exp!(DotDot))
1514                || self.check_noexpect(&token::DotDotDot)
1515                || self.check_keyword(exp!(Underscore))
1516            {
1517                etc = PatFieldsRest::Rest;
1518                let mut etc_sp = self.token.span;
1519                if first_etc_and_maybe_comma_span.is_none() {
1520                    if let Some(comma_tok) =
1521                        self.look_ahead(1, |&t| if t == token::Comma { Some(t) } else { None })
1522                    {
1523                        let nw_span = self
1524                            .psess
1525                            .source_map()
1526                            .span_extend_to_line(comma_tok.span)
1527                            .trim_start(comma_tok.span.shrink_to_lo())
1528                            .map(|s| self.psess.source_map().span_until_non_whitespace(s));
1529                        first_etc_and_maybe_comma_span = nw_span.map(|s| etc_sp.to(s));
1530                    } else {
1531                        first_etc_and_maybe_comma_span =
1532                            Some(self.psess.source_map().span_until_non_whitespace(etc_sp));
1533                    }
1534                }
1535
1536                self.recover_bad_dot_dot();
1537                self.bump(); // `..` || `...` || `_`
1538
1539                if self.token == token::CloseBrace {
1540                    break;
1541                }
1542                let token_str = super::token_descr(&self.token);
1543                let msg = format!("expected `}}`, found {token_str}");
1544                let mut err = self.dcx().struct_span_err(self.token.span, msg);
1545
1546                err.span_label(self.token.span, "expected `}`");
1547                let mut comma_sp = None;
1548                if self.token == token::Comma {
1549                    // Issue #49257
1550                    let nw_span =
1551                        self.psess.source_map().span_until_non_whitespace(self.token.span);
1552                    etc_sp = etc_sp.to(nw_span);
1553                    err.span_label(
1554                        etc_sp,
1555                        "`..` must be at the end and cannot have a trailing comma",
1556                    );
1557                    comma_sp = Some(self.token.span);
1558                    self.bump();
1559                    ate_comma = true;
1560                }
1561
1562                if self.token == token::CloseBrace {
1563                    // If the struct looks otherwise well formed, recover and continue.
1564                    if let Some(sp) = comma_sp {
1565                        err.span_suggestion_short(
1566                            sp,
1567                            "remove this comma",
1568                            "",
1569                            Applicability::MachineApplicable,
1570                        );
1571                    }
1572                    err.emit();
1573                    break;
1574                } else if self.token.is_ident() && ate_comma {
1575                    // Accept fields coming after `..,`.
1576                    // This way we avoid "pattern missing fields" errors afterwards.
1577                    // We delay this error until the end in order to have a span for a
1578                    // suggested fix.
1579                    if let Some(delayed_err) = delayed_err {
1580                        delayed_err.emit();
1581                        return Err(err);
1582                    } else {
1583                        delayed_err = Some(err);
1584                    }
1585                } else {
1586                    if let Some(err) = delayed_err {
1587                        err.emit();
1588                    }
1589                    return Err(err);
1590                }
1591            }
1592
1593            let attrs = match self.parse_outer_attributes() {
1594                Ok(attrs) => attrs,
1595                Err(err) => {
1596                    if let Some(delayed) = delayed_err {
1597                        delayed.emit();
1598                    }
1599                    return Err(err);
1600                }
1601            };
1602            let lo = self.token.span;
1603
1604            let field = self.collect_tokens(None, attrs, ForceCollect::No, |this, attrs| {
1605                let field = match this.parse_pat_field(lo, attrs) {
1606                    Ok(field) => Ok(field),
1607                    Err(err) => {
1608                        if let Some(delayed_err) = delayed_err.take() {
1609                            delayed_err.emit();
1610                        }
1611                        return Err(err);
1612                    }
1613                }?;
1614                ate_comma = this.eat(exp!(Comma));
1615
1616                last_non_comma_dotdot_span = Some(this.prev_token.span);
1617
1618                // We just ate a comma, so there's no need to capture a trailing token.
1619                Ok((field, Trailing::No, UsePreAttrPos::No))
1620            })?;
1621
1622            fields.push(field)
1623        }
1624
1625        if let Some(mut err) = delayed_err {
1626            if let Some(first_etc_span) = first_etc_and_maybe_comma_span {
1627                if self.prev_token == token::DotDot {
1628                    // We have `.., x, ..`.
1629                    err.multipart_suggestion(
1630                        "remove the starting `..`",
1631                        vec![(first_etc_span, String::new())],
1632                        Applicability::MachineApplicable,
1633                    );
1634                } else if let Some(last_non_comma_dotdot_span) = last_non_comma_dotdot_span {
1635                    // We have `.., x`.
1636                    err.multipart_suggestion(
1637                        "move the `..` to the end of the field list",
1638                        vec![
1639                            (first_etc_span, String::new()),
1640                            (
1641                                self.token.span.to(last_non_comma_dotdot_span.shrink_to_hi()),
1642                                format!("{} .. }}", if ate_comma { "" } else { "," }),
1643                            ),
1644                        ],
1645                        Applicability::MachineApplicable,
1646                    );
1647                }
1648            }
1649            err.emit();
1650        }
1651        Ok((fields, etc))
1652    }
1653
1654    #[deny(rustc::untranslatable_diagnostic)]
1655    fn report_misplaced_at_in_struct_pat(&self, prev_field: Ident) -> Diag<'a> {
1656        debug_assert_eq!(self.token, token::At);
1657        let span = prev_field.span.to(self.token.span);
1658        if let Some(dot_dot_span) =
1659            self.look_ahead(1, |t| if t == &token::DotDot { Some(t.span) } else { None })
1660        {
1661            self.dcx().create_err(AtDotDotInStructPattern {
1662                span: span.to(dot_dot_span),
1663                remove: span.until(dot_dot_span),
1664                ident: prev_field,
1665            })
1666        } else {
1667            self.dcx().create_err(AtInStructPattern { span })
1668        }
1669    }
1670
1671    /// If the user writes `S { ref field: name }` instead of `S { field: ref name }`, we suggest
1672    /// the correct code.
1673    fn recover_misplaced_pattern_modifiers(&self, fields: &ThinVec<PatField>, err: &mut Diag<'a>) {
1674        if let Some(last) = fields.iter().last()
1675            && last.is_shorthand
1676            && let PatKind::Ident(binding, ident, None) = last.pat.kind
1677            && binding != BindingMode::NONE
1678            && self.token == token::Colon
1679            // We found `ref mut? ident:`, try to parse a `name,` or `name }`.
1680            && let Some(name_span) = self.look_ahead(1, |t| t.is_ident().then(|| t.span))
1681            && self.look_ahead(2, |t| {
1682                t == &token::Comma || t == &token::CloseBrace
1683            })
1684        {
1685            let span = last.pat.span.with_hi(ident.span.lo());
1686            // We have `S { ref field: name }` instead of `S { field: ref name }`
1687            err.multipart_suggestion(
1688                "the pattern modifiers belong after the `:`",
1689                vec![
1690                    (span, String::new()),
1691                    (name_span.shrink_to_lo(), binding.prefix_str().to_string()),
1692                ],
1693                Applicability::MachineApplicable,
1694            );
1695        }
1696    }
1697
1698    /// Recover on `...` or `_` as if it were `..` to avoid further errors.
1699    /// See issue #46718.
1700    fn recover_bad_dot_dot(&self) {
1701        if self.token == token::DotDot {
1702            return;
1703        }
1704
1705        let token_str = pprust::token_to_string(&self.token);
1706        self.dcx().emit_err(DotDotDotForRemainingFields { span: self.token.span, token_str });
1707    }
1708
1709    fn parse_pat_field(&mut self, lo: Span, attrs: AttrVec) -> PResult<'a, PatField> {
1710        // Check if a colon exists one ahead. This means we're parsing a fieldname.
1711        let hi;
1712        let (subpat, fieldname, is_shorthand) = if self.look_ahead(1, |t| t == &token::Colon) {
1713            // Parsing a pattern of the form `fieldname: pat`.
1714            let fieldname = self.parse_field_name()?;
1715            self.bump();
1716            let pat = self.parse_pat_allow_top_guard(
1717                None,
1718                RecoverComma::No,
1719                RecoverColon::No,
1720                CommaRecoveryMode::EitherTupleOrPipe,
1721            )?;
1722            hi = pat.span;
1723            (pat, fieldname, false)
1724        } else {
1725            // Parsing a pattern of the form `(box) (ref) (mut) fieldname`.
1726            let is_box = self.eat_keyword(exp!(Box));
1727            let boxed_span = self.token.span;
1728            let mutability = self.parse_mutability();
1729            let by_ref = self.parse_byref();
1730
1731            let fieldname = self.parse_field_name()?;
1732            hi = self.prev_token.span;
1733            let ann = BindingMode(by_ref, mutability);
1734            let fieldpat = self.mk_pat_ident(boxed_span.to(hi), ann, fieldname);
1735            let subpat =
1736                if is_box { self.mk_pat(lo.to(hi), PatKind::Box(fieldpat)) } else { fieldpat };
1737            (subpat, fieldname, true)
1738        };
1739
1740        Ok(PatField {
1741            ident: fieldname,
1742            pat: subpat,
1743            is_shorthand,
1744            attrs,
1745            id: ast::DUMMY_NODE_ID,
1746            span: lo.to(hi),
1747            is_placeholder: false,
1748        })
1749    }
1750
1751    pub(super) fn mk_pat_ident(&self, span: Span, ann: BindingMode, ident: Ident) -> Box<Pat> {
1752        self.mk_pat(span, PatKind::Ident(ann, ident, None))
1753    }
1754
1755    pub(super) fn mk_pat(&self, span: Span, kind: PatKind) -> Box<Pat> {
1756        Box::new(Pat { kind, span, id: ast::DUMMY_NODE_ID, tokens: None })
1757    }
1758}