1
Fork 0

Rollup merge of #57540 - estebank:eval-more, r=petrochenkov

Modify some parser diagnostics to continue evaluating beyond the parser

Continue evaluating further errors after parser errors on:
 - trailing type argument attribute
 - lifetime in incorrect location
 - incorrect binary literal
 - missing `for` in `impl Trait for Foo`
 - type argument in `where` clause
 - incorrect float literal
 - incorrect `..` in pattern
 - associated types
 - incorrect discriminator value variant error

and others. All of these were found by making `continue-parse-after-error` `true` by default to identify errors that would need few changes. There are now only a handful of errors that have any change with `continue-parse-after-error` enabled.

These changes make it so `rust` _won't_ stop evaluation after finishing parsing, enabling type checking errors to be displayed on the existing code without having to fix the parse errors.

Each commit has an individual diagnostic change with their corresponding tests.

CC #48724.
This commit is contained in:
Mazdak Farrokhzad 2019-01-14 20:31:55 +01:00 committed by GitHub
commit 2f7a226c4d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
41 changed files with 344 additions and 141 deletions

View file

@ -299,7 +299,7 @@ impl<'a> StringReader<'a> {
/// Report a lexical error with a given span. /// Report a lexical error with a given span.
fn err_span(&self, sp: Span, m: &str) { fn err_span(&self, sp: Span, m: &str) {
self.sess.span_diagnostic.span_err(sp, m) self.sess.span_diagnostic.struct_span_err(sp, m).emit();
} }

View file

@ -520,6 +520,7 @@ fn filtered_float_lit(data: Symbol, suffix: Option<Symbol>, diag: Option<(Span,
} else { } else {
let msg = format!("invalid suffix `{}` for float literal", suf); let msg = format!("invalid suffix `{}` for float literal", suf);
diag.struct_span_err(span, &msg) diag.struct_span_err(span, &msg)
.span_label(span, format!("invalid suffix `{}`", suf))
.help("valid suffixes are `f32` and `f64`") .help("valid suffixes are `f32` and `f64`")
.emit(); .emit();
} }
@ -673,7 +674,11 @@ fn integer_lit(s: &str, suffix: Option<Symbol>, diag: Option<(Span, &Handler)>)
_ => None, _ => None,
}; };
if let Some(err) = err { if let Some(err) = err {
err!(diag, |span, diag| diag.span_err(span, err)); err!(diag, |span, diag| {
diag.struct_span_err(span, err)
.span_label(span, "not supported")
.emit();
});
} }
return filtered_float_lit(Symbol::intern(s), Some(suf), diag) return filtered_float_lit(Symbol::intern(s), Some(suf), diag)
} }
@ -712,6 +717,7 @@ fn integer_lit(s: &str, suffix: Option<Symbol>, diag: Option<(Span, &Handler)>)
} else { } else {
let msg = format!("invalid suffix `{}` for numeric literal", suf); let msg = format!("invalid suffix `{}` for numeric literal", suf);
diag.struct_span_err(span, &msg) diag.struct_span_err(span, &msg)
.span_label(span, format!("invalid suffix `{}`", suf))
.help("the suffix must be one of the integral types \ .help("the suffix must be one of the integral types \
(`u32`, `isize`, etc)") (`u32`, `isize`, etc)")
.emit(); .emit();

View file

@ -1012,7 +1012,10 @@ impl<'a> Parser<'a> {
if text.is_empty() { if text.is_empty() {
self.span_bug(sp, "found empty literal suffix in Some") self.span_bug(sp, "found empty literal suffix in Some")
} }
self.span_err(sp, &format!("{} with a suffix is invalid", kind)); let msg = format!("{} with a suffix is invalid", kind);
self.struct_span_err(sp, &msg)
.span_label(sp, msg)
.emit();
} }
} }
} }
@ -1768,9 +1771,11 @@ impl<'a> Parser<'a> {
Mutability::Immutable Mutability::Immutable
} else { } else {
let span = self.prev_span; let span = self.prev_span;
self.span_err(span, let msg = "expected mut or const in raw pointer type";
"expected mut or const in raw pointer type (use \ self.struct_span_err(span, msg)
`*mut T` or `*const T` as appropriate)"); .span_label(span, msg)
.help("use `*mut T` or `*const T` as appropriate")
.emit();
Mutability::Immutable Mutability::Immutable
}; };
let t = self.parse_ty_no_plus()?; let t = self.parse_ty_no_plus()?;
@ -3815,8 +3820,12 @@ impl<'a> Parser<'a> {
ddpos = Some(fields.len()); ddpos = Some(fields.len());
} else { } else {
// Emit a friendly error, ignore `..` and continue parsing // Emit a friendly error, ignore `..` and continue parsing
self.span_err(self.prev_span, self.struct_span_err(
"`..` can only be used once per tuple or tuple struct pattern"); self.prev_span,
"`..` can only be used once per tuple or tuple struct pattern",
)
.span_label(self.prev_span, "can only be used once per pattern")
.emit();
} }
} else if !self.check(&token::CloseDelim(token::Paren)) { } else if !self.check(&token::CloseDelim(token::Paren)) {
fields.push(self.parse_pat(None)?); fields.push(self.parse_pat(None)?);
@ -3832,7 +3841,10 @@ impl<'a> Parser<'a> {
if ddpos == Some(fields.len()) && trailing_comma { if ddpos == Some(fields.len()) && trailing_comma {
// `..` needs to be followed by `)` or `, pat`, `..,)` is disallowed. // `..` needs to be followed by `)` or `, pat`, `..,)` is disallowed.
self.span_err(self.prev_span, "trailing comma is not permitted after `..`"); let msg = "trailing comma is not permitted after `..`";
self.struct_span_err(self.prev_span, msg)
.span_label(self.prev_span, msg)
.emit();
} }
Ok((fields, ddpos, trailing_comma)) Ok((fields, ddpos, trailing_comma))
@ -5256,8 +5268,12 @@ impl<'a> Parser<'a> {
// Check for trailing attributes and stop parsing. // Check for trailing attributes and stop parsing.
if !attrs.is_empty() { if !attrs.is_empty() {
let param_kind = if seen_ty_param.is_some() { "type" } else { "lifetime" }; let param_kind = if seen_ty_param.is_some() { "type" } else { "lifetime" };
self.span_err(attrs[0].span, self.struct_span_err(
&format!("trailing attribute after {} parameters", param_kind)); attrs[0].span,
&format!("trailing attribute after {} parameters", param_kind),
)
.span_label(attrs[0].span, "attributes must go before parameters")
.emit();
} }
break break
} }
@ -5315,19 +5331,28 @@ impl<'a> Parser<'a> {
/// Parses (possibly empty) list of lifetime and type arguments and associated type bindings, /// Parses (possibly empty) list of lifetime and type arguments and associated type bindings,
/// possibly including trailing comma. /// possibly including trailing comma.
fn parse_generic_args(&mut self) fn parse_generic_args(&mut self) -> PResult<'a, (Vec<GenericArg>, Vec<TypeBinding>)> {
-> PResult<'a, (Vec<GenericArg>, Vec<TypeBinding>)> {
let mut args = Vec::new(); let mut args = Vec::new();
let mut bindings = Vec::new(); let mut bindings = Vec::new();
let mut seen_type = false; let mut seen_type = false;
let mut seen_binding = false; let mut seen_binding = false;
let mut first_type_or_binding_span: Option<Span> = None;
let mut bad_lifetime_pos = vec![];
let mut last_comma_span = None;
let mut suggestions = vec![];
loop { loop {
if self.check_lifetime() && self.look_ahead(1, |t| !t.is_like_plus()) { if self.check_lifetime() && self.look_ahead(1, |t| !t.is_like_plus()) {
// Parse lifetime argument. // Parse lifetime argument.
args.push(GenericArg::Lifetime(self.expect_lifetime())); args.push(GenericArg::Lifetime(self.expect_lifetime()));
if seen_type || seen_binding { if seen_type || seen_binding {
self.span_err(self.prev_span, let remove_sp = last_comma_span.unwrap_or(self.prev_span).to(self.prev_span);
"lifetime parameters must be declared prior to type parameters"); bad_lifetime_pos.push(self.prev_span);
if let Ok(snippet) = self.sess.source_map().span_to_snippet(self.prev_span) {
suggestions.push((remove_sp, String::new()));
suggestions.push((
first_type_or_binding_span.unwrap().shrink_to_lo(),
format!("{}, ", snippet)));
}
} }
} else if self.check_ident() && self.look_ahead(1, |t| t == &token::Eq) { } else if self.check_ident() && self.look_ahead(1, |t| t == &token::Eq) {
// Parse associated type binding. // Parse associated type binding.
@ -5335,19 +5360,33 @@ impl<'a> Parser<'a> {
let ident = self.parse_ident()?; let ident = self.parse_ident()?;
self.bump(); self.bump();
let ty = self.parse_ty()?; let ty = self.parse_ty()?;
let span = lo.to(self.prev_span);
bindings.push(TypeBinding { bindings.push(TypeBinding {
id: ast::DUMMY_NODE_ID, id: ast::DUMMY_NODE_ID,
ident, ident,
ty, ty,
span: lo.to(self.prev_span), span,
}); });
seen_binding = true; seen_binding = true;
if first_type_or_binding_span.is_none() {
first_type_or_binding_span = Some(span);
}
} else if self.check_type() { } else if self.check_type() {
// Parse type argument. // Parse type argument.
let ty_param = self.parse_ty()?; let ty_param = self.parse_ty()?;
if seen_binding { if seen_binding {
self.span_err(ty_param.span, self.struct_span_err(
"type parameters must be declared prior to associated type bindings"); ty_param.span,
"type parameters must be declared prior to associated type bindings"
)
.span_label(
ty_param.span,
"must be declared prior to associated type bindings",
)
.emit();
}
if first_type_or_binding_span.is_none() {
first_type_or_binding_span = Some(ty_param.span);
} }
args.push(GenericArg::Type(ty_param)); args.push(GenericArg::Type(ty_param));
seen_type = true; seen_type = true;
@ -5357,8 +5396,30 @@ impl<'a> Parser<'a> {
if !self.eat(&token::Comma) { if !self.eat(&token::Comma) {
break break
} else {
last_comma_span = Some(self.prev_span);
} }
} }
if !bad_lifetime_pos.is_empty() {
let mut err = self.struct_span_err(
bad_lifetime_pos.clone(),
"lifetime parameters must be declared prior to type parameters"
);
for sp in &bad_lifetime_pos {
err.span_label(*sp, "must be declared prior to type parameters");
}
if !suggestions.is_empty() {
err.multipart_suggestion_with_applicability(
&format!(
"move the lifetime parameter{} prior to the first type parameter",
if bad_lifetime_pos.len() > 1 { "s" } else { "" },
),
suggestions,
Applicability::MachineApplicable,
);
}
err.emit();
}
Ok((args, bindings)) Ok((args, bindings))
} }
@ -5386,8 +5447,12 @@ impl<'a> Parser<'a> {
// change we parse those generics now, but report an error. // change we parse those generics now, but report an error.
if self.choose_generics_over_qpath() { if self.choose_generics_over_qpath() {
let generics = self.parse_generics()?; let generics = self.parse_generics()?;
self.span_err(generics.span, self.struct_span_err(
"generic parameters on `where` clauses are reserved for future use"); generics.span,
"generic parameters on `where` clauses are reserved for future use",
)
.span_label(generics.span, "currently unsupported")
.emit();
} }
loop { loop {
@ -5587,15 +5652,20 @@ impl<'a> Parser<'a> {
// *mut self // *mut self
// *not_self // *not_self
// Emit special error for `self` cases. // Emit special error for `self` cases.
let msg = "cannot pass `self` by raw pointer";
(if isolated_self(self, 1) { (if isolated_self(self, 1) {
self.bump(); self.bump();
self.span_err(self.span, "cannot pass `self` by raw pointer"); self.struct_span_err(self.span, msg)
.span_label(self.span, msg)
.emit();
SelfKind::Value(Mutability::Immutable) SelfKind::Value(Mutability::Immutable)
} else if self.look_ahead(1, |t| t.is_mutability()) && } else if self.look_ahead(1, |t| t.is_mutability()) &&
isolated_self(self, 2) { isolated_self(self, 2) {
self.bump(); self.bump();
self.bump(); self.bump();
self.span_err(self.span, "cannot pass `self` by raw pointer"); self.struct_span_err(self.span, msg)
.span_label(self.span, msg)
.emit();
SelfKind::Value(Mutability::Immutable) SelfKind::Value(Mutability::Immutable)
} else { } else {
return Ok(None); return Ok(None);
@ -5932,7 +6002,10 @@ impl<'a> Parser<'a> {
tps.where_clause = self.parse_where_clause()?; tps.where_clause = self.parse_where_clause()?;
self.expect(&token::Semi)?; self.expect(&token::Semi)?;
if unsafety != Unsafety::Normal { if unsafety != Unsafety::Normal {
self.span_err(self.prev_span, "trait aliases cannot be unsafe"); let msg = "trait aliases cannot be unsafe";
self.struct_span_err(self.prev_span, msg)
.span_label(self.prev_span, msg)
.emit();
} }
Ok((ident, ItemKind::TraitAlias(tps, bounds), None)) Ok((ident, ItemKind::TraitAlias(tps, bounds), None))
} else { } else {
@ -6048,7 +6121,13 @@ impl<'a> Parser<'a> {
Some(ty_second) => { Some(ty_second) => {
// impl Trait for Type // impl Trait for Type
if !has_for { if !has_for {
self.span_err(missing_for_span, "missing `for` in a trait impl"); self.struct_span_err(missing_for_span, "missing `for` in a trait impl")
.span_suggestion_short_with_applicability(
missing_for_span,
"add `for` here",
" for ".to_string(),
Applicability::MachineApplicable,
).emit();
} }
let ty_first = ty_first.into_inner(); let ty_first = ty_first.into_inner();
@ -6939,7 +7018,7 @@ impl<'a> Parser<'a> {
fn parse_enum_def(&mut self, _generics: &ast::Generics) -> PResult<'a, EnumDef> { fn parse_enum_def(&mut self, _generics: &ast::Generics) -> PResult<'a, EnumDef> {
let mut variants = Vec::new(); let mut variants = Vec::new();
let mut all_nullary = true; let mut all_nullary = true;
let mut any_disr = None; let mut any_disr = vec![];
while self.token != token::CloseDelim(token::Brace) { while self.token != token::CloseDelim(token::Brace) {
let variant_attrs = self.parse_outer_attributes()?; let variant_attrs = self.parse_outer_attributes()?;
let vlo = self.span; let vlo = self.span;
@ -6961,7 +7040,9 @@ impl<'a> Parser<'a> {
id: ast::DUMMY_NODE_ID, id: ast::DUMMY_NODE_ID,
value: self.parse_expr()?, value: self.parse_expr()?,
}); });
any_disr = disr_expr.as_ref().map(|c| c.value.span); if let Some(sp) = disr_expr.as_ref().map(|c| c.value.span) {
any_disr.push(sp);
}
struct_def = VariantData::Unit(ast::DUMMY_NODE_ID); struct_def = VariantData::Unit(ast::DUMMY_NODE_ID);
} else { } else {
struct_def = VariantData::Unit(ast::DUMMY_NODE_ID); struct_def = VariantData::Unit(ast::DUMMY_NODE_ID);
@ -6978,11 +7059,15 @@ impl<'a> Parser<'a> {
if !self.eat(&token::Comma) { break; } if !self.eat(&token::Comma) { break; }
} }
self.expect(&token::CloseDelim(token::Brace))?; self.expect(&token::CloseDelim(token::Brace))?;
match any_disr { if !any_disr.is_empty() && !all_nullary {
Some(disr_span) if !all_nullary => let mut err =self.struct_span_err(
self.span_err(disr_span, any_disr.clone(),
"discriminator values can only be used with a field-less enum"), "discriminator values can only be used with a field-less enum",
_ => () );
for sp in any_disr {
err.span_label(sp, "only valid in field-less enums");
}
err.emit();
} }
Ok(ast::EnumDef { variants }) Ok(ast::EnumDef { variants })

View file

@ -2,7 +2,7 @@ error: trailing attribute after lifetime parameters
--> $DIR/attrs-with-no-formal-in-generics-1.rs:9:25 --> $DIR/attrs-with-no-formal-in-generics-1.rs:9:25
| |
LL | impl<#[rustc_1] 'a, 'b, #[oops]> RefIntPair<'a, 'b> { LL | impl<#[rustc_1] 'a, 'b, #[oops]> RefIntPair<'a, 'b> {
| ^^^^^^^ | ^^^^^^^ attributes must go before parameters
error: aborting due to previous error error: aborting due to previous error

View file

@ -2,7 +2,7 @@ error: trailing attribute after type parameters
--> $DIR/attrs-with-no-formal-in-generics-2.rs:9:35 --> $DIR/attrs-with-no-formal-in-generics-2.rs:9:35
| |
LL | impl<#[rustc_1] 'a, #[rustc_2] T, #[oops]> RefAny<'a, T> {} LL | impl<#[rustc_1] 'a, #[rustc_2] T, #[oops]> RefAny<'a, T> {}
| ^^^^^^^ | ^^^^^^^ attributes must go before parameters
error: aborting due to previous error error: aborting due to previous error

View file

@ -5,12 +5,8 @@
struct RefIntPair<'a, 'b>(&'a u32, &'b u32); struct RefIntPair<'a, 'b>(&'a u32, &'b u32);
fn hof_lt<Q>(_: Q) fn hof_lt<Q>(_: Q)
where Q: for <#[rustc_1] 'a, 'b, #[oops]> Fn(RefIntPair<'a,'b>) -> &'b u32 where Q: for <#[allow(unused)] 'a, 'b, #[oops]> Fn(RefIntPair<'a,'b>) -> &'b u32
//~^ ERROR trailing attribute after lifetime parameters //~^ ERROR trailing attribute after lifetime parameters
{ {}
} fn main() {}
fn main() {
}

View file

@ -1,8 +1,8 @@
error: trailing attribute after lifetime parameters error: trailing attribute after lifetime parameters
--> $DIR/attrs-with-no-formal-in-generics-3.rs:8:38 --> $DIR/attrs-with-no-formal-in-generics-3.rs:8:44
| |
LL | where Q: for <#[rustc_1] 'a, 'b, #[oops]> Fn(RefIntPair<'a,'b>) -> &'b u32 LL | where Q: for <#[allow(unused)] 'a, 'b, #[oops]> Fn(RefIntPair<'a,'b>) -> &'b u32
| ^^^^^^^ | ^^^^^^^ attributes must go before parameters
error: aborting due to previous error error: aborting due to previous error

View file

@ -2,7 +2,7 @@ error: invalid suffix `is` for numeric literal
--> $DIR/old-suffixes-are-really-forbidden.rs:2:13 --> $DIR/old-suffixes-are-really-forbidden.rs:2:13
| |
LL | let a = 1_is; //~ ERROR invalid suffix LL | let a = 1_is; //~ ERROR invalid suffix
| ^^^^ | ^^^^ invalid suffix `is`
| |
= help: the suffix must be one of the integral types (`u32`, `isize`, etc) = help: the suffix must be one of the integral types (`u32`, `isize`, etc)
@ -10,7 +10,7 @@ error: invalid suffix `us` for numeric literal
--> $DIR/old-suffixes-are-really-forbidden.rs:3:13 --> $DIR/old-suffixes-are-really-forbidden.rs:3:13
| |
LL | let b = 2_us; //~ ERROR invalid suffix LL | let b = 2_us; //~ ERROR invalid suffix
| ^^^^ | ^^^^ invalid suffix `us`
| |
= help: the suffix must be one of the integral types (`u32`, `isize`, etc) = help: the suffix must be one of the integral types (`u32`, `isize`, etc)

View file

@ -2,49 +2,49 @@ error: ABI spec with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:5:5 --> $DIR/bad-lit-suffixes.rs:5:5
| |
LL | "C"suffix //~ ERROR ABI spec with a suffix is invalid LL | "C"suffix //~ ERROR ABI spec with a suffix is invalid
| ^^^^^^^^^ | ^^^^^^^^^ ABI spec with a suffix is invalid
error: ABI spec with a suffix is invalid error: ABI spec with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:9:5 --> $DIR/bad-lit-suffixes.rs:9:5
| |
LL | "C"suffix //~ ERROR ABI spec with a suffix is invalid LL | "C"suffix //~ ERROR ABI spec with a suffix is invalid
| ^^^^^^^^^ | ^^^^^^^^^ ABI spec with a suffix is invalid
error: string literal with a suffix is invalid error: string literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:13:5 --> $DIR/bad-lit-suffixes.rs:13:5
| |
LL | ""suffix; //~ ERROR string literal with a suffix is invalid LL | ""suffix; //~ ERROR string literal with a suffix is invalid
| ^^^^^^^^ | ^^^^^^^^ string literal with a suffix is invalid
error: byte string literal with a suffix is invalid error: byte string literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:14:5 --> $DIR/bad-lit-suffixes.rs:14:5
| |
LL | b""suffix; //~ ERROR byte string literal with a suffix is invalid LL | b""suffix; //~ ERROR byte string literal with a suffix is invalid
| ^^^^^^^^^ | ^^^^^^^^^ byte string literal with a suffix is invalid
error: string literal with a suffix is invalid error: string literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:15:5 --> $DIR/bad-lit-suffixes.rs:15:5
| |
LL | r#""#suffix; //~ ERROR string literal with a suffix is invalid LL | r#""#suffix; //~ ERROR string literal with a suffix is invalid
| ^^^^^^^^^^^ | ^^^^^^^^^^^ string literal with a suffix is invalid
error: byte string literal with a suffix is invalid error: byte string literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:16:5 --> $DIR/bad-lit-suffixes.rs:16:5
| |
LL | br#""#suffix; //~ ERROR byte string literal with a suffix is invalid LL | br#""#suffix; //~ ERROR byte string literal with a suffix is invalid
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^ byte string literal with a suffix is invalid
error: char literal with a suffix is invalid error: char literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:17:5 --> $DIR/bad-lit-suffixes.rs:17:5
| |
LL | 'a'suffix; //~ ERROR char literal with a suffix is invalid LL | 'a'suffix; //~ ERROR char literal with a suffix is invalid
| ^^^^^^^^^ | ^^^^^^^^^ char literal with a suffix is invalid
error: byte literal with a suffix is invalid error: byte literal with a suffix is invalid
--> $DIR/bad-lit-suffixes.rs:18:5 --> $DIR/bad-lit-suffixes.rs:18:5
| |
LL | b'a'suffix; //~ ERROR byte literal with a suffix is invalid LL | b'a'suffix; //~ ERROR byte literal with a suffix is invalid
| ^^^^^^^^^^ | ^^^^^^^^^^ byte literal with a suffix is invalid
error: invalid width `1024` for integer literal error: invalid width `1024` for integer literal
--> $DIR/bad-lit-suffixes.rs:20:5 --> $DIR/bad-lit-suffixes.rs:20:5
@ -82,7 +82,7 @@ error: invalid suffix `suffix` for numeric literal
--> $DIR/bad-lit-suffixes.rs:25:5 --> $DIR/bad-lit-suffixes.rs:25:5
| |
LL | 1234suffix; //~ ERROR invalid suffix `suffix` for numeric literal LL | 1234suffix; //~ ERROR invalid suffix `suffix` for numeric literal
| ^^^^^^^^^^ | ^^^^^^^^^^ invalid suffix `suffix`
| |
= help: the suffix must be one of the integral types (`u32`, `isize`, etc) = help: the suffix must be one of the integral types (`u32`, `isize`, etc)
@ -90,7 +90,7 @@ error: invalid suffix `suffix` for numeric literal
--> $DIR/bad-lit-suffixes.rs:26:5 --> $DIR/bad-lit-suffixes.rs:26:5
| |
LL | 0b101suffix; //~ ERROR invalid suffix `suffix` for numeric literal LL | 0b101suffix; //~ ERROR invalid suffix `suffix` for numeric literal
| ^^^^^^^^^^^ | ^^^^^^^^^^^ invalid suffix `suffix`
| |
= help: the suffix must be one of the integral types (`u32`, `isize`, etc) = help: the suffix must be one of the integral types (`u32`, `isize`, etc)
@ -98,7 +98,7 @@ error: invalid suffix `suffix` for float literal
--> $DIR/bad-lit-suffixes.rs:27:5 --> $DIR/bad-lit-suffixes.rs:27:5
| |
LL | 1.0suffix; //~ ERROR invalid suffix `suffix` for float literal LL | 1.0suffix; //~ ERROR invalid suffix `suffix` for float literal
| ^^^^^^^^^ | ^^^^^^^^^ invalid suffix `suffix`
| |
= help: valid suffixes are `f32` and `f64` = help: valid suffixes are `f32` and `f64`
@ -106,7 +106,7 @@ error: invalid suffix `suffix` for float literal
--> $DIR/bad-lit-suffixes.rs:28:5 --> $DIR/bad-lit-suffixes.rs:28:5
| |
LL | 1.0e10suffix; //~ ERROR invalid suffix `suffix` for float literal LL | 1.0e10suffix; //~ ERROR invalid suffix `suffix` for float literal
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^ invalid suffix `suffix`
| |
= help: valid suffixes are `f32` and `f64` = help: valid suffixes are `f32` and `f64`

View file

@ -1,5 +1,5 @@
fn foo(_: *()) { fn foo(_: *()) {
//~^ expected mut or const in raw pointer type (use `*mut T` or `*const T` as appropriate) //~^ ERROR expected mut or const in raw pointer type
} }
fn main() {} fn main() {}

View file

@ -1,8 +1,10 @@
error: expected mut or const in raw pointer type (use `*mut T` or `*const T` as appropriate) error: expected mut or const in raw pointer type
--> $DIR/bad-pointer-type.rs:1:11 --> $DIR/bad-pointer-type.rs:1:11
| |
LL | fn foo(_: *()) { LL | fn foo(_: *()) {
| ^ | ^ expected mut or const in raw pointer type
|
= help: use `*mut T` or `*const T` as appropriate
error: aborting due to previous error error: aborting due to previous error

View file

@ -2,13 +2,13 @@ error: missing `for` in a trait impl
--> $DIR/impl-parsing.rs:6:11 --> $DIR/impl-parsing.rs:6:11
| |
LL | impl Trait Type {} //~ ERROR missing `for` in a trait impl LL | impl Trait Type {} //~ ERROR missing `for` in a trait impl
| ^ | ^ help: add `for` here
error: missing `for` in a trait impl error: missing `for` in a trait impl
--> $DIR/impl-parsing.rs:7:11 --> $DIR/impl-parsing.rs:7:11
| |
LL | impl Trait .. {} //~ ERROR missing `for` in a trait impl LL | impl Trait .. {} //~ ERROR missing `for` in a trait impl
| ^ | ^ help: add `for` here
error: expected a trait, found type error: expected a trait, found type
--> $DIR/impl-parsing.rs:8:6 --> $DIR/impl-parsing.rs:8:6

View file

@ -1,6 +1,17 @@
fn main() { // can't run rustfix because it doesn't handle multipart suggestions correctly
(0..4) // compile-flags: -Zborrowck=mir
.map(|x| x * 2) // we need the above to avoid ast borrowck failure in recovered code
.collect::<Vec<'a, usize, 'b>>()
struct S<'a, T> {
a: &'a T,
b: &'a T,
}
fn foo<'a, 'b>(start: &'a usize, end: &'a usize) {
let _x = (*start..*end)
.map(|x| S { a: start, b: end })
.collect::<Vec<S<_, 'a>>>();
//~^ ERROR lifetime parameters must be declared prior to type parameters //~^ ERROR lifetime parameters must be declared prior to type parameters
} }
fn main() {}

View file

@ -1,8 +1,12 @@
error: lifetime parameters must be declared prior to type parameters error: lifetime parameters must be declared prior to type parameters
--> $DIR/issue-14303-fncall.rs:4:31 --> $DIR/issue-14303-fncall.rs:13:29
| |
LL | .collect::<Vec<'a, usize, 'b>>() LL | .collect::<Vec<S<_, 'a>>>();
| ^^ | ^^ must be declared prior to type parameters
help: move the lifetime parameter prior to the first type parameter
|
LL | .collect::<Vec<S<'a, _>>>();
| ^^^ --
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,4 +1,13 @@
fn bar<'a, T>(x: mymodule::X<'a, T, 'b, 'c>) {} mod foo {
pub struct X<'a, 'b, 'c, T> {
a: &'a str,
b: &'b str,
c: &'c str,
t: T,
}
}
fn bar<'a, 'b, 'c, T>(x: foo::X<'a, T, 'b, 'c>) {}
//~^ ERROR lifetime parameters must be declared prior to type parameters //~^ ERROR lifetime parameters must be declared prior to type parameters
fn main() {} fn main() {}

View file

@ -1,8 +1,14 @@
error: lifetime parameters must be declared prior to type parameters error: lifetime parameters must be declared prior to type parameters
--> $DIR/issue-14303-path.rs:1:37 --> $DIR/issue-14303-path.rs:10:40
| |
LL | fn bar<'a, T>(x: mymodule::X<'a, T, 'b, 'c>) {} LL | fn bar<'a, 'b, 'c, T>(x: foo::X<'a, T, 'b, 'c>) {}
| ^^ | ^^ ^^ must be declared prior to type parameters
| |
| must be declared prior to type parameters
help: move the lifetime parameters prior to the first type parameter
|
LL | fn bar<'a, 'b, 'c, T>(x: foo::X<'a, 'b, 'c, T>) {}
| ^^^ ^^^ --
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,8 +1,7 @@
enum X { enum X {
A = A = 3,
b'a' //~ ERROR discriminator values can only be used with a field-less enum //~^ ERROR discriminator values can only be used with a field-less enum
, B(usize)
B(isize)
} }
fn main() {} fn main() {}

View file

@ -1,8 +1,8 @@
error: discriminator values can only be used with a field-less enum error: discriminator values can only be used with a field-less enum
--> $DIR/issue-17383.rs:3:9 --> $DIR/issue-17383.rs:2:9
| |
LL | b'a' //~ ERROR discriminator values can only be used with a field-less enum LL | A = 3,
| ^^^^ | ^ only valid in field-less enums
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,4 +1,7 @@
// error-pattern:no valid digits found for number fn log(a: i32, b: i32) {}
fn main() { fn main() {
let error = 42;
log(error, 0b); log(error, 0b);
//~^ ERROR no valid digits found for number
} }

View file

@ -1,5 +1,5 @@
error: no valid digits found for number error: no valid digits found for number
--> $DIR/issue-1802-1.rs:3:16 --> $DIR/issue-1802-1.rs:5:16
| |
LL | log(error, 0b); LL | log(error, 0b);
| ^^ | ^^

View file

@ -1,4 +1,7 @@
// error-pattern:no valid digits found for number fn log(a: i32, b: i32) {}
fn main() { fn main() {
log(error, 0b_usize); let error = 42;
log(error, 0b);
//~^ ERROR no valid digits found for number
} }

View file

@ -1,8 +1,8 @@
error: no valid digits found for number error: no valid digits found for number
--> $DIR/issue-1802-2.rs:3:16 --> $DIR/issue-1802-2.rs:5:16
| |
LL | log(error, 0b_usize); LL | log(error, 0b);
| ^^^ | ^^
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,3 +1,10 @@
impl A .. {} //~ ERROR trait A {}
impl A .. {}
//~^ ERROR missing `for` in a trait impl
//~| ERROR `impl Trait for .. {}` is an obsolete syntax
impl A usize {}
//~^ ERROR missing `for` in a trait impl
fn main() {} fn main() {}

View file

@ -1,8 +1,22 @@
error: missing `for` in a trait impl error: missing `for` in a trait impl
--> $DIR/issue-27255.rs:1:7 --> $DIR/issue-27255.rs:3:7
| |
LL | impl A .. {} //~ ERROR LL | impl A .. {}
| ^ | ^ help: add `for` here
error: aborting due to previous error error: missing `for` in a trait impl
--> $DIR/issue-27255.rs:7:7
|
LL | impl A usize {}
| ^^^^^^ help: add `for` here
error: `impl Trait for .. {}` is an obsolete syntax
--> $DIR/issue-27255.rs:3:1
|
LL | impl A .. {}
| ^^^^^^^^^^^^
|
= help: use `auto trait Trait {}` instead
error: aborting due to 3 previous errors

View file

@ -2,7 +2,7 @@ error: type parameters must be declared prior to associated type bindings
--> $DIR/issue-32214.rs:5:34 --> $DIR/issue-32214.rs:5:34
| |
LL | pub fn test<W, I: Trait<Item=(), W> >() {} LL | pub fn test<W, I: Trait<Item=(), W> >() {}
| ^ | ^ must be declared prior to associated type bindings
error: aborting due to previous error error: aborting due to previous error

View file

@ -110,7 +110,7 @@ error: octal float literal is not supported
--> $DIR/lex-bad-numeric-literals.rs:5:5 --> $DIR/lex-bad-numeric-literals.rs:5:5
| |
LL | 0o2f32; //~ ERROR: octal float literal is not supported LL | 0o2f32; //~ ERROR: octal float literal is not supported
| ^^^^^^ | ^^^^^^ not supported
error: int literal is too large error: int literal is too large
--> $DIR/lex-bad-numeric-literals.rs:16:5 --> $DIR/lex-bad-numeric-literals.rs:16:5
@ -128,13 +128,13 @@ error: octal float literal is not supported
--> $DIR/lex-bad-numeric-literals.rs:23:5 --> $DIR/lex-bad-numeric-literals.rs:23:5
| |
LL | 0o123f64; //~ ERROR: octal float literal is not supported LL | 0o123f64; //~ ERROR: octal float literal is not supported
| ^^^^^^^^ | ^^^^^^^^ not supported
error: binary float literal is not supported error: binary float literal is not supported
--> $DIR/lex-bad-numeric-literals.rs:25:5 --> $DIR/lex-bad-numeric-literals.rs:25:5
| |
LL | 0b101f64; //~ ERROR: binary float literal is not supported LL | 0b101f64; //~ ERROR: binary float literal is not supported
| ^^^^^^^^ | ^^^^^^^^ not supported
error: aborting due to 23 previous errors error: aborting due to 23 previous errors

View file

@ -1,7 +1,8 @@
// error-pattern:binary float literal is not supported
fn main() { fn main() {
0b101010f64; 0b101010f64;
//~^ ERROR binary float literal is not supported
0b101.010; 0b101.010;
//~^ ERROR binary float literal is not supported
0b101p4f64; 0b101p4f64;
//~^ ERROR invalid suffix `p4f64` for numeric literal
} }

View file

@ -1,8 +1,22 @@
error: binary float literal is not supported error: binary float literal is not supported
--> $DIR/no-binary-float-literal.rs:5:5 --> $DIR/no-binary-float-literal.rs:4:5
| |
LL | 0b101.010; LL | 0b101.010;
| ^^^^^^^^^ | ^^^^^^^^^
error: aborting due to previous error error: binary float literal is not supported
--> $DIR/no-binary-float-literal.rs:2:5
|
LL | 0b101010f64;
| ^^^^^^^^^^^ not supported
error: invalid suffix `p4f64` for numeric literal
--> $DIR/no-binary-float-literal.rs:6:5
|
LL | 0b101p4f64;
| ^^^^^^^^^^ invalid suffix `p4f64`
|
= help: the suffix must be one of the integral types (`u32`, `isize`, etc)
error: aborting due to 3 previous errors

View file

@ -1,7 +1,9 @@
// error-pattern:hexadecimal float literal is not supported
fn main() { fn main() {
0xABC.Df; 0xABC.Df;
//~^ ERROR `{integer}` is a primitive type and therefore doesn't have fields
0x567.89; 0x567.89;
//~^ ERROR hexadecimal float literal is not supported
0xDEAD.BEEFp-2f; 0xDEAD.BEEFp-2f;
//~^ ERROR invalid suffix `f` for float literal
//~| ERROR `{integer}` is a primitive type and therefore doesn't have fields
} }

View file

@ -1,8 +1,29 @@
error: hexadecimal float literal is not supported error: hexadecimal float literal is not supported
--> $DIR/no-hex-float-literal.rs:5:5 --> $DIR/no-hex-float-literal.rs:4:5
| |
LL | 0x567.89; LL | 0x567.89;
| ^^^^^^^^ | ^^^^^^^^
error: aborting due to previous error error: invalid suffix `f` for float literal
--> $DIR/no-hex-float-literal.rs:6:18
|
LL | 0xDEAD.BEEFp-2f;
| ^^ invalid suffix `f`
|
= help: valid suffixes are `f32` and `f64`
error[E0610]: `{integer}` is a primitive type and therefore doesn't have fields
--> $DIR/no-hex-float-literal.rs:2:11
|
LL | 0xABC.Df;
| ^^
error[E0610]: `{integer}` is a primitive type and therefore doesn't have fields
--> $DIR/no-hex-float-literal.rs:6:12
|
LL | 0xDEAD.BEEFp-2f;
| ^^^^^
error: aborting due to 4 previous errors
For more information about this error, try `rustc --explain E0610`.

View file

@ -2,37 +2,37 @@ error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:4:17 --> $DIR/no-unsafe-self.rs:4:17
| |
LL | fn foo(*mut self); //~ ERROR cannot pass `self` by raw pointer LL | fn foo(*mut self); //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: cannot pass `self` by raw pointer error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:5:19 --> $DIR/no-unsafe-self.rs:5:19
| |
LL | fn baz(*const self); //~ ERROR cannot pass `self` by raw pointer LL | fn baz(*const self); //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: cannot pass `self` by raw pointer error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:6:13 --> $DIR/no-unsafe-self.rs:6:13
| |
LL | fn bar(*self); //~ ERROR cannot pass `self` by raw pointer LL | fn bar(*self); //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: cannot pass `self` by raw pointer error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:11:17 --> $DIR/no-unsafe-self.rs:11:17
| |
LL | fn foo(*mut self) { } //~ ERROR cannot pass `self` by raw pointer LL | fn foo(*mut self) { } //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: cannot pass `self` by raw pointer error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:12:19 --> $DIR/no-unsafe-self.rs:12:19
| |
LL | fn baz(*const self) { } //~ ERROR cannot pass `self` by raw pointer LL | fn baz(*const self) { } //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: cannot pass `self` by raw pointer error: cannot pass `self` by raw pointer
--> $DIR/no-unsafe-self.rs:13:13 --> $DIR/no-unsafe-self.rs:13:13
| |
LL | fn bar(*self) { } //~ ERROR cannot pass `self` by raw pointer LL | fn bar(*self) { } //~ ERROR cannot pass `self` by raw pointer
| ^^^^ | ^^^^ cannot pass `self` by raw pointer
error: aborting due to 6 previous errors error: aborting due to 6 previous errors

View file

@ -1,5 +1,6 @@
fn main() { fn main() {
match 0 { match (0, 1, 2) {
(pat, ..,) => {} //~ ERROR trailing comma is not permitted after `..` (pat, ..,) => {}
//~^ ERROR trailing comma is not permitted after `..`
} }
} }

View file

@ -1,8 +1,8 @@
error: trailing comma is not permitted after `..` error: trailing comma is not permitted after `..`
--> $DIR/pat-tuple-2.rs:3:17 --> $DIR/pat-tuple-2.rs:3:17
| |
LL | (pat, ..,) => {} //~ ERROR trailing comma is not permitted after `..` LL | (pat, ..,) => {}
| ^ | ^ trailing comma is not permitted after `..`
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,5 +1,6 @@
fn main() { fn main() {
match 0 { match (0, 1, 2) {
(.., pat, ..) => {} //~ ERROR `..` can only be used once per tuple or tuple struct pattern (.., pat, ..) => {}
//~^ ERROR `..` can only be used once per tuple or tuple struct pattern
} }
} }

View file

@ -1,8 +1,8 @@
error: `..` can only be used once per tuple or tuple struct pattern error: `..` can only be used once per tuple or tuple struct pattern
--> $DIR/pat-tuple-3.rs:3:19 --> $DIR/pat-tuple-3.rs:3:19
| |
LL | (.., pat, ..) => {} //~ ERROR `..` can only be used once per tuple or tuple struct pattern LL | (.., pat, ..) => {}
| ^^ | ^^ can only be used once per pattern
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,12 +1,11 @@
//error-pattern: discriminator values can only be used with a field-less enum enum Color {
Red = 0xff0000,
enum color { //~^ ERROR discriminator values can only be used with a field-less enum
red = 0xff0000, Green = 0x00ff00,
green = 0x00ff00, Blue = 0x0000ff,
blue = 0x0000ff, Black = 0x000000,
black = 0x000000, White = 0xffffff,
white = 0xffffff, Other(usize),
other (str),
} }
fn main() {} fn main() {}

View file

@ -1,8 +1,17 @@
error: discriminator values can only be used with a field-less enum error: discriminator values can only be used with a field-less enum
--> $DIR/tag-variant-disr-non-nullary.rs:8:13 --> $DIR/tag-variant-disr-non-nullary.rs:2:11
| |
LL | white = 0xffffff, LL | Red = 0xff0000,
| ^^^^^^^^ | ^^^^^^^^ only valid in field-less enums
LL | //~^ ERROR discriminator values can only be used with a field-less enum
LL | Green = 0x00ff00,
| ^^^^^^^^ only valid in field-less enums
LL | Blue = 0x0000ff,
| ^^^^^^^^ only valid in field-less enums
LL | Black = 0x000000,
| ^^^^^^^^ only valid in field-less enums
LL | White = 0xffffff,
| ^^^^^^^^ only valid in field-less enums
error: aborting due to previous error error: aborting due to previous error

View file

@ -1,4 +1,5 @@
fn foo<T>() where <T>::Item: ToString, T: Iterator { } fn foo<T>() where <T>::Item: ToString, T: Iterator { }
//~^ ERROR generic parameters on `where` clauses are reserved for future use //~^ ERROR generic parameters on `where` clauses are reserved for future use
//~| ERROR cannot find type `Item` in the crate root
fn main() {} fn main() {}

View file

@ -2,7 +2,14 @@ error: generic parameters on `where` clauses are reserved for future use
--> $DIR/where_with_bound.rs:1:19 --> $DIR/where_with_bound.rs:1:19
| |
LL | fn foo<T>() where <T>::Item: ToString, T: Iterator { } LL | fn foo<T>() where <T>::Item: ToString, T: Iterator { }
| ^^^ | ^^^ currently unsupported
error: aborting due to previous error error[E0412]: cannot find type `Item` in the crate root
--> $DIR/where_with_bound.rs:1:24
|
LL | fn foo<T>() where <T>::Item: ToString, T: Iterator { }
| ^^^^ not found in the crate root
error: aborting due to 2 previous errors
For more information about this error, try `rustc --explain E0412`.

View file

@ -1,8 +1,6 @@
// A few contrived examples where lifetime should (or should not) be parsed as an object type. // A few contrived examples where lifetime should (or should not) be parsed as an object type.
// Lifetimes parsed as types are still rejected later by semantic checks. // Lifetimes parsed as types are still rejected later by semantic checks.
// compile-flags: -Z continue-parse-after-error
struct S<'a, T>(&'a u8, T); struct S<'a, T>(&'a u8, T);
fn main() { fn main() {

View file

@ -1,29 +1,33 @@
error: lifetime parameters must be declared prior to type parameters error: lifetime parameters must be declared prior to type parameters
--> $DIR/trait-object-vs-lifetime.rs:16:25 --> $DIR/trait-object-vs-lifetime.rs:14:25
| |
LL | let _: S<'static +, 'static>; LL | let _: S<'static +, 'static>;
| ^^^^^^^ | ^^^^^^^ must be declared prior to type parameters
help: move the lifetime parameter prior to the first type parameter
|
LL | let _: S<'static, 'static +>;
| ^^^^^^^^ --
error[E0224]: at least one non-builtin trait is required for an object type error[E0224]: at least one non-builtin trait is required for an object type
--> $DIR/trait-object-vs-lifetime.rs:11:23 --> $DIR/trait-object-vs-lifetime.rs:9:23
| |
LL | let _: S<'static, 'static +>; LL | let _: S<'static, 'static +>;
| ^^^^^^^^^ | ^^^^^^^^^
error[E0107]: wrong number of lifetime arguments: expected 1, found 2 error[E0107]: wrong number of lifetime arguments: expected 1, found 2
--> $DIR/trait-object-vs-lifetime.rs:13:23 --> $DIR/trait-object-vs-lifetime.rs:11:23
| |
LL | let _: S<'static, 'static>; LL | let _: S<'static, 'static>;
| ^^^^^^^ unexpected lifetime argument | ^^^^^^^ unexpected lifetime argument
error[E0107]: wrong number of type arguments: expected 1, found 0 error[E0107]: wrong number of type arguments: expected 1, found 0
--> $DIR/trait-object-vs-lifetime.rs:13:12 --> $DIR/trait-object-vs-lifetime.rs:11:12
| |
LL | let _: S<'static, 'static>; LL | let _: S<'static, 'static>;
| ^^^^^^^^^^^^^^^^^^^ expected 1 type argument | ^^^^^^^^^^^^^^^^^^^ expected 1 type argument
error[E0224]: at least one non-builtin trait is required for an object type error[E0224]: at least one non-builtin trait is required for an object type
--> $DIR/trait-object-vs-lifetime.rs:16:14 --> $DIR/trait-object-vs-lifetime.rs:14:14
| |
LL | let _: S<'static +, 'static>; LL | let _: S<'static +, 'static>;
| ^^^^^^^^^ | ^^^^^^^^^