improve diagnostics for lifetime after &mut

This commit is contained in:
SNCPlay42 2020-06-21 21:37:17 +01:00 committed by Mark Rousskov
parent 90b1f5ae59
commit 4de9a53d98
3 changed files with 101 additions and 1 deletions

View File

@ -276,8 +276,34 @@ impl<'a> Parser<'a> {
} }
fn parse_borrowed_pointee(&mut self) -> PResult<'a, TyKind> { fn parse_borrowed_pointee(&mut self) -> PResult<'a, TyKind> {
let opt_lifetime = if self.check_lifetime() { Some(self.expect_lifetime()) } else { None }; let and_span = self.prev_token.span;
let mut opt_lifetime =
if self.check_lifetime() { Some(self.expect_lifetime()) } else { None };
let mutbl = self.parse_mutability(); let mutbl = self.parse_mutability();
if self.token.is_lifetime() && mutbl == Mutability::Mut && opt_lifetime.is_none() {
// A lifetime is invalid here: it would be part of a bare trait bound, which requires
// it to be followed by a plus, but we disallow plus in the pointee type.
// So we can handle this case as an error here, and suggest `'a mut`.
// If there *is* a plus next though, handling the error later provides better suggestions
// (like adding parentheses)
if !self.look_ahead(1, |t| t.is_like_plus()) {
let lifetime_span = self.token.span;
let span = and_span.to(lifetime_span);
let mut err = self.struct_span_err(span, "lifetime must precede `mut`");
if let Ok(lifetime_src) = self.span_to_snippet(lifetime_span) {
err.span_suggestion(
span,
"place the lifetime before `mut`",
format!("&{} mut", lifetime_src),
Applicability::MaybeIncorrect,
);
}
err.emit();
opt_lifetime = Some(self.expect_lifetime());
}
}
let ty = self.parse_ty_no_plus()?; let ty = self.parse_ty_no_plus()?;
Ok(TyKind::Rptr(opt_lifetime, MutTy { ty, mutbl })) Ok(TyKind::Rptr(opt_lifetime, MutTy { ty, mutbl }))
} }

View File

@ -0,0 +1,21 @@
#![crate_type="lib"]
fn x<'a>(x: &mut 'a i32){} //~ ERROR lifetime must precede `mut`
macro_rules! mac {
($lt:lifetime) => {
fn w<$lt>(w: &mut $lt i32) {}
//~^ ERROR lifetime must precede `mut`
}
}
mac!('a);
// avoid false positives
fn y<'a>(y: &mut 'a + Send) {
//~^ ERROR expected a path on the left-hand side of `+`, not `&mut 'a`
//~| WARNING trait objects without an explicit `dyn` are deprecated
//~| ERROR at least one trait is required for an object type
let z = y as &mut 'a + Send;
//~^ ERROR expected value, found trait `Send`
//~| WARNING trait objects without an explicit `dyn` are deprecated
}

View File

@ -0,0 +1,53 @@
error: lifetime must precede `mut`
--> $DIR/issue-73568-lifetime-after-mut.rs:2:13
|
LL | fn x<'a>(x: &mut 'a i32){}
| ^^^^^^^ help: place the lifetime before `mut`: `&'a mut`
error[E0178]: expected a path on the left-hand side of `+`, not `&mut 'a`
--> $DIR/issue-73568-lifetime-after-mut.rs:14:13
|
LL | fn y<'a>(y: &mut 'a + Send) {
| ^^^^^^^^^^^^^^ help: try adding parentheses: `&mut ('a + Send)`
error: lifetime must precede `mut`
--> $DIR/issue-73568-lifetime-after-mut.rs:6:22
|
LL | fn w<$lt>(w: &mut $lt i32) {}
| ^^^^^^^^ help: place the lifetime before `mut`: `&$lt mut`
...
LL | mac!('a);
| --------- in this macro invocation
|
= note: this error originates in a macro (in Nightly builds, run with -Z macro-backtrace for more info)
error[E0423]: expected value, found trait `Send`
--> $DIR/issue-73568-lifetime-after-mut.rs:18:28
|
LL | let z = y as &mut 'a + Send;
| ^^^^ not a value
warning: trait objects without an explicit `dyn` are deprecated
--> $DIR/issue-73568-lifetime-after-mut.rs:14:18
|
LL | fn y<'a>(y: &mut 'a + Send) {
| ^^ help: use `dyn`: `dyn 'a`
|
= note: `#[warn(bare_trait_objects)]` on by default
warning: trait objects without an explicit `dyn` are deprecated
--> $DIR/issue-73568-lifetime-after-mut.rs:18:23
|
LL | let z = y as &mut 'a + Send;
| ^^ help: use `dyn`: `dyn 'a`
error[E0224]: at least one trait is required for an object type
--> $DIR/issue-73568-lifetime-after-mut.rs:14:18
|
LL | fn y<'a>(y: &mut 'a + Send) {
| ^^
error: aborting due to 5 previous errors; 2 warnings emitted
Some errors have detailed explanations: E0178, E0224, E0423.
For more information about an error, try `rustc --explain E0178`.