Skip to content

Commit 48ddb5d

Browse files
Rollup merge of rust-lang#110133 - compiler-errors:issue-110131, r=petrochenkov
Do not use ImplDerivedObligationCause for inherent impl method error reporting We were constructing a `TraitRef` out of impl substs, for an *inherent* impl that has no corresponding trait. Instead of doing that, let's construct a meaningful obligation cause code, and instead adjust the error reporting machinery to handle that correctly. Fixes rust-lang#110131 cc rust-lang#106702, which introduced this regression
2 parents b912d4c + b369c8e commit 48ddb5d

File tree

4 files changed

+124
-35
lines changed

4 files changed

+124
-35
lines changed

compiler/rustc_hir_typeck/src/method/probe.rs

+12-17
Original file line numberDiff line numberDiff line change
@@ -1531,23 +1531,18 @@ impl<'a, 'tcx> ProbeContext<'a, 'tcx> {
15311531

15321532
// Convert the bounds into obligations.
15331533
let impl_obligations = traits::predicates_for_generics(
1534-
|_idx, span| {
1535-
let misc = traits::ObligationCause::misc(span, self.body_id);
1536-
let parent_trait_pred = ty::Binder::dummy(ty::TraitPredicate {
1537-
trait_ref: ty::TraitRef::from_method(self.tcx, impl_def_id, substs),
1538-
constness: ty::BoundConstness::NotConst,
1539-
polarity: ty::ImplPolarity::Positive,
1540-
});
1541-
misc.derived_cause(parent_trait_pred, |derived| {
1542-
traits::ImplDerivedObligation(Box::new(
1543-
traits::ImplDerivedObligationCause {
1544-
derived,
1545-
impl_or_alias_def_id: impl_def_id,
1546-
impl_def_predicate_index: None,
1547-
span,
1548-
},
1549-
))
1550-
})
1534+
|idx, span| {
1535+
let code = if span.is_dummy() {
1536+
traits::ExprItemObligation(impl_def_id, self.scope_expr_id, idx)
1537+
} else {
1538+
traits::ExprBindingObligation(
1539+
impl_def_id,
1540+
span,
1541+
self.scope_expr_id,
1542+
idx,
1543+
)
1544+
};
1545+
ObligationCause::new(self.span, self.body_id, code)
15511546
},
15521547
self.param_env,
15531548
impl_bounds,

compiler/rustc_hir_typeck/src/method/suggest.rs

+25-18
Original file line numberDiff line numberDiff line change
@@ -661,19 +661,26 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
661661
// Find all the requirements that come from a local `impl` block.
662662
let mut skip_list: FxHashSet<_> = Default::default();
663663
let mut spanned_predicates = FxHashMap::default();
664-
for (p, parent_p, impl_def_id, cause) in unsatisfied_predicates
665-
.iter()
666-
.filter_map(|(p, parent, c)| c.as_ref().map(|c| (p, parent, c)))
667-
.filter_map(|(p, parent, c)| match c.code() {
668-
ObligationCauseCode::ImplDerivedObligation(data)
669-
if matches!(p.kind().skip_binder(), ty::PredicateKind::Clause(_)) =>
670-
{
671-
Some((p, parent, data.impl_or_alias_def_id, data))
664+
for (p, parent_p, cause) in unsatisfied_predicates {
665+
// Extract the predicate span and parent def id of the cause,
666+
// if we have one.
667+
let (item_def_id, cause_span) = match cause.as_ref().map(|cause| cause.code()) {
668+
Some(ObligationCauseCode::ImplDerivedObligation(data)) => {
669+
(data.impl_or_alias_def_id, data.span)
672670
}
673-
_ => None,
674-
})
675-
{
676-
match self.tcx.hir().get_if_local(impl_def_id) {
671+
Some(
672+
ObligationCauseCode::ExprBindingObligation(def_id, span, _, _)
673+
| ObligationCauseCode::BindingObligation(def_id, span),
674+
) => (*def_id, *span),
675+
_ => continue,
676+
};
677+
678+
// Don't point out the span of `WellFormed` predicates.
679+
if !matches!(p.kind().skip_binder(), ty::PredicateKind::Clause(_)) {
680+
continue;
681+
};
682+
683+
match self.tcx.hir().get_if_local(item_def_id) {
677684
// Unmet obligation comes from a `derive` macro, point at it once to
678685
// avoid multiple span labels pointing at the same place.
679686
Some(Node::Item(hir::Item {
@@ -718,7 +725,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
718725
}
719726
});
720727
for param in generics.params {
721-
if param.span == cause.span && sized_pred {
728+
if param.span == cause_span && sized_pred {
722729
let (sp, sugg) = match param.colon_span {
723730
Some(sp) => (sp.shrink_to_hi(), " ?Sized +"),
724731
None => (param.span.shrink_to_hi(), ": ?Sized"),
@@ -741,9 +748,9 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
741748
(FxHashSet::default(), FxHashSet::default(), Vec::new())
742749
});
743750
entry.2.push(p);
744-
if cause.span != *item_span {
745-
entry.0.insert(cause.span);
746-
entry.1.insert((cause.span, "unsatisfied trait bound introduced here"));
751+
if cause_span != *item_span {
752+
entry.0.insert(cause_span);
753+
entry.1.insert((cause_span, "unsatisfied trait bound introduced here"));
747754
} else {
748755
if let Some(trait_ref) = of_trait {
749756
entry.0.insert(trait_ref.path.span);
@@ -775,9 +782,9 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
775782
let entry = entry.or_insert_with(|| {
776783
(FxHashSet::default(), FxHashSet::default(), Vec::new())
777784
});
778-
entry.0.insert(cause.span);
785+
entry.0.insert(cause_span);
779786
entry.1.insert((ident.span, ""));
780-
entry.1.insert((cause.span, "unsatisfied trait bound introduced here"));
787+
entry.1.insert((cause_span, "unsatisfied trait bound introduced here"));
781788
entry.2.push(p);
782789
}
783790
Some(node) => unreachable!("encountered `{node:?}`"),
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
// normalize-stderr-test: "long-type-\d+" -> "long-type-hash"
2+
3+
// Fixes #110131
4+
//
5+
// The issue is that we were constructing an `ImplDerived` cause code for the
6+
// `&'a T: IntoIterator<Item = &'a u8>` obligation for `Helper::new`, which is
7+
// incorrect because derived obligations are only expected to come from *traits*.
8+
9+
struct SeqBuffer<'a, T>
10+
where
11+
&'a T: IntoIterator<Item = &'a u8>,
12+
{
13+
iter: <&'a T as IntoIterator>::IntoIter,
14+
}
15+
16+
struct Helper<'a, T>
17+
where
18+
&'a T: IntoIterator<Item = &'a u8>,
19+
{
20+
buf: SeqBuffer<'a, T>,
21+
}
22+
23+
impl<'a, T> Helper<'a, T>
24+
where
25+
&'a T: IntoIterator<Item = &'a u8>,
26+
{
27+
fn new(sq: &'a T) -> Self {
28+
loop {}
29+
}
30+
}
31+
32+
struct BitReaderWrapper<T>(T);
33+
34+
impl<'a, T> IntoIterator for &'a BitReaderWrapper<T>
35+
where
36+
&'a T: IntoIterator<Item = &'a u8>,
37+
{
38+
type Item = u32;
39+
40+
type IntoIter = Helper<'a, T>;
41+
//~^ ERROR `Helper<'a, T>` is not an iterator
42+
43+
fn into_iter(self) -> Self::IntoIter {
44+
Helper::new(&self.0)
45+
//~^ ERROR overflow evaluating the requirement `&_: IntoIterator`
46+
}
47+
}
48+
49+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
error[E0277]: `Helper<'a, T>` is not an iterator
2+
--> $DIR/inherent-bound-in-probe.rs:40:21
3+
|
4+
LL | type IntoIter = Helper<'a, T>;
5+
| ^^^^^^^^^^^^^ `Helper<'a, T>` is not an iterator
6+
|
7+
= help: the trait `Iterator` is not implemented for `Helper<'a, T>`
8+
note: required by a bound in `std::iter::IntoIterator::IntoIter`
9+
--> $SRC_DIR/core/src/iter/traits/collect.rs:LL:COL
10+
11+
error[E0275]: overflow evaluating the requirement `&_: IntoIterator`
12+
--> $DIR/inherent-bound-in-probe.rs:44:17
13+
|
14+
LL | Helper::new(&self.0)
15+
| ^^^
16+
|
17+
= help: consider increasing the recursion limit by adding a `#![recursion_limit = "256"]` attribute to your crate (`inherent_bound_in_probe`)
18+
note: required for `&BitReaderWrapper<_>` to implement `IntoIterator`
19+
--> $DIR/inherent-bound-in-probe.rs:34:13
20+
|
21+
LL | impl<'a, T> IntoIterator for &'a BitReaderWrapper<T>
22+
| ^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^
23+
LL | where
24+
LL | &'a T: IntoIterator<Item = &'a u8>,
25+
| ------------- unsatisfied trait bound introduced here
26+
= note: 126 redundant requirements hidden
27+
= note: required for `&BitReaderWrapper<BitReaderWrapper<BitReaderWrapper<BitReaderWrapper<BitReaderWrapper<...>>>>>` to implement `IntoIterator`
28+
= note: the full type name has been written to '$TEST_BUILD_DIR/methods/inherent-bound-in-probe/inherent-bound-in-probe.long-type-hash.txt'
29+
note: required by a bound in `Helper<'a, T>`
30+
--> $DIR/inherent-bound-in-probe.rs:25:25
31+
|
32+
LL | &'a T: IntoIterator<Item = &'a u8>,
33+
| ^^^^^^^^^^^^^ required by this bound in `Helper<'a, T>`
34+
35+
error: aborting due to 2 previous errors
36+
37+
Some errors have detailed explanations: E0275, E0277.
38+
For more information about an error, try `rustc --explain E0275`.

0 commit comments

Comments
 (0)