//! # Lattice variables //! //! Generic code for operating on [lattices] of inference variables //! that are characterized by an upper- and lower-bound. //! //! The code is defined quite generically so that it can be //! applied both to type variables, which represent types being inferred, //! and fn variables, which represent function types being inferred. //! (It may eventually be applied to their types as well.) //! In some cases, the functions are also generic with respect to the //! operation on the lattice (GLB vs LUB). //! //! ## Note //! //! Although all the functions are generic, for simplicity, comments in the source code //! generally refer to type variables and the LUB operation. //! //! [lattices]: https://en.wikipedia.org/wiki/Lattice_(order) use rustc_type_ir::{ AliasRelationDirection, Interner, TypeVisitableExt, Upcast, Variance, inherent::{IntoKind, Span as _}, relate::{ Relate, StructurallyRelateAliases, TypeRelation, VarianceDiagInfo, combine::{ PredicateEmittingRelation, combine_ty_args, super_combine_consts, super_combine_tys, }, }, }; use crate::next_solver::{ AliasTy, Binder, Const, DbInterner, GenericArgs, Goal, ParamEnv, Predicate, PredicateKind, Region, SolverDefId, Span, Ty, TyKind, infer::{ InferCtxt, TypeTrace, relate::RelateResult, traits::{Obligation, PredicateObligations}, }, }; #[derive(Clone, Copy)] pub(crate) enum LatticeOpKind { Glb, Lub, } impl LatticeOpKind { fn invert(self) -> Self { match self { LatticeOpKind::Glb => LatticeOpKind::Lub, LatticeOpKind::Lub => LatticeOpKind::Glb, } } } /// A greatest lower bound" (common subtype) or least upper bound (common supertype). pub(crate) struct LatticeOp<'infcx, 'db> { infcx: &'infcx InferCtxt<'db>, // Immutable fields trace: TypeTrace<'db>, param_env: ParamEnv<'db>, // Mutable fields kind: LatticeOpKind, obligations: PredicateObligations<'db>, } impl<'infcx, 'db> LatticeOp<'infcx, 'db> { pub(crate) fn new( infcx: &'infcx InferCtxt<'db>, trace: TypeTrace<'db>, param_env: ParamEnv<'db>, kind: LatticeOpKind, ) -> LatticeOp<'infcx, 'db> { LatticeOp { infcx, trace, param_env, kind, obligations: PredicateObligations::new() } } pub(crate) fn into_obligations(self) -> PredicateObligations<'db> { self.obligations } } impl<'db> TypeRelation> for LatticeOp<'_, 'db> { fn cx(&self) -> DbInterner<'db> { self.infcx.interner } fn relate_ty_args( &mut self, a_ty: Ty<'db>, b_ty: Ty<'db>, def_id: SolverDefId, a_args: GenericArgs<'db>, b_args: GenericArgs<'db>, mk: impl FnOnce(GenericArgs<'db>) -> Ty<'db>, ) -> RelateResult<'db, Ty<'db>> { let variances = self.cx().variances_of(def_id); combine_ty_args(self.infcx, self, a_ty, b_ty, variances, a_args, b_args, mk) } fn relate_with_variance>>( &mut self, variance: Variance, _info: VarianceDiagInfo>, a: T, b: T, ) -> RelateResult<'db, T> { match variance { Variance::Invariant => { self.obligations.extend( self.infcx.at(&self.trace.cause, self.param_env).eq(a, b)?.into_obligations(), ); Ok(a) } Variance::Covariant => self.relate(a, b), // FIXME(#41044) -- not correct, need test Variance::Bivariant => Ok(a), Variance::Contravariant => { self.kind = self.kind.invert(); let res = self.relate(a, b); self.kind = self.kind.invert(); res } } } /// Relates two types using a given lattice. fn tys(&mut self, a: Ty<'db>, b: Ty<'db>) -> RelateResult<'db, Ty<'db>> { if a == b { return Ok(a); } let infcx = self.infcx; let a = infcx.shallow_resolve(a); let b = infcx.shallow_resolve(b); match (a.kind(), b.kind()) { // If one side is known to be a variable and one is not, // create a variable (`v`) to represent the LUB. Make sure to // relate `v` to the non-type-variable first (by passing it // first to `relate_bound`). Otherwise, we would produce a // subtype obligation that must then be processed. // // Example: if the LHS is a type variable, and RHS is // `Box`, then we current compare `v` to the RHS first, // which will instantiate `v` with `Box`. Then when `v` // is compared to the LHS, we instantiate LHS with `Box`. // But if we did in reverse order, we would create a `v <: // LHS` (or vice versa) constraint and then instantiate // `v`. This would require further processing to achieve same // end-result; in particular, this screws up some of the logic // in coercion, which expects LUB to figure out that the LHS // is (e.g.) `Box`. A more obvious solution might be to // iterate on the subtype obligations that are returned, but I // think this suffices. -nmatsakis (TyKind::Infer(rustc_type_ir::TyVar(..)), _) => { let v = infcx.next_ty_var(); self.relate_bound(v, b, a)?; Ok(v) } (_, TyKind::Infer(rustc_type_ir::TyVar(..))) => { let v = infcx.next_ty_var(); self.relate_bound(v, a, b)?; Ok(v) } ( TyKind::Alias(rustc_type_ir::Opaque, AliasTy { def_id: a_def_id, .. }), TyKind::Alias(rustc_type_ir::Opaque, AliasTy { def_id: b_def_id, .. }), ) if a_def_id == b_def_id => super_combine_tys(infcx, self, a, b), _ => super_combine_tys(infcx, self, a, b), } } fn regions(&mut self, a: Region<'db>, b: Region<'db>) -> RelateResult<'db, Region<'db>> { let mut inner = self.infcx.inner.borrow_mut(); let mut constraints = inner.unwrap_region_constraints(); Ok(match self.kind { // GLB(&'static u8, &'a u8) == &RegionLUB('static, 'a) u8 == &'static u8 LatticeOpKind::Glb => constraints.lub_regions(self.cx(), a, b), // LUB(&'static u8, &'a u8) == &RegionGLB('static, 'a) u8 == &'a u8 LatticeOpKind::Lub => constraints.glb_regions(self.cx(), a, b), }) } fn consts(&mut self, a: Const<'db>, b: Const<'db>) -> RelateResult<'db, Const<'db>> { super_combine_consts(self.infcx, self, a, b) } fn binders( &mut self, a: Binder<'db, T>, b: Binder<'db, T>, ) -> RelateResult<'db, Binder<'db, T>> where T: Relate>, { // GLB/LUB of a binder and itself is just itself if a == b { return Ok(a); } if a.skip_binder().has_escaping_bound_vars() || b.skip_binder().has_escaping_bound_vars() { // When higher-ranked types are involved, computing the GLB/LUB is // very challenging, switch to invariance. This is obviously // overly conservative but works ok in practice. self.relate_with_variance(Variance::Invariant, VarianceDiagInfo::default(), a, b)?; Ok(a) } else { Ok(Binder::dummy(self.relate(a.skip_binder(), b.skip_binder())?)) } } } impl<'infcx, 'db> LatticeOp<'infcx, 'db> { // Relates the type `v` to `a` and `b` such that `v` represents // the LUB/GLB of `a` and `b` as appropriate. // // Subtle hack: ordering *may* be significant here. This method // relates `v` to `a` first, which may help us to avoid unnecessary // type variable obligations. See caller for details. fn relate_bound(&mut self, v: Ty<'db>, a: Ty<'db>, b: Ty<'db>) -> RelateResult<'db, ()> { let at = self.infcx.at(&self.trace.cause, self.param_env); match self.kind { LatticeOpKind::Glb => { self.obligations.extend(at.sub(v, a)?.into_obligations()); self.obligations.extend(at.sub(v, b)?.into_obligations()); } LatticeOpKind::Lub => { self.obligations.extend(at.sub(a, v)?.into_obligations()); self.obligations.extend(at.sub(b, v)?.into_obligations()); } } Ok(()) } } impl<'db> PredicateEmittingRelation> for LatticeOp<'_, 'db> { fn span(&self) -> Span { Span::dummy() } fn structurally_relate_aliases(&self) -> StructurallyRelateAliases { StructurallyRelateAliases::No } fn param_env(&self) -> ParamEnv<'db> { self.param_env } fn register_predicates( &mut self, preds: impl IntoIterator, Predicate<'db>>>, ) { self.obligations.extend(preds.into_iter().map(|pred| { Obligation::new(self.infcx.interner, self.trace.cause.clone(), self.param_env, pred) })) } fn register_goals(&mut self, goals: impl IntoIterator>>) { self.obligations.extend(goals.into_iter().map(|goal| { Obligation::new( self.infcx.interner, self.trace.cause.clone(), goal.param_env, goal.predicate, ) })) } fn register_alias_relate_predicate(&mut self, a: Ty<'db>, b: Ty<'db>) { self.register_predicates([Binder::dummy(PredicateKind::AliasRelate( a.into(), b.into(), // FIXME(deferred_projection_equality): This isn't right, I think? AliasRelationDirection::Equate, ))]); } }