Skip to content

Commit 079b290

Browse files
committed
Do not suggest 'Trait<Assoc=arg>' when in trait impl
because that would be illegal syntax
1 parent 37fda98 commit 079b290

File tree

3 files changed

+210
-14
lines changed

3 files changed

+210
-14
lines changed

compiler/rustc_hir_analysis/src/structured_errors/wrong_number_of_generic_args.rs

+58-14
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,44 @@ impl<'a, 'tcx> WrongNumberOfGenericArgs<'a, 'tcx> {
129129
if self.missing_lifetimes() { "lifetime" } else { "generic" }
130130
}
131131

132+
/// Returns true if the generic type is a trait
133+
/// and is being referred to from one of its trait impls
134+
fn is_in_trait_impl(&self) -> bool {
135+
if self.tcx.is_trait(self.def_id) {
136+
// Here we check if the reference to the generic type
137+
// is from the 'of_trait' field of the enclosing impl
138+
139+
let parent = self.tcx.hir().get_parent(self.path_segment.hir_id);
140+
let parent_item = self
141+
.tcx
142+
.hir()
143+
.get_by_def_id(self.tcx.hir().get_parent_item(self.path_segment.hir_id).def_id);
144+
145+
// Get the HIR id of the trait ref
146+
let hir::Node::TraitRef(hir::TraitRef { hir_ref_id: trait_ref_id, .. }) = parent else {
147+
return false;
148+
};
149+
150+
// Get the HIR id of the 'of_trait' field of the impl
151+
let hir::Node::Item(hir::Item {
152+
kind:
153+
hir::ItemKind::Impl(hir::Impl {
154+
of_trait: Some(hir::TraitRef { hir_ref_id: id_in_of_trait, .. }),
155+
..
156+
}),
157+
..
158+
}) = parent_item
159+
else {
160+
return false;
161+
};
162+
163+
// Check that trait is referred to from the of_trait field of impl
164+
trait_ref_id == id_in_of_trait
165+
} else {
166+
false
167+
}
168+
}
169+
132170
fn num_provided_args(&self) -> usize {
133171
if self.missing_lifetimes() {
134172
self.num_provided_lifetime_args()
@@ -948,20 +986,26 @@ impl<'a, 'tcx> WrongNumberOfGenericArgs<'a, 'tcx> {
948986
// If there is a single unbound associated type and a single excess generic param
949987
// suggest replacing the generic param with the associated type bound
950988
if provided_args_matches_unbound_traits && !unbound_types.is_empty() {
951-
let unused_generics = &self.gen_args.args[self.num_expected_type_or_const_args()..];
952-
let suggestions = iter::zip(unused_generics, &unbound_types)
953-
.map(|(potential, name)| (potential.span().shrink_to_lo(), format!("{name} = ")))
954-
.collect::<Vec<_>>();
955-
956-
if !suggestions.is_empty() {
957-
err.multipart_suggestion_verbose(
958-
format!(
959-
"replace the generic bound{s} with the associated type{s}",
960-
s = pluralize!(unbound_types.len())
961-
),
962-
suggestions,
963-
Applicability::MaybeIncorrect,
964-
);
989+
// Don't suggest if we're in a trait impl as
990+
// that would result in invalid syntax (fixes #116464)
991+
if !self.is_in_trait_impl() {
992+
let unused_generics = &self.gen_args.args[self.num_expected_type_or_const_args()..];
993+
let suggestions = iter::zip(unused_generics, &unbound_types)
994+
.map(|(potential, name)| {
995+
(potential.span().shrink_to_lo(), format!("{name} = "))
996+
})
997+
.collect::<Vec<_>>();
998+
999+
if !suggestions.is_empty() {
1000+
err.multipart_suggestion_verbose(
1001+
format!(
1002+
"replace the generic bound{s} with the associated type{s}",
1003+
s = pluralize!(unbound_types.len())
1004+
),
1005+
suggestions,
1006+
Applicability::MaybeIncorrect,
1007+
);
1008+
}
9651009
}
9661010
} else if remove_entire_generics {
9671011
let span = self
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
// Regression test for #116464
2+
// Checks that we do not suggest Trait<..., Assoc=arg> when the trait
3+
// is referred to from one of its impls but do so at all other places
4+
5+
pub trait Trait<T> {
6+
type Assoc;
7+
}
8+
9+
impl<T, S> Trait<T> for i32 {
10+
type Assoc = String;
11+
}
12+
13+
// Should not not trigger suggestion here...
14+
impl<T, S> Trait<T, S> for () {}
15+
//~^ ERROR trait takes 1 generic argument but 2 generic arguments were supplied
16+
17+
//... but should do so in all of the below cases except the last one
18+
fn func<T: Trait<u32, String>>(t: T) -> impl Trait<(), i32> {
19+
//~^ ERROR trait takes 1 generic argument but 2 generic arguments were supplied
20+
//~| ERROR trait takes 1 generic argument but 2 generic arguments were supplied
21+
3
22+
}
23+
24+
struct Struct<T: Trait<u32, String>> {
25+
//~^ ERROR trait takes 1 generic argument but 2 generic arguments were supplied
26+
a: T
27+
}
28+
29+
trait AnotherTrait<T: Trait<T, i32>> {}
30+
//~^ ERROR trait takes 1 generic argument but 2 generic arguments were supplied
31+
32+
impl<T: Trait<u32, String>> Struct<T> {}
33+
//~^ ERROR trait takes 1 generic argument but 2 generic arguments were supplied
34+
35+
// Test for self type. Should not trigger suggestion as it doesn't have an
36+
// associated type
37+
trait YetAnotherTrait {}
38+
impl<T: Trait<u32, Assoc=String>, U> YetAnotherTrait for Struct<T, U> {}
39+
//~^ ERROR struct takes 1 generic argument but 2 generic arguments were supplied
40+
41+
42+
fn main() {
43+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,109 @@
1+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
2+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:14:12
3+
|
4+
LL | impl<T, S> Trait<T, S> for () {}
5+
| ^^^^^ expected 1 generic argument
6+
|
7+
note: trait defined here, with 1 generic parameter: `T`
8+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
9+
|
10+
LL | pub trait Trait<T> {
11+
| ^^^^^ -
12+
13+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
14+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:18:12
15+
|
16+
LL | fn func<T: Trait<u32, String>>(t: T) -> impl Trait<(), i32> {
17+
| ^^^^^ expected 1 generic argument
18+
|
19+
note: trait defined here, with 1 generic parameter: `T`
20+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
21+
|
22+
LL | pub trait Trait<T> {
23+
| ^^^^^ -
24+
help: replace the generic bound with the associated type
25+
|
26+
LL | fn func<T: Trait<u32, Assoc = String>>(t: T) -> impl Trait<(), i32> {
27+
| +++++++
28+
29+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
30+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:18:46
31+
|
32+
LL | fn func<T: Trait<u32, String>>(t: T) -> impl Trait<(), i32> {
33+
| ^^^^^ expected 1 generic argument
34+
|
35+
note: trait defined here, with 1 generic parameter: `T`
36+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
37+
|
38+
LL | pub trait Trait<T> {
39+
| ^^^^^ -
40+
help: replace the generic bound with the associated type
41+
|
42+
LL | fn func<T: Trait<u32, String>>(t: T) -> impl Trait<(), Assoc = i32> {
43+
| +++++++
44+
45+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
46+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:24:18
47+
|
48+
LL | struct Struct<T: Trait<u32, String>> {
49+
| ^^^^^ expected 1 generic argument
50+
|
51+
note: trait defined here, with 1 generic parameter: `T`
52+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
53+
|
54+
LL | pub trait Trait<T> {
55+
| ^^^^^ -
56+
help: replace the generic bound with the associated type
57+
|
58+
LL | struct Struct<T: Trait<u32, Assoc = String>> {
59+
| +++++++
60+
61+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
62+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:29:23
63+
|
64+
LL | trait AnotherTrait<T: Trait<T, i32>> {}
65+
| ^^^^^ expected 1 generic argument
66+
|
67+
note: trait defined here, with 1 generic parameter: `T`
68+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
69+
|
70+
LL | pub trait Trait<T> {
71+
| ^^^^^ -
72+
help: replace the generic bound with the associated type
73+
|
74+
LL | trait AnotherTrait<T: Trait<T, Assoc = i32>> {}
75+
| +++++++
76+
77+
error[E0107]: trait takes 1 generic argument but 2 generic arguments were supplied
78+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:32:9
79+
|
80+
LL | impl<T: Trait<u32, String>> Struct<T> {}
81+
| ^^^^^ expected 1 generic argument
82+
|
83+
note: trait defined here, with 1 generic parameter: `T`
84+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:5:11
85+
|
86+
LL | pub trait Trait<T> {
87+
| ^^^^^ -
88+
help: replace the generic bound with the associated type
89+
|
90+
LL | impl<T: Trait<u32, Assoc = String>> Struct<T> {}
91+
| +++++++
92+
93+
error[E0107]: struct takes 1 generic argument but 2 generic arguments were supplied
94+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:38:58
95+
|
96+
LL | impl<T: Trait<u32, Assoc=String>, U> YetAnotherTrait for Struct<T, U> {}
97+
| ^^^^^^ - help: remove this generic argument
98+
| |
99+
| expected 1 generic argument
100+
|
101+
note: struct defined here, with 1 generic parameter: `T`
102+
--> $DIR/116464-invalid-assoc-type-suggestion-in-trait-impl.rs:24:8
103+
|
104+
LL | struct Struct<T: Trait<u32, String>> {
105+
| ^^^^^^ -
106+
107+
error: aborting due to 7 previous errors
108+
109+
For more information about this error, try `rustc --explain E0107`.

0 commit comments

Comments
 (0)