Skip to content

Commit f9f6709

Browse files
authored
Unrolled build for rust-lang#121912
Rollup merge of rust-lang#121912 - fmease:diag-method-chains-gat, r=compiler-errors,estebank Properly deal with GATs when looking for method chains to point at Fixes rust-lang#121898. ~~While it prevents an ICE and the structured suggestion is correct, the method chain diagnostic notes are weird / useless / incorrect judging by a quick look. I guess I should improve that in this PR.~~ Sufficiently taken care of. r? estebank or compiler-errors (rust-lang#105332, rust-lang#105674).
2 parents f7cb53e + 6035e87 commit f9f6709

File tree

4 files changed

+63
-16
lines changed

4 files changed

+63
-16
lines changed

compiler/rustc_trait_selection/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
#![allow(internal_features)]
1717
#![allow(rustc::diagnostic_outside_of_impl)]
1818
#![allow(rustc::untranslatable_diagnostic)]
19+
#![feature(assert_matches)]
1920
#![feature(associated_type_bounds)]
2021
#![feature(box_patterns)]
2122
#![feature(control_flow_enum)]

compiler/rustc_trait_selection/src/traits/error_reporting/suggestions.rs

+13-16
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ use rustc_span::def_id::LocalDefId;
3838
use rustc_span::symbol::{kw, sym, Ident, Symbol};
3939
use rustc_span::{BytePos, DesugaringKind, ExpnKind, MacroKind, Span, DUMMY_SP};
4040
use rustc_target::spec::abi;
41+
use std::assert_matches::debug_assert_matches;
4142
use std::borrow::Cow;
4243
use std::iter;
4344

@@ -4219,30 +4220,25 @@ impl<'tcx> TypeErrCtxt<'_, 'tcx> {
42194220
};
42204221

42214222
let origin = TypeVariableOrigin { kind: TypeVariableOriginKind::TypeInference, span };
4222-
let trait_def_id = proj.trait_def_id(self.tcx);
42234223
// Make `Self` be equivalent to the type of the call chain
42244224
// expression we're looking at now, so that we can tell what
42254225
// for example `Iterator::Item` is at this point in the chain.
4226-
let args = GenericArgs::for_item(self.tcx, trait_def_id, |param, _| {
4227-
match param.kind {
4228-
ty::GenericParamDefKind::Type { .. } => {
4229-
if param.index == 0 {
4230-
return prev_ty.into();
4231-
}
4232-
}
4233-
ty::GenericParamDefKind::Lifetime | ty::GenericParamDefKind::Const { .. } => {}
4226+
let args = GenericArgs::for_item(self.tcx, proj.def_id, |param, _| {
4227+
if param.index == 0 {
4228+
debug_assert_matches!(param.kind, ty::GenericParamDefKind::Type { .. });
4229+
return prev_ty.into();
42344230
}
42354231
self.var_for_def(span, param)
42364232
});
42374233
// This will hold the resolved type of the associated type, if the
42384234
// current expression implements the trait that associated type is
42394235
// in. For example, this would be what `Iterator::Item` is here.
4240-
let ty_var = self.infcx.next_ty_var(origin);
4236+
let ty = self.infcx.next_ty_var(origin);
42414237
// This corresponds to `<ExprTy as Iterator>::Item = _`.
42424238
let projection = ty::Binder::dummy(ty::PredicateKind::Clause(
42434239
ty::ClauseKind::Projection(ty::ProjectionPredicate {
42444240
projection_ty: ty::AliasTy::new(self.tcx, proj.def_id, args),
4245-
term: ty_var.into(),
4241+
term: ty.into(),
42464242
}),
42474243
));
42484244
let body_def_id = self.tcx.hir().enclosing_body_owner(body_id);
@@ -4254,14 +4250,15 @@ impl<'tcx> TypeErrCtxt<'_, 'tcx> {
42544250
param_env,
42554251
projection,
42564252
));
4257-
if ocx.select_where_possible().is_empty() {
4258-
// `ty_var` now holds the type that `Item` is for `ExprTy`.
4259-
let ty_var = self.resolve_vars_if_possible(ty_var);
4260-
assocs_in_this_method.push(Some((span, (proj.def_id, ty_var))));
4253+
if ocx.select_where_possible().is_empty()
4254+
&& let ty = self.resolve_vars_if_possible(ty)
4255+
&& !ty.is_ty_var()
4256+
{
4257+
assocs_in_this_method.push(Some((span, (proj.def_id, ty))));
42614258
} else {
42624259
// `<ExprTy as Iterator>` didn't select, so likely we've
42634260
// reached the end of the iterator chain, like the originating
4264-
// `Vec<_>`.
4261+
// `Vec<_>` or the `ty` couldn't be determined.
42654262
// Keep the space consistent for later zipping.
42664263
assocs_in_this_method.push(None);
42674264
}

tests/ui/typeck/method-chain-gats.rs

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
// Regression test for issue #121898.
2+
3+
trait Base {
4+
type Base<B>;
5+
}
6+
7+
trait Functor<A>: Base {
8+
fn fmap<B>(self, f: impl Fn(A) -> B) -> Self::Base<B>
9+
where
10+
Self::Base<B>: Functor<B>;
11+
}
12+
13+
fn fmap2<T, A, B, C>(input: T, f1: impl Fn(A) -> B, f2: impl Fn(B) -> C) -> T::Base<C>
14+
where
15+
T: Functor<A>,
16+
T::Base<B>: Functor<B, Base<C> = T::Base<C>>,
17+
{
18+
input.fmap(f1).fmap(f2)
19+
//~^ ERROR the trait bound `<T as Base>::Base<C>: Functor<C>` is not satisfied
20+
}
21+
22+
fn main() {}
+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
error[E0277]: the trait bound `<T as Base>::Base<C>: Functor<C>` is not satisfied
2+
--> $DIR/method-chain-gats.rs:18:20
3+
|
4+
LL | input.fmap(f1).fmap(f2)
5+
| ^^^^ the trait `Functor<C>` is not implemented for `<T as Base>::Base<C>`
6+
|
7+
note: the method call chain might not have had the expected associated types
8+
--> $DIR/method-chain-gats.rs:13:29
9+
|
10+
LL | fn fmap2<T, A, B, C>(input: T, f1: impl Fn(A) -> B, f2: impl Fn(B) -> C) -> T::Base<C>
11+
| ^ `Base::Base` is `<T as Base>::Base<_>` here
12+
note: required by a bound in `Functor::fmap`
13+
--> $DIR/method-chain-gats.rs:10:24
14+
|
15+
LL | fn fmap<B>(self, f: impl Fn(A) -> B) -> Self::Base<B>
16+
| ---- required by a bound in this associated function
17+
LL | where
18+
LL | Self::Base<B>: Functor<B>;
19+
| ^^^^^^^^^^ required by this bound in `Functor::fmap`
20+
help: consider further restricting the associated type
21+
|
22+
LL | T::Base<B>: Functor<B, Base<C> = T::Base<C>>, <T as Base>::Base<C>: Functor<C>
23+
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
24+
25+
error: aborting due to 1 previous error
26+
27+
For more information about this error, try `rustc --explain E0277`.

0 commit comments

Comments
 (0)