| use rustc_hir::def_id::LocalDefId; |
| use rustc_infer::infer::outlives::env::OutlivesEnvironment; |
| use rustc_infer::infer::{InferCtxt, RegionResolutionError}; |
| use rustc_macros::extension; |
| use rustc_middle::traits::ObligationCause; |
| use rustc_middle::traits::query::NoSolution; |
| use rustc_middle::ty::{self, Ty, elaborate}; |
| |
| use crate::traits::ScrubbedTraitError; |
| use crate::traits::outlives_bounds::InferCtxtExt; |
| |
| #[extension(pub trait OutlivesEnvironmentBuildExt<'tcx>)] |
| impl<'tcx> OutlivesEnvironment<'tcx> { |
| fn new( |
| infcx: &InferCtxt<'tcx>, |
| body_id: LocalDefId, |
| param_env: ty::ParamEnv<'tcx>, |
| assumed_wf_tys: impl IntoIterator<Item = Ty<'tcx>>, |
| ) -> Self { |
| Self::new_with_implied_bounds_compat(infcx, body_id, param_env, assumed_wf_tys, false) |
| } |
| |
| fn new_with_implied_bounds_compat( |
| infcx: &InferCtxt<'tcx>, |
| body_id: LocalDefId, |
| param_env: ty::ParamEnv<'tcx>, |
| assumed_wf_tys: impl IntoIterator<Item = Ty<'tcx>>, |
| disable_implied_bounds_hack: bool, |
| ) -> Self { |
| let mut bounds = vec![]; |
| |
| for bound in param_env.caller_bounds() { |
| if let Some(mut type_outlives) = bound.as_type_outlives_clause() { |
| if infcx.next_trait_solver() { |
| match crate::solve::deeply_normalize::<_, ScrubbedTraitError<'tcx>>( |
| infcx.at(&ObligationCause::dummy(), param_env), |
| type_outlives, |
| ) { |
| Ok(new) => type_outlives = new, |
| Err(_) => { |
| infcx.dcx().delayed_bug(format!("could not normalize `{bound}`")); |
| } |
| } |
| } |
| bounds.push(type_outlives); |
| } |
| } |
| |
| // FIXME(-Znext-trait-solver): Normalize these. |
| let higher_ranked_assumptions = infcx.take_registered_region_assumptions(); |
| let higher_ranked_assumptions = |
| elaborate::elaborate_outlives_assumptions(infcx.tcx, higher_ranked_assumptions); |
| |
| // FIXME: This needs to be modified so that we normalize the known type |
| // outlives obligations then elaborate them into their region/type components. |
| // Otherwise, `<W<'a> as Mirror>::Assoc: 'b` will not imply `'a: 'b` even |
| // if we can normalize `'a`. |
| OutlivesEnvironment::from_normalized_bounds( |
| param_env, |
| bounds, |
| infcx.implied_bounds_tys( |
| body_id, |
| param_env, |
| assumed_wf_tys, |
| disable_implied_bounds_hack, |
| ), |
| higher_ranked_assumptions, |
| ) |
| } |
| } |
| |
| #[extension(pub trait InferCtxtRegionExt<'tcx>)] |
| impl<'tcx> InferCtxt<'tcx> { |
| /// Resolve regions, using the deep normalizer to normalize any type-outlives |
| /// obligations in the process. This is in `rustc_trait_selection` because |
| /// we need to normalize. |
| /// |
| /// Prefer this method over `resolve_regions_with_normalize`, unless you are |
| /// doing something specific for normalization. |
| fn resolve_regions( |
| &self, |
| body_id: LocalDefId, |
| param_env: ty::ParamEnv<'tcx>, |
| assumed_wf_tys: impl IntoIterator<Item = Ty<'tcx>>, |
| ) -> Vec<RegionResolutionError<'tcx>> { |
| self.resolve_regions_with_outlives_env(&OutlivesEnvironment::new( |
| self, |
| body_id, |
| param_env, |
| assumed_wf_tys, |
| )) |
| } |
| |
| /// Don't call this directly unless you know what you're doing. |
| fn resolve_regions_with_outlives_env( |
| &self, |
| outlives_env: &OutlivesEnvironment<'tcx>, |
| ) -> Vec<RegionResolutionError<'tcx>> { |
| self.resolve_regions_with_normalize(&outlives_env, |ty, origin| { |
| let ty = self.resolve_vars_if_possible(ty); |
| |
| if self.next_trait_solver() { |
| crate::solve::deeply_normalize( |
| self.at( |
| &ObligationCause::dummy_with_span(origin.span()), |
| outlives_env.param_env, |
| ), |
| ty, |
| ) |
| .map_err(|_: Vec<ScrubbedTraitError<'tcx>>| NoSolution) |
| } else { |
| Ok(ty) |
| } |
| }) |
| } |
| } |