From 80a4e1b5e35a2f8ebb1392098ba28e24e50eace6 Mon Sep 17 00:00:00 2001 From: Michael Goulet Date: Sun, 12 Mar 2023 00:12:52 +0000 Subject: [PATCH] Do not emit alias-eq when instantiating type var --- compiler/rustc_infer/src/infer/combine.rs | 22 +++++++--- compiler/rustc_infer/src/infer/equate.rs | 7 ++++ compiler/rustc_infer/src/infer/mod.rs | 1 + compiler/rustc_infer/src/infer/sub.rs | 8 ++++ .../generalize-shouldnt-emit-alias-eq.rs | 40 +++++++++++++++++++ 5 files changed, 73 insertions(+), 5 deletions(-) create mode 100644 tests/ui/traits/new-solver/generalize-shouldnt-emit-alias-eq.rs diff --git a/compiler/rustc_infer/src/infer/combine.rs b/compiler/rustc_infer/src/infer/combine.rs index a2332797e8680..987775685099b 100644 --- a/compiler/rustc_infer/src/infer/combine.rs +++ b/compiler/rustc_infer/src/infer/combine.rs @@ -58,6 +58,10 @@ pub struct CombineFields<'infcx, 'tcx> { /// matching from matching anything against opaque /// types. pub define_opaque_types: bool, + /// Used in [`CombineFields::instantiate`] to ensure that when we relate + /// a type and its generalized type, we never emit an alias-eq goal, but + /// instead relate via substs. + pub relate_projections_via_substs: bool, } #[derive(Copy, Clone, Debug)] @@ -420,16 +424,16 @@ impl<'infcx, 'tcx> CombineFields<'infcx, 'tcx> { // relations wind up attributed to the same spans. We need // to associate causes/spans with each of the relations in // the stack to get this right. - match dir { - EqTo => self.equate(a_is_expected).relate(a_ty, b_ty), - SubtypeOf => self.sub(a_is_expected).relate(a_ty, b_ty), - SupertypeOf => self.sub(a_is_expected).relate_with_variance( + self.eager_relate_projections_via_substs(|this| match dir { + EqTo => this.equate(a_is_expected).relate(a_ty, b_ty), + SubtypeOf => this.sub(a_is_expected).relate(a_ty, b_ty), + SupertypeOf => this.sub(a_is_expected).relate_with_variance( ty::Contravariant, ty::VarianceDiagInfo::default(), a_ty, b_ty, ), - }?; + })?; Ok(()) } @@ -492,6 +496,14 @@ impl<'infcx, 'tcx> CombineFields<'infcx, 'tcx> { Ok(Generalization { ty, needs_wf }) } + // Relate projections via their substs eagerly, and never via an `AliasEq` goal + pub fn eager_relate_projections_via_substs(&mut self, f: impl FnOnce(&mut Self) -> T) -> T { + let old = std::mem::replace(&mut self.relate_projections_via_substs, true); + let v = f(self); + self.relate_projections_via_substs = old; + v + } + pub fn register_obligations(&mut self, obligations: PredicateObligations<'tcx>) { self.obligations.extend(obligations.into_iter()); } diff --git a/compiler/rustc_infer/src/infer/equate.rs b/compiler/rustc_infer/src/infer/equate.rs index 54a62326ef7bd..075b86813451d 100644 --- a/compiler/rustc_infer/src/infer/equate.rs +++ b/compiler/rustc_infer/src/infer/equate.rs @@ -144,6 +144,13 @@ impl<'tcx> TypeRelation<'tcx> for Equate<'_, '_, 'tcx> { } } + // Inside the generalizer, we want to related projections by substs eagerly. + (&ty::Alias(ty::Projection, a_data), &ty::Alias(ty::Projection, b_data)) + if self.fields.relate_projections_via_substs => + { + self.relate(a_data, b_data)?; + } + _ => { self.fields.infcx.super_combine_tys(self, a, b)?; } diff --git a/compiler/rustc_infer/src/infer/mod.rs b/compiler/rustc_infer/src/infer/mod.rs index 4a834957959db..d2f89e4a71de7 100644 --- a/compiler/rustc_infer/src/infer/mod.rs +++ b/compiler/rustc_infer/src/infer/mod.rs @@ -738,6 +738,7 @@ impl<'tcx> InferCtxt<'tcx> { param_env, obligations: PredicateObligations::new(), define_opaque_types, + relate_projections_via_substs: false, } } diff --git a/compiler/rustc_infer/src/infer/sub.rs b/compiler/rustc_infer/src/infer/sub.rs index 3e8c2052de89d..7501552b3a687 100644 --- a/compiler/rustc_infer/src/infer/sub.rs +++ b/compiler/rustc_infer/src/infer/sub.rs @@ -172,6 +172,14 @@ impl<'tcx> TypeRelation<'tcx> for Sub<'_, '_, 'tcx> { } } + // Inside the generalizer, we want to related projections by substs eagerly. + (&ty::Alias(ty::Projection, a_data), &ty::Alias(ty::Projection, b_data)) + if self.fields.relate_projections_via_substs => + { + let projection_ty = self.relate(a_data, b_data)?; + Ok(self.tcx().mk_alias(ty::Projection, projection_ty)) + } + _ => { self.fields.infcx.super_combine_tys(self, a, b)?; Ok(a) diff --git a/tests/ui/traits/new-solver/generalize-shouldnt-emit-alias-eq.rs b/tests/ui/traits/new-solver/generalize-shouldnt-emit-alias-eq.rs new file mode 100644 index 0000000000000..4bfb6323a5366 --- /dev/null +++ b/tests/ui/traits/new-solver/generalize-shouldnt-emit-alias-eq.rs @@ -0,0 +1,40 @@ +// check-pass +// compile-flags: -Ztrait-solver=next + +trait Foo { + type Gat<'a> + where + Self: 'a; + fn bar(&self) -> Self::Gat<'_>; +} + +enum Option { + Some(T), + None, +} + +impl Option { + fn as_ref(&self) -> Option<&T> { + match self { + Option::Some(t) => Option::Some(t), + Option::None => Option::None, + } + } + + fn map(self, f: impl FnOnce(T) -> U) -> Option { + match self { + Option::Some(t) => Option::Some(f(t)), + Option::None => Option::None, + } + } +} + +impl Foo for Option { + type Gat<'a> = Option<::Gat<'a>> where Self: 'a; + + fn bar(&self) -> Self::Gat<'_> { + self.as_ref().map(Foo::bar) + } +} + +fn main() {}