|
1 | 1 | use super::ty::AllowPlus; |
2 | 2 | use super::{BlockMode, Parser, PathStyle, SemiColonMode, SeqSep, TokenExpectType, TokenType}; |
3 | 3 |
|
4 | | -use rustc_ast::ast::{self, BinOpKind, BindingMode, BlockCheckMode, Expr, ExprKind, Item, Param}; |
5 | 4 | use rustc_ast::ast::{ |
6 | | - AngleBracketedArgs, AttrVec, ItemKind, Mutability, Pat, PatKind, PathSegment, QSelf, Ty, TyKind, |
| 5 | + self, AngleBracketedArgs, AttrVec, BinOpKind, BindingMode, BlockCheckMode, Expr, ExprKind, |
| 6 | + Item, ItemKind, Mutability, Param, Pat, PatKind, PathSegment, QSelf, Ty, TyKind, |
7 | 7 | }; |
8 | 8 | use rustc_ast::ptr::P; |
9 | 9 | use rustc_ast::token::{self, Lit, LitKind, TokenKind}; |
@@ -498,29 +498,42 @@ impl<'a> Parser<'a> { |
498 | 498 | self.bump(); |
499 | 499 | let lo = self.token.span; |
500 | 500 | match self.parse_angle_args() { |
501 | | - Ok(args) if self.token.kind == token::OpenDelim(token::Paren) => { |
502 | | - // Recover from bad turbofish: `foo.collect::Vec<_>()`. |
| 501 | + Ok(args) => { |
503 | 502 | let span = lo.to(self.prev_token.span); |
504 | | - let args = AngleBracketedArgs { args, span }.into(); |
505 | | - segment.args = args; |
506 | | - self.struct_span_err( |
507 | | - span, |
508 | | - "generic parameters without surrounding angle brackets", |
509 | | - ) |
510 | | - .multipart_suggestion( |
511 | | - "surround the type parameters with angle brackets", |
512 | | - vec![ |
513 | | - (span.shrink_to_lo(), "<".to_string()), |
514 | | - (span.shrink_to_hi(), ">".to_string()), |
515 | | - ], |
516 | | - Applicability::MachineApplicable, |
517 | | - ) |
518 | | - .emit(); |
519 | | - } |
520 | | - Ok(_) => { |
521 | | - *self = snapshot; |
| 503 | + // Detect trailing `>` like in `x.collect::Vec<_>>()`. |
| 504 | + let mut trailing_span = self.prev_token.span.shrink_to_hi(); |
| 505 | + while self.token.kind == token::BinOp(token::Shr) |
| 506 | + || self.token.kind == token::Gt |
| 507 | + { |
| 508 | + trailing_span = trailing_span.to(self.token.span); |
| 509 | + self.bump(); |
| 510 | + } |
| 511 | + if self.token.kind == token::OpenDelim(token::Paren) { |
| 512 | + // Recover from bad turbofish: `foo.collect::Vec<_>()`. |
| 513 | + let args = AngleBracketedArgs { args, span }.into(); |
| 514 | + segment.args = args; |
| 515 | + |
| 516 | + self.struct_span_err( |
| 517 | + span, |
| 518 | + "generic parameters without surrounding angle brackets", |
| 519 | + ) |
| 520 | + .multipart_suggestion( |
| 521 | + "surround the type parameters with angle brackets", |
| 522 | + vec![ |
| 523 | + (span.shrink_to_lo(), "<".to_string()), |
| 524 | + (trailing_span, ">".to_string()), |
| 525 | + ], |
| 526 | + Applicability::MachineApplicable, |
| 527 | + ) |
| 528 | + .emit(); |
| 529 | + } else { |
| 530 | + // This doesn't look like an invalid turbofish, can't recover parse state. |
| 531 | + *self = snapshot; |
| 532 | + } |
522 | 533 | } |
523 | 534 | Err(mut err) => { |
| 535 | + // We could't parse generic parameters, unlikely to be a turbofish. Rely on |
| 536 | + // generic parse error instead. |
524 | 537 | err.cancel(); |
525 | 538 | *self = snapshot; |
526 | 539 | } |
|
0 commit comments