Skip to main content

rustc_parse/parser/
mod.rs

1pub mod attr;
2mod attr_wrapper;
3mod diagnostics;
4mod expr;
5mod generics;
6mod item;
7mod nonterminal;
8mod pat;
9mod path;
10mod stmt;
11pub mod token_type;
12mod ty;
13
14// Parsers for non-functionlike builtin macros are defined in rustc_parse so they can be used by
15// both rustc_builtin_macros and rustfmt.
16pub mod asm;
17pub mod cfg_select;
18
19use std::{debug_assert_matches, fmt, mem, slice};
20
21use attr_wrapper::{AttrWrapper, UsePreAttrPos};
22pub use diagnostics::AttemptLocalParseRecovery;
23// Public to use it for custom `if` expressions in rustfmt forks like https://github.com/tucant/rustfmt
24pub use expr::LetChainsPolicy;
25pub(crate) use item::{FnContext, FnParseMode};
26pub use pat::{CommaRecoveryMode, RecoverColon, RecoverComma};
27pub use path::PathStyle;
28use rustc_ast::token::{
29    self, IdentIsRaw, InvisibleOrigin, MetaVarKind, NtExprKind, NtPatKind, Token, TokenKind,
30};
31use rustc_ast::tokenstream::{
32    ParserRange, ParserReplacement, Spacing, TokenCursor, TokenStream, TokenTree, TokenTreeCursor,
33};
34use rustc_ast::util::case::Case;
35use rustc_ast::util::classify;
36use rustc_ast::{
37    self as ast, AnonConst, AttrArgs, AttrId, BinOpKind, ByRef, Const, CoroutineKind,
38    DUMMY_NODE_ID, DelimArgs, Expr, ExprKind, Extern, HasAttrs, HasTokens, ImplRestriction,
39    MgcaDisambiguation, MutRestriction, Mutability, Recovered, RestrictionKind, Safety, StrLit,
40    Visibility, VisibilityKind,
41};
42use rustc_ast_pretty::pprust;
43use rustc_data_structures::fx::FxHashMap;
44use rustc_errors::{Applicability, Diag, FatalError, MultiSpan, PResult};
45use rustc_index::interval::IntervalSet;
46use rustc_session::parse::ParseSess;
47use rustc_span::{ErrorGuaranteed, Ident, Span, Symbol, kw, sym};
48use thin_vec::ThinVec;
49use token_type::TokenTypeSet;
50pub use token_type::{ExpKeywordPair, ExpTokenPair, TokenType};
51use tracing::debug;
52
53use crate::errors::{
54    self, IncorrectImplRestriction, IncorrectMutRestriction, IncorrectVisibilityRestriction,
55    NonStringAbiLiteral, TokenDescription,
56};
57use crate::exp;
58
59#[cfg(test)]
60mod tests;
61
62// Ideally, these tests would be in `rustc_ast`. But they depend on having a
63// parser, so they are here.
64#[cfg(test)]
65mod tokenstream {
66    mod tests;
67}
68
69bitflags::bitflags! {
70    /// Restrictions applied while parsing.
71    ///
72    /// The parser maintains a bitset of restrictions it will honor while
73    /// parsing. This is essentially used as a way of tracking state of what
74    /// is being parsed and to change behavior based on that.
75    #[derive(#[automatically_derived]
impl ::core::clone::Clone for Restrictions {
    #[inline]
    fn clone(&self) -> Restrictions {
        let _:
                ::core::clone::AssertParamIsClone<<Restrictions as
                ::bitflags::__private::PublicFlags>::Internal>;
        *self
    }
}
impl Restrictions {
    #[doc = r" Restricts expressions for use in statement position."]
    #[doc = r""]
    #[doc =
    r" When expressions are used in various places, like statements or"]
    #[doc =
    r" match arms, this is used to stop parsing once certain tokens are"]
    #[doc = r" reached."]
    #[doc = r""]
    #[doc =
    r" For example, `if true {} & 1` with `STMT_EXPR` in effect is parsed"]
    #[doc =
    r" as two separate expression statements (`if` and a reference to 1)."]
    #[doc =
    r" Otherwise it is parsed as a bitwise AND where `if` is on the left"]
    #[doc = r" and 1 is on the right."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const STMT_EXPR: Self = Self::from_bits_retain(1 << 0);
    #[doc = r" Do not allow struct literals."]
    #[doc = r""]
    #[doc =
    r" There are several places in the grammar where we don't want to"]
    #[doc = r" allow struct literals because they can require lookahead, or"]
    #[doc = r" otherwise could be ambiguous or cause confusion. For example,"]
    #[doc =
    r" `if Foo {} {}` isn't clear if it is `Foo{}` struct literal, or"]
    #[doc = r" just `Foo` is the condition, followed by a consequent block,"]
    #[doc = r" followed by an empty block."]
    #[doc = r""]
    #[doc =
    r" See [RFC 92](https://rust-lang.github.io/rfcs/0092-struct-grammar.html)."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const NO_STRUCT_LITERAL: Self = Self::from_bits_retain(1 << 1);
    #[doc =
    r" Used to provide better error messages for const generic arguments."]
    #[doc = r""]
    #[doc =
    r" An un-braced const generic argument is limited to a very small"]
    #[doc =
    r" subset of expressions. This is used to detect the situation where"]
    #[doc =
    r" an expression outside of that subset is used, and to suggest to"]
    #[doc = r" wrap the expression in braces."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const CONST_EXPR: Self = Self::from_bits_retain(1 << 2);
    #[doc = r" Allows `let` expressions."]
    #[doc = r""]
    #[doc =
    r" `let pattern = scrutinee` is parsed as an expression, but it is"]
    #[doc = r" only allowed in let chains (`if` and `while` conditions)."]
    #[doc =
    r" Otherwise it is not an expression (note that `let` in statement"]
    #[doc =
    r" positions is treated as a `StmtKind::Let` statement, which has a"]
    #[doc = r" slightly different grammar)."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const ALLOW_LET: Self = Self::from_bits_retain(1 << 3);
    #[doc = r" Used to detect a missing `=>` in a match guard."]
    #[doc = r""]
    #[doc =
    r" This is used for error handling in a match guard to give a better"]
    #[doc =
    r" error message if the `=>` is missing. It is set when parsing the"]
    #[doc = r" guard expression."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const IN_IF_GUARD: Self = Self::from_bits_retain(1 << 4);
    #[doc = r" Used to detect the incorrect use of expressions in patterns."]
    #[doc = r""]
    #[doc =
    r" This is used for error handling while parsing a pattern. During"]
    #[doc =
    r" error recovery, this will be set to try to parse the pattern as an"]
    #[doc =
    r" expression, but halts parsing the expression when reaching certain"]
    #[doc = r" tokens like `=`."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const IS_PAT: Self = Self::from_bits_retain(1 << 5);
}
impl ::bitflags::Flags for Restrictions {
    const FLAGS: &'static [::bitflags::Flag<Restrictions>] =
        &[{

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("STMT_EXPR", Restrictions::STMT_EXPR)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("NO_STRUCT_LITERAL",
                            Restrictions::NO_STRUCT_LITERAL)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("CONST_EXPR",
                            Restrictions::CONST_EXPR)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("ALLOW_LET", Restrictions::ALLOW_LET)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("IN_IF_GUARD",
                            Restrictions::IN_IF_GUARD)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("IS_PAT", Restrictions::IS_PAT)
                    }];
    type Bits = u8;
    fn bits(&self) -> u8 { Restrictions::bits(self) }
    fn from_bits_retain(bits: u8) -> Restrictions {
        Restrictions::from_bits_retain(bits)
    }
}
#[allow(dead_code, deprecated, unused_doc_comments, unused_attributes,
unused_mut, unused_imports, non_upper_case_globals, clippy ::
assign_op_pattern, clippy :: indexing_slicing, clippy :: same_name_method,
clippy :: iter_without_into_iter,)]
const _: () =
    {
        #[repr(transparent)]
        struct InternalBitFlags(u8);
        #[automatically_derived]
        #[doc(hidden)]
        unsafe impl ::core::clone::TrivialClone for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::clone::Clone for InternalBitFlags {
            #[inline]
            fn clone(&self) -> InternalBitFlags {
                let _: ::core::clone::AssertParamIsClone<u8>;
                *self
            }
        }
        #[automatically_derived]
        impl ::core::marker::Copy for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::marker::StructuralPartialEq for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::cmp::PartialEq for InternalBitFlags {
            #[inline]
            fn eq(&self, other: &InternalBitFlags) -> bool {
                self.0 == other.0
            }
        }
        #[automatically_derived]
        impl ::core::cmp::Eq for InternalBitFlags {
            #[inline]
            #[doc(hidden)]
            #[coverage(off)]
            fn assert_fields_are_eq(&self) {
                let _: ::core::cmp::AssertParamIsEq<u8>;
            }
        }
        #[automatically_derived]
        impl ::core::cmp::PartialOrd for InternalBitFlags {
            #[inline]
            fn partial_cmp(&self, other: &InternalBitFlags)
                -> ::core::option::Option<::core::cmp::Ordering> {
                ::core::option::Option::Some(::core::cmp::Ord::cmp(self,
                        other))
            }
        }
        #[automatically_derived]
        impl ::core::cmp::Ord for InternalBitFlags {
            #[inline]
            fn cmp(&self, other: &InternalBitFlags) -> ::core::cmp::Ordering {
                ::core::cmp::Ord::cmp(&self.0, &other.0)
            }
        }
        #[automatically_derived]
        impl ::core::hash::Hash for InternalBitFlags {
            #[inline]
            fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
                ::core::hash::Hash::hash(&self.0, state)
            }
        }
        impl ::bitflags::__private::PublicFlags for Restrictions {
            type Primitive = u8;
            type Internal = InternalBitFlags;
        }
        impl ::bitflags::__private::core::default::Default for
            InternalBitFlags {
            #[inline]
            fn default() -> Self { InternalBitFlags::empty() }
        }
        impl ::bitflags::__private::core::fmt::Debug for InternalBitFlags {
            fn fmt(&self,
                f: &mut ::bitflags::__private::core::fmt::Formatter<'_>)
                -> ::bitflags::__private::core::fmt::Result {
                if self.is_empty() {
                    f.write_fmt(format_args!("{0:#x}",
                            <u8 as ::bitflags::Bits>::EMPTY))
                } else {
                    ::bitflags::__private::core::fmt::Display::fmt(self, f)
                }
            }
        }
        impl ::bitflags::__private::core::fmt::Display for InternalBitFlags {
            fn fmt(&self,
                f: &mut ::bitflags::__private::core::fmt::Formatter<'_>)
                -> ::bitflags::__private::core::fmt::Result {
                ::bitflags::parser::to_writer(&Restrictions(*self), f)
            }
        }
        impl ::bitflags::__private::core::str::FromStr for InternalBitFlags {
            type Err = ::bitflags::parser::ParseError;
            fn from_str(s: &str)
                ->
                    ::bitflags::__private::core::result::Result<Self,
                    Self::Err> {
                ::bitflags::parser::from_str::<Restrictions>(s).map(|flags|
                        flags.0)
            }
        }
        impl ::bitflags::__private::core::convert::AsRef<u8> for
            InternalBitFlags {
            fn as_ref(&self) -> &u8 { &self.0 }
        }
        impl ::bitflags::__private::core::convert::From<u8> for
            InternalBitFlags {
            fn from(bits: u8) -> Self { Self::from_bits_retain(bits) }
        }
        #[allow(dead_code, deprecated, unused_attributes)]
        impl InternalBitFlags {
            /// Get a flags value with all bits unset.
            #[inline]
            pub const fn empty() -> Self {
                Self(<u8 as ::bitflags::Bits>::EMPTY)
            }
            /// Get a flags value with all known bits set.
            #[inline]
            pub const fn all() -> Self {
                let mut truncated = <u8 as ::bitflags::Bits>::EMPTY;
                let mut i = 0;
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <Restrictions as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                let _ = i;
                Self(truncated)
            }
            /// Get the underlying bits value.
            ///
            /// The returned value is exactly the bits set in this flags value.
            #[inline]
            pub const fn bits(&self) -> u8 { self.0 }
            /// Convert from a bits value.
            ///
            /// This method will return `None` if any unknown bits are set.
            #[inline]
            pub const fn from_bits(bits: u8)
                -> ::bitflags::__private::core::option::Option<Self> {
                let truncated = Self::from_bits_truncate(bits).0;
                if truncated == bits {
                    ::bitflags::__private::core::option::Option::Some(Self(bits))
                } else { ::bitflags::__private::core::option::Option::None }
            }
            /// Convert from a bits value, unsetting any unknown bits.
            #[inline]
            pub const fn from_bits_truncate(bits: u8) -> Self {
                Self(bits & Self::all().0)
            }
            /// Convert from a bits value exactly.
            #[inline]
            pub const fn from_bits_retain(bits: u8) -> Self { Self(bits) }
            /// Get a flags value with the bits of a flag with the given name set.
            ///
            /// This method will return `None` if `name` is empty or doesn't
            /// correspond to any named flag.
            #[inline]
            pub fn from_name(name: &str)
                -> ::bitflags::__private::core::option::Option<Self> {
                {
                    if name == "STMT_EXPR" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::STMT_EXPR.bits()));
                    }
                };
                ;
                {
                    if name == "NO_STRUCT_LITERAL" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::NO_STRUCT_LITERAL.bits()));
                    }
                };
                ;
                {
                    if name == "CONST_EXPR" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::CONST_EXPR.bits()));
                    }
                };
                ;
                {
                    if name == "ALLOW_LET" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::ALLOW_LET.bits()));
                    }
                };
                ;
                {
                    if name == "IN_IF_GUARD" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::IN_IF_GUARD.bits()));
                    }
                };
                ;
                {
                    if name == "IS_PAT" {
                        return ::bitflags::__private::core::option::Option::Some(Self(Restrictions::IS_PAT.bits()));
                    }
                };
                ;
                let _ = name;
                ::bitflags::__private::core::option::Option::None
            }
            /// Whether all bits in this flags value are unset.
            #[inline]
            pub const fn is_empty(&self) -> bool {
                self.0 == <u8 as ::bitflags::Bits>::EMPTY
            }
            /// Whether all known bits in this flags value are set.
            #[inline]
            pub const fn is_all(&self) -> bool {
                Self::all().0 | self.0 == self.0
            }
            /// Whether any set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn intersects(&self, other: Self) -> bool {
                self.0 & other.0 != <u8 as ::bitflags::Bits>::EMPTY
            }
            /// Whether all set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn contains(&self, other: Self) -> bool {
                self.0 & other.0 == other.0
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            pub fn insert(&mut self, other: Self) {
                *self = Self(self.0).union(other);
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `remove` won't truncate `other`, but the `!` operator will.
            #[inline]
            pub fn remove(&mut self, other: Self) {
                *self = Self(self.0).difference(other);
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            pub fn toggle(&mut self, other: Self) {
                *self = Self(self.0).symmetric_difference(other);
            }
            /// Call `insert` when `value` is `true` or `remove` when `value` is `false`.
            #[inline]
            pub fn set(&mut self, other: Self, value: bool) {
                if value { self.insert(other); } else { self.remove(other); }
            }
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn intersection(self, other: Self) -> Self {
                Self(self.0 & other.0)
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn union(self, other: Self) -> Self {
                Self(self.0 | other.0)
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            #[must_use]
            pub const fn difference(self, other: Self) -> Self {
                Self(self.0 & !other.0)
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn symmetric_difference(self, other: Self) -> Self {
                Self(self.0 ^ other.0)
            }
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            #[must_use]
            pub const fn complement(self) -> Self {
                Self::from_bits_truncate(!self.0)
            }
        }
        impl ::bitflags::__private::core::fmt::Binary for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Binary::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::Octal for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Octal::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::LowerHex for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::LowerHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::UpperHex for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::UpperHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::ops::BitOr for InternalBitFlags {
            type Output = Self;
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor(self, other: InternalBitFlags) -> Self {
                self.union(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitOrAssign for
            InternalBitFlags {
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor_assign(&mut self, other: Self) { self.insert(other); }
        }
        impl ::bitflags::__private::core::ops::BitXor for InternalBitFlags {
            type Output = Self;
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor(self, other: Self) -> Self {
                self.symmetric_difference(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitXorAssign for
            InternalBitFlags {
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor_assign(&mut self, other: Self) { self.toggle(other); }
        }
        impl ::bitflags::__private::core::ops::BitAnd for InternalBitFlags {
            type Output = Self;
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand(self, other: Self) -> Self { self.intersection(other) }
        }
        impl ::bitflags::__private::core::ops::BitAndAssign for
            InternalBitFlags {
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand_assign(&mut self, other: Self) {
                *self =
                    Self::from_bits_retain(self.bits()).intersection(other);
            }
        }
        impl ::bitflags::__private::core::ops::Sub for InternalBitFlags {
            type Output = Self;
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub(self, other: Self) -> Self { self.difference(other) }
        }
        impl ::bitflags::__private::core::ops::SubAssign for InternalBitFlags
            {
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub_assign(&mut self, other: Self) { self.remove(other); }
        }
        impl ::bitflags::__private::core::ops::Not for InternalBitFlags {
            type Output = Self;
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            fn not(self) -> Self { self.complement() }
        }
        impl ::bitflags::__private::core::iter::Extend<InternalBitFlags> for
            InternalBitFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn extend<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(&mut self, iterator: T) {
                for item in iterator { self.insert(item) }
            }
        }
        impl ::bitflags::__private::core::iter::FromIterator<InternalBitFlags>
            for InternalBitFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn from_iter<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(iterator: T) -> Self {
                use ::bitflags::__private::core::iter::Extend;
                let mut result = Self::empty();
                result.extend(iterator);
                result
            }
        }
        impl InternalBitFlags {
            /// Yield a set of contained flags values.
            ///
            /// Each yielded flags value will correspond to a defined named flag. Any unknown bits
            /// will be yielded together as a final flags value.
            #[inline]
            pub const fn iter(&self) -> ::bitflags::iter::Iter<Restrictions> {
                ::bitflags::iter::Iter::__private_const_new(<Restrictions as
                        ::bitflags::Flags>::FLAGS,
                    Restrictions::from_bits_retain(self.bits()),
                    Restrictions::from_bits_retain(self.bits()))
            }
            /// Yield a set of contained named flags values.
            ///
            /// This method is like [`iter`](#method.iter), except only yields bits in contained named flags.
            /// Any unknown bits, or bits not corresponding to a contained flag will not be yielded.
            #[inline]
            pub const fn iter_names(&self)
                -> ::bitflags::iter::IterNames<Restrictions> {
                ::bitflags::iter::IterNames::__private_const_new(<Restrictions
                        as ::bitflags::Flags>::FLAGS,
                    Restrictions::from_bits_retain(self.bits()),
                    Restrictions::from_bits_retain(self.bits()))
            }
        }
        impl ::bitflags::__private::core::iter::IntoIterator for
            InternalBitFlags {
            type Item = Restrictions;
            type IntoIter = ::bitflags::iter::Iter<Restrictions>;
            fn into_iter(self) -> Self::IntoIter { self.iter() }
        }
        impl InternalBitFlags {
            /// Returns a mutable reference to the raw value of the flags currently stored.
            #[inline]
            pub fn bits_mut(&mut self) -> &mut u8 { &mut self.0 }
        }
        #[allow(dead_code, deprecated, unused_attributes)]
        impl Restrictions {
            /// Get a flags value with all bits unset.
            #[inline]
            pub const fn empty() -> Self { Self(InternalBitFlags::empty()) }
            /// Get a flags value with all known bits set.
            #[inline]
            pub const fn all() -> Self { Self(InternalBitFlags::all()) }
            /// Get the underlying bits value.
            ///
            /// The returned value is exactly the bits set in this flags value.
            #[inline]
            pub const fn bits(&self) -> u8 { self.0.bits() }
            /// Convert from a bits value.
            ///
            /// This method will return `None` if any unknown bits are set.
            #[inline]
            pub const fn from_bits(bits: u8)
                -> ::bitflags::__private::core::option::Option<Self> {
                match InternalBitFlags::from_bits(bits) {
                    ::bitflags::__private::core::option::Option::Some(bits) =>
                        ::bitflags::__private::core::option::Option::Some(Self(bits)),
                    ::bitflags::__private::core::option::Option::None =>
                        ::bitflags::__private::core::option::Option::None,
                }
            }
            /// Convert from a bits value, unsetting any unknown bits.
            #[inline]
            pub const fn from_bits_truncate(bits: u8) -> Self {
                Self(InternalBitFlags::from_bits_truncate(bits))
            }
            /// Convert from a bits value exactly.
            #[inline]
            pub const fn from_bits_retain(bits: u8) -> Self {
                Self(InternalBitFlags::from_bits_retain(bits))
            }
            /// Get a flags value with the bits of a flag with the given name set.
            ///
            /// This method will return `None` if `name` is empty or doesn't
            /// correspond to any named flag.
            #[inline]
            pub fn from_name(name: &str)
                -> ::bitflags::__private::core::option::Option<Self> {
                match InternalBitFlags::from_name(name) {
                    ::bitflags::__private::core::option::Option::Some(bits) =>
                        ::bitflags::__private::core::option::Option::Some(Self(bits)),
                    ::bitflags::__private::core::option::Option::None =>
                        ::bitflags::__private::core::option::Option::None,
                }
            }
            /// Whether all bits in this flags value are unset.
            #[inline]
            pub const fn is_empty(&self) -> bool { self.0.is_empty() }
            /// Whether all known bits in this flags value are set.
            #[inline]
            pub const fn is_all(&self) -> bool { self.0.is_all() }
            /// Whether any set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn intersects(&self, other: Self) -> bool {
                self.0.intersects(other.0)
            }
            /// Whether all set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn contains(&self, other: Self) -> bool {
                self.0.contains(other.0)
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            pub fn insert(&mut self, other: Self) { self.0.insert(other.0) }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `remove` won't truncate `other`, but the `!` operator will.
            #[inline]
            pub fn remove(&mut self, other: Self) { self.0.remove(other.0) }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            pub fn toggle(&mut self, other: Self) { self.0.toggle(other.0) }
            /// Call `insert` when `value` is `true` or `remove` when `value` is `false`.
            #[inline]
            pub fn set(&mut self, other: Self, value: bool) {
                self.0.set(other.0, value)
            }
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn intersection(self, other: Self) -> Self {
                Self(self.0.intersection(other.0))
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn union(self, other: Self) -> Self {
                Self(self.0.union(other.0))
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            #[must_use]
            pub const fn difference(self, other: Self) -> Self {
                Self(self.0.difference(other.0))
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn symmetric_difference(self, other: Self) -> Self {
                Self(self.0.symmetric_difference(other.0))
            }
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            #[must_use]
            pub const fn complement(self) -> Self {
                Self(self.0.complement())
            }
        }
        impl ::bitflags::__private::core::fmt::Binary for Restrictions {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Binary::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::Octal for Restrictions {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Octal::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::LowerHex for Restrictions {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::LowerHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::UpperHex for Restrictions {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::UpperHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::ops::BitOr for Restrictions {
            type Output = Self;
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor(self, other: Restrictions) -> Self { self.union(other) }
        }
        impl ::bitflags::__private::core::ops::BitOrAssign for Restrictions {
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor_assign(&mut self, other: Self) { self.insert(other); }
        }
        impl ::bitflags::__private::core::ops::BitXor for Restrictions {
            type Output = Self;
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor(self, other: Self) -> Self {
                self.symmetric_difference(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitXorAssign for Restrictions {
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor_assign(&mut self, other: Self) { self.toggle(other); }
        }
        impl ::bitflags::__private::core::ops::BitAnd for Restrictions {
            type Output = Self;
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand(self, other: Self) -> Self { self.intersection(other) }
        }
        impl ::bitflags::__private::core::ops::BitAndAssign for Restrictions {
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand_assign(&mut self, other: Self) {
                *self =
                    Self::from_bits_retain(self.bits()).intersection(other);
            }
        }
        impl ::bitflags::__private::core::ops::Sub for Restrictions {
            type Output = Self;
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub(self, other: Self) -> Self { self.difference(other) }
        }
        impl ::bitflags::__private::core::ops::SubAssign for Restrictions {
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub_assign(&mut self, other: Self) { self.remove(other); }
        }
        impl ::bitflags::__private::core::ops::Not for Restrictions {
            type Output = Self;
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            fn not(self) -> Self { self.complement() }
        }
        impl ::bitflags::__private::core::iter::Extend<Restrictions> for
            Restrictions {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn extend<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(&mut self, iterator: T) {
                for item in iterator { self.insert(item) }
            }
        }
        impl ::bitflags::__private::core::iter::FromIterator<Restrictions> for
            Restrictions {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn from_iter<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(iterator: T) -> Self {
                use ::bitflags::__private::core::iter::Extend;
                let mut result = Self::empty();
                result.extend(iterator);
                result
            }
        }
        impl Restrictions {
            /// Yield a set of contained flags values.
            ///
            /// Each yielded flags value will correspond to a defined named flag. Any unknown bits
            /// will be yielded together as a final flags value.
            #[inline]
            pub const fn iter(&self) -> ::bitflags::iter::Iter<Restrictions> {
                ::bitflags::iter::Iter::__private_const_new(<Restrictions as
                        ::bitflags::Flags>::FLAGS,
                    Restrictions::from_bits_retain(self.bits()),
                    Restrictions::from_bits_retain(self.bits()))
            }
            /// Yield a set of contained named flags values.
            ///
            /// This method is like [`iter`](#method.iter), except only yields bits in contained named flags.
            /// Any unknown bits, or bits not corresponding to a contained flag will not be yielded.
            #[inline]
            pub const fn iter_names(&self)
                -> ::bitflags::iter::IterNames<Restrictions> {
                ::bitflags::iter::IterNames::__private_const_new(<Restrictions
                        as ::bitflags::Flags>::FLAGS,
                    Restrictions::from_bits_retain(self.bits()),
                    Restrictions::from_bits_retain(self.bits()))
            }
        }
        impl ::bitflags::__private::core::iter::IntoIterator for Restrictions
            {
            type Item = Restrictions;
            type IntoIter = ::bitflags::iter::Iter<Restrictions>;
            fn into_iter(self) -> Self::IntoIter { self.iter() }
        }
    };Clone, #[automatically_derived]
impl ::core::marker::Copy for Restrictions { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for Restrictions {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Restrictions",
            &&self.0)
    }
}Debug)]
76    struct Restrictions: u8 {
77        /// Restricts expressions for use in statement position.
78        ///
79        /// When expressions are used in various places, like statements or
80        /// match arms, this is used to stop parsing once certain tokens are
81        /// reached.
82        ///
83        /// For example, `if true {} & 1` with `STMT_EXPR` in effect is parsed
84        /// as two separate expression statements (`if` and a reference to 1).
85        /// Otherwise it is parsed as a bitwise AND where `if` is on the left
86        /// and 1 is on the right.
87        const STMT_EXPR         = 1 << 0;
88        /// Do not allow struct literals.
89        ///
90        /// There are several places in the grammar where we don't want to
91        /// allow struct literals because they can require lookahead, or
92        /// otherwise could be ambiguous or cause confusion. For example,
93        /// `if Foo {} {}` isn't clear if it is `Foo{}` struct literal, or
94        /// just `Foo` is the condition, followed by a consequent block,
95        /// followed by an empty block.
96        ///
97        /// See [RFC 92](https://rust-lang.github.io/rfcs/0092-struct-grammar.html).
98        const NO_STRUCT_LITERAL = 1 << 1;
99        /// Used to provide better error messages for const generic arguments.
100        ///
101        /// An un-braced const generic argument is limited to a very small
102        /// subset of expressions. This is used to detect the situation where
103        /// an expression outside of that subset is used, and to suggest to
104        /// wrap the expression in braces.
105        const CONST_EXPR        = 1 << 2;
106        /// Allows `let` expressions.
107        ///
108        /// `let pattern = scrutinee` is parsed as an expression, but it is
109        /// only allowed in let chains (`if` and `while` conditions).
110        /// Otherwise it is not an expression (note that `let` in statement
111        /// positions is treated as a `StmtKind::Let` statement, which has a
112        /// slightly different grammar).
113        const ALLOW_LET         = 1 << 3;
114        /// Used to detect a missing `=>` in a match guard.
115        ///
116        /// This is used for error handling in a match guard to give a better
117        /// error message if the `=>` is missing. It is set when parsing the
118        /// guard expression.
119        const IN_IF_GUARD       = 1 << 4;
120        /// Used to detect the incorrect use of expressions in patterns.
121        ///
122        /// This is used for error handling while parsing a pattern. During
123        /// error recovery, this will be set to try to parse the pattern as an
124        /// expression, but halts parsing the expression when reaching certain
125        /// tokens like `=`.
126        const IS_PAT            = 1 << 5;
127    }
128}
129
130#[derive(#[automatically_derived]
impl ::core::clone::Clone for SemiColonMode {
    #[inline]
    fn clone(&self) -> SemiColonMode { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for SemiColonMode { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for SemiColonMode {
    #[inline]
    fn eq(&self, other: &SemiColonMode) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::fmt::Debug for SemiColonMode {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                SemiColonMode::Break => "Break",
                SemiColonMode::Ignore => "Ignore",
                SemiColonMode::Comma => "Comma",
            })
    }
}Debug)]
131enum SemiColonMode {
132    Break,
133    Ignore,
134    Comma,
135}
136
137#[derive(#[automatically_derived]
impl ::core::clone::Clone for BlockMode {
    #[inline]
    fn clone(&self) -> BlockMode { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for BlockMode { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for BlockMode {
    #[inline]
    fn eq(&self, other: &BlockMode) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::fmt::Debug for BlockMode {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                BlockMode::Break => "Break",
                BlockMode::Ignore => "Ignore",
            })
    }
}Debug)]
138enum BlockMode {
139    Break,
140    Ignore,
141}
142
143/// Whether or not we should force collection of tokens for an AST node,
144/// regardless of whether or not it has attributes
145#[derive(#[automatically_derived]
impl ::core::clone::Clone for ForceCollect {
    #[inline]
    fn clone(&self) -> ForceCollect { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for ForceCollect { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for ForceCollect {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                ForceCollect::Yes => "Yes",
                ForceCollect::No => "No",
            })
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for ForceCollect {
    #[inline]
    fn eq(&self, other: &ForceCollect) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq)]
146pub enum ForceCollect {
147    Yes,
148    No,
149}
150
151/// Whether to accept `const { ... }` as a shorthand for `const _: () = const { ... }`.
152#[derive(#[automatically_derived]
impl ::core::clone::Clone for AllowConstBlockItems {
    #[inline]
    fn clone(&self) -> AllowConstBlockItems { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for AllowConstBlockItems { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for AllowConstBlockItems {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                AllowConstBlockItems::Yes => "Yes",
                AllowConstBlockItems::No => "No",
                AllowConstBlockItems::DoesNotMatter => "DoesNotMatter",
            })
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for AllowConstBlockItems {
    #[inline]
    fn eq(&self, other: &AllowConstBlockItems) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for AllowConstBlockItems {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq)]
153pub enum AllowConstBlockItems {
154    Yes,
155    No,
156    DoesNotMatter,
157}
158
159/// If the next tokens are ill-formed `$ty::` recover them as `<$ty>::`.
160#[macro_export]
161macro_rules! maybe_recover_from_interpolated_ty_qpath {
162    ($self: expr, $allow_qpath_recovery: expr) => {
163        if $allow_qpath_recovery
164            && $self.may_recover()
165            && let Some(mv_kind) = $self.token.is_metavar_seq()
166            && let token::MetaVarKind::Ty { .. } = mv_kind
167            && $self.check_noexpect_past_close_delim(&token::PathSep)
168        {
169            // Reparse the type, then move to recovery.
170            let ty = $self
171                .eat_metavar_seq(mv_kind, |this| this.parse_ty_no_question_mark_recover())
172                .expect("metavar seq ty");
173
174            return $self.maybe_recover_from_bad_qpath_stage_2($self.prev_token.span, ty);
175        }
176    };
177}
178
179#[derive(#[automatically_derived]
impl ::core::clone::Clone for Recovery {
    #[inline]
    fn clone(&self) -> Recovery { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for Recovery { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for Recovery {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                Recovery::Allowed => "Allowed",
                Recovery::Forbidden => "Forbidden",
            })
    }
}Debug)]
180pub enum Recovery {
181    Allowed,
182    Forbidden,
183}
184
185#[derive(#[automatically_derived]
impl<'a> ::core::clone::Clone for Parser<'a> {
    #[inline]
    fn clone(&self) -> Parser<'a> {
        Parser {
            psess: ::core::clone::Clone::clone(&self.psess),
            token: ::core::clone::Clone::clone(&self.token),
            token_spacing: ::core::clone::Clone::clone(&self.token_spacing),
            prev_token: ::core::clone::Clone::clone(&self.prev_token),
            capture_cfg: ::core::clone::Clone::clone(&self.capture_cfg),
            restrictions: ::core::clone::Clone::clone(&self.restrictions),
            expected_token_types: ::core::clone::Clone::clone(&self.expected_token_types),
            token_cursor: ::core::clone::Clone::clone(&self.token_cursor),
            num_bump_calls: ::core::clone::Clone::clone(&self.num_bump_calls),
            break_last_token: ::core::clone::Clone::clone(&self.break_last_token),
            unmatched_angle_bracket_count: ::core::clone::Clone::clone(&self.unmatched_angle_bracket_count),
            angle_bracket_nesting: ::core::clone::Clone::clone(&self.angle_bracket_nesting),
            parsing_generics: ::core::clone::Clone::clone(&self.parsing_generics),
            last_unexpected_token_span: ::core::clone::Clone::clone(&self.last_unexpected_token_span),
            subparser_name: ::core::clone::Clone::clone(&self.subparser_name),
            capture_state: ::core::clone::Clone::clone(&self.capture_state),
            current_closure: ::core::clone::Clone::clone(&self.current_closure),
            recovery: ::core::clone::Clone::clone(&self.recovery),
            in_fn_body: ::core::clone::Clone::clone(&self.in_fn_body),
            fn_body_missing_semi_guar: ::core::clone::Clone::clone(&self.fn_body_missing_semi_guar),
        }
    }
}Clone)]
186pub struct Parser<'a> {
187    pub psess: &'a ParseSess,
188    /// The current token.
189    pub token: Token = Token::dummy(),
190    /// The spacing for the current token.
191    token_spacing: Spacing = Spacing::Alone,
192    /// The previous token.
193    pub prev_token: Token = Token::dummy(),
194    pub capture_cfg: bool = false,
195    restrictions: Restrictions = Restrictions::empty(),
196    expected_token_types: TokenTypeSet = TokenTypeSet::new(),
197    token_cursor: TokenCursor,
198    // The number of calls to `bump`, i.e. the position in the token stream.
199    num_bump_calls: u32 = 0,
200    // During parsing we may sometimes need to "unglue" a glued token into two
201    // or three component tokens (e.g. `>>` into `>` and `>`, or `>>=` into `>`
202    // and `>` and `=`), so the parser can consume them one at a time. This
203    // process bypasses the normal capturing mechanism (e.g. `num_bump_calls`
204    // will not be incremented), since the "unglued" tokens due not exist in
205    // the original `TokenStream`.
206    //
207    // If we end up consuming all the component tokens, this is not an issue,
208    // because we'll end up capturing the single "glued" token.
209    //
210    // However, sometimes we may want to capture not all of the original
211    // token. For example, capturing the `Vec<u8>` in `Option<Vec<u8>>`
212    // requires us to unglue the trailing `>>` token. The `break_last_token`
213    // field is used to track these tokens. They get appended to the captured
214    // stream when we evaluate a `LazyAttrTokenStream`.
215    //
216    // This value is always 0, 1, or 2. It can only reach 2 when splitting
217    // `>>=` or `<<=`.
218    break_last_token: u32 = 0,
219    /// This field is used to keep track of how many left angle brackets we have seen. This is
220    /// required in order to detect extra leading left angle brackets (`<` characters) and error
221    /// appropriately.
222    ///
223    /// See the comments in the `parse_path_segment` function for more details.
224    unmatched_angle_bracket_count: u16 = 0,
225    angle_bracket_nesting: u16 = 0,
226    /// Keep track of when we're within `<...>` for proper error recovery.
227    parsing_generics: bool = false,
228
229    last_unexpected_token_span: Option<Span> = None,
230    /// If present, this `Parser` is not parsing Rust code but rather a macro call.
231    subparser_name: Option<&'static str>,
232    capture_state: CaptureState,
233    /// This allows us to recover when the user forget to add braces around
234    /// multiple statements in the closure body.
235    current_closure: Option<ClosureSpans> = None,
236    /// Whether the parser is allowed to do recovery.
237    /// This is disabled when parsing macro arguments, see #103534
238    recovery: Recovery = Recovery::Allowed,
239    /// Whether we're parsing a function body.
240    in_fn_body: bool = false,
241    /// Whether we have detected a missing semicolon in function body.
242    pub fn_body_missing_semi_guar: Option<ErrorGuaranteed> = None,
243}
244
245// This type is used a lot, e.g. it's cloned when matching many declarative macro rules with
246// nonterminals. Make sure it doesn't unintentionally get bigger. We only check a few arches
247// though, because `TokenTypeSet(u128)` alignment varies on others, changing the total size.
248#[cfg(all(target_pointer_width = "64", any(target_arch = "aarch64", target_arch = "x86_64")))]
249const _: [(); 288] = [(); ::std::mem::size_of::<Parser<'_>>()];rustc_data_structures::static_assert_size!(Parser<'_>, 288);
250
251/// Stores span information about a closure.
252#[derive(#[automatically_derived]
impl ::core::clone::Clone for ClosureSpans {
    #[inline]
    fn clone(&self) -> ClosureSpans {
        ClosureSpans {
            whole_closure: ::core::clone::Clone::clone(&self.whole_closure),
            closing_pipe: ::core::clone::Clone::clone(&self.closing_pipe),
            body: ::core::clone::Clone::clone(&self.body),
        }
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for ClosureSpans {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f, "ClosureSpans",
            "whole_closure", &self.whole_closure, "closing_pipe",
            &self.closing_pipe, "body", &&self.body)
    }
}Debug)]
253struct ClosureSpans {
254    whole_closure: Span,
255    closing_pipe: Span,
256    body: Span,
257}
258
259/// Controls how we capture tokens. Capturing can be expensive,
260/// so we try to avoid performing capturing in cases where
261/// we will never need an `AttrTokenStream`.
262#[derive(#[automatically_derived]
impl ::core::marker::Copy for Capturing { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Capturing {
    #[inline]
    fn clone(&self) -> Capturing { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Capturing {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self { Capturing::No => "No", Capturing::Yes => "Yes", })
    }
}Debug)]
263enum Capturing {
264    /// We aren't performing any capturing - this is the default mode.
265    No,
266    /// We are capturing tokens
267    Yes,
268}
269
270// This state is used by `Parser::collect_tokens`.
271#[derive(#[automatically_derived]
impl ::core::clone::Clone for CaptureState {
    #[inline]
    fn clone(&self) -> CaptureState {
        CaptureState {
            capturing: ::core::clone::Clone::clone(&self.capturing),
            parser_replacements: ::core::clone::Clone::clone(&self.parser_replacements),
            inner_attr_parser_ranges: ::core::clone::Clone::clone(&self.inner_attr_parser_ranges),
            seen_attrs: ::core::clone::Clone::clone(&self.seen_attrs),
        }
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for CaptureState {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field4_finish(f, "CaptureState",
            "capturing", &self.capturing, "parser_replacements",
            &self.parser_replacements, "inner_attr_parser_ranges",
            &self.inner_attr_parser_ranges, "seen_attrs", &&self.seen_attrs)
    }
}Debug)]
272struct CaptureState {
273    capturing: Capturing,
274    parser_replacements: Vec<ParserReplacement>,
275    inner_attr_parser_ranges: FxHashMap<AttrId, ParserRange>,
276    // `IntervalSet` is good for perf because attrs are mostly added to this
277    // set in contiguous ranges.
278    seen_attrs: IntervalSet<AttrId>,
279}
280
281/// A sequence separator.
282#[derive(#[automatically_derived]
impl ::core::fmt::Debug for SeqSep {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f, "SeqSep", "sep",
            &self.sep, "trailing_sep_allowed", &&self.trailing_sep_allowed)
    }
}Debug)]
283struct SeqSep {
284    /// The separator token.
285    sep: Option<ExpTokenPair>,
286    /// `true` if a trailing separator is allowed.
287    trailing_sep_allowed: bool,
288}
289
290impl SeqSep {
291    fn trailing_allowed(sep: ExpTokenPair) -> SeqSep {
292        SeqSep { sep: Some(sep), trailing_sep_allowed: true }
293    }
294
295    fn none() -> SeqSep {
296        SeqSep { sep: None, trailing_sep_allowed: false }
297    }
298}
299
300/// Whether parsing `impl` or `mut` restrictions.
301#[derive(#[automatically_derived]
impl ::core::clone::Clone for ParsingRestrictionKind {
    #[inline]
    fn clone(&self) -> ParsingRestrictionKind { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for ParsingRestrictionKind { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for ParsingRestrictionKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                ParsingRestrictionKind::Impl => "Impl",
                ParsingRestrictionKind::Mut => "Mut",
            })
    }
}Debug)]
302enum ParsingRestrictionKind {
303    Impl,
304    Mut,
305}
306
307#[derive(#[automatically_derived]
impl ::core::fmt::Debug for FollowedByType {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                FollowedByType::Yes => "Yes",
                FollowedByType::No => "No",
            })
    }
}Debug)]
308pub enum FollowedByType {
309    Yes,
310    No,
311}
312
313#[derive(#[automatically_derived]
impl ::core::marker::Copy for Trailing { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Trailing {
    #[inline]
    fn clone(&self) -> Trailing { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Trailing {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self { Trailing::No => "No", Trailing::Yes => "Yes", })
    }
}Debug)]
314pub enum Trailing {
315    No,
316    Yes,
317}
318
319impl From<bool> for Trailing {
320    fn from(b: bool) -> Trailing {
321        if b { Trailing::Yes } else { Trailing::No }
322    }
323}
324
325pub fn token_descr(token: &Token) -> String {
326    let s = pprust::token_to_string(token).to_string();
327
328    match (TokenDescription::from_token(token), &token.kind) {
329        (Some(TokenDescription::ReservedIdentifier), _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("reserved identifier `{0}`", s))
    })format!("reserved identifier `{s}`"),
330        (Some(TokenDescription::Keyword), _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("keyword `{0}`", s))
    })format!("keyword `{s}`"),
331        (Some(TokenDescription::ReservedKeyword), _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("reserved keyword `{0}`", s))
    })format!("reserved keyword `{s}`"),
332        (Some(TokenDescription::DocComment), _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("doc comment `{0}`", s))
    })format!("doc comment `{s}`"),
333        // Deliberately doesn't print `s`, which is empty.
334        (Some(TokenDescription::MetaVar(kind)), _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` metavariable", kind))
    })format!("`{kind}` metavariable"),
335        (None, TokenKind::NtIdent(..)) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("identifier `{0}`", s))
    })format!("identifier `{s}`"),
336        (None, TokenKind::NtLifetime(..)) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("lifetime `{0}`", s))
    })format!("lifetime `{s}`"),
337        (None, _) => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`", s))
    })format!("`{s}`"),
338    }
339}
340
341impl<'a> Parser<'a> {
342    pub fn new(
343        psess: &'a ParseSess,
344        stream: TokenStream,
345        subparser_name: Option<&'static str>,
346    ) -> Self {
347        let mut parser = Parser {
348            psess,
349            token_cursor: TokenCursor { curr: TokenTreeCursor::new(stream), stack: Vec::new() },
350            subparser_name,
351            capture_state: CaptureState {
352                capturing: Capturing::No,
353                parser_replacements: Vec::new(),
354                inner_attr_parser_ranges: Default::default(),
355                seen_attrs: IntervalSet::new(u32::MAX as usize),
356            },
357            ..
358        };
359
360        // Make parser point to the first token.
361        parser.bump();
362
363        // Change this from 1 back to 0 after the bump. This eases debugging of
364        // `Parser::collect_tokens` because 0-indexed token positions are nicer
365        // than 1-indexed token positions.
366        parser.num_bump_calls = 0;
367
368        parser
369    }
370
371    #[inline]
372    pub fn recovery(mut self, recovery: Recovery) -> Self {
373        self.recovery = recovery;
374        self
375    }
376
377    #[inline]
378    fn with_recovery<T>(&mut self, recovery: Recovery, f: impl FnOnce(&mut Self) -> T) -> T {
379        let old = mem::replace(&mut self.recovery, recovery);
380        let res = f(self);
381        self.recovery = old;
382        res
383    }
384
385    /// Whether the parser is allowed to recover from broken code.
386    ///
387    /// If this returns false, recovering broken code into valid code (especially if this recovery does lookahead)
388    /// is not allowed. All recovery done by the parser must be gated behind this check.
389    ///
390    /// Technically, this only needs to restrict eager recovery by doing lookahead at more tokens.
391    /// But making the distinction is very subtle, and simply forbidding all recovery is a lot simpler to uphold.
392    #[inline]
393    fn may_recover(&self) -> bool {
394        #[allow(non_exhaustive_omitted_patterns)] match self.recovery {
    Recovery::Allowed => true,
    _ => false,
}matches!(self.recovery, Recovery::Allowed)
395    }
396
397    /// Version of [`unexpected`](Parser::unexpected) that "returns" any type in the `Ok`
398    /// (both those functions never return "Ok", and so can lie like that in the type).
399    pub fn unexpected_any<T>(&mut self) -> PResult<'a, T> {
400        match self.expect_one_of(&[], &[]) {
401            Err(e) => Err(e),
402            // We can get `Ok(true)` from `recover_closing_delimiter`
403            // which is called in `expected_one_of_not_found`.
404            Ok(_) => FatalError.raise(),
405        }
406    }
407
408    pub fn unexpected(&mut self) -> PResult<'a, ()> {
409        self.unexpected_any()
410    }
411
412    /// Expects and consumes the token `t`. Signals an error if the next token is not `t`.
413    pub fn expect(&mut self, exp: ExpTokenPair) -> PResult<'a, Recovered> {
414        if self.expected_token_types.is_empty() {
415            if self.token == exp.tok {
416                self.bump();
417                Ok(Recovered::No)
418            } else {
419                Err(self.unexpected_err(&exp.tok))
420            }
421        } else {
422            self.expect_one_of(slice::from_ref(&exp), &[])
423        }
424    }
425
426    /// Expect next token to be edible or inedible token. If edible,
427    /// then consume it; if inedible, then return without consuming
428    /// anything. Signal a fatal error if next token is unexpected.
429    fn expect_one_of(
430        &mut self,
431        edible: &[ExpTokenPair],
432        inedible: &[ExpTokenPair],
433    ) -> PResult<'a, Recovered> {
434        if edible.iter().any(|exp| exp.tok == self.token.kind) {
435            self.bump();
436            Ok(Recovered::No)
437        } else if inedible.iter().any(|exp| exp.tok == self.token.kind) {
438            // leave it in the input
439            Ok(Recovered::No)
440        } else if self.token != token::Eof
441            && self.last_unexpected_token_span == Some(self.token.span)
442        {
443            FatalError.raise();
444        } else {
445            self.expected_one_of_not_found(edible, inedible)
446                .map(|error_guaranteed| Recovered::Yes(error_guaranteed))
447        }
448    }
449
450    // Public for rustfmt usage.
451    pub fn parse_ident(&mut self) -> PResult<'a, Ident> {
452        self.parse_ident_common(self.may_recover())
453    }
454
455    pub(crate) fn parse_ident_common(&mut self, recover: bool) -> PResult<'a, Ident> {
456        let (ident, is_raw) = self.ident_or_err(recover)?;
457
458        if is_raw == IdentIsRaw::No && ident.is_reserved() {
459            let err = self.expected_ident_found_err();
460            if recover {
461                err.emit();
462            } else {
463                return Err(err);
464            }
465        }
466        self.bump();
467        Ok(ident)
468    }
469
470    fn ident_or_err(&mut self, recover: bool) -> PResult<'a, (Ident, IdentIsRaw)> {
471        match self.token.ident() {
472            Some(ident) => Ok(ident),
473            None => self.expected_ident_found(recover),
474        }
475    }
476
477    /// Checks if the next token is `tok`, and returns `true` if so.
478    ///
479    /// This method will automatically add `tok` to `expected_token_types` if `tok` is not
480    /// encountered.
481    #[inline]
482    pub fn check(&mut self, exp: ExpTokenPair) -> bool {
483        let is_present = self.token == exp.tok;
484        if !is_present {
485            self.expected_token_types.insert(exp.token_type);
486        }
487        is_present
488    }
489
490    #[inline]
491    #[must_use]
492    fn check_noexpect(&self, tok: &TokenKind) -> bool {
493        self.token == *tok
494    }
495
496    // Check the first token after the delimiter that closes the current
497    // delimited sequence. (Panics if used in the outermost token stream, which
498    // has no delimiters.) It uses a clone of the relevant tree cursor to skip
499    // past the entire `TokenTree::Delimited` in a single step, avoiding the
500    // need for unbounded token lookahead.
501    //
502    // Primarily used when `self.token` matches `OpenInvisible(_))`, to look
503    // ahead through the current metavar expansion.
504    fn check_noexpect_past_close_delim(&self, tok: &TokenKind) -> bool {
505        let mut tree_cursor = self.token_cursor.stack.last().unwrap().clone();
506        tree_cursor.bump();
507        #[allow(non_exhaustive_omitted_patterns)] match tree_cursor.curr() {
    Some(TokenTree::Token(token::Token { kind, .. }, _)) if kind == tok =>
        true,
    _ => false,
}matches!(
508            tree_cursor.curr(),
509            Some(TokenTree::Token(token::Token { kind, .. }, _)) if kind == tok
510        )
511    }
512
513    /// Consumes a token 'tok' if it exists. Returns whether the given token was present.
514    ///
515    /// the main purpose of this function is to reduce the cluttering of the suggestions list
516    /// which using the normal eat method could introduce in some cases.
517    #[inline]
518    #[must_use]
519    fn eat_noexpect(&mut self, tok: &TokenKind) -> bool {
520        let is_present = self.check_noexpect(tok);
521        if is_present {
522            self.bump()
523        }
524        is_present
525    }
526
527    /// Consumes a token 'tok' if it exists. Returns whether the given token was present.
528    #[inline]
529    #[must_use]
530    pub fn eat(&mut self, exp: ExpTokenPair) -> bool {
531        let is_present = self.check(exp);
532        if is_present {
533            self.bump()
534        }
535        is_present
536    }
537
538    /// If the next token is the given keyword, returns `true` without eating it.
539    /// An expectation is also added for diagnostics purposes.
540    #[inline]
541    #[must_use]
542    fn check_keyword(&mut self, exp: ExpKeywordPair) -> bool {
543        let is_keyword = self.token.is_keyword(exp.kw);
544        if !is_keyword {
545            self.expected_token_types.insert(exp.token_type);
546        }
547        is_keyword
548    }
549
550    #[inline]
551    #[must_use]
552    fn check_keyword_case(&mut self, exp: ExpKeywordPair, case: Case) -> bool {
553        if self.check_keyword(exp) {
554            true
555        } else if case == Case::Insensitive
556            && let Some((ident, IdentIsRaw::No)) = self.token.ident()
557            // Do an ASCII case-insensitive match, because all keywords are ASCII.
558            && ident.as_str().eq_ignore_ascii_case(exp.kw.as_str())
559        {
560            true
561        } else {
562            false
563        }
564    }
565
566    /// If the next token is the given keyword, eats it and returns `true`.
567    /// Otherwise, returns `false`. An expectation is also added for diagnostics purposes.
568    // Public for rustc_builtin_macros and rustfmt usage.
569    #[inline]
570    #[must_use]
571    pub fn eat_keyword(&mut self, exp: ExpKeywordPair) -> bool {
572        let is_keyword = self.check_keyword(exp);
573        if is_keyword {
574            self.bump();
575        }
576        is_keyword
577    }
578
579    /// Eats a keyword, optionally ignoring the case.
580    /// If the case differs (and is ignored) an error is issued.
581    /// This is useful for recovery.
582    #[inline]
583    #[must_use]
584    fn eat_keyword_case(&mut self, exp: ExpKeywordPair, case: Case) -> bool {
585        if self.eat_keyword(exp) {
586            true
587        } else if case == Case::Insensitive
588            && let Some((ident, IdentIsRaw::No)) = self.token.ident()
589            // Do an ASCII case-insensitive match, because all keywords are ASCII.
590            && ident.as_str().eq_ignore_ascii_case(exp.kw.as_str())
591        {
592            let kw = exp.kw.as_str();
593            let is_upper = kw.chars().all(char::is_uppercase);
594            let is_lower = kw.chars().all(char::is_lowercase);
595
596            let case = match (is_upper, is_lower) {
597                (true, true) => {
598                    {
    ::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
            format_args!("keyword that is both fully upper- and fully lowercase")));
}unreachable!("keyword that is both fully upper- and fully lowercase")
599                }
600                (true, false) => errors::Case::Upper,
601                (false, true) => errors::Case::Lower,
602                (false, false) => errors::Case::Mixed,
603            };
604
605            self.dcx().emit_err(errors::KwBadCase { span: ident.span, kw, case });
606            self.bump();
607            true
608        } else {
609            false
610        }
611    }
612
613    /// If the next token is the given keyword, eats it and returns `true`.
614    /// Otherwise, returns `false`. No expectation is added.
615    // Public for rustc_builtin_macros usage.
616    #[inline]
617    #[must_use]
618    pub fn eat_keyword_noexpect(&mut self, kw: Symbol) -> bool {
619        let is_keyword = self.token.is_keyword(kw);
620        if is_keyword {
621            self.bump();
622        }
623        is_keyword
624    }
625
626    /// If the given word is not a keyword, signals an error.
627    /// If the next token is not the given word, signals an error.
628    /// Otherwise, eats it.
629    pub fn expect_keyword(&mut self, exp: ExpKeywordPair) -> PResult<'a, ()> {
630        if !self.eat_keyword(exp) { self.unexpected() } else { Ok(()) }
631    }
632
633    /// Consume a sequence produced by a metavar expansion, if present.
634    pub fn eat_metavar_seq<T>(
635        &mut self,
636        mv_kind: MetaVarKind,
637        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
638    ) -> Option<T> {
639        self.eat_metavar_seq_with_matcher(|mvk| mvk == mv_kind, f)
640    }
641
642    /// A slightly more general form of `eat_metavar_seq`, for use with the
643    /// `MetaVarKind` variants that have parameters, where an exact match isn't
644    /// desired.
645    fn eat_metavar_seq_with_matcher<T>(
646        &mut self,
647        match_mv_kind: impl Fn(MetaVarKind) -> bool,
648        mut f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
649    ) -> Option<T> {
650        if let token::OpenInvisible(InvisibleOrigin::MetaVar(mv_kind)) = self.token.kind
651            && match_mv_kind(mv_kind)
652        {
653            self.bump();
654
655            // Recovery is disabled when parsing macro arguments, so it must
656            // also be disabled when reparsing pasted macro arguments,
657            // otherwise we get inconsistent results (e.g. #137874).
658            let res = self.with_recovery(Recovery::Forbidden, |this| f(this));
659
660            let res = match res {
661                Ok(res) => res,
662                Err(err) => {
663                    // This can occur in unusual error cases, e.g. #139445.
664                    err.delay_as_bug();
665                    return None;
666                }
667            };
668
669            if let token::CloseInvisible(InvisibleOrigin::MetaVar(mv_kind)) = self.token.kind
670                && match_mv_kind(mv_kind)
671            {
672                self.bump();
673                Some(res)
674            } else {
675                // This can occur when invalid syntax is passed to a decl macro. E.g. see #139248,
676                // where the reparse attempt of an invalid expr consumed the trailing invisible
677                // delimiter.
678                self.dcx()
679                    .span_delayed_bug(self.token.span, "no close delim with reparsing {mv_kind:?}");
680                None
681            }
682        } else {
683            None
684        }
685    }
686
687    /// Is the given keyword `kw` followed by a non-reserved identifier?
688    fn is_kw_followed_by_ident(&self, kw: Symbol) -> bool {
689        self.token.is_keyword(kw) && self.look_ahead(1, |t| t.is_non_reserved_ident())
690    }
691
692    #[inline]
693    fn check_or_expected(&mut self, ok: bool, token_type: TokenType) -> bool {
694        if !ok {
695            self.expected_token_types.insert(token_type);
696        }
697        ok
698    }
699
700    fn check_ident(&mut self) -> bool {
701        self.check_or_expected(self.token.is_ident(), TokenType::Ident)
702    }
703
704    fn check_path(&mut self) -> bool {
705        self.check_or_expected(self.token.is_path_start(), TokenType::Path)
706    }
707
708    fn check_type(&mut self) -> bool {
709        self.check_or_expected(self.token.can_begin_type(), TokenType::Type)
710    }
711
712    fn check_const_arg(&mut self) -> bool {
713        let is_mcg_arg = self.check_or_expected(self.token.can_begin_const_arg(), TokenType::Const);
714        let is_mgca_arg = self.is_keyword_ahead(0, &[kw::Const])
715            && self.look_ahead(1, |t| *t == token::OpenBrace);
716        is_mcg_arg || is_mgca_arg
717    }
718
719    fn check_const_closure(&self) -> bool {
720        self.is_keyword_ahead(0, &[kw::Const])
721            && self.look_ahead(1, |t| match &t.kind {
722                // async closures do not work with const closures, so we do not parse that here.
723                token::Ident(kw::Move | kw::Use | kw::Static, IdentIsRaw::No)
724                | token::OrOr
725                | token::Or => true,
726                _ => false,
727            })
728    }
729
730    fn check_inline_const(&self, dist: usize) -> bool {
731        self.is_keyword_ahead(dist, &[kw::Const])
732            && self.look_ahead(dist + 1, |t| match &t.kind {
733                token::OpenBrace => true,
734                token::OpenInvisible(InvisibleOrigin::MetaVar(MetaVarKind::Block)) => true,
735                _ => false,
736            })
737    }
738
739    /// Checks to see if the next token is either `+` or `+=`.
740    /// Otherwise returns `false`.
741    #[inline]
742    fn check_plus(&mut self) -> bool {
743        self.check_or_expected(self.token.is_like_plus(), TokenType::Plus)
744    }
745
746    /// Eats the expected token if it's present possibly breaking
747    /// compound tokens like multi-character operators in process.
748    /// Returns `true` if the token was eaten.
749    fn break_and_eat(&mut self, exp: ExpTokenPair) -> bool {
750        if self.token == exp.tok {
751            self.bump();
752            return true;
753        }
754        match self.token.kind.break_two_token_op(1) {
755            Some((first, second)) if first == exp.tok => {
756                let first_span = self.psess.source_map().start_point(self.token.span);
757                let second_span = self.token.span.with_lo(first_span.hi());
758                self.token = Token::new(first, first_span);
759                // Keep track of this token - if we end token capturing now,
760                // we'll want to append this token to the captured stream.
761                //
762                // If we consume any additional tokens, then this token
763                // is not needed (we'll capture the entire 'glued' token),
764                // and `bump` will set this field to 0.
765                self.break_last_token += 1;
766                // Use the spacing of the glued token as the spacing of the
767                // unglued second token.
768                self.bump_with((Token::new(second, second_span), self.token_spacing));
769                true
770            }
771            _ => {
772                self.expected_token_types.insert(exp.token_type);
773                false
774            }
775        }
776    }
777
778    /// Eats `+` possibly breaking tokens like `+=` in process.
779    fn eat_plus(&mut self) -> bool {
780        self.break_and_eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Plus,
    token_type: crate::parser::token_type::TokenType::Plus,
}exp!(Plus))
781    }
782
783    /// Eats `&` possibly breaking tokens like `&&` in process.
784    /// Signals an error if `&` is not eaten.
785    fn expect_and(&mut self) -> PResult<'a, ()> {
786        if self.break_and_eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::And,
    token_type: crate::parser::token_type::TokenType::And,
}exp!(And)) { Ok(()) } else { self.unexpected() }
787    }
788
789    /// Eats `|` possibly breaking tokens like `||` in process.
790    /// Signals an error if `|` was not eaten.
791    fn expect_or(&mut self) -> PResult<'a, ()> {
792        if self.break_and_eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Or,
    token_type: crate::parser::token_type::TokenType::Or,
}exp!(Or)) { Ok(()) } else { self.unexpected() }
793    }
794
795    /// Eats `<` possibly breaking tokens like `<<` in process.
796    fn eat_lt(&mut self) -> bool {
797        let ate = self.break_and_eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Lt,
    token_type: crate::parser::token_type::TokenType::Lt,
}exp!(Lt));
798        if ate {
799            // See doc comment for `unmatched_angle_bracket_count`.
800            self.unmatched_angle_bracket_count += 1;
801            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_parse/src/parser/mod.rs:801",
                        "rustc_parse::parser", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(801u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_parse::parser"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("eat_lt: (increment) count={0:?}",
                                                    self.unmatched_angle_bracket_count) as &dyn Value))])
            });
    } else { ; }
};debug!("eat_lt: (increment) count={:?}", self.unmatched_angle_bracket_count);
802        }
803        ate
804    }
805
806    /// Eats `<` possibly breaking tokens like `<<` in process.
807    /// Signals an error if `<` was not eaten.
808    fn expect_lt(&mut self) -> PResult<'a, ()> {
809        if self.eat_lt() { Ok(()) } else { self.unexpected() }
810    }
811
812    /// Eats `>` possibly breaking tokens like `>>` in process.
813    /// Signals an error if `>` was not eaten.
814    fn expect_gt(&mut self) -> PResult<'a, ()> {
815        if self.break_and_eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Gt,
    token_type: crate::parser::token_type::TokenType::Gt,
}exp!(Gt)) {
816            // See doc comment for `unmatched_angle_bracket_count`.
817            if self.unmatched_angle_bracket_count > 0 {
818                self.unmatched_angle_bracket_count -= 1;
819                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_parse/src/parser/mod.rs:819",
                        "rustc_parse::parser", ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(819u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_parse::parser"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&format_args!("expect_gt: (decrement) count={0:?}",
                                                    self.unmatched_angle_bracket_count) as &dyn Value))])
            });
    } else { ; }
};debug!("expect_gt: (decrement) count={:?}", self.unmatched_angle_bracket_count);
820            }
821            Ok(())
822        } else {
823            self.unexpected()
824        }
825    }
826
827    /// Checks if the next token is contained within `closes`, and returns `true` if so.
828    fn expect_any_with_type(
829        &mut self,
830        closes_expected: &[ExpTokenPair],
831        closes_not_expected: &[&TokenKind],
832    ) -> bool {
833        closes_expected.iter().any(|&close| self.check(close))
834            || closes_not_expected.iter().any(|k| self.check_noexpect(k))
835    }
836
837    /// Parses a sequence until the specified delimiters. The function
838    /// `f` must consume tokens until reaching the next separator or
839    /// closing bracket.
840    fn parse_seq_to_before_tokens<T>(
841        &mut self,
842        closes_expected: &[ExpTokenPair],
843        closes_not_expected: &[&TokenKind],
844        sep: SeqSep,
845        mut f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
846    ) -> PResult<'a, (ThinVec<T>, Trailing, Recovered)> {
847        let mut first = true;
848        let mut recovered = Recovered::No;
849        let mut trailing = Trailing::No;
850        let mut v = ThinVec::new();
851
852        while !self.expect_any_with_type(closes_expected, closes_not_expected) {
853            if self.token.kind.is_close_delim_or_eof() {
854                break;
855            }
856            if let Some(exp) = sep.sep {
857                if first {
858                    // no separator for the first element
859                    first = false;
860                } else {
861                    // check for separator
862                    match self.expect(exp) {
863                        Ok(Recovered::No) => {
864                            self.current_closure.take();
865                        }
866                        Ok(Recovered::Yes(guar)) => {
867                            self.current_closure.take();
868                            recovered = Recovered::Yes(guar);
869                            break;
870                        }
871                        Err(mut expect_err) => {
872                            let sp = self.prev_token.span.shrink_to_hi();
873                            let token_str = pprust::token_kind_to_string(&exp.tok);
874
875                            match self.current_closure.take() {
876                                Some(closure_spans) if self.token == TokenKind::Semi => {
877                                    // Finding a semicolon instead of a comma
878                                    // after a closure body indicates that the
879                                    // closure body may be a block but the user
880                                    // forgot to put braces around its
881                                    // statements.
882
883                                    self.recover_missing_braces_around_closure_body(
884                                        closure_spans,
885                                        expect_err,
886                                    )?;
887
888                                    continue;
889                                }
890
891                                _ => {
892                                    // Attempt to keep parsing if it was a similar separator.
893                                    if exp.tok.similar_tokens().contains(&self.token.kind) {
894                                        self.bump();
895                                    }
896                                }
897                            }
898
899                            // If this was a missing `@` in a binding pattern
900                            // bail with a suggestion
901                            // https://github.com/rust-lang/rust/issues/72373
902                            if self.prev_token.is_ident() && self.token == token::DotDot {
903                                let msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("if you meant to bind the contents of the rest of the array pattern into `{0}`, use `@`",
                pprust::token_to_string(&self.prev_token)))
    })format!(
904                                    "if you meant to bind the contents of the rest of the array \
905                                     pattern into `{}`, use `@`",
906                                    pprust::token_to_string(&self.prev_token)
907                                );
908                                expect_err
909                                    .with_span_suggestion_verbose(
910                                        self.prev_token.span.shrink_to_hi().until(self.token.span),
911                                        msg,
912                                        " @ ",
913                                        Applicability::MaybeIncorrect,
914                                    )
915                                    .emit();
916                                break;
917                            }
918
919                            // Attempt to keep parsing if it was an omitted separator.
920                            // `&raw <expr>` already has a specific suggestion for missing
921                            // `const`/`mut`, so don't recover `<expr>` as the next element in
922                            // a comma-separated list.
923                            if exp.token_type == TokenType::Comma && self.is_expected_raw_ref_mut()
924                            {
925                                return Err(expect_err);
926                            }
927                            self.last_unexpected_token_span = None;
928                            match f(self) {
929                                Ok(t) => {
930                                    // Parsed successfully, therefore most probably the code only
931                                    // misses a separator.
932                                    expect_err
933                                        .with_span_suggestion_short(
934                                            sp,
935                                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("missing `{0}`", token_str))
    })format!("missing `{token_str}`"),
936                                            token_str,
937                                            Applicability::MaybeIncorrect,
938                                        )
939                                        .emit();
940
941                                    v.push(t);
942                                    continue;
943                                }
944                                Err(e) => {
945                                    // Parsing failed, therefore it must be something more serious
946                                    // than just a missing separator.
947                                    for xx in &e.children {
948                                        // Propagate the help message from sub error `e` to main
949                                        // error `expect_err`.
950                                        expect_err.children.push(xx.clone());
951                                    }
952                                    e.cancel();
953                                    if self.token == token::Colon {
954                                        // We will try to recover in
955                                        // `maybe_recover_struct_lit_bad_delims`.
956                                        return Err(expect_err);
957                                    } else if let [exp] = closes_expected
958                                        && exp.token_type == TokenType::CloseParen
959                                    {
960                                        return Err(expect_err);
961                                    } else {
962                                        expect_err.emit();
963                                        break;
964                                    }
965                                }
966                            }
967                        }
968                    }
969                }
970            }
971            if sep.trailing_sep_allowed
972                && self.expect_any_with_type(closes_expected, closes_not_expected)
973            {
974                trailing = Trailing::Yes;
975                break;
976            }
977
978            let t = f(self)?;
979            v.push(t);
980        }
981
982        Ok((v, trailing, recovered))
983    }
984
985    fn recover_missing_braces_around_closure_body(
986        &mut self,
987        closure_spans: ClosureSpans,
988        mut expect_err: Diag<'_>,
989    ) -> PResult<'a, ()> {
990        let initial_semicolon = self.token.span;
991
992        while self.eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Semi,
    token_type: crate::parser::token_type::TokenType::Semi,
}exp!(Semi)) {
993            let _ = self
994                .parse_stmt_without_recovery(false, ForceCollect::No, false)
995                .unwrap_or_else(|e| {
996                    e.cancel();
997                    None
998                });
999        }
1000
1001        expect_err
1002            .primary_message("closure bodies that contain statements must be surrounded by braces");
1003
1004        let preceding_pipe_span = closure_spans.closing_pipe;
1005        let following_token_span = self.token.span;
1006
1007        let mut first_note = MultiSpan::from(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [initial_semicolon]))vec![initial_semicolon]);
1008        first_note.push_span_label(
1009            initial_semicolon,
1010            "this `;` turns the preceding closure into a statement",
1011        );
1012        first_note.push_span_label(
1013            closure_spans.body,
1014            "this expression is a statement because of the trailing semicolon",
1015        );
1016        expect_err.span_note(first_note, "statement found outside of a block");
1017
1018        let mut second_note = MultiSpan::from(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [closure_spans.whole_closure]))vec![closure_spans.whole_closure]);
1019        second_note.push_span_label(closure_spans.whole_closure, "this is the parsed closure...");
1020        second_note.push_span_label(
1021            following_token_span,
1022            "...but likely you meant the closure to end here",
1023        );
1024        expect_err.span_note(second_note, "the closure body may be incorrectly delimited");
1025
1026        expect_err.span(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [preceding_pipe_span, following_token_span]))vec![preceding_pipe_span, following_token_span]);
1027
1028        let opening_suggestion_str = " {".to_string();
1029        let closing_suggestion_str = "}".to_string();
1030
1031        expect_err.multipart_suggestion(
1032            "try adding braces",
1033            ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(preceding_pipe_span.shrink_to_hi(), opening_suggestion_str),
                (following_token_span.shrink_to_lo(),
                    closing_suggestion_str)]))vec![
1034                (preceding_pipe_span.shrink_to_hi(), opening_suggestion_str),
1035                (following_token_span.shrink_to_lo(), closing_suggestion_str),
1036            ],
1037            Applicability::MaybeIncorrect,
1038        );
1039
1040        expect_err.emit();
1041
1042        Ok(())
1043    }
1044
1045    /// Parses a sequence, not including the delimiters. The function
1046    /// `f` must consume tokens until reaching the next separator or
1047    /// closing bracket.
1048    fn parse_seq_to_before_end<T>(
1049        &mut self,
1050        close: ExpTokenPair,
1051        sep: SeqSep,
1052        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
1053    ) -> PResult<'a, (ThinVec<T>, Trailing, Recovered)> {
1054        self.parse_seq_to_before_tokens(&[close], &[], sep, f)
1055    }
1056
1057    /// Parses a sequence, including only the closing delimiter. The function
1058    /// `f` must consume tokens until reaching the next separator or
1059    /// closing bracket.
1060    fn parse_seq_to_end<T>(
1061        &mut self,
1062        close: ExpTokenPair,
1063        sep: SeqSep,
1064        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
1065    ) -> PResult<'a, (ThinVec<T>, Trailing)> {
1066        let (val, trailing, recovered) = self.parse_seq_to_before_end(close, sep, f)?;
1067        if #[allow(non_exhaustive_omitted_patterns)] match recovered {
    Recovered::No => true,
    _ => false,
}matches!(recovered, Recovered::No) && !self.eat(close) {
1068            self.dcx().span_delayed_bug(
1069                self.token.span,
1070                "recovered but `parse_seq_to_before_end` did not give us the close token",
1071            );
1072        }
1073        Ok((val, trailing))
1074    }
1075
1076    /// Parses a sequence, including both delimiters. The function
1077    /// `f` must consume tokens until reaching the next separator or
1078    /// closing bracket.
1079    fn parse_unspanned_seq<T>(
1080        &mut self,
1081        open: ExpTokenPair,
1082        close: ExpTokenPair,
1083        sep: SeqSep,
1084        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
1085    ) -> PResult<'a, (ThinVec<T>, Trailing)> {
1086        self.expect(open)?;
1087        self.parse_seq_to_end(close, sep, f)
1088    }
1089
1090    /// Parses a comma-separated sequence, including both delimiters.
1091    /// The function `f` must consume tokens until reaching the next separator or
1092    /// closing bracket.
1093    fn parse_delim_comma_seq<T>(
1094        &mut self,
1095        open: ExpTokenPair,
1096        close: ExpTokenPair,
1097        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
1098    ) -> PResult<'a, (ThinVec<T>, Trailing)> {
1099        self.parse_unspanned_seq(open, close, SeqSep::trailing_allowed(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Comma,
    token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma)), f)
1100    }
1101
1102    /// Parses a comma-separated sequence delimited by parentheses (e.g. `(x, y)`).
1103    /// The function `f` must consume tokens until reaching the next separator or
1104    /// closing bracket.
1105    pub fn parse_paren_comma_seq<T>(
1106        &mut self,
1107        f: impl FnMut(&mut Parser<'a>) -> PResult<'a, T>,
1108    ) -> PResult<'a, (ThinVec<T>, Trailing)> {
1109        self.parse_delim_comma_seq(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenParen,
    token_type: crate::parser::token_type::TokenType::OpenParen,
}exp!(OpenParen), crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen), f)
1110    }
1111
1112    /// Advance the parser by one token using provided token as the next one.
1113    fn bump_with(&mut self, next: (Token, Spacing)) {
1114        self.inlined_bump_with(next)
1115    }
1116
1117    /// This always-inlined version should only be used on hot code paths.
1118    #[inline(always)]
1119    fn inlined_bump_with(&mut self, (next_token, next_spacing): (Token, Spacing)) {
1120        // Update the current and previous tokens.
1121        self.prev_token = mem::replace(&mut self.token, next_token);
1122        self.token_spacing = next_spacing;
1123
1124        // Diagnostics.
1125        self.expected_token_types.clear();
1126    }
1127
1128    /// Advance the parser by one token.
1129    pub fn bump(&mut self) {
1130        // Note: destructuring here would give nicer code, but it was found in #96210 to be slower
1131        // than `.0`/`.1` access.
1132        let mut next = self.token_cursor.inlined_next();
1133        self.num_bump_calls += 1;
1134        // We got a token from the underlying cursor and no longer need to
1135        // worry about an unglued token. See `break_and_eat` for more details.
1136        self.break_last_token = 0;
1137        if next.0.span.is_dummy() {
1138            // Tweak the location for better diagnostics, but keep syntactic context intact.
1139            let fallback_span = self.token.span;
1140            next.0.span = fallback_span.with_ctxt(next.0.span.ctxt());
1141        }
1142        if true {
    if !!#[allow(non_exhaustive_omitted_patterns)] match next.0.kind {
                    token::OpenInvisible(origin) | token::CloseInvisible(origin)
                        if origin.skip() => true,
                    _ => false,
                } {
        ::core::panicking::panic("assertion failed: !matches!(next.0.kind, token::OpenInvisible(origin) |\n        token::CloseInvisible(origin) if origin.skip())")
    };
};debug_assert!(!matches!(
1143            next.0.kind,
1144            token::OpenInvisible(origin) | token::CloseInvisible(origin) if origin.skip()
1145        ));
1146        self.inlined_bump_with(next)
1147    }
1148
1149    /// Look-ahead `dist` tokens of `self.token` and get access to that token there.
1150    /// When `dist == 0` then the current token is looked at. `Eof` will be
1151    /// returned if the look-ahead is any distance past the end of the tokens.
1152    pub fn look_ahead<R>(&self, dist: usize, looker: impl FnOnce(&Token) -> R) -> R {
1153        if dist == 0 {
1154            return looker(&self.token);
1155        }
1156
1157        // Typically around 98% of the `dist > 0` cases have `dist == 1`, so we
1158        // have a fast special case for that.
1159        if dist == 1 {
1160            // The index is zero because the tree cursor's index always points
1161            // to the next token to be gotten.
1162            match self.token_cursor.curr.curr() {
1163                Some(tree) => {
1164                    // Indexing stayed within the current token tree.
1165                    match tree {
1166                        TokenTree::Token(token, _) => return looker(token),
1167                        &TokenTree::Delimited(dspan, _, delim, _) => {
1168                            if !delim.skip() {
1169                                return looker(&Token::new(delim.as_open_token_kind(), dspan.open));
1170                            }
1171                        }
1172                    }
1173                }
1174                None => {
1175                    // The tree cursor lookahead went (one) past the end of the
1176                    // current token tree. Try to return a close delimiter.
1177                    if let Some(last) = self.token_cursor.stack.last()
1178                        && let Some(&TokenTree::Delimited(span, _, delim, _)) = last.curr()
1179                        && !delim.skip()
1180                    {
1181                        // We are not in the outermost token stream, so we have
1182                        // delimiters. Also, those delimiters are not skipped.
1183                        return looker(&Token::new(delim.as_close_token_kind(), span.close));
1184                    }
1185                }
1186            }
1187        }
1188
1189        // Just clone the token cursor and use `next`, skipping delimiters as
1190        // necessary. Slow but simple.
1191        let mut cursor = self.token_cursor.clone();
1192        let mut i = 0;
1193        let mut token = Token::dummy();
1194        while i < dist {
1195            token = cursor.next().0;
1196            if let token::OpenInvisible(origin) | token::CloseInvisible(origin) = token.kind
1197                && origin.skip()
1198            {
1199                continue;
1200            }
1201            i += 1;
1202        }
1203        looker(&token)
1204    }
1205
1206    /// Like `lookahead`, but skips over token trees rather than tokens. Useful
1207    /// when looking past possible metavariable pasting sites.
1208    pub fn tree_look_ahead<R>(
1209        &self,
1210        dist: usize,
1211        looker: impl FnOnce(&TokenTree) -> R,
1212    ) -> Option<R> {
1213        match (&dist, &0) {
    (left_val, right_val) => {
        if *left_val == *right_val {
            let kind = ::core::panicking::AssertKind::Ne;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_ne!(dist, 0);
1214        self.token_cursor.curr.look_ahead(dist - 1).map(looker)
1215    }
1216
1217    /// Returns whether any of the given keywords are `dist` tokens ahead of the current one.
1218    pub(crate) fn is_keyword_ahead(&self, dist: usize, kws: &[Symbol]) -> bool {
1219        self.look_ahead(dist, |t| kws.iter().any(|&kw| t.is_keyword(kw)))
1220    }
1221
1222    /// Parses asyncness: `async` or nothing.
1223    fn parse_coroutine_kind(&mut self, case: Case) -> Option<CoroutineKind> {
1224        let span = self.token_uninterpolated_span();
1225        if self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Async,
    token_type: crate::parser::token_type::TokenType::KwAsync,
}exp!(Async), case) {
1226            // FIXME(gen_blocks): Do we want to unconditionally parse `gen` and then
1227            // error if edition <= 2024, like we do with async and edition <= 2018?
1228            if self.token_uninterpolated_span().at_least_rust_2024()
1229                && self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Gen,
    token_type: crate::parser::token_type::TokenType::KwGen,
}exp!(Gen), case)
1230            {
1231                let gen_span = self.prev_token_uninterpolated_span();
1232                Some(CoroutineKind::AsyncGen {
1233                    span: span.to(gen_span),
1234                    closure_id: DUMMY_NODE_ID,
1235                    return_impl_trait_id: DUMMY_NODE_ID,
1236                })
1237            } else {
1238                Some(CoroutineKind::Async {
1239                    span,
1240                    closure_id: DUMMY_NODE_ID,
1241                    return_impl_trait_id: DUMMY_NODE_ID,
1242                })
1243            }
1244        } else if self.token_uninterpolated_span().at_least_rust_2024()
1245            && self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Gen,
    token_type: crate::parser::token_type::TokenType::KwGen,
}exp!(Gen), case)
1246        {
1247            Some(CoroutineKind::Gen {
1248                span,
1249                closure_id: DUMMY_NODE_ID,
1250                return_impl_trait_id: DUMMY_NODE_ID,
1251            })
1252        } else {
1253            None
1254        }
1255    }
1256
1257    /// Parses fn unsafety: `unsafe`, `safe` or nothing.
1258    fn parse_safety(&mut self, case: Case) -> Safety {
1259        if self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Unsafe,
    token_type: crate::parser::token_type::TokenType::KwUnsafe,
}exp!(Unsafe), case) {
1260            Safety::Unsafe(self.prev_token_uninterpolated_span())
1261        } else if self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Safe,
    token_type: crate::parser::token_type::TokenType::KwSafe,
}exp!(Safe), case) {
1262            Safety::Safe(self.prev_token_uninterpolated_span())
1263        } else {
1264            Safety::Default
1265        }
1266    }
1267
1268    /// Parses constness: `const` or nothing.
1269    fn parse_constness(&mut self, case: Case) -> Const {
1270        self.parse_constness_(case, false)
1271    }
1272
1273    /// Parses constness for closures (case sensitive, feature-gated)
1274    fn parse_closure_constness(&mut self) -> Const {
1275        let constness = self.parse_constness_(Case::Sensitive, true);
1276        if let Const::Yes(span) = constness {
1277            self.psess.gated_spans.gate(sym::const_closures, span);
1278        }
1279        constness
1280    }
1281
1282    fn parse_constness_(&mut self, case: Case, is_closure: bool) -> Const {
1283        // Avoid const blocks and const closures to be parsed as const items
1284        if (self.check_const_closure() == is_closure)
1285            && !self.look_ahead(1, |t| *t == token::OpenBrace || t.is_metavar_block())
1286            && self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Const,
    token_type: crate::parser::token_type::TokenType::KwConst,
}exp!(Const), case)
1287        {
1288            Const::Yes(self.prev_token_uninterpolated_span())
1289        } else {
1290            Const::No
1291        }
1292    }
1293
1294    /// Parses inline const expressions.
1295    fn parse_const_block(&mut self, span: Span, pat: bool) -> PResult<'a, Box<Expr>> {
1296        self.expect_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Const,
    token_type: crate::parser::token_type::TokenType::KwConst,
}exp!(Const))?;
1297        let (attrs, blk) = self.parse_inner_attrs_and_block(None)?;
1298        let anon_const = AnonConst {
1299            id: DUMMY_NODE_ID,
1300            value: self.mk_expr(blk.span, ExprKind::Block(blk, None)),
1301            mgca_disambiguation: MgcaDisambiguation::AnonConst,
1302        };
1303        let blk_span = anon_const.value.span;
1304        let kind = if pat {
1305            let guar = self
1306                .dcx()
1307                .struct_span_err(blk_span, "const blocks cannot be used as patterns")
1308                .with_help(
1309                    "use a named `const`-item or an `if`-guard (`x if x == const { ... }`) instead",
1310                )
1311                .emit();
1312            ExprKind::Err(guar)
1313        } else {
1314            ExprKind::ConstBlock(anon_const)
1315        };
1316        Ok(self.mk_expr_with_attrs(span.to(blk_span), kind, attrs))
1317    }
1318
1319    /// Parse nothing or `mut`.
1320    fn parse_mutability(&mut self) -> Mutability {
1321        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Mut,
    token_type: crate::parser::token_type::TokenType::KwMut,
}exp!(Mut)) { Mutability::Mut } else { Mutability::Not }
1322    }
1323
1324    /// Parse nothing or a by-reference mode.
1325    ///
1326    /// ```ebnf
1327    /// ByRef = "ref" PinAndMut?
1328    /// ```
1329    fn parse_byref(&mut self) -> ByRef {
1330        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Ref,
    token_type: crate::parser::token_type::TokenType::KwRef,
}exp!(Ref)) {
1331            let (pinnedness, mutability) = self.parse_pin_and_mut();
1332            ByRef::Yes(pinnedness, mutability)
1333        } else {
1334            ByRef::No
1335        }
1336    }
1337
1338    /// Parse nothing or "explicit" mutability.
1339    ///
1340    /// ```ebnf
1341    /// MutOrConst = "mut" | "const"
1342    /// ```
1343    fn parse_mut_or_const(&mut self) -> Option<Mutability> {
1344        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Mut,
    token_type: crate::parser::token_type::TokenType::KwMut,
}exp!(Mut)) {
1345            Some(Mutability::Mut)
1346        } else if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Const,
    token_type: crate::parser::token_type::TokenType::KwConst,
}exp!(Const)) {
1347            Some(Mutability::Not)
1348        } else {
1349            None
1350        }
1351    }
1352
1353    /// Parse a field name.
1354    ///
1355    /// ```enbf
1356    /// FieldName = IntLit | Ident
1357    /// ```
1358    fn parse_field_name(&mut self) -> PResult<'a, Ident> {
1359        if let token::Literal(token::Lit { kind: token::Integer, symbol, suffix }) = self.token.kind
1360        {
1361            if let Some(suffix) = suffix {
1362                self.dcx().emit_err(errors::InvalidLiteralSuffixOnTupleIndex {
1363                    span: self.token.span,
1364                    suffix,
1365                });
1366            }
1367            self.bump();
1368            Ok(Ident::new(symbol, self.prev_token.span))
1369        } else {
1370            self.parse_ident_common(true)
1371        }
1372    }
1373
1374    fn parse_delim_args(&mut self) -> PResult<'a, Box<DelimArgs>> {
1375        if let Some(args) = self.parse_delim_args_inner() {
1376            Ok(Box::new(args))
1377        } else {
1378            self.unexpected_any()
1379        }
1380    }
1381
1382    fn parse_attr_args(&mut self) -> PResult<'a, AttrArgs> {
1383        Ok(if let Some(args) = self.parse_delim_args_inner() {
1384            AttrArgs::Delimited(args)
1385        } else if self.eat(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::Eq,
    token_type: crate::parser::token_type::TokenType::Eq,
}exp!(Eq)) {
1386            let eq_span = self.prev_token.span;
1387            let expr = self.parse_expr_force_collect()?;
1388            AttrArgs::Eq { eq_span, expr }
1389        } else {
1390            AttrArgs::Empty
1391        })
1392    }
1393
1394    fn parse_delim_args_inner(&mut self) -> Option<DelimArgs> {
1395        let delimited = self.check(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenParen,
    token_type: crate::parser::token_type::TokenType::OpenParen,
}exp!(OpenParen))
1396            || self.check(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenBracket,
    token_type: crate::parser::token_type::TokenType::OpenBracket,
}exp!(OpenBracket))
1397            || self.check(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenBrace,
    token_type: crate::parser::token_type::TokenType::OpenBrace,
}exp!(OpenBrace));
1398
1399        delimited.then(|| {
1400            let TokenTree::Delimited(dspan, _, delim, tokens) = self.parse_token_tree() else {
1401                ::core::panicking::panic("internal error: entered unreachable code")unreachable!()
1402            };
1403            DelimArgs { dspan, delim, tokens }
1404        })
1405    }
1406
1407    /// Parses a single token tree from the input.
1408    pub fn parse_token_tree(&mut self) -> TokenTree {
1409        if self.token.kind.open_delim().is_some() {
1410            // Clone the `TokenTree::Delimited` that we are currently
1411            // within. That's what we are going to return.
1412            let tree = self.token_cursor.stack.last().unwrap().curr().unwrap().clone();
1413            if true {
    {
        match tree {
            TokenTree::Delimited(..) => {}
            ref left_val => {
                ::core::panicking::assert_matches_failed(left_val,
                    "TokenTree::Delimited(..)", ::core::option::Option::None);
            }
        }
    };
};debug_assert_matches!(tree, TokenTree::Delimited(..));
1414
1415            // Advance the token cursor through the entire delimited
1416            // sequence. After getting the `OpenDelim` we are *within* the
1417            // delimited sequence, i.e. at depth `d`. After getting the
1418            // matching `CloseDelim` we are *after* the delimited sequence,
1419            // i.e. at depth `d - 1`.
1420            let target_depth = self.token_cursor.stack.len() - 1;
1421
1422            if let Capturing::No = self.capture_state.capturing {
1423                // We are not capturing tokens, so skip to the end of the
1424                // delimited sequence. This is a perf win when dealing with
1425                // declarative macros that pass large `tt` fragments through
1426                // multiple rules, as seen in the uom-0.37.0 crate.
1427                self.token_cursor.curr.bump_to_end();
1428                self.bump();
1429                if true {
    match (&self.token_cursor.stack.len(), &target_depth) {
        (left_val, right_val) => {
            if !(*left_val == *right_val) {
                let kind = ::core::panicking::AssertKind::Eq;
                ::core::panicking::assert_failed(kind, &*left_val,
                    &*right_val, ::core::option::Option::None);
            }
        }
    };
};debug_assert_eq!(self.token_cursor.stack.len(), target_depth);
1430            } else {
1431                loop {
1432                    // Advance one token at a time, so `TokenCursor::next()`
1433                    // can capture these tokens if necessary.
1434                    self.bump();
1435                    if self.token_cursor.stack.len() == target_depth {
1436                        break;
1437                    }
1438                }
1439            }
1440            if true {
    if !self.token.kind.close_delim().is_some() {
        ::core::panicking::panic("assertion failed: self.token.kind.close_delim().is_some()")
    };
};debug_assert!(self.token.kind.close_delim().is_some());
1441
1442            // Consume close delimiter
1443            self.bump();
1444            tree
1445        } else {
1446            if !!self.token.kind.is_close_delim_or_eof() {
    ::core::panicking::panic("assertion failed: !self.token.kind.is_close_delim_or_eof()")
};assert!(!self.token.kind.is_close_delim_or_eof());
1447            let prev_spacing = self.token_spacing;
1448            self.bump();
1449            TokenTree::Token(self.prev_token, prev_spacing)
1450        }
1451    }
1452
1453    pub fn parse_tokens(&mut self) -> TokenStream {
1454        let mut result = Vec::new();
1455        loop {
1456            if self.token.kind.is_close_delim_or_eof() {
1457                break;
1458            } else {
1459                result.push(self.parse_token_tree());
1460            }
1461        }
1462        TokenStream::new(result)
1463    }
1464
1465    /// Evaluates the closure with restrictions in place.
1466    ///
1467    /// Afters the closure is evaluated, restrictions are reset.
1468    fn with_res<T>(&mut self, res: Restrictions, f: impl FnOnce(&mut Self) -> T) -> T {
1469        let old = self.restrictions;
1470        self.restrictions = res;
1471        let res = f(self);
1472        self.restrictions = old;
1473        res
1474    }
1475
1476    /// Parses `pub` and `pub(in path)` plus shortcuts `pub(crate)` for `pub(in crate)`, `pub(self)`
1477    /// for `pub(in self)` and `pub(super)` for `pub(in super)`.
1478    /// If the following element can't be a tuple (i.e., it's a function definition), then
1479    /// it's not a tuple struct field), and the contents within the parentheses aren't valid,
1480    /// so emit a proper diagnostic.
1481    // Public for rustfmt usage.
1482    pub fn parse_visibility(&mut self, fbt: FollowedByType) -> PResult<'a, Visibility> {
1483        if let Some(vis) = self
1484            .eat_metavar_seq(MetaVarKind::Vis, |this| this.parse_visibility(FollowedByType::Yes))
1485        {
1486            return Ok(vis);
1487        }
1488
1489        if !self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Pub,
    token_type: crate::parser::token_type::TokenType::KwPub,
}exp!(Pub)) {
1490            // We need a span for our `Spanned<VisibilityKind>`, but there's inherently no
1491            // keyword to grab a span from for inherited visibility; an empty span at the
1492            // beginning of the current token would seem to be the "Schelling span".
1493            return Ok(Visibility {
1494                span: self.token.span.shrink_to_lo(),
1495                kind: VisibilityKind::Inherited,
1496                tokens: None,
1497            });
1498        }
1499        let lo = self.prev_token.span;
1500
1501        if self.check(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenParen,
    token_type: crate::parser::token_type::TokenType::OpenParen,
}exp!(OpenParen)) {
1502            // We don't `self.bump()` the `(` yet because this might be a struct definition where
1503            // `()` or a tuple might be allowed. For example, `struct Struct(pub (), pub (usize));`.
1504            // Because of this, we only `bump` the `(` if we're assured it is appropriate to do so
1505            // by the following tokens.
1506            if self.is_keyword_ahead(1, &[kw::In]) {
1507                // Parse `pub(in path)`.
1508                self.bump(); // `(`
1509                self.bump(); // `in`
1510                let path = self.parse_path(PathStyle::Mod)?; // `path`
1511                self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1512                let vis = VisibilityKind::Restricted {
1513                    path: Box::new(path),
1514                    id: ast::DUMMY_NODE_ID,
1515                    shorthand: false,
1516                };
1517                return Ok(Visibility {
1518                    span: lo.to(self.prev_token.span),
1519                    kind: vis,
1520                    tokens: None,
1521                });
1522            } else if self.look_ahead(2, |t| t == &token::CloseParen)
1523                && self.is_keyword_ahead(1, &[kw::Crate, kw::Super, kw::SelfLower])
1524            {
1525                // Parse `pub(crate)`, `pub(self)`, or `pub(super)`.
1526                self.bump(); // `(`
1527                let path = self.parse_path(PathStyle::Mod)?; // `crate`/`super`/`self`
1528                self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1529                let vis = VisibilityKind::Restricted {
1530                    path: Box::new(path),
1531                    id: ast::DUMMY_NODE_ID,
1532                    shorthand: true,
1533                };
1534                return Ok(Visibility {
1535                    span: lo.to(self.prev_token.span),
1536                    kind: vis,
1537                    tokens: None,
1538                });
1539            } else if let FollowedByType::No = fbt {
1540                // Provide this diagnostic if a type cannot follow;
1541                // in particular, if this is not a tuple struct.
1542                self.recover_incorrect_vis_restriction()?;
1543                // Emit diagnostic, but continue with public visibility.
1544            }
1545        }
1546
1547        Ok(Visibility { span: lo, kind: VisibilityKind::Public, tokens: None })
1548    }
1549
1550    /// Recovery for e.g. `pub(something) fn ...` or `struct X { pub(something) y: Z }`
1551    fn recover_incorrect_vis_restriction(&mut self) -> PResult<'a, ()> {
1552        self.bump(); // `(`
1553        let path = self.parse_path(PathStyle::Mod)?;
1554        self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1555
1556        let path_str = pprust::path_to_string(&path);
1557        self.dcx()
1558            .emit_err(IncorrectVisibilityRestriction { span: path.span, inner_str: path_str });
1559
1560        Ok(())
1561    }
1562
1563    /// Parses an optional `impl` restriction.
1564    /// Enforces the `impl_restriction` feature gate whenever an explicit restriction is encountered.
1565    fn parse_impl_restriction(&mut self) -> PResult<'a, ImplRestriction> {
1566        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Impl,
    token_type: crate::parser::token_type::TokenType::KwImpl,
}exp!(Impl)) {
1567            let (kind, span, gated_span) = self.parse_restriction(ParsingRestrictionKind::Impl)?;
1568            self.psess.gated_spans.gate(sym::impl_restriction, gated_span);
1569            return Ok(ImplRestriction { kind, span, tokens: None });
1570        }
1571        Ok(ImplRestriction {
1572            kind: RestrictionKind::Unrestricted,
1573            span: self.token.span.shrink_to_lo(),
1574            tokens: None,
1575        })
1576    }
1577
1578    /// Parses an optional `mut` restriction.
1579    /// Enforces the `mut_restriction` feature gate whenever an explicit restriction is encountered.
1580    fn parse_mut_restriction(&mut self) -> PResult<'a, MutRestriction> {
1581        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Mut,
    token_type: crate::parser::token_type::TokenType::KwMut,
}exp!(Mut)) {
1582            let (kind, span, gated_span) = self.parse_restriction(ParsingRestrictionKind::Mut)?;
1583            self.psess.gated_spans.gate(sym::mut_restriction, gated_span);
1584            return Ok(MutRestriction { kind, span, tokens: None });
1585        }
1586        Ok(MutRestriction {
1587            kind: RestrictionKind::Unrestricted,
1588            span: self.token.span.shrink_to_lo(),
1589            tokens: None,
1590        })
1591    }
1592
1593    /// Parses `impl` or `mut` restrictions.
1594    /// Returns the parsed restriction and its span, as well as the gated span.
1595    fn parse_restriction(
1596        &mut self,
1597        restriction_kind: ParsingRestrictionKind,
1598    ) -> PResult<'a, (RestrictionKind, Span, Span)> {
1599        let lo = self.prev_token.span;
1600        // No units or tuples are allowed to follow `impl` or `mut` here, so we can safely bump `(`.
1601        self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::OpenParen,
    token_type: crate::parser::token_type::TokenType::OpenParen,
}exp!(OpenParen))?;
1602        if self.eat_keyword(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::In,
    token_type: crate::parser::token_type::TokenType::KwIn,
}exp!(In)) {
1603            let path = self.parse_path(PathStyle::Mod)?; // `in path`
1604            self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1605            let restriction = RestrictionKind::Restricted {
1606                path: Box::new(path),
1607                id: ast::DUMMY_NODE_ID,
1608                shorthand: false,
1609            };
1610            let span = lo.to(self.prev_token.span);
1611            Ok((restriction, span, span))
1612        } else if self.look_ahead(1, |t| t == &token::CloseParen)
1613            && self.is_keyword_ahead(0, &[kw::Crate, kw::Super, kw::SelfLower])
1614        {
1615            let path = self.parse_path(PathStyle::Mod)?; // `crate`/`super`/`self`
1616            self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1617            let restriction = RestrictionKind::Restricted {
1618                path: Box::new(path),
1619                id: ast::DUMMY_NODE_ID,
1620                shorthand: true,
1621            };
1622            let span = lo.to(self.prev_token.span);
1623            Ok((restriction, span, span))
1624        } else {
1625            // Emit diagnostic, but continue with no restrictions.
1626            // Recovery for `impl(something) trait` or `mut (something) field`.
1627            let path = self.parse_path(PathStyle::Mod)?;
1628            self.expect(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::CloseParen,
    token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?; // `)`
1629            let path_str = pprust::path_to_string(&path);
1630            let end = self.prev_token.span;
1631            match restriction_kind {
1632                ParsingRestrictionKind::Impl => {
1633                    self.dcx().emit_err(IncorrectImplRestriction {
1634                        span: path.span,
1635                        inner_str: path_str,
1636                    });
1637                }
1638                ParsingRestrictionKind::Mut => {
1639                    self.dcx()
1640                        .emit_err(IncorrectMutRestriction { span: path.span, inner_str: path_str });
1641                }
1642            }
1643            Ok((RestrictionKind::Unrestricted, self.token.span.shrink_to_lo(), lo.to(end)))
1644        }
1645    }
1646
1647    /// Parses `extern string_literal?`.
1648    fn parse_extern(&mut self, case: Case) -> Extern {
1649        if self.eat_keyword_case(crate::parser::token_type::ExpKeywordPair {
    kw: rustc_span::symbol::kw::Extern,
    token_type: crate::parser::token_type::TokenType::KwExtern,
}exp!(Extern), case) {
1650            let mut extern_span = self.prev_token.span;
1651            let abi = self.parse_abi();
1652            if let Some(abi) = abi {
1653                extern_span = extern_span.to(abi.span);
1654            }
1655            Extern::from_abi(abi, extern_span)
1656        } else {
1657            Extern::None
1658        }
1659    }
1660
1661    /// Parses a string literal as an ABI spec.
1662    fn parse_abi(&mut self) -> Option<StrLit> {
1663        match self.parse_str_lit() {
1664            Ok(str_lit) => Some(str_lit),
1665            Err(Some(lit)) => match lit.kind {
1666                ast::LitKind::Err(_) => None,
1667                _ => {
1668                    self.dcx().emit_err(NonStringAbiLiteral { span: lit.span });
1669                    None
1670                }
1671            },
1672            Err(None) => None,
1673        }
1674    }
1675
1676    fn collect_tokens_no_attrs<R: HasAttrs + HasTokens>(
1677        &mut self,
1678        f: impl FnOnce(&mut Self) -> PResult<'a, R>,
1679    ) -> PResult<'a, R> {
1680        // The only reason to call `collect_tokens_no_attrs` is if you want tokens, so use
1681        // `ForceCollect::Yes`
1682        self.collect_tokens(None, AttrWrapper::empty(), ForceCollect::Yes, |this, _attrs| {
1683            Ok((f(this)?, Trailing::No, UsePreAttrPos::No))
1684        })
1685    }
1686
1687    /// Checks for `::` or, potentially, `:::` and then look ahead after it.
1688    fn check_path_sep_and_look_ahead(&mut self, looker: impl Fn(&Token) -> bool) -> bool {
1689        if self.check(crate::parser::token_type::ExpTokenPair {
    tok: rustc_ast::token::PathSep,
    token_type: crate::parser::token_type::TokenType::PathSep,
}exp!(PathSep)) {
1690            if self.may_recover() && self.look_ahead(1, |t| t.kind == token::Colon) {
1691                if true {
    if !!self.look_ahead(1, &looker) {
        {
            ::core::panicking::panic_fmt(format_args!("Looker must not match on colon"));
        }
    };
};debug_assert!(!self.look_ahead(1, &looker), "Looker must not match on colon");
1692                self.look_ahead(2, looker)
1693            } else {
1694                self.look_ahead(1, looker)
1695            }
1696        } else {
1697            false
1698        }
1699    }
1700
1701    /// `::{` or `::*`
1702    fn is_import_coupler(&mut self) -> bool {
1703        self.check_path_sep_and_look_ahead(|t| #[allow(non_exhaustive_omitted_patterns)] match t.kind {
    token::OpenBrace | token::Star => true,
    _ => false,
}matches!(t.kind, token::OpenBrace | token::Star))
1704    }
1705
1706    // Debug view of the parser's token stream, up to `{lookahead}` tokens.
1707    // Only used when debugging.
1708    #[allow(unused)]
1709    pub(crate) fn debug_lookahead(&self, lookahead: usize) -> impl fmt::Debug {
1710        fmt::from_fn(move |f| {
1711            let mut dbg_fmt = f.debug_struct("Parser"); // or at least, one view of
1712
1713            // we don't need N spans, but we want at least one, so print all of prev_token
1714            dbg_fmt.field("prev_token", &self.prev_token);
1715            let mut tokens = ::alloc::vec::Vec::new()vec![];
1716            for i in 0..lookahead {
1717                let tok = self.look_ahead(i, |tok| tok.kind);
1718                let is_eof = tok == TokenKind::Eof;
1719                tokens.push(tok);
1720                if is_eof {
1721                    // Don't look ahead past EOF.
1722                    break;
1723                }
1724            }
1725            dbg_fmt.field_with("tokens", |field| field.debug_list().entries(tokens).finish());
1726            dbg_fmt.field("approx_token_stream_pos", &self.num_bump_calls);
1727
1728            // some fields are interesting for certain values, as they relate to macro parsing
1729            if let Some(subparser) = self.subparser_name {
1730                dbg_fmt.field("subparser_name", &subparser);
1731            }
1732            if let Recovery::Forbidden = self.recovery {
1733                dbg_fmt.field("recovery", &self.recovery);
1734            }
1735
1736            // imply there's "more to know" than this view
1737            dbg_fmt.finish_non_exhaustive()
1738        })
1739    }
1740
1741    pub fn clear_expected_token_types(&mut self) {
1742        self.expected_token_types.clear();
1743    }
1744
1745    pub fn approx_token_stream_pos(&self) -> u32 {
1746        self.num_bump_calls
1747    }
1748
1749    /// For interpolated `self.token`, returns a span of the fragment to which
1750    /// the interpolated token refers. For all other tokens this is just a
1751    /// regular span. It is particularly important to use this for identifiers
1752    /// and lifetimes for which spans affect name resolution and edition
1753    /// checks. Note that keywords are also identifiers, so they should use
1754    /// this if they keep spans or perform edition checks.
1755    pub fn token_uninterpolated_span(&self) -> Span {
1756        match &self.token.kind {
1757            token::NtIdent(ident, _) | token::NtLifetime(ident, _) => ident.span,
1758            token::OpenInvisible(InvisibleOrigin::MetaVar(_)) => self.look_ahead(1, |t| t.span),
1759            _ => self.token.span,
1760        }
1761    }
1762
1763    /// Like `token_uninterpolated_span`, but works on `self.prev_token`.
1764    pub fn prev_token_uninterpolated_span(&self) -> Span {
1765        match &self.prev_token.kind {
1766            token::NtIdent(ident, _) | token::NtLifetime(ident, _) => ident.span,
1767            token::OpenInvisible(InvisibleOrigin::MetaVar(_)) => self.look_ahead(0, |t| t.span),
1768            _ => self.prev_token.span,
1769        }
1770    }
1771
1772    fn missing_semi_from_binop(
1773        &self,
1774        kind_desc: &str,
1775        expr: &Expr,
1776        decl_lo: Option<Span>,
1777    ) -> Option<(Span, ErrorGuaranteed)> {
1778        if self.token == TokenKind::Semi {
1779            return None;
1780        }
1781        if !self.may_recover() || expr.span.from_expansion() {
1782            return None;
1783        }
1784        let sm = self.psess.source_map();
1785        if let ExprKind::Binary(op, lhs, rhs) = &expr.kind
1786            && sm.is_multiline(lhs.span.shrink_to_hi().until(rhs.span.shrink_to_lo()))
1787            && #[allow(non_exhaustive_omitted_patterns)] match op.node {
    BinOpKind::Mul | BinOpKind::BitAnd => true,
    _ => false,
}matches!(op.node, BinOpKind::Mul | BinOpKind::BitAnd)
1788            && classify::expr_requires_semi_to_be_stmt(rhs)
1789        {
1790            let lhs_end_span = lhs.span.shrink_to_hi();
1791            let token_str = token_descr(&self.token);
1792            let mut err = self
1793                .dcx()
1794                .struct_span_err(lhs_end_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected `;`, found {0}",
                token_str))
    })format!("expected `;`, found {token_str}"));
1795            err.span_label(self.token.span, "unexpected token");
1796
1797            // Use the declaration start if provided, otherwise fall back to lhs_end_span.
1798            let continuation_start = decl_lo.unwrap_or(lhs_end_span);
1799            let continuation_span = continuation_start.until(rhs.span.shrink_to_hi());
1800            err.span_label(
1801                continuation_span,
1802                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("to finish parsing this {0}, expected this to be followed by a `;`",
                kind_desc))
    })format!(
1803                    "to finish parsing this {kind_desc}, expected this to be followed by a `;`",
1804                ),
1805            );
1806            let op_desc = match op.node {
1807                BinOpKind::BitAnd => "a bit-and",
1808                BinOpKind::Mul => "a multiplication",
1809                _ => "a binary",
1810            };
1811            let mut note_spans = MultiSpan::new();
1812            note_spans.push_span_label(lhs.span, "parsed as the left-hand expression");
1813            note_spans.push_span_label(rhs.span, "parsed as the right-hand expression");
1814            note_spans.push_span_label(op.span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this was parsed as {0}", op_desc))
    })format!("this was parsed as {op_desc}"));
1815            err.span_note(
1816                note_spans,
1817                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the {0} was parsed as having {1} binary expression",
                kind_desc, op_desc))
    })format!("the {kind_desc} was parsed as having {op_desc} binary expression"),
1818            );
1819
1820            err.span_suggestion(
1821                lhs_end_span,
1822                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("you may have meant to write a `;` to terminate the {0} earlier",
                kind_desc))
    })format!("you may have meant to write a `;` to terminate the {kind_desc} earlier"),
1823                ";",
1824                Applicability::MaybeIncorrect,
1825            );
1826            return Some((lhs.span, err.emit()));
1827        }
1828        None
1829    }
1830}
1831
1832// Metavar captures of various kinds.
1833#[derive(#[automatically_derived]
impl ::core::clone::Clone for ParseNtResult {
    #[inline]
    fn clone(&self) -> ParseNtResult {
        match self {
            ParseNtResult::Tt(__self_0) =>
                ParseNtResult::Tt(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Ident(__self_0, __self_1) =>
                ParseNtResult::Ident(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            ParseNtResult::Lifetime(__self_0, __self_1) =>
                ParseNtResult::Lifetime(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            ParseNtResult::Item(__self_0) =>
                ParseNtResult::Item(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Block(__self_0) =>
                ParseNtResult::Block(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Stmt(__self_0) =>
                ParseNtResult::Stmt(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Pat(__self_0, __self_1) =>
                ParseNtResult::Pat(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            ParseNtResult::Expr(__self_0, __self_1) =>
                ParseNtResult::Expr(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            ParseNtResult::Literal(__self_0) =>
                ParseNtResult::Literal(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Ty(__self_0) =>
                ParseNtResult::Ty(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Meta(__self_0) =>
                ParseNtResult::Meta(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Path(__self_0) =>
                ParseNtResult::Path(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Vis(__self_0) =>
                ParseNtResult::Vis(::core::clone::Clone::clone(__self_0)),
            ParseNtResult::Guard(__self_0) =>
                ParseNtResult::Guard(::core::clone::Clone::clone(__self_0)),
        }
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for ParseNtResult {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            ParseNtResult::Tt(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Tt",
                    &__self_0),
            ParseNtResult::Ident(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f, "Ident",
                    __self_0, &__self_1),
            ParseNtResult::Lifetime(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f,
                    "Lifetime", __self_0, &__self_1),
            ParseNtResult::Item(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Item",
                    &__self_0),
            ParseNtResult::Block(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Block",
                    &__self_0),
            ParseNtResult::Stmt(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Stmt",
                    &__self_0),
            ParseNtResult::Pat(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f, "Pat",
                    __self_0, &__self_1),
            ParseNtResult::Expr(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f, "Expr",
                    __self_0, &__self_1),
            ParseNtResult::Literal(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "Literal", &__self_0),
            ParseNtResult::Ty(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Ty",
                    &__self_0),
            ParseNtResult::Meta(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Meta",
                    &__self_0),
            ParseNtResult::Path(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Path",
                    &__self_0),
            ParseNtResult::Vis(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Vis",
                    &__self_0),
            ParseNtResult::Guard(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Guard",
                    &__self_0),
        }
    }
}Debug)]
1834pub enum ParseNtResult {
1835    Tt(TokenTree),
1836    Ident(Ident, IdentIsRaw),
1837    Lifetime(Ident, IdentIsRaw),
1838    Item(Box<ast::Item>),
1839    Block(Box<ast::Block>),
1840    Stmt(Box<ast::Stmt>),
1841    Pat(Box<ast::Pat>, NtPatKind),
1842    Expr(Box<ast::Expr>, NtExprKind),
1843    Literal(Box<ast::Expr>),
1844    Ty(Box<ast::Ty>),
1845    Meta(Box<ast::AttrItem>),
1846    Path(Box<ast::Path>),
1847    Vis(Box<ast::Visibility>),
1848    Guard(Box<ast::Guard>),
1849}