| //! The new trait solver, currently still WIP. |
| //! |
| //! As a user of the trait system, you can use `TyCtxt::evaluate_goal` to |
| //! interact with this solver. |
| //! |
| //! For a high-level overview of how this solver works, check out the relevant |
| //! section of the rustc-dev-guide. |
| //! |
| //! FIXME(@lcnr): Write that section. If you read this before then ask me |
| //! about it on zulip. |
| |
| use rustc_hir::def_id::DefId; |
| use rustc_infer::infer::canonical::{Canonical, CanonicalVarValues}; |
| use rustc_infer::traits::query::NoSolution; |
| use rustc_middle::traits::solve::{ |
| CanonicalResponse, Certainty, ExternalConstraintsData, Goal, QueryResult, Response, |
| }; |
| use rustc_middle::ty::{self, Ty, TyCtxt}; |
| use rustc_middle::ty::{ |
| CoercePredicate, RegionOutlivesPredicate, SubtypePredicate, TypeOutlivesPredicate, |
| }; |
| |
| mod assembly; |
| mod canonicalize; |
| mod eval_ctxt; |
| mod fulfill; |
| mod opaques; |
| mod project_goals; |
| mod search_graph; |
| mod trait_goals; |
| |
| pub use eval_ctxt::{EvalCtxt, InferCtxtEvalExt}; |
| pub use fulfill::FulfillmentCtxt; |
| |
| #[derive(Debug, Clone, Copy)] |
| enum SolverMode { |
| /// Ordinary trait solving, using everywhere except for coherence. |
| Normal, |
| /// Trait solving during coherence. There are a few notable differences |
| /// between coherence and ordinary trait solving. |
| /// |
| /// Most importantly, trait solving during coherence must not be incomplete, |
| /// i.e. return `Err(NoSolution)` for goals for which a solution exists. |
| /// This means that we must not make any guesses or arbitrary choices. |
| Coherence, |
| } |
| |
| trait CanonicalResponseExt { |
| fn has_no_inference_or_external_constraints(&self) -> bool; |
| |
| fn has_only_region_constraints(&self) -> bool; |
| } |
| |
| impl<'tcx> CanonicalResponseExt for Canonical<'tcx, Response<'tcx>> { |
| fn has_no_inference_or_external_constraints(&self) -> bool { |
| self.value.external_constraints.region_constraints.is_empty() |
| && self.value.var_values.is_identity() |
| && self.value.external_constraints.opaque_types.is_empty() |
| } |
| |
| fn has_only_region_constraints(&self) -> bool { |
| self.value.var_values.is_identity_modulo_regions() |
| && self.value.external_constraints.opaque_types.is_empty() |
| } |
| } |
| |
| impl<'a, 'tcx> EvalCtxt<'a, 'tcx> { |
| #[instrument(level = "debug", skip(self))] |
| fn compute_type_outlives_goal( |
| &mut self, |
| goal: Goal<'tcx, TypeOutlivesPredicate<'tcx>>, |
| ) -> QueryResult<'tcx> { |
| let ty::OutlivesPredicate(ty, lt) = goal.predicate; |
| self.register_ty_outlives(ty, lt); |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_region_outlives_goal( |
| &mut self, |
| goal: Goal<'tcx, RegionOutlivesPredicate<'tcx>>, |
| ) -> QueryResult<'tcx> { |
| let ty::OutlivesPredicate(a, b) = goal.predicate; |
| self.register_region_outlives(a, b); |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_coerce_goal( |
| &mut self, |
| goal: Goal<'tcx, CoercePredicate<'tcx>>, |
| ) -> QueryResult<'tcx> { |
| self.compute_subtype_goal(Goal { |
| param_env: goal.param_env, |
| predicate: SubtypePredicate { |
| a_is_expected: false, |
| a: goal.predicate.a, |
| b: goal.predicate.b, |
| }, |
| }) |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_subtype_goal( |
| &mut self, |
| goal: Goal<'tcx, SubtypePredicate<'tcx>>, |
| ) -> QueryResult<'tcx> { |
| if goal.predicate.a.is_ty_var() && goal.predicate.b.is_ty_var() { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS) |
| } else { |
| self.sub(goal.param_env, goal.predicate.a, goal.predicate.b)?; |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_closure_kind_goal( |
| &mut self, |
| goal: Goal<'tcx, (DefId, ty::SubstsRef<'tcx>, ty::ClosureKind)>, |
| ) -> QueryResult<'tcx> { |
| let (_, substs, expected_kind) = goal.predicate; |
| let found_kind = substs.as_closure().kind_ty().to_opt_closure_kind(); |
| |
| let Some(found_kind) = found_kind else { |
| return self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS); |
| }; |
| if found_kind.extends(expected_kind) { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } else { |
| Err(NoSolution) |
| } |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_object_safe_goal(&mut self, trait_def_id: DefId) -> QueryResult<'tcx> { |
| if self.tcx().check_is_object_safe(trait_def_id) { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } else { |
| Err(NoSolution) |
| } |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn compute_well_formed_goal( |
| &mut self, |
| goal: Goal<'tcx, ty::GenericArg<'tcx>>, |
| ) -> QueryResult<'tcx> { |
| match self.well_formed_goals(goal.param_env, goal.predicate) { |
| Some(goals) => { |
| self.add_goals(goals); |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| None => self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS), |
| } |
| } |
| |
| #[instrument(level = "debug", skip(self), ret)] |
| fn compute_alias_relate_goal( |
| &mut self, |
| goal: Goal<'tcx, (ty::Term<'tcx>, ty::Term<'tcx>, ty::AliasRelationDirection)>, |
| ) -> QueryResult<'tcx> { |
| let tcx = self.tcx(); |
| // We may need to invert the alias relation direction if dealing an alias on the RHS. |
| #[derive(Debug)] |
| enum Invert { |
| No, |
| Yes, |
| } |
| let evaluate_normalizes_to = |
| |ecx: &mut EvalCtxt<'_, 'tcx>, alias, other, direction, invert| { |
| let span = tracing::span!( |
| tracing::Level::DEBUG, |
| "compute_alias_relate_goal(evaluate_normalizes_to)", |
| ?alias, |
| ?other, |
| ?direction, |
| ?invert |
| ); |
| let _enter = span.enter(); |
| let result = ecx.probe(|ecx| { |
| let other = match direction { |
| // This is purely an optimization. |
| ty::AliasRelationDirection::Equate => other, |
| |
| ty::AliasRelationDirection::Subtype => { |
| let fresh = ecx.next_term_infer_of_kind(other); |
| let (sub, sup) = match invert { |
| Invert::No => (fresh, other), |
| Invert::Yes => (other, fresh), |
| }; |
| ecx.sub(goal.param_env, sub, sup)?; |
| fresh |
| } |
| }; |
| ecx.add_goal(goal.with( |
| tcx, |
| ty::Binder::dummy(ty::ProjectionPredicate { |
| projection_ty: alias, |
| term: other, |
| }), |
| )); |
| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| }); |
| debug!(?result); |
| result |
| }; |
| |
| let (lhs, rhs, direction) = goal.predicate; |
| |
| if lhs.is_infer() || rhs.is_infer() { |
| bug!( |
| "`AliasRelate` goal with an infer var on lhs or rhs which should have been instantiated" |
| ); |
| } |
| |
| match (lhs.to_alias_ty(tcx), rhs.to_alias_ty(tcx)) { |
| (None, None) => bug!("`AliasRelate` goal without an alias on either lhs or rhs"), |
| |
| // RHS is not a projection, only way this is true is if LHS normalizes-to RHS |
| (Some(alias_lhs), None) => { |
| evaluate_normalizes_to(self, alias_lhs, rhs, direction, Invert::No) |
| } |
| |
| // LHS is not a projection, only way this is true is if RHS normalizes-to LHS |
| (None, Some(alias_rhs)) => { |
| evaluate_normalizes_to(self, alias_rhs, lhs, direction, Invert::Yes) |
| } |
| |
| (Some(alias_lhs), Some(alias_rhs)) => { |
| debug!("both sides are aliases"); |
| |
| let mut candidates = Vec::new(); |
| // LHS normalizes-to RHS |
| candidates.extend(evaluate_normalizes_to( |
| self, |
| alias_lhs, |
| rhs, |
| direction, |
| Invert::No, |
| )); |
| // RHS normalizes-to RHS |
| candidates.extend(evaluate_normalizes_to( |
| self, |
| alias_rhs, |
| lhs, |
| direction, |
| Invert::Yes, |
| )); |
| // Relate via substs |
| let subst_relate_response = self.probe(|ecx| { |
| let span = tracing::span!( |
| tracing::Level::DEBUG, |
| "compute_alias_relate_goal(relate_via_substs)", |
| ?alias_lhs, |
| ?alias_rhs, |
| ?direction |
| ); |
| let _enter = span.enter(); |
| |
| match direction { |
| ty::AliasRelationDirection::Equate => { |
| ecx.eq(goal.param_env, alias_lhs, alias_rhs)?; |
| } |
| ty::AliasRelationDirection::Subtype => { |
| ecx.sub(goal.param_env, alias_lhs, alias_rhs)?; |
| } |
| } |
| |
| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| }); |
| candidates.extend(subst_relate_response); |
| debug!(?candidates); |
| |
| if let Some(merged) = self.try_merge_responses(&candidates) { |
| Ok(merged) |
| } else { |
| // When relating two aliases and we have ambiguity, we prefer |
| // relating the generic arguments of the aliases over normalizing |
| // them. This is necessary for inference during typeck. |
| // |
| // As this is incomplete, we must not do so during coherence. |
| match (self.solver_mode(), subst_relate_response) { |
| (SolverMode::Normal, Ok(response)) => Ok(response), |
| (SolverMode::Normal, Err(NoSolution)) | (SolverMode::Coherence, _) => { |
| self.flounder(&candidates) |
| } |
| } |
| } |
| } |
| } |
| } |
| |
| #[instrument(level = "debug", skip(self), ret)] |
| fn compute_const_arg_has_type_goal( |
| &mut self, |
| goal: Goal<'tcx, (ty::Const<'tcx>, Ty<'tcx>)>, |
| ) -> QueryResult<'tcx> { |
| let (ct, ty) = goal.predicate; |
| self.eq(goal.param_env, ct.ty(), ty)?; |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| } |
| |
| impl<'tcx> EvalCtxt<'_, 'tcx> { |
| #[instrument(level = "debug", skip(self))] |
| fn set_normalizes_to_hack_goal(&mut self, goal: Goal<'tcx, ty::ProjectionPredicate<'tcx>>) { |
| assert!( |
| self.nested_goals.normalizes_to_hack_goal.is_none(), |
| "attempted to set the projection eq hack goal when one already exists" |
| ); |
| self.nested_goals.normalizes_to_hack_goal = Some(goal); |
| } |
| |
| #[instrument(level = "debug", skip(self))] |
| fn add_goal(&mut self, goal: Goal<'tcx, ty::Predicate<'tcx>>) { |
| self.nested_goals.goals.push(goal); |
| } |
| |
| #[instrument(level = "debug", skip(self, goals))] |
| fn add_goals(&mut self, goals: impl IntoIterator<Item = Goal<'tcx, ty::Predicate<'tcx>>>) { |
| let current_len = self.nested_goals.goals.len(); |
| self.nested_goals.goals.extend(goals); |
| debug!("added_goals={:?}", &self.nested_goals.goals[current_len..]); |
| } |
| |
| /// Try to merge multiple possible ways to prove a goal, if that is not possible returns `None`. |
| /// |
| /// In this case we tend to flounder and return ambiguity by calling `[EvalCtxt::flounder]`. |
| #[instrument(level = "debug", skip(self), ret)] |
| fn try_merge_responses( |
| &mut self, |
| responses: &[CanonicalResponse<'tcx>], |
| ) -> Option<CanonicalResponse<'tcx>> { |
| if responses.is_empty() { |
| return None; |
| } |
| |
| // FIXME(-Ztrait-solver=next): We should instead try to find a `Certainty::Yes` response with |
| // a subset of the constraints that all the other responses have. |
| let one = responses[0]; |
| if responses[1..].iter().all(|&resp| resp == one) { |
| return Some(one); |
| } |
| |
| responses |
| .iter() |
| .find(|response| { |
| response.value.certainty == Certainty::Yes |
| && response.has_no_inference_or_external_constraints() |
| }) |
| .copied() |
| } |
| |
| /// If we fail to merge responses we flounder and return overflow or ambiguity. |
| #[instrument(level = "debug", skip(self), ret)] |
| fn flounder(&mut self, responses: &[CanonicalResponse<'tcx>]) -> QueryResult<'tcx> { |
| if responses.is_empty() { |
| return Err(NoSolution); |
| } |
| |
| let Certainty::Maybe(maybe_cause) = responses.iter().fold( |
| Certainty::AMBIGUOUS, |
| |certainty, response| { |
| certainty.unify_with(response.value.certainty) |
| }, |
| ) else { |
| bug!("expected flounder response to be ambiguous") |
| }; |
| |
| Ok(self.make_ambiguous_response_no_constraints(maybe_cause)) |
| } |
| } |
| |
| pub(super) fn response_no_constraints<'tcx>( |
| tcx: TyCtxt<'tcx>, |
| goal: Canonical<'tcx, impl Sized>, |
| certainty: Certainty, |
| ) -> QueryResult<'tcx> { |
| Ok(Canonical { |
| max_universe: goal.max_universe, |
| variables: goal.variables, |
| value: Response { |
| var_values: CanonicalVarValues::make_identity(tcx, goal.variables), |
| // FIXME: maybe we should store the "no response" version in tcx, like |
| // we do for tcx.types and stuff. |
| external_constraints: tcx.mk_external_constraints(ExternalConstraintsData::default()), |
| certainty, |
| }, |
| }) |
| } |