|
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