| ///! Definition of `InferCtxtLike` from the librarified type layer. |
| use rustc_hir::def_id::DefId; |
| use rustc_middle::traits::ObligationCause; |
| use rustc_middle::ty::relate::RelateResult; |
| use rustc_middle::ty::relate::combine::PredicateEmittingRelation; |
| use rustc_middle::ty::{self, Ty, TyCtxt, TypeFoldable}; |
| use rustc_span::{DUMMY_SP, ErrorGuaranteed, Span}; |
| |
| use super::{ |
| BoundRegionConversionTime, InferCtxt, OpaqueTypeStorageEntries, RegionVariableOrigin, |
| SubregionOrigin, |
| }; |
| |
| impl<'tcx> rustc_type_ir::InferCtxtLike for InferCtxt<'tcx> { |
| type Interner = TyCtxt<'tcx>; |
| |
| fn cx(&self) -> TyCtxt<'tcx> { |
| self.tcx |
| } |
| |
| fn next_trait_solver(&self) -> bool { |
| self.next_trait_solver |
| } |
| |
| fn typing_mode(&self) -> ty::TypingMode<'tcx> { |
| self.typing_mode() |
| } |
| |
| fn universe(&self) -> ty::UniverseIndex { |
| self.universe() |
| } |
| |
| fn create_next_universe(&self) -> ty::UniverseIndex { |
| self.create_next_universe() |
| } |
| |
| fn universe_of_ty(&self, vid: ty::TyVid) -> Option<ty::UniverseIndex> { |
| match self.probe_ty_var(vid) { |
| Err(universe) => Some(universe), |
| Ok(_) => None, |
| } |
| } |
| |
| fn universe_of_lt(&self, lt: ty::RegionVid) -> Option<ty::UniverseIndex> { |
| match self.inner.borrow_mut().unwrap_region_constraints().probe_value(lt) { |
| Err(universe) => Some(universe), |
| Ok(_) => None, |
| } |
| } |
| |
| fn universe_of_ct(&self, ct: ty::ConstVid) -> Option<ty::UniverseIndex> { |
| match self.probe_const_var(ct) { |
| Err(universe) => Some(universe), |
| Ok(_) => None, |
| } |
| } |
| |
| fn root_ty_var(&self, var: ty::TyVid) -> ty::TyVid { |
| self.root_var(var) |
| } |
| |
| fn root_const_var(&self, var: ty::ConstVid) -> ty::ConstVid { |
| self.root_const_var(var) |
| } |
| |
| fn opportunistic_resolve_ty_var(&self, vid: ty::TyVid) -> Ty<'tcx> { |
| match self.probe_ty_var(vid) { |
| Ok(ty) => ty, |
| Err(_) => Ty::new_var(self.tcx, self.root_var(vid)), |
| } |
| } |
| |
| fn opportunistic_resolve_int_var(&self, vid: ty::IntVid) -> Ty<'tcx> { |
| self.opportunistic_resolve_int_var(vid) |
| } |
| |
| fn opportunistic_resolve_float_var(&self, vid: ty::FloatVid) -> Ty<'tcx> { |
| self.opportunistic_resolve_float_var(vid) |
| } |
| |
| fn opportunistic_resolve_ct_var(&self, vid: ty::ConstVid) -> ty::Const<'tcx> { |
| match self.probe_const_var(vid) { |
| Ok(ct) => ct, |
| Err(_) => ty::Const::new_var(self.tcx, self.root_const_var(vid)), |
| } |
| } |
| |
| fn opportunistic_resolve_lt_var(&self, vid: ty::RegionVid) -> ty::Region<'tcx> { |
| self.inner.borrow_mut().unwrap_region_constraints().opportunistic_resolve_var(self.tcx, vid) |
| } |
| |
| fn is_changed_arg(&self, arg: ty::GenericArg<'tcx>) -> bool { |
| match arg.kind() { |
| ty::GenericArgKind::Lifetime(_) => { |
| // Lifetimes should not change affect trait selection. |
| false |
| } |
| ty::GenericArgKind::Type(ty) => { |
| if let ty::Infer(infer_ty) = *ty.kind() { |
| match infer_ty { |
| ty::InferTy::TyVar(vid) => { |
| !self.probe_ty_var(vid).is_err_and(|_| self.root_var(vid) == vid) |
| } |
| ty::InferTy::IntVar(vid) => { |
| let mut inner = self.inner.borrow_mut(); |
| !matches!( |
| inner.int_unification_table().probe_value(vid), |
| ty::IntVarValue::Unknown |
| if inner.int_unification_table().find(vid) == vid |
| ) |
| } |
| ty::InferTy::FloatVar(vid) => { |
| let mut inner = self.inner.borrow_mut(); |
| !matches!( |
| inner.float_unification_table().probe_value(vid), |
| ty::FloatVarValue::Unknown |
| if inner.float_unification_table().find(vid) == vid |
| ) |
| } |
| ty::InferTy::FreshTy(_) |
| | ty::InferTy::FreshIntTy(_) |
| | ty::InferTy::FreshFloatTy(_) => true, |
| } |
| } else { |
| true |
| } |
| } |
| ty::GenericArgKind::Const(ct) => { |
| if let ty::ConstKind::Infer(infer_ct) = ct.kind() { |
| match infer_ct { |
| ty::InferConst::Var(vid) => !self |
| .probe_const_var(vid) |
| .is_err_and(|_| self.root_const_var(vid) == vid), |
| ty::InferConst::Fresh(_) => true, |
| } |
| } else { |
| true |
| } |
| } |
| } |
| } |
| |
| fn next_region_infer(&self) -> ty::Region<'tcx> { |
| self.next_region_var(RegionVariableOrigin::Misc(DUMMY_SP)) |
| } |
| |
| fn next_ty_infer(&self) -> Ty<'tcx> { |
| self.next_ty_var(DUMMY_SP) |
| } |
| |
| fn next_const_infer(&self) -> ty::Const<'tcx> { |
| self.next_const_var(DUMMY_SP) |
| } |
| |
| fn fresh_args_for_item(&self, def_id: DefId) -> ty::GenericArgsRef<'tcx> { |
| self.fresh_args_for_item(DUMMY_SP, def_id) |
| } |
| |
| fn instantiate_binder_with_infer<T: TypeFoldable<TyCtxt<'tcx>> + Copy>( |
| &self, |
| value: ty::Binder<'tcx, T>, |
| ) -> T { |
| self.instantiate_binder_with_fresh_vars( |
| DUMMY_SP, |
| BoundRegionConversionTime::HigherRankedType, |
| value, |
| ) |
| } |
| |
| fn enter_forall<T: TypeFoldable<TyCtxt<'tcx>>, U>( |
| &self, |
| value: ty::Binder<'tcx, T>, |
| f: impl FnOnce(T) -> U, |
| ) -> U { |
| self.enter_forall(value, f) |
| } |
| |
| fn equate_ty_vids_raw(&self, a: ty::TyVid, b: ty::TyVid) { |
| self.inner.borrow_mut().type_variables().equate(a, b); |
| } |
| |
| fn equate_int_vids_raw(&self, a: ty::IntVid, b: ty::IntVid) { |
| self.inner.borrow_mut().int_unification_table().union(a, b); |
| } |
| |
| fn equate_float_vids_raw(&self, a: ty::FloatVid, b: ty::FloatVid) { |
| self.inner.borrow_mut().float_unification_table().union(a, b); |
| } |
| |
| fn equate_const_vids_raw(&self, a: ty::ConstVid, b: ty::ConstVid) { |
| self.inner.borrow_mut().const_unification_table().union(a, b); |
| } |
| |
| fn instantiate_ty_var_raw<R: PredicateEmittingRelation<Self>>( |
| &self, |
| relation: &mut R, |
| target_is_expected: bool, |
| target_vid: ty::TyVid, |
| instantiation_variance: ty::Variance, |
| source_ty: Ty<'tcx>, |
| ) -> RelateResult<'tcx, ()> { |
| self.instantiate_ty_var( |
| relation, |
| target_is_expected, |
| target_vid, |
| instantiation_variance, |
| source_ty, |
| ) |
| } |
| |
| fn instantiate_int_var_raw(&self, vid: ty::IntVid, value: ty::IntVarValue) { |
| self.inner.borrow_mut().int_unification_table().union_value(vid, value); |
| } |
| |
| fn instantiate_float_var_raw(&self, vid: ty::FloatVid, value: ty::FloatVarValue) { |
| self.inner.borrow_mut().float_unification_table().union_value(vid, value); |
| } |
| |
| fn instantiate_const_var_raw<R: PredicateEmittingRelation<Self>>( |
| &self, |
| relation: &mut R, |
| target_is_expected: bool, |
| target_vid: ty::ConstVid, |
| source_ct: ty::Const<'tcx>, |
| ) -> RelateResult<'tcx, ()> { |
| self.instantiate_const_var(relation, target_is_expected, target_vid, source_ct) |
| } |
| |
| fn set_tainted_by_errors(&self, e: ErrorGuaranteed) { |
| self.set_tainted_by_errors(e) |
| } |
| |
| fn shallow_resolve(&self, ty: Ty<'tcx>) -> Ty<'tcx> { |
| self.shallow_resolve(ty) |
| } |
| fn shallow_resolve_const(&self, ct: ty::Const<'tcx>) -> ty::Const<'tcx> { |
| self.shallow_resolve_const(ct) |
| } |
| |
| fn resolve_vars_if_possible<T>(&self, value: T) -> T |
| where |
| T: TypeFoldable<TyCtxt<'tcx>>, |
| { |
| self.resolve_vars_if_possible(value) |
| } |
| |
| fn probe<T>(&self, probe: impl FnOnce() -> T) -> T { |
| self.probe(|_| probe()) |
| } |
| |
| fn sub_regions(&self, sub: ty::Region<'tcx>, sup: ty::Region<'tcx>, span: Span) { |
| self.inner.borrow_mut().unwrap_region_constraints().make_subregion( |
| SubregionOrigin::RelateRegionParamBound(span, None), |
| sub, |
| sup, |
| ); |
| } |
| |
| fn equate_regions(&self, a: ty::Region<'tcx>, b: ty::Region<'tcx>, span: Span) { |
| self.inner.borrow_mut().unwrap_region_constraints().make_eqregion( |
| SubregionOrigin::RelateRegionParamBound(span, None), |
| a, |
| b, |
| ); |
| } |
| |
| fn register_ty_outlives(&self, ty: Ty<'tcx>, r: ty::Region<'tcx>, span: Span) { |
| self.register_type_outlives_constraint(ty, r, &ObligationCause::dummy_with_span(span)); |
| } |
| |
| type OpaqueTypeStorageEntries = OpaqueTypeStorageEntries; |
| fn opaque_types_storage_num_entries(&self) -> OpaqueTypeStorageEntries { |
| self.inner.borrow_mut().opaque_types().num_entries() |
| } |
| fn clone_opaque_types_lookup_table(&self) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> { |
| self.inner.borrow_mut().opaque_types().iter_lookup_table().map(|(k, h)| (k, h.ty)).collect() |
| } |
| fn clone_duplicate_opaque_types(&self) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> { |
| self.inner |
| .borrow_mut() |
| .opaque_types() |
| .iter_duplicate_entries() |
| .map(|(k, h)| (k, h.ty)) |
| .collect() |
| } |
| fn clone_opaque_types_added_since( |
| &self, |
| prev_entries: OpaqueTypeStorageEntries, |
| ) -> Vec<(ty::OpaqueTypeKey<'tcx>, Ty<'tcx>)> { |
| self.inner |
| .borrow_mut() |
| .opaque_types() |
| .opaque_types_added_since(prev_entries) |
| .map(|(k, h)| (k, h.ty)) |
| .collect() |
| } |
| |
| fn register_hidden_type_in_storage( |
| &self, |
| opaque_type_key: ty::OpaqueTypeKey<'tcx>, |
| hidden_ty: Ty<'tcx>, |
| span: Span, |
| ) -> Option<Ty<'tcx>> { |
| self.register_hidden_type_in_storage( |
| opaque_type_key, |
| ty::OpaqueHiddenType { span, ty: hidden_ty }, |
| ) |
| } |
| fn add_duplicate_opaque_type( |
| &self, |
| opaque_type_key: ty::OpaqueTypeKey<'tcx>, |
| hidden_ty: Ty<'tcx>, |
| span: Span, |
| ) { |
| self.inner |
| .borrow_mut() |
| .opaque_types() |
| .add_duplicate(opaque_type_key, ty::OpaqueHiddenType { span, ty: hidden_ty }) |
| } |
| |
| fn reset_opaque_types(&self) { |
| let _ = self.take_opaque_types(); |
| } |
| } |