Skip to content

Commit 13bff3f

Browse files
Improve error message
1 parent f9e007d commit 13bff3f

File tree

8 files changed

+66
-11
lines changed

8 files changed

+66
-11
lines changed

compiler/rustc_parse/messages.ftl

+3
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,9 @@ parse_bare_cr = {$double_quotes ->
6161
6262
parse_bare_cr_in_raw_string = bare CR not allowed in raw string
6363
64+
parse_binder_before_modifiers = `for<...>` binder should be placed before trait bound modifiers
65+
.label = place the `for<...>` binder before any modifiers
66+
6467
parse_bounds_not_allowed_on_trait_aliases = bounds are not allowed on trait aliases
6568
6669
parse_box_not_pat = expected pattern, found {$descr}

compiler/rustc_parse/src/errors.rs

+9
Original file line numberDiff line numberDiff line change
@@ -3028,3 +3028,12 @@ pub struct UnsafeAttrOutsideUnsafeSuggestion {
30283028
#[suggestion_part(code = ")")]
30293029
pub right: Span,
30303030
}
3031+
3032+
#[derive(Diagnostic)]
3033+
#[diag(parse_binder_before_modifiers)]
3034+
pub struct BinderBeforeModifiers {
3035+
#[primary_span]
3036+
pub binder_span: Span,
3037+
#[label]
3038+
pub modifiers_span: Span,
3039+
}

compiler/rustc_parse/src/parser/ty.rs

+8
Original file line numberDiff line numberDiff line change
@@ -989,7 +989,10 @@ impl<'a> Parser<'a> {
989989
leading_token: &Token,
990990
) -> PResult<'a, GenericBound> {
991991
let (mut lifetime_defs, binder_span) = self.parse_late_bound_lifetime_defs()?;
992+
993+
let modifiers_lo = self.token.span;
992994
let modifiers = self.parse_trait_bound_modifiers()?;
995+
let modifiers_span = modifiers_lo.to(self.prev_token.span);
993996

994997
// Recover erroneous lifetime bound with modifiers or binder.
995998
// e.g. `T: for<'a> 'a` or `T: ~const 'a`.
@@ -998,6 +1001,11 @@ impl<'a> Parser<'a> {
9981001
return self.parse_generic_lt_bound(lo, has_parens);
9991002
}
10001003

1004+
if let (more_lifetime_defs, Some(binder_span)) = self.parse_late_bound_lifetime_defs()? {
1005+
lifetime_defs.extend(more_lifetime_defs);
1006+
self.dcx().emit_err(errors::BinderBeforeModifiers { binder_span, modifiers_span });
1007+
}
1008+
10011009
let mut path = if self.token.is_keyword(kw::Fn)
10021010
&& self.look_ahead(1, |tok| tok.kind == TokenKind::OpenDelim(Delimiter::Parenthesis))
10031011
&& let Some(path) = self.recover_path_from_fn()

src/tools/tidy/src/ui_tests.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ use std::path::{Path, PathBuf};
1616
const ENTRY_LIMIT: u32 = 900;
1717
// FIXME: The following limits should be reduced eventually.
1818

19-
const ISSUES_ENTRY_LIMIT: u32 = 1672;
19+
const ISSUES_ENTRY_LIMIT: u32 = 1673;
2020

2121
const EXPECTED_TEST_FILE_EXTENSIONS: &[&str] = &[
2222
"rs", // test source files

tests/ui/impl-trait/normalize-tait-in-const.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ mod foo {
2424
}
2525
use foo::*;
2626

27-
const fn with_positive<F: ~const for<'a> Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
27+
const fn with_positive<F: for<'a> ~const Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
2828
fun(filter_positive());
2929
}
3030

Original file line numberDiff line numberDiff line change
@@ -1,8 +1,41 @@
1-
error: expected a trait, found type
2-
--> $DIR/normalize-tait-in-const.rs:27:34
1+
error: `~const` can only be applied to `#[const_trait]` traits
2+
--> $DIR/normalize-tait-in-const.rs:27:42
33
|
4-
LL | const fn with_positive<F: ~const for<'a> Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
5-
| ^^^^^^^^^^^^^^^^^^^^^^^^^
4+
LL | const fn with_positive<F: for<'a> ~const Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
5+
| ^^^^^^^^^^^^^^^^^
66

7-
error: aborting due to 1 previous error
7+
error: `~const` can only be applied to `#[const_trait]` traits
8+
--> $DIR/normalize-tait-in-const.rs:27:69
9+
|
10+
LL | const fn with_positive<F: for<'a> ~const Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
11+
| ^^^^^^^^
12+
13+
error[E0015]: cannot call non-const closure in constant functions
14+
--> $DIR/normalize-tait-in-const.rs:28:5
15+
|
16+
LL | fun(filter_positive());
17+
| ^^^^^^^^^^^^^^^^^^^^^^
18+
|
19+
= note: calls in constant functions are limited to constant functions, tuple structs and tuple variants
20+
help: consider further restricting this bound
21+
|
22+
LL | const fn with_positive<F: for<'a> ~const Fn(&'a Alias<'a>) + ~const Destruct + ~const Fn(&foo::Alias<'_>)>(fun: F) {
23+
| ++++++++++++++++++++++++++++
24+
help: add `#![feature(effects)]` to the crate attributes to enable
25+
|
26+
LL + #![feature(effects)]
27+
|
28+
29+
error[E0493]: destructor of `F` cannot be evaluated at compile-time
30+
--> $DIR/normalize-tait-in-const.rs:27:79
31+
|
32+
LL | const fn with_positive<F: for<'a> ~const Fn(&'a Alias<'a>) + ~const Destruct>(fun: F) {
33+
| ^^^ the destructor for this type cannot be evaluated in constant functions
34+
LL | fun(filter_positive());
35+
LL | }
36+
| - value is dropped here
37+
38+
error: aborting due to 4 previous errors
839

40+
Some errors have detailed explanations: E0015, E0493.
41+
For more information about an error, try `rustc --explain E0015`.

tests/ui/issues/issue-39089.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
fn f<T: ?for<'a> Sized>() {}
2-
//~^ ERROR expected a trait, found type
2+
//~^ ERROR `for<...>` binder should be placed before trait bound modifiers
33

44
fn main() {}

tests/ui/issues/issue-39089.stderr

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
1-
error: expected a trait, found type
2-
--> $DIR/issue-39089.rs:1:10
1+
error: `for<...>` binder should be placed before trait bound modifiers
2+
--> $DIR/issue-39089.rs:1:13
33
|
44
LL | fn f<T: ?for<'a> Sized>() {}
5-
| ^^^^^^^^^^^^^
5+
| - ^^^^
6+
| |
7+
| place the `for<...>` binder before any modifiers
68

79
error: aborting due to 1 previous error
810

0 commit comments

Comments
 (0)