1use std::mem::take;
2use std::ops::{Deref, DerefMut};
3
4use ast::token::IdentIsRaw;
5use rustc_ast::token::{self, Lit, LitKind, Token, TokenKind};
6use rustc_ast::util::parser::AssocOp;
7use rustc_ast::{
8 self as ast, AngleBracketedArg, AngleBracketedArgs, AnonConst, AttrVec, BinOpKind, BindingMode,
9 Block, BlockCheckMode, Expr, ExprKind, GenericArg, Generics, Item, ItemKind,
10 MgcaDisambiguation, Param, Pat, PatKind, Path, PathSegment, QSelf, Recovered, Ty, TyKind,
11};
12use rustc_ast_pretty::pprust;
13use rustc_data_structures::fx::FxHashSet;
14use rustc_errors::{
15 Applicability, Diag, DiagCtxtHandle, ErrorGuaranteed, PResult, Subdiagnostic, Suggestions, msg,
16 pluralize,
17};
18use rustc_session::errors::ExprParenthesesNeeded;
19use rustc_span::symbol::used_keywords;
20use rustc_span::{BytePos, DUMMY_SP, Ident, Span, SpanSnippetError, Spanned, Symbol, kw, sym};
21use thin_vec::{ThinVec, thin_vec};
22use tracing::{debug, trace};
23
24use super::pat::Expected;
25use super::{
26 BlockMode, CommaRecoveryMode, ExpTokenPair, Parser, PathStyle, Restrictions, SemiColonMode,
27 SeqSep, TokenType,
28};
29use crate::errors::{
30 AddParen, AmbiguousPlus, AsyncMoveBlockIn2015, AsyncUseBlockIn2015, AttributeOnParamType,
31 AwaitSuggestion, BadQPathStage2, BadTypePlus, BadTypePlusSub, ColonAsSemi,
32 ComparisonOperatorsCannotBeChained, ComparisonOperatorsCannotBeChainedSugg,
33 DocCommentDoesNotDocumentAnything, DocCommentOnParamType, DoubleColonInBound,
34 ExpectedIdentifier, ExpectedSemi, ExpectedSemiSugg, GenericParamsWithoutAngleBrackets,
35 GenericParamsWithoutAngleBracketsSugg, HelpIdentifierStartsWithNumber, HelpUseLatestEdition,
36 InInTypo, IncorrectAwait, IncorrectSemicolon, IncorrectUseOfAwait, IncorrectUseOfUse,
37 MisspelledKw, PatternMethodParamWithoutBody, QuestionMarkInType, QuestionMarkInTypeSugg,
38 SelfParamNotFirst, StructLiteralBodyWithoutPath, StructLiteralBodyWithoutPathSugg,
39 SuggAddMissingLetStmt, SuggEscapeIdentifier, SuggRemoveComma, TernaryOperator,
40 TernaryOperatorSuggestion, UnexpectedConstInGenericParam, UnexpectedConstParamDeclaration,
41 UnexpectedConstParamDeclarationSugg, UnmatchedAngleBrackets, UseEqInstead, WrapType,
42};
43use crate::exp;
44use crate::parser::FnContext;
45use crate::parser::attr::InnerAttrPolicy;
46use crate::parser::item::IsDotDotDot;
47
48pub(super) fn dummy_arg(ident: Ident, guar: ErrorGuaranteed) -> Param {
50 let pat = Box::new(Pat {
51 id: ast::DUMMY_NODE_ID,
52 kind: PatKind::Ident(BindingMode::NONE, ident, None),
53 span: ident.span,
54 tokens: None,
55 });
56 let ty = Ty { kind: TyKind::Err(guar), span: ident.span, id: ast::DUMMY_NODE_ID, tokens: None };
57 Param {
58 attrs: AttrVec::default(),
59 id: ast::DUMMY_NODE_ID,
60 pat,
61 span: ident.span,
62 ty: Box::new(ty),
63 is_placeholder: false,
64 }
65}
66
67pub(super) trait RecoverQPath: Sized + 'static {
68 const PATH_STYLE: PathStyle = PathStyle::Expr;
69 fn to_ty(&self) -> Option<Box<Ty>>;
70 fn recovered(qself: Option<Box<QSelf>>, path: ast::Path) -> Self;
71}
72
73impl<T: RecoverQPath> RecoverQPath for Box<T> {
74 const PATH_STYLE: PathStyle = T::PATH_STYLE;
75 fn to_ty(&self) -> Option<Box<Ty>> {
76 T::to_ty(self)
77 }
78 fn recovered(qself: Option<Box<QSelf>>, path: ast::Path) -> Self {
79 Box::new(T::recovered(qself, path))
80 }
81}
82
83impl RecoverQPath for Ty {
84 const PATH_STYLE: PathStyle = PathStyle::Type;
85 fn to_ty(&self) -> Option<Box<Ty>> {
86 Some(Box::new(self.clone()))
87 }
88 fn recovered(qself: Option<Box<QSelf>>, path: ast::Path) -> Self {
89 Self {
90 span: path.span,
91 kind: TyKind::Path(qself, path),
92 id: ast::DUMMY_NODE_ID,
93 tokens: None,
94 }
95 }
96}
97
98impl RecoverQPath for Pat {
99 const PATH_STYLE: PathStyle = PathStyle::Pat;
100 fn to_ty(&self) -> Option<Box<Ty>> {
101 self.to_ty()
102 }
103 fn recovered(qself: Option<Box<QSelf>>, path: ast::Path) -> Self {
104 Self {
105 span: path.span,
106 kind: PatKind::Path(qself, path),
107 id: ast::DUMMY_NODE_ID,
108 tokens: None,
109 }
110 }
111}
112
113impl RecoverQPath for Expr {
114 fn to_ty(&self) -> Option<Box<Ty>> {
115 self.to_ty()
116 }
117 fn recovered(qself: Option<Box<QSelf>>, path: ast::Path) -> Self {
118 Self {
119 span: path.span,
120 kind: ExprKind::Path(qself, path),
121 attrs: AttrVec::new(),
122 id: ast::DUMMY_NODE_ID,
123 tokens: None,
124 }
125 }
126}
127
128pub(crate) enum ConsumeClosingDelim {
130 Yes,
131 No,
132}
133
134#[derive(#[automatically_derived]
impl ::core::clone::Clone for AttemptLocalParseRecovery {
#[inline]
fn clone(&self) -> AttemptLocalParseRecovery { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for AttemptLocalParseRecovery { }Copy)]
135pub enum AttemptLocalParseRecovery {
136 Yes,
137 No,
138}
139
140impl AttemptLocalParseRecovery {
141 pub(super) fn yes(&self) -> bool {
142 match self {
143 AttemptLocalParseRecovery::Yes => true,
144 AttemptLocalParseRecovery::No => false,
145 }
146 }
147
148 pub(super) fn no(&self) -> bool {
149 match self {
150 AttemptLocalParseRecovery::Yes => false,
151 AttemptLocalParseRecovery::No => true,
152 }
153 }
154}
155
156#[derive(#[automatically_derived]
impl ::core::fmt::Debug for IncDecRecovery {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field3_finish(f,
"IncDecRecovery", "standalone", &self.standalone, "op", &self.op,
"fixity", &&self.fixity)
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for IncDecRecovery { }Copy, #[automatically_derived]
impl ::core::clone::Clone for IncDecRecovery {
#[inline]
fn clone(&self) -> IncDecRecovery {
let _: ::core::clone::AssertParamIsClone<IsStandalone>;
let _: ::core::clone::AssertParamIsClone<IncOrDec>;
let _: ::core::clone::AssertParamIsClone<UnaryFixity>;
*self
}
}Clone)]
159struct IncDecRecovery {
160 standalone: IsStandalone,
162 op: IncOrDec,
164 fixity: UnaryFixity,
166}
167
168#[derive(#[automatically_derived]
impl ::core::fmt::Debug for IsStandalone {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
IsStandalone::Standalone => "Standalone",
IsStandalone::Subexpr => "Subexpr",
})
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for IsStandalone { }Copy, #[automatically_derived]
impl ::core::clone::Clone for IsStandalone {
#[inline]
fn clone(&self) -> IsStandalone { *self }
}Clone)]
170enum IsStandalone {
171 Standalone,
173 Subexpr,
175}
176
177#[derive(#[automatically_derived]
impl ::core::fmt::Debug for IncOrDec {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self { IncOrDec::Inc => "Inc", IncOrDec::Dec => "Dec", })
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for IncOrDec { }Copy, #[automatically_derived]
impl ::core::clone::Clone for IncOrDec {
#[inline]
fn clone(&self) -> IncOrDec { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for IncOrDec {
#[inline]
fn eq(&self, other: &IncOrDec) -> 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 IncOrDec {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
178enum IncOrDec {
179 Inc,
180 Dec,
181}
182
183#[derive(#[automatically_derived]
impl ::core::fmt::Debug for UnaryFixity {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
UnaryFixity::Pre => "Pre",
UnaryFixity::Post => "Post",
})
}
}Debug, #[automatically_derived]
impl ::core::marker::Copy for UnaryFixity { }Copy, #[automatically_derived]
impl ::core::clone::Clone for UnaryFixity {
#[inline]
fn clone(&self) -> UnaryFixity { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for UnaryFixity {
#[inline]
fn eq(&self, other: &UnaryFixity) -> 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 UnaryFixity {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
184enum UnaryFixity {
185 Pre,
186 Post,
187}
188
189impl IncOrDec {
190 fn chr(&self) -> char {
191 match self {
192 Self::Inc => '+',
193 Self::Dec => '-',
194 }
195 }
196
197 fn name(&self) -> &'static str {
198 match self {
199 Self::Inc => "increment",
200 Self::Dec => "decrement",
201 }
202 }
203}
204
205impl std::fmt::Display for UnaryFixity {
206 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
207 match self {
208 Self::Pre => f.write_fmt(format_args!("prefix"))write!(f, "prefix"),
209 Self::Post => f.write_fmt(format_args!("postfix"))write!(f, "postfix"),
210 }
211 }
212}
213
214fn find_similar_kw(lookup: Ident, candidates: &[Symbol]) -> Option<MisspelledKw> {
219 lookup.name.find_similar(candidates).map(|(similar_kw, is_incorrect_case)| MisspelledKw {
220 similar_kw: similar_kw.to_string(),
221 is_incorrect_case,
222 span: lookup.span,
223 })
224}
225
226struct MultiSugg {
227 msg: String,
228 patches: Vec<(Span, String)>,
229 applicability: Applicability,
230}
231
232impl MultiSugg {
233 fn emit(self, err: &mut Diag<'_>) {
234 err.multipart_suggestion(self.msg, self.patches, self.applicability);
235 }
236
237 fn emit_verbose(self, err: &mut Diag<'_>) {
238 err.multipart_suggestion(self.msg, self.patches, self.applicability);
239 }
240}
241
242pub struct SnapshotParser<'a> {
246 parser: Parser<'a>,
247}
248
249impl<'a> Deref for SnapshotParser<'a> {
250 type Target = Parser<'a>;
251
252 fn deref(&self) -> &Self::Target {
253 &self.parser
254 }
255}
256
257impl<'a> DerefMut for SnapshotParser<'a> {
258 fn deref_mut(&mut self) -> &mut Self::Target {
259 &mut self.parser
260 }
261}
262
263impl<'a> Parser<'a> {
264 pub fn dcx(&self) -> DiagCtxtHandle<'a> {
265 self.psess.dcx()
266 }
267
268 pub fn restore_snapshot(&mut self, snapshot: SnapshotParser<'a>) {
270 *self = snapshot.parser;
271 }
272
273 pub fn create_snapshot_for_diagnostic(&self) -> SnapshotParser<'a> {
275 let snapshot = self.clone();
276 SnapshotParser { parser: snapshot }
277 }
278
279 pub(super) fn span_to_snippet(&self, span: Span) -> Result<String, SpanSnippetError> {
280 self.psess.source_map().span_to_snippet(span)
281 }
282
283 pub(super) fn expected_ident_found(
287 &mut self,
288 recover: bool,
289 ) -> PResult<'a, (Ident, IdentIsRaw)> {
290 let valid_follow = &[
291 TokenKind::Eq,
292 TokenKind::Colon,
293 TokenKind::Comma,
294 TokenKind::Semi,
295 TokenKind::PathSep,
296 TokenKind::OpenBrace,
297 TokenKind::OpenParen,
298 TokenKind::CloseBrace,
299 TokenKind::CloseParen,
300 ];
301 if let TokenKind::DocComment(..) = self.prev_token.kind
302 && valid_follow.contains(&self.token.kind)
303 {
304 let err = self.dcx().create_err(DocCommentDoesNotDocumentAnything {
305 span: self.prev_token.span,
306 missing_comma: None,
307 });
308 return Err(err);
309 }
310
311 let mut recovered_ident = None;
312 let bad_token = self.token;
315
316 let suggest_raw = if let Some((ident, IdentIsRaw::No)) = self.token.ident()
318 && ident.is_raw_guess()
319 && self.look_ahead(1, |t| valid_follow.contains(&t.kind))
320 {
321 recovered_ident = Some((ident, IdentIsRaw::Yes));
322
323 let ident_name = ident.name.to_string();
326
327 Some(SuggEscapeIdentifier { span: ident.span.shrink_to_lo(), ident_name })
328 } else {
329 None
330 };
331
332 let suggest_remove_comma =
333 if self.token == token::Comma && self.look_ahead(1, |t| t.is_ident()) {
334 if recover {
335 self.bump();
336 recovered_ident = self.ident_or_err(false).ok();
337 };
338
339 Some(SuggRemoveComma { span: bad_token.span })
340 } else {
341 None
342 };
343
344 let help_cannot_start_number = self.is_lit_bad_ident().map(|(len, valid_portion)| {
345 let (invalid, valid) = self.token.span.split_at(len as u32);
346
347 recovered_ident = Some((Ident::new(valid_portion, valid), IdentIsRaw::No));
348
349 HelpIdentifierStartsWithNumber { num_span: invalid }
350 });
351
352 let err = ExpectedIdentifier {
353 span: bad_token.span,
354 token: bad_token,
355 suggest_raw,
356 suggest_remove_comma,
357 help_cannot_start_number,
358 };
359 let mut err = self.dcx().create_err(err);
360
361 if self.token == token::Lt {
365 let valid_prev_keywords =
367 [kw::Fn, kw::Type, kw::Struct, kw::Enum, kw::Union, kw::Trait];
368
369 let maybe_keyword = self.prev_token;
375 if valid_prev_keywords.into_iter().any(|x| maybe_keyword.is_keyword(x)) {
376 match self.parse_generics() {
379 Ok(generic) => {
380 if let TokenKind::Ident(symbol, _) = maybe_keyword.kind {
381 let ident_name = symbol;
382 if !self.look_ahead(1, |t| *t == token::Lt)
388 && let Ok(snippet) =
389 self.psess.source_map().span_to_snippet(generic.span)
390 {
391 err.multipart_suggestion(
392 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("place the generic parameter name after the {0} name",
ident_name))
})format!("place the generic parameter name after the {ident_name} name"),
393 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(self.token.span.shrink_to_hi(), snippet),
(generic.span, String::new())]))vec![
394 (self.token.span.shrink_to_hi(), snippet),
395 (generic.span, String::new())
396 ],
397 Applicability::MaybeIncorrect,
398 );
399 } else {
400 err.help(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("place the generic parameter name after the {0} name",
ident_name))
})format!(
401 "place the generic parameter name after the {ident_name} name"
402 ));
403 }
404 }
405 }
406 Err(err) => {
407 err.cancel();
411 }
412 }
413 }
414 }
415
416 if let Some(recovered_ident) = recovered_ident
417 && recover
418 {
419 err.emit();
420 Ok(recovered_ident)
421 } else {
422 Err(err)
423 }
424 }
425
426 pub(super) fn expected_ident_found_err(&mut self) -> Diag<'a> {
427 self.expected_ident_found(false).unwrap_err()
428 }
429
430 pub(super) fn is_lit_bad_ident(&mut self) -> Option<(usize, Symbol)> {
436 if let token::Literal(Lit {
440 kind: token::LitKind::Integer | token::LitKind::Float,
441 symbol,
442 suffix: Some(suffix), }) = self.token.kind
444 && rustc_ast::MetaItemLit::from_token(&self.token).is_none()
445 {
446 Some((symbol.as_str().len(), suffix))
447 } else {
448 None
449 }
450 }
451
452 pub(super) fn expected_one_of_not_found(
453 &mut self,
454 edible: &[ExpTokenPair],
455 inedible: &[ExpTokenPair],
456 ) -> PResult<'a, ErrorGuaranteed> {
457 {
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/diagnostics.rs:457",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(457u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("expected_one_of_not_found(edible: {0:?}, inedible: {1:?})",
edible, inedible) as &dyn Value))])
});
} else { ; }
};debug!("expected_one_of_not_found(edible: {:?}, inedible: {:?})", edible, inedible);
458 fn tokens_to_string(tokens: &[TokenType]) -> String {
459 let mut i = tokens.iter();
460 let b = i.next().map_or_else(String::new, |t| t.to_string());
462 i.enumerate().fold(b, |mut b, (i, a)| {
463 if tokens.len() > 2 && i == tokens.len() - 2 {
464 b.push_str(", or ");
465 } else if tokens.len() == 2 && i == tokens.len() - 2 {
466 b.push_str(" or ");
467 } else {
468 b.push_str(", ");
469 }
470 b.push_str(&a.to_string());
471 b
472 })
473 }
474
475 for exp in edible.iter().chain(inedible.iter()) {
476 self.expected_token_types.insert(exp.token_type);
477 }
478 let mut expected: Vec<_> = self.expected_token_types.iter().collect();
479 expected.sort_by_cached_key(|x| x.to_string());
480 expected.dedup();
481
482 let sm = self.psess.source_map();
483
484 if expected.contains(&TokenType::Semi) {
486 if self.prev_token == token::Question
489 && let Err(e) = self.maybe_recover_from_ternary_operator(None)
490 {
491 return Err(e);
492 }
493
494 if self.token.span == DUMMY_SP || self.prev_token.span == DUMMY_SP {
495 } else if !sm.is_multiline(self.prev_token.span.until(self.token.span)) {
497 } else if [token::Comma, token::Colon].contains(&self.token.kind)
499 && self.prev_token == token::CloseParen
500 {
501 } else if self.look_ahead(1, |t| {
510 t == &token::CloseBrace || t.can_begin_expr() && *t != token::Colon
511 }) && [token::Comma, token::Colon].contains(&self.token.kind)
512 {
513 let guar = self.dcx().emit_err(ExpectedSemi {
520 span: self.token.span,
521 token: self.token,
522 unexpected_token_label: None,
523 sugg: ExpectedSemiSugg::ChangeToSemi(self.token.span),
524 });
525 self.bump();
526 return Ok(guar);
527 } else if self.look_ahead(0, |t| {
528 t == &token::CloseBrace
529 || ((t.can_begin_expr() || t.can_begin_item())
530 && t != &token::Semi
531 && t != &token::Pound)
532 || (sm.is_multiline(
534 self.prev_token.span.shrink_to_hi().until(self.token.span.shrink_to_lo()),
535 ) && t == &token::Pound)
536 }) && !expected.contains(&TokenType::Comma)
537 {
538 let span = self.prev_token.span.shrink_to_hi();
544 let guar = self.dcx().emit_err(ExpectedSemi {
545 span,
546 token: self.token,
547 unexpected_token_label: Some(self.token.span),
548 sugg: ExpectedSemiSugg::AddSemi(span),
549 });
550 return Ok(guar);
551 }
552 }
553
554 if self.token == TokenKind::EqEq
555 && self.prev_token.is_ident()
556 && expected.contains(&TokenType::Eq)
557 {
558 return Err(self.dcx().create_err(UseEqInstead { span: self.token.span }));
560 }
561
562 if (self.token.is_keyword(kw::Move) || self.token.is_keyword(kw::Use))
563 && self.prev_token.is_keyword(kw::Async)
564 {
565 let span = self.prev_token.span.to(self.token.span);
567 if self.token.is_keyword(kw::Move) {
568 return Err(self.dcx().create_err(AsyncMoveBlockIn2015 { span }));
569 } else {
570 return Err(self.dcx().create_err(AsyncUseBlockIn2015 { span }));
572 }
573 }
574
575 let expect = tokens_to_string(&expected);
576 let actual = super::token_descr(&self.token);
577 let (msg_exp, (label_sp, label_exp)) = if expected.len() > 1 {
578 let fmt = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected one of {0}, found {1}",
expect, actual))
})format!("expected one of {expect}, found {actual}");
579 let short_expect = if expected.len() > 6 {
580 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} possible tokens",
expected.len()))
})format!("{} possible tokens", expected.len())
581 } else {
582 expect
583 };
584 (fmt, (self.prev_token.span.shrink_to_hi(), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected one of {0}",
short_expect))
})format!("expected one of {short_expect}")))
585 } else if expected.is_empty() {
586 (
587 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("unexpected token: {0}", actual))
})format!("unexpected token: {actual}"),
588 (self.prev_token.span, "unexpected token after this".to_string()),
589 )
590 } else {
591 (
592 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected {0}, found {1}", expect,
actual))
})format!("expected {expect}, found {actual}"),
593 (self.prev_token.span.shrink_to_hi(), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected {0}", expect))
})format!("expected {expect}")),
594 )
595 };
596 self.last_unexpected_token_span = Some(self.token.span);
597 let mut err = self.dcx().struct_span_err(self.token.span, msg_exp);
599
600 self.label_expected_raw_ref(&mut err);
601
602 if self.token == token::FatArrow
604 && expected.iter().any(|tok| #[allow(non_exhaustive_omitted_patterns)] match tok {
TokenType::Operator | TokenType::Le => true,
_ => false,
}matches!(tok, TokenType::Operator | TokenType::Le))
605 && !expected.iter().any(|tok| #[allow(non_exhaustive_omitted_patterns)] match tok {
TokenType::FatArrow | TokenType::Comma => true,
_ => false,
}matches!(tok, TokenType::FatArrow | TokenType::Comma))
606 {
607 err.span_suggestion(
608 self.token.span,
609 "you might have meant to write a \"greater than or equal to\" comparison",
610 ">=",
611 Applicability::MaybeIncorrect,
612 );
613 }
614
615 if let TokenKind::Ident(symbol, _) = &self.prev_token.kind {
616 if ["def", "fun", "func", "function"].contains(&symbol.as_str()) {
617 err.span_suggestion_short(
618 self.prev_token.span,
619 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("write `fn` instead of `{0}` to declare a function",
symbol))
})format!("write `fn` instead of `{symbol}` to declare a function"),
620 "fn",
621 Applicability::MachineApplicable,
622 );
623 }
624 }
625
626 if let TokenKind::Ident(prev, _) = &self.prev_token.kind
627 && let TokenKind::Ident(cur, _) = &self.token.kind
628 {
629 let concat = Symbol::intern(&::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}", prev, cur))
})format!("{prev}{cur}"));
630 let ident = Ident::new(concat, DUMMY_SP);
631 if ident.is_used_keyword() || ident.is_reserved() || ident.is_raw_guess() {
632 let concat_span = self.prev_token.span.to(self.token.span);
633 err.span_suggestion_verbose(
634 concat_span,
635 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("consider removing the space to spell keyword `{0}`",
concat))
})format!("consider removing the space to spell keyword `{concat}`"),
636 concat,
637 Applicability::MachineApplicable,
638 );
639 }
640 }
641
642 if ((self.prev_token == TokenKind::Ident(sym::character('c'), IdentIsRaw::No)
650 && #[allow(non_exhaustive_omitted_patterns)] match &self.token.kind {
TokenKind::Literal(token::Lit { kind: token::Str, .. }) => true,
_ => false,
}matches!(&self.token.kind, TokenKind::Literal(token::Lit { kind: token::Str, .. })))
651 || (self.prev_token == TokenKind::Ident(sym::cr, IdentIsRaw::No)
652 && #[allow(non_exhaustive_omitted_patterns)] match &self.token.kind {
TokenKind::Literal(token::Lit { kind: token::Str, .. }) | token::Pound =>
true,
_ => false,
}matches!(
653 &self.token.kind,
654 TokenKind::Literal(token::Lit { kind: token::Str, .. }) | token::Pound
655 )))
656 && self.prev_token.span.hi() == self.token.span.lo()
657 && !self.token.span.at_least_rust_2021()
658 {
659 err.note("you may be trying to write a c-string literal");
660 err.note("c-string literals require Rust 2021 or later");
661 err.subdiagnostic(HelpUseLatestEdition::new());
662 }
663
664 if self.prev_token.is_ident_named(sym::public)
666 && (self.token.can_begin_item() || self.token == TokenKind::OpenParen)
667 {
668 err.span_suggestion_short(
669 self.prev_token.span,
670 "write `pub` instead of `public` to make the item public",
671 "pub",
672 Applicability::MachineApplicable,
673 );
674 }
675
676 if let token::DocComment(kind, style, _) = self.token.kind {
677 if !expected.contains(&TokenType::Comma) {
687 let pos = self.token.span.lo() + BytePos(2);
689 let span = self.token.span.with_lo(pos).with_hi(pos);
690 err.span_suggestion_verbose(
691 span,
692 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("add a space before {0} to write a regular comment",
match (kind, style) {
(token::CommentKind::Line, ast::AttrStyle::Inner) => "`!`",
(token::CommentKind::Block, ast::AttrStyle::Inner) => "`!`",
(token::CommentKind::Line, ast::AttrStyle::Outer) =>
"the last `/`",
(token::CommentKind::Block, ast::AttrStyle::Outer) =>
"the last `*`",
}))
})format!(
693 "add a space before {} to write a regular comment",
694 match (kind, style) {
695 (token::CommentKind::Line, ast::AttrStyle::Inner) => "`!`",
696 (token::CommentKind::Block, ast::AttrStyle::Inner) => "`!`",
697 (token::CommentKind::Line, ast::AttrStyle::Outer) => "the last `/`",
698 (token::CommentKind::Block, ast::AttrStyle::Outer) => "the last `*`",
699 },
700 ),
701 " ".to_string(),
702 Applicability::MaybeIncorrect,
703 );
704 }
705 }
706
707 let sp = if self.token == token::Eof {
708 self.prev_token.span
710 } else {
711 label_sp
712 };
713
714 if self.check_too_many_raw_str_terminators(&mut err) {
715 if expected.contains(&TokenType::Semi) && self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Semi,
token_type: crate::parser::token_type::TokenType::Semi,
}exp!(Semi)) {
716 let guar = err.emit();
717 return Ok(guar);
718 } else {
719 return Err(err);
720 }
721 }
722
723 if self.prev_token.span == DUMMY_SP {
724 err.span_label(self.token.span, label_exp);
727 } else if !sm.is_multiline(self.token.span.shrink_to_hi().until(sp.shrink_to_lo())) {
728 err.span_label(self.token.span, label_exp);
741 } else {
742 err.span_label(sp, label_exp);
743 err.span_label(self.token.span, "unexpected token");
744 }
745
746 if let Suggestions::Enabled(list) = &err.suggestions
748 && list.is_empty()
749 {
750 self.check_for_misspelled_kw(&mut err, &expected);
751 }
752 Err(err)
753 }
754
755 pub(super) fn label_expected_raw_ref(&mut self, err: &mut Diag<'_>) {
760 if self.prev_token.is_keyword(kw::Raw)
761 && self.expected_token_types.contains(TokenType::KwMut)
762 && self.expected_token_types.contains(TokenType::KwConst)
763 && self.token.can_begin_expr()
764 {
765 err.span_suggestions(
766 self.prev_token.span.shrink_to_hi(),
767 "`&raw` must be followed by `const` or `mut` to be a raw reference expression",
768 [" const".to_string(), " mut".to_string()],
769 Applicability::MaybeIncorrect,
770 );
771 }
772 }
773
774 fn check_for_misspelled_kw(&self, err: &mut Diag<'_>, expected: &[TokenType]) {
777 let Some((curr_ident, _)) = self.token.ident() else {
778 return;
779 };
780 let expected_token_types: &[TokenType] =
781 expected.len().checked_sub(10).map_or(&expected, |index| &expected[index..]);
782 let expected_keywords: Vec<Symbol> =
783 expected_token_types.iter().filter_map(|token| token.is_keyword()).collect();
784
785 if !expected_keywords.is_empty()
790 && !curr_ident.is_used_keyword()
791 && let Some(misspelled_kw) = find_similar_kw(curr_ident, &expected_keywords)
792 {
793 err.subdiagnostic(misspelled_kw);
794 err.seal_suggestions();
797 } else if let Some((prev_ident, _)) = self.prev_token.ident()
798 && !prev_ident.is_used_keyword()
799 {
800 let all_keywords = used_keywords(|| prev_ident.span.edition());
805
806 if let Some(misspelled_kw) = find_similar_kw(prev_ident, &all_keywords) {
811 err.subdiagnostic(misspelled_kw);
812 err.seal_suggestions();
815 }
816 }
817 }
818
819 pub(super) fn attr_on_non_tail_expr(&self, expr: &Expr) -> ErrorGuaranteed {
821 let span = self.prev_token.span.shrink_to_hi();
823 let mut err = self.dcx().create_err(ExpectedSemi {
824 span,
825 token: self.token,
826 unexpected_token_label: Some(self.token.span),
827 sugg: ExpectedSemiSugg::AddSemi(span),
828 });
829 let attr_span = match &expr.attrs[..] {
830 [] => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
831 [only] => only.span,
832 [first, rest @ ..] => {
833 for attr in rest {
834 err.span_label(attr.span, "");
835 }
836 first.span
837 }
838 };
839 err.span_label(
840 attr_span,
841 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("only `;` terminated statements or tail expressions are allowed after {0}",
if expr.attrs.len() == 1 {
"this attribute"
} else { "these attributes" }))
})format!(
842 "only `;` terminated statements or tail expressions are allowed after {}",
843 if expr.attrs.len() == 1 { "this attribute" } else { "these attributes" },
844 ),
845 );
846 if self.token == token::Pound && self.look_ahead(1, |t| *t == token::OpenBracket) {
847 err.span_label(span, "expected `;` here");
853 err.multipart_suggestion(
854 "alternatively, consider surrounding the expression with a block",
855 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(expr.span.shrink_to_lo(), "{ ".to_string()),
(expr.span.shrink_to_hi(), " }".to_string())]))vec![
856 (expr.span.shrink_to_lo(), "{ ".to_string()),
857 (expr.span.shrink_to_hi(), " }".to_string()),
858 ],
859 Applicability::MachineApplicable,
860 );
861
862 let mut snapshot = self.create_snapshot_for_diagnostic();
864 if let [attr] = &expr.attrs[..]
865 && let ast::AttrKind::Normal(attr_kind) = &attr.kind
866 && let [segment] = &attr_kind.item.path.segments[..]
867 && segment.ident.name == sym::cfg
868 && let Some(args_span) = attr_kind.item.args.span()
869 && let next_attr = match snapshot.parse_attribute(InnerAttrPolicy::Forbidden(None))
870 {
871 Ok(next_attr) => next_attr,
872 Err(inner_err) => {
873 inner_err.cancel();
874 return err.emit();
875 }
876 }
877 && let ast::AttrKind::Normal(next_attr_kind) = next_attr.kind
878 && let Some(next_attr_args_span) = next_attr_kind.item.args.span()
879 && let [next_segment] = &next_attr_kind.item.path.segments[..]
880 && segment.ident.name == sym::cfg
881 {
882 let next_expr = match snapshot.parse_expr() {
883 Ok(next_expr) => next_expr,
884 Err(inner_err) => {
885 inner_err.cancel();
886 return err.emit();
887 }
888 };
889 let margin = self.psess.source_map().span_to_margin(next_expr.span).unwrap_or(0);
896 let sugg = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(attr.span.with_hi(segment.span().hi()), "if cfg!".to_string()),
(args_span.shrink_to_hi().with_hi(attr.span.hi()),
" {".to_string()),
(expr.span.shrink_to_lo(), " ".to_string()),
(next_attr.span.with_hi(next_segment.span().hi()),
"} else if cfg!".to_string()),
(next_attr_args_span.shrink_to_hi().with_hi(next_attr.span.hi()),
" {".to_string()),
(next_expr.span.shrink_to_lo(), " ".to_string()),
(next_expr.span.shrink_to_hi(),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("\n{0}}}",
" ".repeat(margin)))
}))]))vec![
897 (attr.span.with_hi(segment.span().hi()), "if cfg!".to_string()),
898 (args_span.shrink_to_hi().with_hi(attr.span.hi()), " {".to_string()),
899 (expr.span.shrink_to_lo(), " ".to_string()),
900 (
901 next_attr.span.with_hi(next_segment.span().hi()),
902 "} else if cfg!".to_string(),
903 ),
904 (
905 next_attr_args_span.shrink_to_hi().with_hi(next_attr.span.hi()),
906 " {".to_string(),
907 ),
908 (next_expr.span.shrink_to_lo(), " ".to_string()),
909 (next_expr.span.shrink_to_hi(), format!("\n{}}}", " ".repeat(margin))),
910 ];
911 err.multipart_suggestion(
912 "it seems like you are trying to provide different expressions depending on \
913 `cfg`, consider using `if cfg!(..)`",
914 sugg,
915 Applicability::MachineApplicable,
916 );
917 }
918 }
919
920 err.emit()
921 }
922
923 fn check_too_many_raw_str_terminators(&mut self, err: &mut Diag<'_>) -> bool {
924 let sm = self.psess.source_map();
925 match (&self.prev_token.kind, &self.token.kind) {
926 (
927 TokenKind::Literal(Lit {
928 kind: LitKind::StrRaw(n_hashes) | LitKind::ByteStrRaw(n_hashes),
929 ..
930 }),
931 TokenKind::Pound,
932 ) if !sm.is_multiline(
933 self.prev_token.span.shrink_to_hi().until(self.token.span.shrink_to_lo()),
934 ) =>
935 {
936 let n_hashes: u8 = *n_hashes;
937 err.primary_message("too many `#` when terminating raw string");
938 let str_span = self.prev_token.span;
939 let mut span = self.token.span;
940 let mut count = 0;
941 while self.token == TokenKind::Pound
942 && !sm.is_multiline(span.shrink_to_hi().until(self.token.span.shrink_to_lo()))
943 {
944 span = span.with_hi(self.token.span.hi());
945 self.bump();
946 count += 1;
947 }
948 err.span(span);
949 err.span_suggestion(
950 span,
951 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("remove the extra `#`{0}",
if count == 1 { "" } else { "s" }))
})format!("remove the extra `#`{}", pluralize!(count)),
952 "",
953 Applicability::MachineApplicable,
954 );
955 err.span_label(
956 str_span,
957 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this raw string started with {1} `#`{0}",
if n_hashes == 1 { "" } else { "s" }, n_hashes))
})format!("this raw string started with {n_hashes} `#`{}", pluralize!(n_hashes)),
958 );
959 true
960 }
961 _ => false,
962 }
963 }
964
965 pub(super) fn maybe_suggest_struct_literal(
966 &mut self,
967 lo: Span,
968 s: BlockCheckMode,
969 maybe_struct_name: token::Token,
970 ) -> Option<PResult<'a, Box<Block>>> {
971 if self.token.is_ident() && self.look_ahead(1, |t| t == &token::Colon) {
972 {
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/diagnostics.rs:976",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(976u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::tracing_core::field::FieldSet::new(&["maybe_struct_name",
"self.token"],
::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(&debug(&maybe_struct_name)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&self.token)
as &dyn Value))])
});
} else { ; }
};debug!(?maybe_struct_name, ?self.token);
977 let mut snapshot = self.create_snapshot_for_diagnostic();
978 let path = Path {
979 segments: ThinVec::new(),
980 span: self.prev_token.span.shrink_to_lo(),
981 tokens: None,
982 };
983 let struct_expr = snapshot.parse_expr_struct(None, path, false);
984 let block_tail = self.parse_block_tail(lo, s, AttemptLocalParseRecovery::No);
985 return Some(match (struct_expr, block_tail) {
986 (Ok(expr), Err(err)) => {
987 err.cancel();
996 self.restore_snapshot(snapshot);
997 let guar = self.dcx().emit_err(StructLiteralBodyWithoutPath {
998 span: expr.span,
999 sugg: StructLiteralBodyWithoutPathSugg {
1000 before: expr.span.shrink_to_lo(),
1001 after: expr.span.shrink_to_hi(),
1002 },
1003 });
1004 Ok(self.mk_block(
1005 {
let len = [()].len();
let mut vec = ::thin_vec::ThinVec::with_capacity(len);
vec.push(self.mk_stmt_err(expr.span, guar));
vec
}thin_vec![self.mk_stmt_err(expr.span, guar)],
1006 s,
1007 lo.to(self.prev_token.span),
1008 ))
1009 }
1010 (Err(err), Ok(tail)) => {
1011 err.cancel();
1013 Ok(tail)
1014 }
1015 (Err(snapshot_err), Err(err)) => {
1016 snapshot_err.cancel();
1018 self.consume_block(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::OpenBrace,
token_type: crate::parser::token_type::TokenType::OpenBrace,
}exp!(OpenBrace), crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::CloseBrace,
token_type: crate::parser::token_type::TokenType::CloseBrace,
}exp!(CloseBrace), ConsumeClosingDelim::Yes);
1019 Err(err)
1020 }
1021 (Ok(_), Ok(tail)) => Ok(tail),
1022 });
1023 }
1024 None
1025 }
1026
1027 pub(super) fn recover_closure_body(
1028 &mut self,
1029 mut err: Diag<'a>,
1030 before: token::Token,
1031 prev: token::Token,
1032 token: token::Token,
1033 lo: Span,
1034 decl_hi: Span,
1035 ) -> PResult<'a, Box<Expr>> {
1036 err.span_label(lo.to(decl_hi), "while parsing the body of this closure");
1037 let guar = match before.kind {
1038 token::OpenBrace if token.kind != token::OpenBrace => {
1039 err.multipart_suggestion(
1041 "you might have meant to open the body of the closure, instead of enclosing \
1042 the closure in a block",
1043 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(before.span, String::new()),
(prev.span.shrink_to_hi(), " {".to_string())]))vec![
1044 (before.span, String::new()),
1045 (prev.span.shrink_to_hi(), " {".to_string()),
1046 ],
1047 Applicability::MaybeIncorrect,
1048 );
1049 let guar = err.emit();
1050 self.eat_to_tokens(&[crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::CloseBrace,
token_type: crate::parser::token_type::TokenType::CloseBrace,
}exp!(CloseBrace)]);
1051 guar
1052 }
1053 token::OpenParen if token.kind != token::OpenBrace => {
1054 self.eat_to_tokens(&[crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::CloseParen,
token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen), crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Comma,
token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma)]);
1057
1058 err.multipart_suggestion(
1059 "you might have meant to open the body of the closure",
1060 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(prev.span.shrink_to_hi(), " {".to_string()),
(self.token.span.shrink_to_lo(), "}".to_string())]))vec![
1061 (prev.span.shrink_to_hi(), " {".to_string()),
1062 (self.token.span.shrink_to_lo(), "}".to_string()),
1063 ],
1064 Applicability::MaybeIncorrect,
1065 );
1066 err.emit()
1067 }
1068 _ if token.kind != token::OpenBrace => {
1069 err.multipart_suggestion(
1072 "you might have meant to open the body of the closure",
1073 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(prev.span.shrink_to_hi(), " {".to_string())]))vec![(prev.span.shrink_to_hi(), " {".to_string())],
1074 Applicability::HasPlaceholders,
1075 );
1076 return Err(err);
1077 }
1078 _ => return Err(err),
1079 };
1080 Ok(self.mk_expr_err(lo.to(self.token.span), guar))
1081 }
1082
1083 pub(super) fn eat_to_tokens(&mut self, closes: &[ExpTokenPair]) {
1086 if let Err(err) = self
1087 .parse_seq_to_before_tokens(closes, &[], SeqSep::none(), |p| Ok(p.parse_token_tree()))
1088 {
1089 err.cancel();
1090 }
1091 }
1092
1093 pub(super) fn check_trailing_angle_brackets(
1104 &mut self,
1105 segment: &PathSegment,
1106 end: &[ExpTokenPair],
1107 ) -> Option<ErrorGuaranteed> {
1108 if !self.may_recover() {
1109 return None;
1110 }
1111
1112 let parsed_angle_bracket_args =
1137 segment.args.as_ref().is_some_and(|args| args.is_angle_bracketed());
1138
1139 {
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/diagnostics.rs:1139",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(1139u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("check_trailing_angle_brackets: parsed_angle_bracket_args={0:?}",
parsed_angle_bracket_args) as &dyn Value))])
});
} else { ; }
};debug!(
1140 "check_trailing_angle_brackets: parsed_angle_bracket_args={:?}",
1141 parsed_angle_bracket_args,
1142 );
1143 if !parsed_angle_bracket_args {
1144 return None;
1145 }
1146
1147 let lo = self.token.span;
1150
1151 let mut position = 0;
1155
1156 let mut number_of_shr = 0;
1160 let mut number_of_gt = 0;
1161 while self.look_ahead(position, |t| {
1162 {
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/diagnostics.rs:1162",
"rustc_parse::parser::diagnostics", ::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(1162u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::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!("check_trailing_angle_brackets: t={0:?}",
t) as &dyn Value))])
});
} else { ; }
};trace!("check_trailing_angle_brackets: t={:?}", t);
1163 if *t == token::Shr {
1164 number_of_shr += 1;
1165 true
1166 } else if *t == token::Gt {
1167 number_of_gt += 1;
1168 true
1169 } else {
1170 false
1171 }
1172 }) {
1173 position += 1;
1174 }
1175
1176 {
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/diagnostics.rs:1177",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(1177u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("check_trailing_angle_brackets: number_of_gt={0:?} number_of_shr={1:?}",
number_of_gt, number_of_shr) as &dyn Value))])
});
} else { ; }
};debug!(
1178 "check_trailing_angle_brackets: number_of_gt={:?} number_of_shr={:?}",
1179 number_of_gt, number_of_shr,
1180 );
1181 if number_of_gt < 1 && number_of_shr < 1 {
1182 return None;
1183 }
1184
1185 if self.look_ahead(position, |t| {
1188 {
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/diagnostics.rs:1188",
"rustc_parse::parser::diagnostics", ::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(1188u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::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!("check_trailing_angle_brackets: t={0:?}",
t) as &dyn Value))])
});
} else { ; }
};trace!("check_trailing_angle_brackets: t={:?}", t);
1189 end.iter().any(|exp| exp.tok == t.kind)
1190 }) {
1191 self.eat_to_tokens(end);
1194 let span = lo.to(self.prev_token.span);
1195
1196 let num_extra_brackets = number_of_gt + number_of_shr * 2;
1197 return Some(self.dcx().emit_err(UnmatchedAngleBrackets { span, num_extra_brackets }));
1198 }
1199 None
1200 }
1201
1202 pub(super) fn check_turbofish_missing_angle_brackets(&mut self, segment: &mut PathSegment) {
1205 if !self.may_recover() {
1206 return;
1207 }
1208
1209 if self.token == token::PathSep && segment.args.is_none() {
1210 let snapshot = self.create_snapshot_for_diagnostic();
1211 self.bump();
1212 let lo = self.token.span;
1213 match self.parse_angle_args(None) {
1214 Ok(args) => {
1215 let span = lo.to(self.prev_token.span);
1216 let mut trailing_span = self.prev_token.span.shrink_to_hi();
1218 while self.token == token::Shr || self.token == token::Gt {
1219 trailing_span = trailing_span.to(self.token.span);
1220 self.bump();
1221 }
1222 if self.token == token::OpenParen {
1223 segment.args = Some(AngleBracketedArgs { args, span }.into());
1225
1226 self.dcx().emit_err(GenericParamsWithoutAngleBrackets {
1227 span,
1228 sugg: GenericParamsWithoutAngleBracketsSugg {
1229 left: span.shrink_to_lo(),
1230 right: trailing_span,
1231 },
1232 });
1233 } else {
1234 self.restore_snapshot(snapshot);
1236 }
1237 }
1238 Err(err) => {
1239 err.cancel();
1242 self.restore_snapshot(snapshot);
1243 }
1244 }
1245 }
1246 }
1247
1248 pub(super) fn check_mistyped_turbofish_with_multiple_type_params(
1251 &mut self,
1252 mut e: Diag<'a>,
1253 expr: &mut Box<Expr>,
1254 ) -> PResult<'a, ErrorGuaranteed> {
1255 if let ExprKind::Binary(binop, _, _) = &expr.kind
1256 && let ast::BinOpKind::Lt = binop.node
1257 && self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Comma,
token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma))
1258 {
1259 let x = self.parse_seq_to_before_end(
1260 crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Gt,
token_type: crate::parser::token_type::TokenType::Gt,
}exp!(Gt),
1261 SeqSep::trailing_allowed(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Comma,
token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma)),
1262 |p| match p.parse_generic_arg(None)? {
1263 Some(arg) => Ok(arg),
1264 None => p.unexpected_any(),
1266 },
1267 );
1268 match x {
1269 Ok((_, _, Recovered::No)) => {
1270 if self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Gt,
token_type: crate::parser::token_type::TokenType::Gt,
}exp!(Gt)) {
1271 e.span_suggestion_verbose(
1273 binop.span.shrink_to_lo(),
1274 rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("use `::<...>` instead of `<...>` to specify lifetime, type, or const arguments"))msg!("use `::<...>` instead of `<...>` to specify lifetime, type, or const arguments"),
1275 "::",
1276 Applicability::MaybeIncorrect,
1277 );
1278 match self.parse_expr() {
1279 Ok(_) => {
1280 let guar = e.emit();
1284 *expr = self.mk_expr_err(expr.span.to(self.prev_token.span), guar);
1285 return Ok(guar);
1286 }
1287 Err(err) => {
1288 err.cancel();
1289 }
1290 }
1291 }
1292 }
1293 Ok((_, _, Recovered::Yes(_))) => {}
1294 Err(err) => {
1295 err.cancel();
1296 }
1297 }
1298 }
1299 Err(e)
1300 }
1301
1302 pub(super) fn suggest_add_missing_let_for_stmt(&mut self, err: &mut Diag<'a>) {
1305 if self.token == token::Colon {
1306 let prev_span = self.prev_token.span.shrink_to_lo();
1307 let snapshot = self.create_snapshot_for_diagnostic();
1308 self.bump();
1309 match self.parse_ty() {
1310 Ok(_) => {
1311 if self.token == token::Eq {
1312 let sugg = SuggAddMissingLetStmt { span: prev_span };
1313 sugg.add_to_diag(err);
1314 }
1315 }
1316 Err(e) => {
1317 e.cancel();
1318 }
1319 }
1320 self.restore_snapshot(snapshot);
1321 }
1322 }
1323
1324 fn attempt_chained_comparison_suggestion(
1328 &mut self,
1329 err: &mut ComparisonOperatorsCannotBeChained,
1330 inner_op: &Expr,
1331 outer_op: &Spanned<AssocOp>,
1332 ) -> bool {
1333 if let ExprKind::Binary(op, l1, r1) = &inner_op.kind {
1334 if let ExprKind::Field(_, ident) = l1.kind
1335 && !ident.is_numeric()
1336 && !#[allow(non_exhaustive_omitted_patterns)] match r1.kind {
ExprKind::Lit(_) => true,
_ => false,
}matches!(r1.kind, ExprKind::Lit(_))
1337 {
1338 return false;
1341 }
1342 return match (op.node, &outer_op.node) {
1343 (BinOpKind::Eq, AssocOp::Binary(BinOpKind::Eq)) |
1345 (BinOpKind::Lt, AssocOp::Binary(BinOpKind::Lt | BinOpKind::Le)) |
1347 (BinOpKind::Le, AssocOp::Binary(BinOpKind::Lt | BinOpKind::Le)) |
1348 (BinOpKind::Gt, AssocOp::Binary(BinOpKind::Gt | BinOpKind::Ge)) |
1350 (BinOpKind::Ge, AssocOp::Binary(BinOpKind::Gt | BinOpKind::Ge)) => {
1351 let expr_to_str = |e: &Expr| {
1352 self.span_to_snippet(e.span)
1353 .unwrap_or_else(|_| pprust::expr_to_string(e))
1354 };
1355 err.chaining_sugg = Some(ComparisonOperatorsCannotBeChainedSugg::SplitComparison {
1356 span: inner_op.span.shrink_to_hi(),
1357 middle_term: expr_to_str(r1),
1358 });
1359 false }
1361 (
1363 BinOpKind::Eq,
1364 AssocOp::Binary(BinOpKind::Lt | BinOpKind::Le | BinOpKind::Gt | BinOpKind::Ge)
1365 ) => {
1366 let snapshot = self.create_snapshot_for_diagnostic();
1368 match self.parse_expr() {
1369 Ok(r2) => {
1370 err.chaining_sugg = Some(ComparisonOperatorsCannotBeChainedSugg::Parenthesize {
1373 left: r1.span.shrink_to_lo(),
1374 right: r2.span.shrink_to_hi(),
1375 });
1376 true
1377 }
1378 Err(expr_err) => {
1379 expr_err.cancel();
1380 self.restore_snapshot(snapshot);
1381 true
1382 }
1383 }
1384 }
1385 (
1387 BinOpKind::Lt | BinOpKind::Le | BinOpKind::Gt | BinOpKind::Ge,
1388 AssocOp::Binary(BinOpKind::Eq)
1389 ) => {
1390 let snapshot = self.create_snapshot_for_diagnostic();
1391 match self.parse_expr() {
1394 Ok(_) => {
1395 err.chaining_sugg = Some(ComparisonOperatorsCannotBeChainedSugg::Parenthesize {
1396 left: l1.span.shrink_to_lo(),
1397 right: r1.span.shrink_to_hi(),
1398 });
1399 true
1400 }
1401 Err(expr_err) => {
1402 expr_err.cancel();
1403 self.restore_snapshot(snapshot);
1404 false
1405 }
1406 }
1407 }
1408 _ => false
1409 };
1410 }
1411 false
1412 }
1413
1414 pub(super) fn check_no_chained_comparison(
1433 &mut self,
1434 inner_op: &Expr,
1435 outer_op: &Spanned<AssocOp>,
1436 ) -> PResult<'a, Option<Box<Expr>>> {
1437 if true {
if !outer_op.node.is_comparison() {
{
::core::panicking::panic_fmt(format_args!("check_no_chained_comparison: {0:?} is not comparison",
outer_op.node));
}
};
};debug_assert!(
1438 outer_op.node.is_comparison(),
1439 "check_no_chained_comparison: {:?} is not comparison",
1440 outer_op.node,
1441 );
1442
1443 let mk_err_expr =
1444 |this: &Self, span, guar| Ok(Some(this.mk_expr(span, ExprKind::Err(guar))));
1445
1446 match &inner_op.kind {
1447 ExprKind::Binary(op, l1, r1) if op.node.is_comparison() => {
1448 let mut err = ComparisonOperatorsCannotBeChained {
1449 span: ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[op.span, self.prev_token.span]))vec![op.span, self.prev_token.span],
1450 suggest_turbofish: None,
1451 help_turbofish: false,
1452 chaining_sugg: None,
1453 };
1454
1455 if op.node == BinOpKind::Lt && outer_op.node == AssocOp::Binary(BinOpKind::Lt)
1458 || outer_op.node == AssocOp::Binary(BinOpKind::Gt)
1459 {
1460 if outer_op.node == AssocOp::Binary(BinOpKind::Lt) {
1461 let snapshot = self.create_snapshot_for_diagnostic();
1462 self.bump();
1463 let modifiers = [(token::Lt, 1), (token::Gt, -1), (token::Shr, -2)];
1465 self.consume_tts(1, &modifiers);
1466
1467 if !#[allow(non_exhaustive_omitted_patterns)] match self.token.kind {
token::OpenParen | token::PathSep => true,
_ => false,
}matches!(self.token.kind, token::OpenParen | token::PathSep) {
1468 self.restore_snapshot(snapshot);
1471 }
1472 }
1473 return if self.token == token::PathSep {
1474 if let ExprKind::Binary(o, ..) = inner_op.kind
1477 && o.node == BinOpKind::Lt
1478 {
1479 err.suggest_turbofish = Some(op.span.shrink_to_lo());
1480 } else {
1481 err.help_turbofish = true;
1482 }
1483
1484 let snapshot = self.create_snapshot_for_diagnostic();
1485 self.bump(); match self.parse_expr() {
1489 Ok(_) => {
1490 let guar = self.dcx().emit_err(err);
1492 mk_err_expr(self, inner_op.span.to(self.prev_token.span), guar)
1496 }
1497 Err(expr_err) => {
1498 expr_err.cancel();
1499 self.restore_snapshot(snapshot);
1502 Err(self.dcx().create_err(err))
1503 }
1504 }
1505 } else if self.token == token::OpenParen {
1506 if let ExprKind::Binary(o, ..) = inner_op.kind
1509 && o.node == BinOpKind::Lt
1510 {
1511 err.suggest_turbofish = Some(op.span.shrink_to_lo());
1512 } else {
1513 err.help_turbofish = true;
1514 }
1515 match self.consume_fn_args() {
1517 Err(()) => Err(self.dcx().create_err(err)),
1518 Ok(()) => {
1519 let guar = self.dcx().emit_err(err);
1520 mk_err_expr(self, inner_op.span.to(self.prev_token.span), guar)
1524 }
1525 }
1526 } else {
1527 if !#[allow(non_exhaustive_omitted_patterns)] match l1.kind {
ExprKind::Lit(_) => true,
_ => false,
}matches!(l1.kind, ExprKind::Lit(_))
1528 && !#[allow(non_exhaustive_omitted_patterns)] match r1.kind {
ExprKind::Lit(_) => true,
_ => false,
}matches!(r1.kind, ExprKind::Lit(_))
1529 {
1530 err.help_turbofish = true;
1533 }
1534
1535 let recovered = self
1538 .attempt_chained_comparison_suggestion(&mut err, inner_op, outer_op);
1539 if recovered {
1540 let guar = self.dcx().emit_err(err);
1541 mk_err_expr(self, inner_op.span.to(self.prev_token.span), guar)
1542 } else {
1543 Err(self.dcx().create_err(err))
1545 }
1546 };
1547 }
1548 let recovered =
1549 self.attempt_chained_comparison_suggestion(&mut err, inner_op, outer_op);
1550 let guar = self.dcx().emit_err(err);
1551 if recovered {
1552 return mk_err_expr(self, inner_op.span.to(self.prev_token.span), guar);
1553 }
1554 }
1555 _ => {}
1556 }
1557 Ok(None)
1558 }
1559
1560 fn consume_fn_args(&mut self) -> Result<(), ()> {
1561 let snapshot = self.create_snapshot_for_diagnostic();
1562 self.bump(); let modifiers = [(token::OpenParen, 1), (token::CloseParen, -1)];
1566 self.consume_tts(1, &modifiers);
1567
1568 if self.token == token::Eof {
1569 self.restore_snapshot(snapshot);
1571 Err(())
1572 } else {
1573 Ok(())
1575 }
1576 }
1577
1578 pub(super) fn maybe_report_ambiguous_plus(&mut self, impl_dyn_multi: bool, ty: &Ty) {
1579 if impl_dyn_multi {
1580 self.dcx().emit_err(AmbiguousPlus {
1581 span: ty.span,
1582 suggestion: AddParen { lo: ty.span.shrink_to_lo(), hi: ty.span.shrink_to_hi() },
1583 });
1584 }
1585 }
1586
1587 pub(super) fn maybe_recover_from_question_mark(&mut self, ty: Box<Ty>) -> Box<Ty> {
1589 if self.token == token::Question {
1590 self.bump();
1591 let guar = self.dcx().emit_err(QuestionMarkInType {
1592 span: self.prev_token.span,
1593 sugg: QuestionMarkInTypeSugg {
1594 left: ty.span.shrink_to_lo(),
1595 right: self.prev_token.span,
1596 },
1597 });
1598 self.mk_ty(ty.span.to(self.prev_token.span), TyKind::Err(guar))
1599 } else {
1600 ty
1601 }
1602 }
1603
1604 pub(super) fn maybe_recover_from_ternary_operator(
1610 &mut self,
1611 cond: Option<Span>,
1612 ) -> PResult<'a, ()> {
1613 if self.prev_token != token::Question {
1614 return PResult::Ok(());
1615 }
1616
1617 let question = self.prev_token.span;
1618 let lo = cond.unwrap_or(question).lo();
1619 let snapshot = self.create_snapshot_for_diagnostic();
1620
1621 if match self.parse_expr() {
1622 Ok(_) => true,
1623 Err(err) => {
1624 err.cancel();
1625 self.token == token::Colon
1628 }
1629 } {
1630 if self.eat_noexpect(&token::Colon) {
1631 let colon = self.prev_token.span;
1632 match self.parse_expr() {
1633 Ok(expr) => {
1634 let sugg = cond.map(|cond| TernaryOperatorSuggestion {
1635 before_cond: cond.shrink_to_lo(),
1636 question,
1637 colon,
1638 end: expr.span.shrink_to_hi(),
1639 });
1640 return Err(self.dcx().create_err(TernaryOperator {
1641 span: self.prev_token.span.with_lo(lo),
1642 sugg,
1643 no_sugg: sugg.is_none(),
1644 }));
1645 }
1646 Err(err) => {
1647 err.cancel();
1648 }
1649 };
1650 }
1651 }
1652 self.restore_snapshot(snapshot);
1653 Ok(())
1654 }
1655
1656 pub(super) fn maybe_recover_from_bad_type_plus(&mut self, ty: &Ty) -> PResult<'a, ()> {
1657 if !self.token.is_like_plus() {
1659 return Ok(());
1660 }
1661
1662 self.bump(); let _bounds = self.parse_generic_bounds()?;
1664 let sub = match &ty.kind {
1665 TyKind::Ref(_lifetime, mut_ty) => {
1666 let lo = mut_ty.ty.span.shrink_to_lo();
1667 let hi = self.prev_token.span.shrink_to_hi();
1668 BadTypePlusSub::AddParen { suggestion: AddParen { lo, hi } }
1669 }
1670 TyKind::Ptr(..) | TyKind::FnPtr(..) => {
1671 BadTypePlusSub::ForgotParen { span: ty.span.to(self.prev_token.span) }
1672 }
1673 _ => BadTypePlusSub::ExpectPath { span: ty.span },
1674 };
1675
1676 self.dcx().emit_err(BadTypePlus { span: ty.span, sub });
1677
1678 Ok(())
1679 }
1680
1681 pub(super) fn recover_from_prefix_increment(
1682 &mut self,
1683 operand_expr: Box<Expr>,
1684 op_span: Span,
1685 start_stmt: bool,
1686 ) -> PResult<'a, Box<Expr>> {
1687 let standalone = if start_stmt { IsStandalone::Standalone } else { IsStandalone::Subexpr };
1688 let kind = IncDecRecovery { standalone, op: IncOrDec::Inc, fixity: UnaryFixity::Pre };
1689 self.recover_from_inc_dec(operand_expr, kind, op_span)
1690 }
1691
1692 pub(super) fn recover_from_postfix_increment(
1693 &mut self,
1694 operand_expr: Box<Expr>,
1695 op_span: Span,
1696 start_stmt: bool,
1697 ) -> PResult<'a, Box<Expr>> {
1698 let kind = IncDecRecovery {
1699 standalone: if start_stmt { IsStandalone::Standalone } else { IsStandalone::Subexpr },
1700 op: IncOrDec::Inc,
1701 fixity: UnaryFixity::Post,
1702 };
1703 self.recover_from_inc_dec(operand_expr, kind, op_span)
1704 }
1705
1706 pub(super) fn recover_from_postfix_decrement(
1707 &mut self,
1708 operand_expr: Box<Expr>,
1709 op_span: Span,
1710 start_stmt: bool,
1711 ) -> PResult<'a, Box<Expr>> {
1712 let kind = IncDecRecovery {
1713 standalone: if start_stmt { IsStandalone::Standalone } else { IsStandalone::Subexpr },
1714 op: IncOrDec::Dec,
1715 fixity: UnaryFixity::Post,
1716 };
1717 self.recover_from_inc_dec(operand_expr, kind, op_span)
1718 }
1719
1720 fn recover_from_inc_dec(
1721 &mut self,
1722 base: Box<Expr>,
1723 kind: IncDecRecovery,
1724 op_span: Span,
1725 ) -> PResult<'a, Box<Expr>> {
1726 let mut err = self.dcx().struct_span_err(
1727 op_span,
1728 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("Rust has no {0} {1} operator",
kind.fixity, kind.op.name()))
})format!("Rust has no {} {} operator", kind.fixity, kind.op.name()),
1729 );
1730 err.span_label(op_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("not a valid {0} operator",
kind.fixity))
})format!("not a valid {} operator", kind.fixity));
1731
1732 let help_base_case = |mut err: Diag<'_, _>, base| {
1733 err.help(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("use `{0}= 1` instead",
kind.op.chr()))
})format!("use `{}= 1` instead", kind.op.chr()));
1734 err.emit();
1735 Ok(base)
1736 };
1737
1738 let spans = match kind.fixity {
1740 UnaryFixity::Pre => (op_span, base.span.shrink_to_hi()),
1741 UnaryFixity::Post => (base.span.shrink_to_lo(), op_span),
1742 };
1743
1744 match kind.standalone {
1745 IsStandalone::Standalone => {
1746 self.inc_dec_standalone_suggest(kind, spans).emit_verbose(&mut err)
1747 }
1748 IsStandalone::Subexpr => {
1749 let Ok(base_src) = self.span_to_snippet(base.span) else {
1750 return help_base_case(err, base);
1751 };
1752 match kind.fixity {
1753 UnaryFixity::Pre => {
1754 self.prefix_inc_dec_suggest(base_src, kind, spans).emit(&mut err)
1755 }
1756 UnaryFixity::Post => {
1757 if !#[allow(non_exhaustive_omitted_patterns)] match base.kind {
ExprKind::Binary(_, _, _) => true,
_ => false,
}matches!(base.kind, ExprKind::Binary(_, _, _)) {
1760 self.postfix_inc_dec_suggest(base_src, kind, spans).emit(&mut err)
1761 }
1762 }
1763 }
1764 }
1765 }
1766 Err(err)
1767 }
1768
1769 fn prefix_inc_dec_suggest(
1770 &mut self,
1771 base_src: String,
1772 kind: IncDecRecovery,
1773 (pre_span, post_span): (Span, Span),
1774 ) -> MultiSugg {
1775 MultiSugg {
1776 msg: ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("use `{0}= 1` instead",
kind.op.chr()))
})format!("use `{}= 1` instead", kind.op.chr()),
1777 patches: ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(pre_span, "{ ".to_string()),
(post_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" {0}= 1; {1} }}",
kind.op.chr(), base_src))
}))]))vec![
1778 (pre_span, "{ ".to_string()),
1779 (post_span, format!(" {}= 1; {} }}", kind.op.chr(), base_src)),
1780 ],
1781 applicability: Applicability::MachineApplicable,
1782 }
1783 }
1784
1785 fn postfix_inc_dec_suggest(
1786 &mut self,
1787 base_src: String,
1788 kind: IncDecRecovery,
1789 (pre_span, post_span): (Span, Span),
1790 ) -> MultiSugg {
1791 let tmp_var = if base_src.trim() == "tmp" { "tmp_" } else { "tmp" };
1792 MultiSugg {
1793 msg: ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("use `{0}= 1` instead",
kind.op.chr()))
})format!("use `{}= 1` instead", kind.op.chr()),
1794 patches: ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(pre_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{{ let {0} = ", tmp_var))
})),
(post_span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("; {0} {1}= 1; {2} }}",
base_src, kind.op.chr(), tmp_var))
}))]))vec![
1795 (pre_span, format!("{{ let {tmp_var} = ")),
1796 (post_span, format!("; {} {}= 1; {} }}", base_src, kind.op.chr(), tmp_var)),
1797 ],
1798 applicability: Applicability::HasPlaceholders,
1799 }
1800 }
1801
1802 fn inc_dec_standalone_suggest(
1803 &mut self,
1804 kind: IncDecRecovery,
1805 (pre_span, post_span): (Span, Span),
1806 ) -> MultiSugg {
1807 let mut patches = Vec::new();
1808
1809 if !pre_span.is_empty() {
1810 patches.push((pre_span, String::new()));
1811 }
1812
1813 patches.push((post_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" {0}= 1", kind.op.chr()))
})format!(" {}= 1", kind.op.chr())));
1814 MultiSugg {
1815 msg: ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("use `{0}= 1` instead",
kind.op.chr()))
})format!("use `{}= 1` instead", kind.op.chr()),
1816 patches,
1817 applicability: Applicability::MachineApplicable,
1818 }
1819 }
1820
1821 pub(super) fn maybe_recover_from_bad_qpath<T: RecoverQPath>(
1825 &mut self,
1826 base: T,
1827 ) -> PResult<'a, T> {
1828 if self.may_recover() && self.token == token::PathSep {
1830 return self.recover_from_bad_qpath(base);
1831 }
1832 Ok(base)
1833 }
1834
1835 #[cold]
1836 fn recover_from_bad_qpath<T: RecoverQPath>(&mut self, base: T) -> PResult<'a, T> {
1837 if let Some(ty) = base.to_ty() {
1838 return self.maybe_recover_from_bad_qpath_stage_2(ty.span, ty);
1839 }
1840 Ok(base)
1841 }
1842
1843 pub(super) fn maybe_recover_from_bad_qpath_stage_2<T: RecoverQPath>(
1846 &mut self,
1847 ty_span: Span,
1848 ty: Box<Ty>,
1849 ) -> PResult<'a, T> {
1850 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::PathSep,
token_type: crate::parser::token_type::TokenType::PathSep,
}exp!(PathSep))?;
1851
1852 let mut path = ast::Path { segments: ThinVec::new(), span: DUMMY_SP, tokens: None };
1853 self.parse_path_segments(&mut path.segments, T::PATH_STYLE, None)?;
1854 path.span = ty_span.to(self.prev_token.span);
1855
1856 self.dcx().emit_err(BadQPathStage2 {
1857 span: ty_span,
1858 wrap: WrapType { lo: ty_span.shrink_to_lo(), hi: ty_span.shrink_to_hi() },
1859 });
1860
1861 let path_span = ty_span.shrink_to_hi(); Ok(T::recovered(Some(Box::new(QSelf { ty, path_span, position: 0 })), path))
1863 }
1864
1865 pub fn maybe_consume_incorrect_semicolon(&mut self, previous_item: Option<&Item>) -> bool {
1868 if self.token != TokenKind::Semi {
1869 return false;
1870 }
1871
1872 let err = match previous_item {
1875 Some(previous_item) => {
1876 let name = match previous_item.kind {
1877 ItemKind::Struct(..) => "braced struct",
1880 _ => previous_item.kind.descr(),
1881 };
1882 IncorrectSemicolon { span: self.token.span, name, show_help: true }
1883 }
1884 None => IncorrectSemicolon { span: self.token.span, name: "", show_help: false },
1885 };
1886 self.dcx().emit_err(err);
1887
1888 self.bump();
1889 true
1890 }
1891
1892 pub(super) fn unexpected_try_recover(&mut self, t: &TokenKind) -> PResult<'a, Recovered> {
1895 let token_str = pprust::token_kind_to_string(t);
1896 let this_token_str = super::token_descr(&self.token);
1897 let (prev_sp, sp) = match (&self.token.kind, self.subparser_name) {
1898 (token::Eof, Some(_)) => {
1900 let sp = self.prev_token.span.shrink_to_hi();
1901 (sp, sp)
1902 }
1903 _ if self.prev_token.span == DUMMY_SP => (self.token.span, self.token.span),
1906 (token::Eof, None) => (self.prev_token.span, self.token.span),
1908 _ => (self.prev_token.span.shrink_to_hi(), self.token.span),
1909 };
1910 let msg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected `{0}`, found {1}",
token_str,
match (&self.token.kind, self.subparser_name) {
(token::Eof, Some(origin)) =>
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("end of {0}", origin))
}),
_ => this_token_str,
}))
})format!(
1911 "expected `{}`, found {}",
1912 token_str,
1913 match (&self.token.kind, self.subparser_name) {
1914 (token::Eof, Some(origin)) => format!("end of {origin}"),
1915 _ => this_token_str,
1916 },
1917 );
1918 let mut err = self.dcx().struct_span_err(sp, msg);
1919 let label_exp = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected `{0}`", token_str))
})format!("expected `{token_str}`");
1920 let sm = self.psess.source_map();
1921 if !sm.is_multiline(prev_sp.until(sp)) {
1922 err.span_label(sp, label_exp);
1925 } else {
1926 err.span_label(prev_sp, label_exp);
1927 err.span_label(sp, "unexpected token");
1928 }
1929 Err(err)
1930 }
1931
1932 pub(super) fn expect_semi(&mut self) -> PResult<'a, ()> {
1933 if self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Semi,
token_type: crate::parser::token_type::TokenType::Semi,
}exp!(Semi)) || self.recover_colon_as_semi() {
1934 return Ok(());
1935 }
1936 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Semi,
token_type: crate::parser::token_type::TokenType::Semi,
}exp!(Semi)).map(drop) }
1938
1939 pub(super) fn recover_colon_as_semi(&mut self) -> bool {
1940 let line_idx = |span: Span| {
1941 self.psess
1942 .source_map()
1943 .span_to_lines(span)
1944 .ok()
1945 .and_then(|lines| Some(lines.lines.get(0)?.line_index))
1946 };
1947
1948 if self.may_recover()
1949 && self.token == token::Colon
1950 && self.look_ahead(1, |next| line_idx(self.token.span) < line_idx(next.span))
1951 {
1952 self.dcx().emit_err(ColonAsSemi { span: self.token.span });
1953 self.bump();
1954 return true;
1955 }
1956
1957 false
1958 }
1959
1960 pub(super) fn recover_incorrect_await_syntax(
1963 &mut self,
1964 await_sp: Span,
1965 ) -> PResult<'a, Box<Expr>> {
1966 let (hi, expr, is_question) = if self.token == token::Bang {
1967 self.recover_await_macro()?
1969 } else {
1970 self.recover_await_prefix(await_sp)?
1971 };
1972 let (sp, guar) = self.error_on_incorrect_await(await_sp, hi, &expr, is_question);
1973 let expr = self.mk_expr_err(await_sp.to(sp), guar);
1974 self.maybe_recover_from_bad_qpath(expr)
1975 }
1976
1977 fn recover_await_macro(&mut self) -> PResult<'a, (Span, Box<Expr>, bool)> {
1978 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Bang,
token_type: crate::parser::token_type::TokenType::Bang,
}exp!(Bang))?;
1979 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::OpenParen,
token_type: crate::parser::token_type::TokenType::OpenParen,
}exp!(OpenParen))?;
1980 let expr = self.parse_expr()?;
1981 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::CloseParen,
token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen))?;
1982 Ok((self.prev_token.span, expr, false))
1983 }
1984
1985 fn recover_await_prefix(&mut self, await_sp: Span) -> PResult<'a, (Span, Box<Expr>, bool)> {
1986 let is_question = self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Question,
token_type: crate::parser::token_type::TokenType::Question,
}exp!(Question)); let expr = if self.token == token::OpenBrace {
1988 self.parse_expr_block(None, self.token.span, BlockCheckMode::Default)
1992 } else {
1993 self.parse_expr()
1994 }
1995 .map_err(|mut err| {
1996 err.span_label(await_sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("while parsing this incorrect await expression"))
})format!("while parsing this incorrect await expression"));
1997 err
1998 })?;
1999 Ok((expr.span, expr, is_question))
2000 }
2001
2002 fn error_on_incorrect_await(
2003 &self,
2004 lo: Span,
2005 hi: Span,
2006 expr: &Expr,
2007 is_question: bool,
2008 ) -> (Span, ErrorGuaranteed) {
2009 let span = lo.to(hi);
2010 let guar = self.dcx().emit_err(IncorrectAwait {
2011 span,
2012 suggestion: AwaitSuggestion {
2013 removal: lo.until(expr.span),
2014 dot_await: expr.span.shrink_to_hi(),
2015 question_mark: if is_question { "?" } else { "" },
2016 },
2017 });
2018 (span, guar)
2019 }
2020
2021 pub(super) fn recover_from_await_method_call(&mut self) {
2023 if self.token == token::OpenParen && self.look_ahead(1, |t| t == &token::CloseParen) {
2024 let lo = self.token.span;
2026 self.bump(); let span = lo.to(self.token.span);
2028 self.bump(); self.dcx().emit_err(IncorrectUseOfAwait { span });
2031 }
2032 }
2033 pub(super) fn recover_from_use(&mut self) {
2036 if self.token == token::OpenParen && self.look_ahead(1, |t| t == &token::CloseParen) {
2037 let lo = self.token.span;
2039 self.bump(); let span = lo.to(self.token.span);
2041 self.bump(); self.dcx().emit_err(IncorrectUseOfUse { span });
2044 }
2045 }
2046
2047 pub(super) fn try_macro_suggestion(&mut self) -> PResult<'a, Box<Expr>> {
2048 let is_try = self.token.is_keyword(kw::Try);
2049 let is_questionmark = self.look_ahead(1, |t| t == &token::Bang); let is_open = self.look_ahead(2, |t| t == &token::OpenParen); if is_try && is_questionmark && is_open {
2053 let lo = self.token.span;
2054 self.bump(); self.bump(); let try_span = lo.to(self.token.span); self.bump(); let is_empty = self.token == token::CloseParen; self.consume_block(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), ConsumeClosingDelim::No); let hi = self.token.span;
2061 self.bump(); let mut err = self.dcx().struct_span_err(lo.to(hi), "use of deprecated `try` macro");
2063 err.note("in the 2018 edition `try` is a reserved keyword, and the `try!()` macro is deprecated");
2064 let prefix = if is_empty { "" } else { "alternatively, " };
2065 if !is_empty {
2066 err.multipart_suggestion(
2067 "you can use the `?` operator instead",
2068 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(try_span, "".to_owned()), (hi, "?".to_owned())]))vec![(try_span, "".to_owned()), (hi, "?".to_owned())],
2069 Applicability::MachineApplicable,
2070 );
2071 }
2072 err.span_suggestion(lo.shrink_to_lo(), ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}you can still access the deprecated `try!()` macro using the \"raw identifier\" syntax",
prefix))
})format!("{prefix}you can still access the deprecated `try!()` macro using the \"raw identifier\" syntax"), "r#", Applicability::MachineApplicable);
2073 let guar = err.emit();
2074 Ok(self.mk_expr_err(lo.to(hi), guar))
2075 } else {
2076 Err(self.expected_expression_found()) }
2078 }
2079
2080 pub(super) fn expect_gt_or_maybe_suggest_closing_generics(
2087 &mut self,
2088 params: &[ast::GenericParam],
2089 ) -> PResult<'a, ()> {
2090 let Err(mut err) = self.expect_gt() else {
2091 return Ok(());
2092 };
2093 if let [.., ast::GenericParam { bounds, .. }] = params
2095 && let Some(poly) = bounds
2096 .iter()
2097 .filter_map(|bound| match bound {
2098 ast::GenericBound::Trait(poly) => Some(poly),
2099 _ => None,
2100 })
2101 .next_back()
2102 {
2103 err.span_suggestion_verbose(
2104 poly.span.shrink_to_hi(),
2105 "you might have meant to end the type parameters here",
2106 ">",
2107 Applicability::MaybeIncorrect,
2108 );
2109 }
2110 Err(err)
2111 }
2112
2113 pub(super) fn recover_seq_parse_error(
2114 &mut self,
2115 open: ExpTokenPair,
2116 close: ExpTokenPair,
2117 lo: Span,
2118 err: Diag<'a>,
2119 ) -> Box<Expr> {
2120 let guar = err.emit();
2121 self.consume_block(open, close, ConsumeClosingDelim::Yes);
2123 self.mk_expr(lo.to(self.prev_token.span), ExprKind::Err(guar))
2124 }
2125
2126 pub(super) fn recover_stmt(&mut self) {
2131 self.recover_stmt_(SemiColonMode::Ignore, BlockMode::Ignore)
2132 }
2133
2134 pub(super) fn recover_stmt_(
2142 &mut self,
2143 break_on_semi: SemiColonMode,
2144 break_on_block: BlockMode,
2145 ) {
2146 let mut brace_depth = 0;
2147 let mut bracket_depth = 0;
2148 let mut in_block = false;
2149 {
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/diagnostics.rs:2149",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2149u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ enter loop (semi={0:?}, block={1:?})",
break_on_semi, break_on_block) as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ enter loop (semi={:?}, block={:?})", break_on_semi, break_on_block);
2150 loop {
2151 {
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/diagnostics.rs:2151",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2151u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ loop {0:?}",
self.token) as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ loop {:?}", self.token);
2152 match self.token.kind {
2153 token::OpenBrace => {
2154 brace_depth += 1;
2155 self.bump();
2156 if break_on_block == BlockMode::Break && brace_depth == 1 && bracket_depth == 0
2157 {
2158 in_block = true;
2159 }
2160 }
2161 token::OpenBracket => {
2162 bracket_depth += 1;
2163 self.bump();
2164 }
2165 token::CloseBrace => {
2166 if brace_depth == 0 {
2167 {
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/diagnostics.rs:2167",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2167u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ return - close delim {0:?}",
self.token) as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ return - close delim {:?}", self.token);
2168 break;
2169 }
2170 brace_depth -= 1;
2171 self.bump();
2172 if in_block && bracket_depth == 0 && brace_depth == 0 {
2173 {
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/diagnostics.rs:2173",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2173u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ return - block end {0:?}",
self.token) as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ return - block end {:?}", self.token);
2174 break;
2175 }
2176 }
2177 token::CloseBracket => {
2178 bracket_depth -= 1;
2179 if bracket_depth < 0 {
2180 bracket_depth = 0;
2181 }
2182 self.bump();
2183 }
2184 token::Eof => {
2185 {
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/diagnostics.rs:2185",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2185u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ return - Eof")
as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ return - Eof");
2186 break;
2187 }
2188 token::Semi => {
2189 self.bump();
2190 if break_on_semi == SemiColonMode::Break
2191 && brace_depth == 0
2192 && bracket_depth == 0
2193 {
2194 {
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/diagnostics.rs:2194",
"rustc_parse::parser::diagnostics", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_parse/src/parser/diagnostics.rs"),
::tracing_core::__macro_support::Option::Some(2194u32),
::tracing_core::__macro_support::Option::Some("rustc_parse::parser::diagnostics"),
::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!("recover_stmt_ return - Semi")
as &dyn Value))])
});
} else { ; }
};debug!("recover_stmt_ return - Semi");
2195 break;
2196 }
2197 }
2198 token::Comma
2199 if break_on_semi == SemiColonMode::Comma
2200 && brace_depth == 0
2201 && bracket_depth == 0 =>
2202 {
2203 break;
2204 }
2205 _ => self.bump(),
2206 }
2207 }
2208 }
2209
2210 pub(super) fn check_for_for_in_in_typo(&mut self, in_span: Span) {
2211 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)) {
2212 self.dcx().emit_err(InInTypo {
2214 span: self.prev_token.span,
2215 sugg_span: in_span.until(self.prev_token.span),
2216 });
2217 }
2218 }
2219
2220 pub(super) fn eat_incorrect_doc_comment_for_param_type(&mut self) {
2221 if let token::DocComment(..) = self.token.kind {
2222 self.dcx().emit_err(DocCommentOnParamType { span: self.token.span });
2223 self.bump();
2224 } else if self.token == token::Pound && self.look_ahead(1, |t| *t == token::OpenBracket) {
2225 let lo = self.token.span;
2226 while self.token != token::CloseBracket {
2228 self.bump();
2229 }
2230 let sp = lo.to(self.token.span);
2231 self.bump();
2232 self.dcx().emit_err(AttributeOnParamType { span: sp });
2233 }
2234 }
2235
2236 pub(super) fn parameter_without_type(
2237 &mut self,
2238 err: &mut Diag<'_>,
2239 pat: Box<ast::Pat>,
2240 require_name: bool,
2241 first_param: bool,
2242 fn_parse_mode: &crate::parser::item::FnParseMode,
2243 ) -> Option<Ident> {
2244 if self.check_ident()
2247 && self.look_ahead(1, |t| *t == token::Comma || *t == token::CloseParen)
2248 {
2249 let ident = self.parse_ident_common(true).unwrap();
2251 let span = pat.span.with_hi(ident.span.hi());
2252
2253 err.span_suggestion(
2254 span,
2255 "declare the type after the parameter binding",
2256 "<identifier>: <type>",
2257 Applicability::HasPlaceholders,
2258 );
2259 return Some(ident);
2260 } else if require_name
2261 && (self.token == token::Comma
2262 || self.token == token::Lt
2263 || self.token == token::CloseParen)
2264 {
2265 let maybe_emit_anon_params_note = |this: &mut Self, err: &mut Diag<'_>| {
2266 let ed = this.token.span.with_neighbor(this.prev_token.span).edition();
2267 if #[allow(non_exhaustive_omitted_patterns)] match fn_parse_mode.context {
crate::parser::item::FnContext::Trait => true,
_ => false,
}matches!(fn_parse_mode.context, crate::parser::item::FnContext::Trait)
2268 && (fn_parse_mode.req_name)(ed, IsDotDotDot::No)
2269 {
2270 err.note("anonymous parameters are removed in the 2018 edition (see RFC 1685)");
2271 }
2272 };
2273
2274 let (ident, self_sugg, param_sugg, type_sugg, self_span, param_span, type_span) =
2275 match pat.kind {
2276 PatKind::Ident(_, ident, _) => (
2277 ident,
2278 "self: ",
2279 ": TypeName".to_string(),
2280 "_: ",
2281 pat.span.shrink_to_lo(),
2282 pat.span.shrink_to_hi(),
2283 pat.span.shrink_to_lo(),
2284 ),
2285 PatKind::Ref(ref inner_pat, _, _)
2286 if let PatKind::Ref(_, _, _) = &inner_pat.kind
2289 && let PatKind::Path(_, path) = &pat.peel_refs().kind
2290 && let [a, ..] = path.segments.as_slice()
2291 && a.ident.name == kw::SelfLower =>
2292 {
2293 let mut inner = inner_pat;
2294 let mut span_vec = ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[pat.span]))vec![pat.span];
2295
2296 while let PatKind::Ref(ref inner_type, _, _) = inner.kind {
2297 inner = inner_type;
2298 span_vec.push(inner.span.shrink_to_lo());
2299 }
2300
2301 let span = match span_vec.len() {
2302 0 | 1 => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
2304 2 => span_vec[0].until(inner_pat.span.shrink_to_lo()),
2305 _ => span_vec[0].until(span_vec[span_vec.len() - 2].shrink_to_lo()),
2306 };
2307
2308 err.span_suggestion_verbose(
2309 span,
2310 "`self` should be `self`, `&self` or `&mut self`, consider removing extra references",
2311 "".to_string(),
2312 Applicability::MachineApplicable,
2313 );
2314
2315 return None;
2316 }
2317 PatKind::Ref(ref inner_pat, pinned, mutab)
2319 if let PatKind::Ident(_, ident, _) = inner_pat.clone().kind =>
2320 {
2321 let mutab = pinned.prefix_str(mutab);
2322 (
2323 ident,
2324 "self: ",
2325 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: &{1}TypeName", ident, mutab))
})format!("{ident}: &{mutab}TypeName"),
2326 "_: ",
2327 pat.span.shrink_to_lo(),
2328 pat.span,
2329 pat.span.shrink_to_lo(),
2330 )
2331 }
2332 _ => {
2333 if let Some(_) = pat.to_ty() {
2335 err.span_suggestion_verbose(
2336 pat.span.shrink_to_lo(),
2337 "explicitly ignore the parameter name",
2338 "_: ".to_string(),
2339 Applicability::MachineApplicable,
2340 );
2341 maybe_emit_anon_params_note(self, err);
2342 }
2343
2344 return None;
2345 }
2346 };
2347
2348 if first_param
2350 && #[allow(non_exhaustive_omitted_patterns)] match fn_parse_mode.context {
FnContext::Trait | FnContext::Impl => true,
_ => false,
}matches!(
2352 fn_parse_mode.context,
2353 FnContext::Trait | FnContext::Impl
2354 )
2355 {
2356 err.span_suggestion_verbose(
2357 self_span,
2358 "if this is a `self` type, give it a parameter name",
2359 self_sugg,
2360 Applicability::MaybeIncorrect,
2361 );
2362 }
2363 if self.token != token::Lt {
2366 err.span_suggestion_verbose(
2367 param_span,
2368 "if this is a parameter name, give it a type",
2369 param_sugg,
2370 Applicability::HasPlaceholders,
2371 );
2372 }
2373 err.span_suggestion_verbose(
2374 type_span,
2375 "if this is a type, explicitly ignore the parameter name",
2376 type_sugg,
2377 Applicability::MachineApplicable,
2378 );
2379 maybe_emit_anon_params_note(self, err);
2380
2381 return if self.token == token::Lt { None } else { Some(ident) };
2383 }
2384 None
2385 }
2386
2387 #[cold]
2388 pub(super) fn recover_arg_parse(&mut self) -> PResult<'a, (Box<ast::Pat>, Box<ast::Ty>)> {
2389 let pat = self.parse_pat_no_top_alt(Some(Expected::ArgumentName), None)?;
2390 self.expect(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Colon,
token_type: crate::parser::token_type::TokenType::Colon,
}exp!(Colon))?;
2391 let ty = self.parse_ty()?;
2392
2393 self.dcx().emit_err(PatternMethodParamWithoutBody { span: pat.span });
2394
2395 let pat = Box::new(Pat {
2397 kind: PatKind::Wild,
2398 span: pat.span,
2399 id: ast::DUMMY_NODE_ID,
2400 tokens: None,
2401 });
2402 Ok((pat, ty))
2403 }
2404
2405 pub(super) fn recover_bad_self_param(&mut self, mut param: Param) -> PResult<'a, Param> {
2406 let span = param.pat.span;
2407 let guar = self.dcx().emit_err(SelfParamNotFirst { span });
2408 param.ty.kind = TyKind::Err(guar);
2409 Ok(param)
2410 }
2411
2412 pub(super) fn consume_block(
2413 &mut self,
2414 open: ExpTokenPair,
2415 close: ExpTokenPair,
2416 consume_close: ConsumeClosingDelim,
2417 ) {
2418 let mut brace_depth = 0;
2419 loop {
2420 if self.eat(open) {
2421 brace_depth += 1;
2422 } else if self.check(close) {
2423 if brace_depth == 0 {
2424 if let ConsumeClosingDelim::Yes = consume_close {
2425 self.bump();
2429 }
2430 return;
2431 } else {
2432 self.bump();
2433 brace_depth -= 1;
2434 continue;
2435 }
2436 } else if self.token == token::Eof {
2437 return;
2438 } else {
2439 self.bump();
2440 }
2441 }
2442 }
2443
2444 pub(super) fn expected_expression_found(&self) -> Diag<'a> {
2445 let (span, msg) = match (&self.token.kind, self.subparser_name) {
2446 (&token::Eof, Some(origin)) => {
2447 let sp = self.prev_token.span.shrink_to_hi();
2448 (sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected expression, found end of {0}",
origin))
})format!("expected expression, found end of {origin}"))
2449 }
2450 _ => (
2451 self.token.span,
2452 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected expression, found {0}",
super::token_descr(&self.token)))
})format!("expected expression, found {}", super::token_descr(&self.token)),
2453 ),
2454 };
2455 let mut err = self.dcx().struct_span_err(span, msg);
2456 let sp = self.psess.source_map().start_point(self.token.span);
2457 if let Some(sp) = self.psess.ambiguous_block_expr_parse.borrow().get(&sp) {
2458 err.subdiagnostic(ExprParenthesesNeeded::surrounding(*sp));
2459 }
2460 err.span_label(span, "expected expression");
2461 err
2462 }
2463
2464 fn consume_tts(
2465 &mut self,
2466 mut acc: i64, modifier: &[(token::TokenKind, i64)],
2469 ) {
2470 while acc > 0 {
2471 if let Some((_, val)) = modifier.iter().find(|(t, _)| self.token == *t) {
2472 acc += *val;
2473 }
2474 if self.token == token::Eof {
2475 break;
2476 }
2477 self.bump();
2478 }
2479 }
2480
2481 pub(super) fn deduplicate_recovered_params_names(&self, fn_inputs: &mut ThinVec<Param>) {
2490 let mut seen_inputs = FxHashSet::default();
2491 for input in fn_inputs.iter_mut() {
2492 let opt_ident = if let (PatKind::Ident(_, ident, _), TyKind::Err(_)) =
2493 (&input.pat.kind, &input.ty.kind)
2494 {
2495 Some(*ident)
2496 } else {
2497 None
2498 };
2499 if let Some(ident) = opt_ident {
2500 if seen_inputs.contains(&ident) {
2501 input.pat.kind = PatKind::Wild;
2502 }
2503 seen_inputs.insert(ident);
2504 }
2505 }
2506 }
2507
2508 pub(super) fn handle_ambiguous_unbraced_const_arg(
2512 &mut self,
2513 args: &mut ThinVec<AngleBracketedArg>,
2514 ) -> PResult<'a, bool> {
2515 let arg = args.pop().unwrap();
2519 let mut err = self.dcx().struct_span_err(
2525 self.token.span,
2526 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected one of `,` or `>`, found {0}",
super::token_descr(&self.token)))
})format!("expected one of `,` or `>`, found {}", super::token_descr(&self.token)),
2527 );
2528 err.span_label(self.token.span, "expected one of `,` or `>`");
2529 match self.recover_const_arg(arg.span(), err) {
2530 Ok(arg) => {
2531 args.push(AngleBracketedArg::Arg(arg));
2532 if self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Comma,
token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma)) {
2533 return Ok(true); }
2535 }
2536 Err(err) => {
2537 args.push(arg);
2538 err.delay_as_bug();
2540 }
2541 }
2542 Ok(false) }
2544
2545 fn recover_const_param_decl(&mut self, ty_generics: Option<&Generics>) -> Option<GenericArg> {
2546 let snapshot = self.create_snapshot_for_diagnostic();
2547 let param = match self.parse_const_param(AttrVec::new()) {
2548 Ok(param) => param,
2549 Err(err) => {
2550 err.cancel();
2551 self.restore_snapshot(snapshot);
2552 return None;
2553 }
2554 };
2555
2556 let ident = param.ident.to_string();
2557 let sugg = match (ty_generics, self.psess.source_map().span_to_snippet(param.span())) {
2558 (Some(Generics { params, span: impl_generics, .. }), Ok(snippet)) => {
2559 Some(match ¶ms[..] {
2560 [] => UnexpectedConstParamDeclarationSugg::AddParam {
2561 impl_generics: *impl_generics,
2562 incorrect_decl: param.span(),
2563 snippet,
2564 ident,
2565 },
2566 [.., generic] => UnexpectedConstParamDeclarationSugg::AppendParam {
2567 impl_generics_end: generic.span().shrink_to_hi(),
2568 incorrect_decl: param.span(),
2569 snippet,
2570 ident,
2571 },
2572 })
2573 }
2574 _ => None,
2575 };
2576 let guar =
2577 self.dcx().emit_err(UnexpectedConstParamDeclaration { span: param.span(), sugg });
2578
2579 let value = self.mk_expr_err(param.span(), guar);
2580 Some(GenericArg::Const(AnonConst {
2581 id: ast::DUMMY_NODE_ID,
2582 value,
2583 mgca_disambiguation: MgcaDisambiguation::Direct,
2584 }))
2585 }
2586
2587 pub(super) fn recover_const_param_declaration(
2588 &mut self,
2589 ty_generics: Option<&Generics>,
2590 ) -> PResult<'a, Option<GenericArg>> {
2591 if let Some(arg) = self.recover_const_param_decl(ty_generics) {
2593 return Ok(Some(arg));
2594 }
2595
2596 let start = self.token.span;
2598 self.bump(); let mut err = UnexpectedConstInGenericParam { span: start, to_remove: None };
2602 if self.check_const_arg() {
2603 err.to_remove = Some(start.until(self.token.span));
2604 self.dcx().emit_err(err);
2605 Ok(Some(GenericArg::Const(self.parse_const_arg()?)))
2606 } else {
2607 let after_kw_const = self.token.span;
2608 self.recover_const_arg(after_kw_const, self.dcx().create_err(err)).map(Some)
2609 }
2610 }
2611
2612 pub(super) fn recover_const_arg(
2618 &mut self,
2619 start: Span,
2620 mut err: Diag<'a>,
2621 ) -> PResult<'a, GenericArg> {
2622 let is_op_or_dot = AssocOp::from_token(&self.token)
2623 .and_then(|op| {
2624 if let AssocOp::Binary(
2625 BinOpKind::Gt
2626 | BinOpKind::Lt
2627 | BinOpKind::Shr
2628 | BinOpKind::Ge
2629 )
2630 | AssocOp::Assign
2633 | AssocOp::AssignOp(_) = op
2634 {
2635 None
2636 } else {
2637 Some(op)
2638 }
2639 })
2640 .is_some()
2641 || self.token == TokenKind::Dot;
2642 let was_op = #[allow(non_exhaustive_omitted_patterns)] match self.prev_token.kind {
token::Plus | token::Shr | token::Gt => true,
_ => false,
}matches!(self.prev_token.kind, token::Plus | token::Shr | token::Gt);
2645 if !is_op_or_dot && !was_op {
2646 return Err(err);
2648 }
2649 let snapshot = self.create_snapshot_for_diagnostic();
2650 if is_op_or_dot {
2651 self.bump();
2652 }
2653 match (|| {
2654 let attrs = self.parse_outer_attributes()?;
2655 self.parse_expr_res(Restrictions::CONST_EXPR, attrs)
2656 })() {
2657 Ok((expr, _)) => {
2658 if snapshot.token == token::EqEq {
2660 err.span_suggestion(
2661 snapshot.token.span,
2662 "if you meant to use an associated type binding, replace `==` with `=`",
2663 "=",
2664 Applicability::MaybeIncorrect,
2665 );
2666 let guar = err.emit();
2667 let value = self.mk_expr_err(start.to(expr.span), guar);
2668 return Ok(GenericArg::Const(AnonConst {
2669 id: ast::DUMMY_NODE_ID,
2670 value,
2671 mgca_disambiguation: MgcaDisambiguation::Direct,
2672 }));
2673 } else if snapshot.token == token::Colon
2674 && expr.span.lo() == snapshot.token.span.hi()
2675 && #[allow(non_exhaustive_omitted_patterns)] match expr.kind {
ExprKind::Path(..) => true,
_ => false,
}matches!(expr.kind, ExprKind::Path(..))
2676 {
2677 err.span_suggestion(
2679 snapshot.token.span,
2680 "write a path separator here",
2681 "::",
2682 Applicability::MaybeIncorrect,
2683 );
2684 let guar = err.emit();
2685 return Ok(GenericArg::Type(
2686 self.mk_ty(start.to(expr.span), TyKind::Err(guar)),
2687 ));
2688 } else if self.token == token::Comma || self.token.kind.should_end_const_arg() {
2689 return Ok(self.dummy_const_arg_needs_braces(err, start.to(expr.span)));
2696 }
2697 }
2698 Err(err) => {
2699 err.cancel();
2700 }
2701 }
2702 self.restore_snapshot(snapshot);
2703 Err(err)
2704 }
2705
2706 pub(crate) fn recover_unbraced_const_arg_that_can_begin_ty(
2710 &mut self,
2711 mut snapshot: SnapshotParser<'a>,
2712 ) -> Option<Box<ast::Expr>> {
2713 match (|| {
2714 let attrs = self.parse_outer_attributes()?;
2715 snapshot.parse_expr_res(Restrictions::CONST_EXPR, attrs)
2716 })() {
2717 Ok((expr, _)) if let token::Comma | token::Gt = snapshot.token.kind => {
2720 self.restore_snapshot(snapshot);
2721 Some(expr)
2722 }
2723 Ok(_) => None,
2724 Err(err) => {
2725 err.cancel();
2726 None
2727 }
2728 }
2729 }
2730
2731 pub(super) fn dummy_const_arg_needs_braces(&self, mut err: Diag<'a>, span: Span) -> GenericArg {
2733 err.multipart_suggestion(
2734 "expressions must be enclosed in braces to be used as const generic \
2735 arguments",
2736 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(span.shrink_to_lo(), "{ ".to_string()),
(span.shrink_to_hi(), " }".to_string())]))vec![(span.shrink_to_lo(), "{ ".to_string()), (span.shrink_to_hi(), " }".to_string())],
2737 Applicability::MaybeIncorrect,
2738 );
2739 let guar = err.emit();
2740 let value = self.mk_expr_err(span, guar);
2741 GenericArg::Const(AnonConst {
2742 id: ast::DUMMY_NODE_ID,
2743 value,
2744 mgca_disambiguation: MgcaDisambiguation::Direct,
2745 })
2746 }
2747
2748 #[cold]
2751 pub(crate) fn recover_colon_colon_in_pat_typo(
2752 &mut self,
2753 mut first_pat: Pat,
2754 expected: Option<Expected>,
2755 ) -> Pat {
2756 if token::Colon != self.token.kind {
2757 return first_pat;
2758 }
2759
2760 let colon_span = self.token.span;
2763 let mut snapshot_pat = self.create_snapshot_for_diagnostic();
2766 let mut snapshot_type = self.create_snapshot_for_diagnostic();
2767
2768 match self.expected_one_of_not_found(&[], &[]) {
2770 Err(mut err) => {
2771 snapshot_pat.bump();
2773 snapshot_type.bump();
2774 match snapshot_pat.parse_pat_no_top_alt(expected, None) {
2775 Err(inner_err) => {
2776 inner_err.cancel();
2777 }
2778 Ok(mut pat) => {
2779 let new_span = first_pat.span.to(pat.span);
2781 let mut show_sugg = false;
2782 match &mut pat.kind {
2784 PatKind::Struct(qself @ None, path, ..)
2785 | PatKind::TupleStruct(qself @ None, path, _)
2786 | PatKind::Path(qself @ None, path) => match &first_pat.kind {
2787 PatKind::Ident(_, ident, _) => {
2788 path.segments.insert(0, PathSegment::from_ident(*ident));
2789 path.span = new_span;
2790 show_sugg = true;
2791 first_pat = pat;
2792 }
2793 PatKind::Path(old_qself, old_path) => {
2794 path.segments = old_path
2795 .segments
2796 .iter()
2797 .cloned()
2798 .chain(take(&mut path.segments))
2799 .collect();
2800 path.span = new_span;
2801 *qself = old_qself.clone();
2802 first_pat = pat;
2803 show_sugg = true;
2804 }
2805 _ => {}
2806 },
2807 PatKind::Ident(BindingMode::NONE, ident, None) => {
2808 match &first_pat.kind {
2809 PatKind::Ident(_, old_ident, _) => {
2810 let path = PatKind::Path(
2811 None,
2812 Path {
2813 span: new_span,
2814 segments: {
let len = [(), ()].len();
let mut vec = ::thin_vec::ThinVec::with_capacity(len);
vec.push(PathSegment::from_ident(*old_ident));
vec.push(PathSegment::from_ident(*ident));
vec
}thin_vec![
2815 PathSegment::from_ident(*old_ident),
2816 PathSegment::from_ident(*ident),
2817 ],
2818 tokens: None,
2819 },
2820 );
2821 first_pat = self.mk_pat(new_span, path);
2822 show_sugg = true;
2823 }
2824 PatKind::Path(old_qself, old_path) => {
2825 let mut segments = old_path.segments.clone();
2826 segments.push(PathSegment::from_ident(*ident));
2827 let path = PatKind::Path(
2828 old_qself.clone(),
2829 Path { span: new_span, segments, tokens: None },
2830 );
2831 first_pat = self.mk_pat(new_span, path);
2832 show_sugg = true;
2833 }
2834 _ => {}
2835 }
2836 }
2837 _ => {}
2838 }
2839 if show_sugg {
2840 err.span_suggestion_verbose(
2841 colon_span.until(self.look_ahead(1, |t| t.span)),
2842 "maybe write a path separator here",
2843 "::",
2844 Applicability::MaybeIncorrect,
2845 );
2846 } else {
2847 first_pat = self.mk_pat(
2848 new_span,
2849 PatKind::Err(
2850 self.dcx()
2851 .span_delayed_bug(colon_span, "recovered bad path pattern"),
2852 ),
2853 );
2854 }
2855 self.restore_snapshot(snapshot_pat);
2856 }
2857 }
2858 match snapshot_type.parse_ty() {
2859 Err(inner_err) => {
2860 inner_err.cancel();
2861 }
2862 Ok(ty) => {
2863 err.span_label(ty.span, "specifying the type of a pattern isn't supported");
2864 self.restore_snapshot(snapshot_type);
2865 let new_span = first_pat.span.to(ty.span);
2866 first_pat =
2867 self.mk_pat(
2868 new_span,
2869 PatKind::Err(self.dcx().span_delayed_bug(
2870 colon_span,
2871 "recovered bad pattern with type",
2872 )),
2873 );
2874 }
2875 }
2876 err.emit();
2877 }
2878 _ => {
2879 }
2881 };
2882 first_pat
2883 }
2884
2885 pub(crate) fn maybe_recover_unexpected_block_label(
2888 &mut self,
2889 loop_header: Option<Span>,
2890 ) -> bool {
2891 if !(self.check_lifetime()
2893 && self.look_ahead(1, |t| *t == token::Colon)
2894 && self.look_ahead(2, |t| *t == token::OpenBrace))
2895 {
2896 return false;
2897 }
2898 let label = self.eat_label().expect("just checked if a label exists");
2899 self.bump(); let span = label.ident.span.to(self.prev_token.span);
2901 let mut diag = self
2902 .dcx()
2903 .struct_span_err(span, "block label not supported here")
2904 .with_span_label(span, "not supported here");
2905 if let Some(loop_header) = loop_header {
2906 diag.multipart_suggestion(
2907 "if you meant to label the loop, move this label before the loop",
2908 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(label.ident.span.until(self.token.span), String::from("")),
(loop_header.shrink_to_lo(),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: ", label.ident))
}))]))vec![
2909 (label.ident.span.until(self.token.span), String::from("")),
2910 (loop_header.shrink_to_lo(), format!("{}: ", label.ident)),
2911 ],
2912 Applicability::MachineApplicable,
2913 );
2914 } else {
2915 diag.tool_only_span_suggestion(
2916 label.ident.span.until(self.token.span),
2917 "remove this block label",
2918 "",
2919 Applicability::MachineApplicable,
2920 );
2921 }
2922 diag.emit();
2923 true
2924 }
2925
2926 pub(crate) fn maybe_recover_unexpected_comma(
2929 &mut self,
2930 lo: Span,
2931 rt: CommaRecoveryMode,
2932 ) -> PResult<'a, ()> {
2933 if self.token != token::Comma {
2934 return Ok(());
2935 }
2936 self.recover_unexpected_comma(lo, rt)
2937 }
2938
2939 #[cold]
2940 fn recover_unexpected_comma(&mut self, lo: Span, rt: CommaRecoveryMode) -> PResult<'a, ()> {
2941 let comma_span = self.token.span;
2946 self.bump();
2947 if let Err(err) = self.skip_pat_list() {
2948 err.cancel();
2951 }
2952 let seq_span = lo.to(self.prev_token.span);
2953 let mut err = self.dcx().struct_span_err(comma_span, "unexpected `,` in pattern");
2954 err.multipart_suggestion(
2955 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("try adding parentheses to match on a tuple{0}",
if let CommaRecoveryMode::LikelyTuple = rt {
""
} else { "..." }))
})format!(
2956 "try adding parentheses to match on a tuple{}",
2957 if let CommaRecoveryMode::LikelyTuple = rt { "" } else { "..." },
2958 ),
2959 ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(seq_span.shrink_to_lo(), "(".to_string()),
(seq_span.shrink_to_hi(), ")".to_string())]))vec![
2960 (seq_span.shrink_to_lo(), "(".to_string()),
2961 (seq_span.shrink_to_hi(), ")".to_string()),
2962 ],
2963 Applicability::MachineApplicable,
2964 );
2965 if let CommaRecoveryMode::EitherTupleOrPipe = rt {
2966 err.span_suggestion(
2967 comma_span,
2968 "...or a vertical bar to match on alternatives",
2969 " |",
2970 Applicability::MachineApplicable,
2971 );
2972 }
2973 Err(err)
2974 }
2975
2976 pub(crate) fn maybe_recover_bounds_doubled_colon(&mut self, ty: &Ty) -> PResult<'a, ()> {
2977 let TyKind::Path(qself, path) = &ty.kind else { return Ok(()) };
2978 let qself_position = qself.as_ref().map(|qself| qself.position);
2979 for (i, segments) in path.segments.windows(2).enumerate() {
2980 if qself_position.is_some_and(|pos| i < pos) {
2981 continue;
2982 }
2983 if let [a, b] = segments {
2984 let (a_span, b_span) = (a.span(), b.span());
2985 let between_span = a_span.shrink_to_hi().to(b_span.shrink_to_lo());
2986 if self.span_to_snippet(between_span).as_deref() == Ok(":: ") {
2987 return Err(self.dcx().create_err(DoubleColonInBound {
2988 span: path.span.shrink_to_hi(),
2989 between: between_span,
2990 }));
2991 }
2992 }
2993 }
2994 Ok(())
2995 }
2996
2997 pub(crate) fn maybe_err_dotdotlt_syntax(&self, maybe_lt: Token, mut err: Diag<'a>) -> Diag<'a> {
2999 if maybe_lt == token::Lt
3000 && (self.expected_token_types.contains(TokenType::Gt)
3001 || #[allow(non_exhaustive_omitted_patterns)] match self.token.kind {
token::Literal(..) => true,
_ => false,
}matches!(self.token.kind, token::Literal(..)))
3002 {
3003 err.span_suggestion(
3004 maybe_lt.span,
3005 "remove the `<` to write an exclusive range",
3006 "",
3007 Applicability::MachineApplicable,
3008 );
3009 }
3010 err
3011 }
3012
3013 pub(super) fn is_vcs_conflict_marker(
3021 &mut self,
3022 long_kind: &TokenKind,
3023 short_kind: &TokenKind,
3024 ) -> bool {
3025 if long_kind == short_kind {
3026 (0..7).all(|i| self.look_ahead(i, |tok| tok == long_kind))
3028 } else {
3029 (0..3).all(|i| self.look_ahead(i, |tok| tok == long_kind))
3031 && self.look_ahead(3, |tok| tok == short_kind || tok == long_kind)
3032 }
3033 }
3034
3035 fn conflict_marker(
3036 &mut self,
3037 long_kind: &TokenKind,
3038 short_kind: &TokenKind,
3039 expected: Option<usize>,
3040 ) -> Option<(Span, usize)> {
3041 if self.is_vcs_conflict_marker(long_kind, short_kind) {
3042 let lo = self.token.span;
3043 if self.psess.source_map().span_to_margin(lo) != Some(0) {
3044 return None;
3045 }
3046 let mut len = 0;
3047 while self.token.kind == *long_kind || self.token.kind == *short_kind {
3048 if self.token.kind.break_two_token_op(1).is_some() {
3049 len += 2;
3050 } else {
3051 len += 1;
3052 }
3053 self.bump();
3054 if expected == Some(len) {
3055 break;
3056 }
3057 }
3058 if expected.is_some() && expected != Some(len) {
3059 return None;
3060 }
3061 return Some((lo.to(self.prev_token.span), len));
3062 }
3063 None
3064 }
3065
3066 pub(super) fn recover_vcs_conflict_marker(&mut self) {
3067 let Some((start, len)) = self.conflict_marker(&TokenKind::Shl, &TokenKind::Lt, None) else {
3069 return;
3070 };
3071 let mut spans = Vec::with_capacity(2);
3072 spans.push(start);
3073 let mut middlediff3 = None;
3075 let mut middle = None;
3077 let mut end = None;
3079 loop {
3080 if self.token == TokenKind::Eof {
3081 break;
3082 }
3083 if let Some((span, _)) =
3084 self.conflict_marker(&TokenKind::OrOr, &TokenKind::Or, Some(len))
3085 {
3086 middlediff3 = Some(span);
3087 }
3088 if let Some((span, _)) =
3089 self.conflict_marker(&TokenKind::EqEq, &TokenKind::Eq, Some(len))
3090 {
3091 middle = Some(span);
3092 }
3093 if let Some((span, _)) =
3094 self.conflict_marker(&TokenKind::Shr, &TokenKind::Gt, Some(len))
3095 {
3096 spans.push(span);
3097 end = Some(span);
3098 break;
3099 }
3100 self.bump();
3101 }
3102
3103 let mut err = self.dcx().struct_span_fatal(spans, "encountered diff marker");
3104 let middle_marker = match middlediff3 {
3105 Some(middlediff3) => {
3107 err.span_label(
3108 middlediff3,
3109 "between this marker and `=======` is the base code (what the two refs \
3110 diverged from)",
3111 );
3112 "|||||||"
3113 }
3114 None => "=======",
3115 };
3116 err.span_label(
3117 start,
3118 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("between this marker and `{0}` is the code that you are merging into",
middle_marker))
})format!(
3119 "between this marker and `{middle_marker}` is the code that you are merging into",
3120 ),
3121 );
3122
3123 if let Some(middle) = middle {
3124 err.span_label(middle, "between this marker and `>>>>>>>` is the incoming code");
3125 }
3126 if let Some(end) = end {
3127 err.span_label(end, "this marker concludes the conflict region");
3128 }
3129 err.note(
3130 "conflict markers indicate that a merge was started but could not be completed due \
3131 to merge conflicts\n\
3132 to resolve a conflict, keep only the code you want and then delete the lines \
3133 containing conflict markers",
3134 );
3135 err.help(
3136 "if you are in a merge, the top section is the code you already had checked out and \
3137 the bottom section is the new code\n\
3138 if you are in a rebase, the top section is the code being rebased onto and the bottom \
3139 section is the code you had checked out which is being rebased",
3140 );
3141
3142 err.note(
3143 "for an explanation on these markers from the `git` documentation, visit \
3144 <https://git-scm.com/book/en/v2/Git-Tools-Advanced-Merging#_checking_out_conflicts>",
3145 );
3146
3147 err.emit();
3148 }
3149
3150 fn skip_pat_list(&mut self) -> PResult<'a, ()> {
3153 while !self.check(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::CloseParen,
token_type: crate::parser::token_type::TokenType::CloseParen,
}exp!(CloseParen)) {
3154 self.parse_pat_no_top_alt(None, None)?;
3155 if !self.eat(crate::parser::token_type::ExpTokenPair {
tok: rustc_ast::token::Comma,
token_type: crate::parser::token_type::TokenType::Comma,
}exp!(Comma)) {
3156 return Ok(());
3157 }
3158 }
3159 Ok(())
3160 }
3161}