Skip to content

Commit cea82ed

Browse files
committed
Don't suggest .into_iter() on iterators
1 parent 59cec72 commit cea82ed

File tree

3 files changed

+56
-8
lines changed

3 files changed

+56
-8
lines changed

compiler/rustc_hir_typeck/src/method/suggest.rs

+10-8
Original file line numberDiff line numberDiff line change
@@ -100,9 +100,9 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
100100
ty: Ty<'tcx>,
101101
span: Span,
102102
unsatisfied_predicates: &Vec<(
103-
ty::Predicate<'_>,
104-
Option<ty::Predicate<'_>>,
105-
Option<ObligationCause<'_>>,
103+
ty::Predicate<'tcx>,
104+
Option<ty::Predicate<'tcx>>,
105+
Option<ObligationCause<'tcx>>,
106106
)>,
107107
) -> bool {
108108
fn predicate_bounds_generic_param<'tcx>(
@@ -131,15 +131,17 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
131131
}
132132
}
133133

134-
fn is_iterator_predicate(predicate: ty::Predicate<'_>, tcx: TyCtxt<'_>) -> bool {
134+
let is_iterator_predicate = |predicate: ty::Predicate<'tcx>| -> bool {
135135
if let ty::PredicateKind::Clause(ty::ClauseKind::Trait(trait_pred)) =
136136
predicate.kind().as_ref().skip_binder()
137137
{
138-
tcx.is_diagnostic_item(sym::Iterator, trait_pred.trait_ref.def_id)
138+
self.tcx.is_diagnostic_item(sym::Iterator, trait_pred.trait_ref.def_id)
139+
// ignore unsatisfied predicates generated from trying to auto-ref ty (#127511)
140+
&& trait_pred.trait_ref.self_ty() == ty
139141
} else {
140142
false
141143
}
142-
}
144+
};
143145

144146
// Does the `ty` implement `IntoIterator`?
145147
let Some(into_iterator_trait) = self.tcx.get_diagnostic_item(sym::IntoIterator) else {
@@ -164,15 +166,15 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
164166
generics,
165167
generic_param,
166168
self.tcx,
167-
) && is_iterator_predicate(unsatisfied.0, self.tcx)
169+
) && is_iterator_predicate(unsatisfied.0)
168170
{
169171
return true;
170172
}
171173
}
172174
}
173175
ty::Slice(..) | ty::Adt(..) | ty::Alias(ty::Opaque, _) => {
174176
for unsatisfied in unsatisfied_predicates.iter() {
175-
if is_iterator_predicate(unsatisfied.0, self.tcx) {
177+
if is_iterator_predicate(unsatisfied.0) {
176178
return true;
177179
}
178180
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
//! regression test for #127511: don't suggest `.into_iter()` on iterators
2+
3+
trait Missing {}
4+
trait HasMethod {
5+
fn foo(self);
6+
}
7+
impl<T: Iterator + Missing> HasMethod for T {
8+
fn foo(self) {}
9+
}
10+
11+
fn get_iter() -> impl Iterator {
12+
core::iter::once(())
13+
}
14+
15+
fn main() {
16+
get_iter().foo();
17+
//~^ ERROR the method `foo` exists for opaque type `impl Iterator`, but its trait bounds were not satisfied [E0599]
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
error[E0599]: the method `foo` exists for opaque type `impl Iterator`, but its trait bounds were not satisfied
2+
--> $DIR/iterator-does-not-need-into-iter.rs:16:16
3+
|
4+
LL | get_iter().foo();
5+
| ^^^ method cannot be called on `impl Iterator` due to unsatisfied trait bounds
6+
|
7+
note: the following trait bounds were not satisfied:
8+
`&impl Iterator: Iterator`
9+
`&impl Iterator: Missing`
10+
`&mut impl Iterator: Missing`
11+
`impl Iterator: Missing`
12+
--> $DIR/iterator-does-not-need-into-iter.rs:7:9
13+
|
14+
LL | impl<T: Iterator + Missing> HasMethod for T {
15+
| ^^^^^^^^ ^^^^^^^ --------- -
16+
| | |
17+
| | unsatisfied trait bound introduced here
18+
| unsatisfied trait bound introduced here
19+
= help: items from traits can only be used if the trait is implemented and in scope
20+
note: `HasMethod` defines an item `foo`, perhaps you need to implement it
21+
--> $DIR/iterator-does-not-need-into-iter.rs:4:1
22+
|
23+
LL | trait HasMethod {
24+
| ^^^^^^^^^^^^^^^
25+
26+
error: aborting due to 1 previous error
27+
28+
For more information about this error, try `rustc --explain E0599`.

0 commit comments

Comments
 (0)