| //! The next-generation trait solver, currently still WIP. |
| //! |
| //! As a user of rust, you can use `-Znext-solver` to enable the new trait solver. |
| //! |
| //! As a developer of rustc, you shouldn't be using the new trait |
| //! solver without asking the trait-system-refactor-initiative, but it can |
| //! be enabled with `InferCtxtBuilder::with_next_trait_solver`. This will |
| //! ensure that trait solving using that inference context will be routed |
| //! to the new trait solver. |
| //! |
| //! For a high-level overview of how this solver works, check out the relevant |
| //! section of the rustc-dev-guide. |
| |
| mod alias_relate; |
| mod assembly; |
| mod effect_goals; |
| mod eval_ctxt; |
| pub mod inspect; |
| mod normalizes_to; |
| mod project_goals; |
| mod search_graph; |
| mod trait_goals; |
| |
| use derive_where::derive_where; |
| use rustc_type_ir::inherent::*; |
| pub use rustc_type_ir::solve::*; |
| use rustc_type_ir::{self as ty, Interner, TypingMode}; |
| use tracing::instrument; |
| |
| pub use self::eval_ctxt::{EvalCtxt, GenerateProofTree, SolverDelegateEvalExt}; |
| use crate::delegate::SolverDelegate; |
| use crate::solve::assembly::Candidate; |
| |
| /// How many fixpoint iterations we should attempt inside of the solver before bailing |
| /// with overflow. |
| /// |
| /// We previously used `cx.recursion_limit().0.checked_ilog2().unwrap_or(0)` for this. |
| /// However, it feels unlikely that uncreasing the recursion limit by a power of two |
| /// to get one more itereation is every useful or desirable. We now instead used a constant |
| /// here. If there ever ends up some use-cases where a bigger number of fixpoint iterations |
| /// is required, we can add a new attribute for that or revert this to be dependant on the |
| /// recursion limit again. However, this feels very unlikely. |
| const FIXPOINT_STEP_LIMIT: usize = 8; |
| |
| #[derive(Debug, Copy, Clone, PartialEq, Eq)] |
| enum GoalEvaluationKind { |
| Root, |
| Nested, |
| } |
| |
| /// Whether evaluating this goal ended up changing the |
| /// inference state. |
| #[derive(PartialEq, Eq, Debug, Hash, Clone, Copy)] |
| pub enum HasChanged { |
| Yes, |
| No, |
| } |
| |
| // FIXME(trait-system-refactor-initiative#117): we don't detect whether a response |
| // ended up pulling down any universes. |
| fn has_no_inference_or_external_constraints<I: Interner>( |
| response: ty::Canonical<I, Response<I>>, |
| ) -> bool { |
| let ExternalConstraintsData { |
| ref region_constraints, |
| ref opaque_types, |
| ref normalization_nested_goals, |
| } = *response.value.external_constraints; |
| response.value.var_values.is_identity() |
| && region_constraints.is_empty() |
| && opaque_types.is_empty() |
| && normalization_nested_goals.is_empty() |
| } |
| |
| fn has_only_region_constraints<I: Interner>(response: ty::Canonical<I, Response<I>>) -> bool { |
| let ExternalConstraintsData { |
| region_constraints: _, |
| ref opaque_types, |
| ref normalization_nested_goals, |
| } = *response.value.external_constraints; |
| response.value.var_values.is_identity_modulo_regions() |
| && opaque_types.is_empty() |
| && normalization_nested_goals.is_empty() |
| } |
| |
| impl<'a, D, I> EvalCtxt<'a, D> |
| where |
| D: SolverDelegate<Interner = I>, |
| I: Interner, |
| { |
| #[instrument(level = "trace", skip(self))] |
| fn compute_type_outlives_goal( |
| &mut self, |
| goal: Goal<I, ty::OutlivesPredicate<I, I::Ty>>, |
| ) -> QueryResult<I> { |
| 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 = "trace", skip(self))] |
| fn compute_region_outlives_goal( |
| &mut self, |
| goal: Goal<I, ty::OutlivesPredicate<I, I::Region>>, |
| ) -> QueryResult<I> { |
| 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 = "trace", skip(self))] |
| fn compute_coerce_goal(&mut self, goal: Goal<I, ty::CoercePredicate<I>>) -> QueryResult<I> { |
| self.compute_subtype_goal(Goal { |
| param_env: goal.param_env, |
| predicate: ty::SubtypePredicate { |
| a_is_expected: false, |
| a: goal.predicate.a, |
| b: goal.predicate.b, |
| }, |
| }) |
| } |
| |
| #[instrument(level = "trace", skip(self))] |
| fn compute_subtype_goal(&mut self, goal: Goal<I, ty::SubtypePredicate<I>>) -> QueryResult<I> { |
| 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) |
| } |
| } |
| |
| fn compute_dyn_compatible_goal(&mut self, trait_def_id: I::TraitId) -> QueryResult<I> { |
| if self.cx().trait_is_dyn_compatible(trait_def_id) { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } else { |
| Err(NoSolution) |
| } |
| } |
| |
| #[instrument(level = "trace", skip(self))] |
| fn compute_well_formed_goal(&mut self, goal: Goal<I, I::Term>) -> QueryResult<I> { |
| match self.well_formed_goals(goal.param_env, goal.predicate) { |
| Some(goals) => { |
| self.add_goals(GoalSource::Misc, goals); |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| None => self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS), |
| } |
| } |
| |
| fn compute_unstable_feature_goal( |
| &mut self, |
| param_env: <I as Interner>::ParamEnv, |
| symbol: <I as Interner>::Symbol, |
| ) -> QueryResult<I> { |
| if self.may_use_unstable_feature(param_env, symbol) { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } else { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Maybe( |
| MaybeCause::Ambiguity, |
| )) |
| } |
| } |
| |
| #[instrument(level = "trace", skip(self))] |
| fn compute_const_evaluatable_goal( |
| &mut self, |
| Goal { param_env, predicate: ct }: Goal<I, I::Const>, |
| ) -> QueryResult<I> { |
| match ct.kind() { |
| ty::ConstKind::Unevaluated(uv) => { |
| // We never return `NoSolution` here as `evaluate_const` emits an |
| // error itself when failing to evaluate, so emitting an additional fulfillment |
| // error in that case is unnecessary noise. This may change in the future once |
| // evaluation failures are allowed to impact selection, e.g. generic const |
| // expressions in impl headers or `where`-clauses. |
| |
| // FIXME(generic_const_exprs): Implement handling for generic |
| // const expressions here. |
| if let Some(_normalized) = self.evaluate_const(param_env, uv) { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } else { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS) |
| } |
| } |
| ty::ConstKind::Infer(_) => { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS) |
| } |
| ty::ConstKind::Placeholder(_) | ty::ConstKind::Value(_) | ty::ConstKind::Error(_) => { |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| // We can freely ICE here as: |
| // - `Param` gets replaced with a placeholder during canonicalization |
| // - `Bound` cannot exist as we don't have a binder around the self Type |
| // - `Expr` is part of `feature(generic_const_exprs)` and is not implemented yet |
| ty::ConstKind::Param(_) | ty::ConstKind::Bound(_, _) | ty::ConstKind::Expr(_) => { |
| panic!("unexpected const kind: {:?}", ct) |
| } |
| } |
| } |
| |
| #[instrument(level = "trace", skip(self), ret)] |
| fn compute_const_arg_has_type_goal( |
| &mut self, |
| goal: Goal<I, (I::Const, I::Ty)>, |
| ) -> QueryResult<I> { |
| let (ct, ty) = goal.predicate; |
| let ct = self.structurally_normalize_const(goal.param_env, ct)?; |
| |
| let ct_ty = match ct.kind() { |
| ty::ConstKind::Infer(_) => { |
| return self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS); |
| } |
| ty::ConstKind::Error(_) => { |
| return self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes); |
| } |
| ty::ConstKind::Unevaluated(uv) => { |
| self.cx().type_of(uv.def).instantiate(self.cx(), uv.args) |
| } |
| ty::ConstKind::Expr(_) => unimplemented!( |
| "`feature(generic_const_exprs)` is not supported in the new trait solver" |
| ), |
| ty::ConstKind::Param(_) => { |
| unreachable!("`ConstKind::Param` should have been canonicalized to `Placeholder`") |
| } |
| ty::ConstKind::Bound(_, _) => panic!("escaping bound vars in {:?}", ct), |
| ty::ConstKind::Value(cv) => cv.ty(), |
| ty::ConstKind::Placeholder(placeholder) => { |
| placeholder.find_const_ty_from_env(goal.param_env) |
| } |
| }; |
| |
| self.eq(goal.param_env, ct_ty, ty)?; |
| self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes) |
| } |
| } |
| |
| #[derive(Debug)] |
| enum MergeCandidateInfo { |
| AlwaysApplicable(usize), |
| EqualResponse, |
| } |
| |
| impl<D, I> EvalCtxt<'_, D> |
| where |
| D: SolverDelegate<Interner = I>, |
| I: Interner, |
| { |
| /// 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 = "trace", skip(self), ret)] |
| fn try_merge_candidates( |
| &mut self, |
| candidates: &[Candidate<I>], |
| ) -> Option<(CanonicalResponse<I>, MergeCandidateInfo)> { |
| if candidates.is_empty() { |
| return None; |
| } |
| |
| let always_applicable = candidates.iter().enumerate().find(|(_, candidate)| { |
| candidate.result.value.certainty == Certainty::Yes |
| && has_no_inference_or_external_constraints(candidate.result) |
| }); |
| if let Some((i, c)) = always_applicable { |
| return Some((c.result, MergeCandidateInfo::AlwaysApplicable(i))); |
| } |
| |
| let one: CanonicalResponse<I> = candidates[0].result; |
| if candidates[1..].iter().all(|candidate| candidate.result == one) { |
| return Some((one, MergeCandidateInfo::EqualResponse)); |
| } |
| |
| None |
| } |
| |
| fn bail_with_ambiguity(&mut self, candidates: &[Candidate<I>]) -> CanonicalResponse<I> { |
| debug_assert!(candidates.len() > 1); |
| let maybe_cause = |
| candidates.iter().fold(MaybeCause::Ambiguity, |maybe_cause, candidates| { |
| // Pull down the certainty of `Certainty::Yes` to ambiguity when combining |
| // these responses, b/c we're combining more than one response and this we |
| // don't know which one applies. |
| let candidate = match candidates.result.value.certainty { |
| Certainty::Yes => MaybeCause::Ambiguity, |
| Certainty::Maybe(candidate) => candidate, |
| }; |
| maybe_cause.or(candidate) |
| }); |
| self.make_ambiguous_response_no_constraints(maybe_cause) |
| } |
| |
| /// If we fail to merge responses we flounder and return overflow or ambiguity. |
| #[instrument(level = "trace", skip(self), ret)] |
| fn flounder(&mut self, candidates: &[Candidate<I>]) -> QueryResult<I> { |
| if candidates.is_empty() { |
| return Err(NoSolution); |
| } else { |
| Ok(self.bail_with_ambiguity(candidates)) |
| } |
| } |
| |
| /// Normalize a type for when it is structurally matched on. |
| /// |
| /// This function is necessary in nearly all cases before matching on a type. |
| /// Not doing so is likely to be incomplete and therefore unsound during |
| /// coherence. |
| #[instrument(level = "trace", skip(self, param_env), ret)] |
| fn structurally_normalize_ty( |
| &mut self, |
| param_env: I::ParamEnv, |
| ty: I::Ty, |
| ) -> Result<I::Ty, NoSolution> { |
| self.structurally_normalize_term(param_env, ty.into()).map(|term| term.expect_ty()) |
| } |
| |
| /// Normalize a const for when it is structurally matched on, or more likely |
| /// when it needs `.try_to_*` called on it (e.g. to turn it into a usize). |
| /// |
| /// This function is necessary in nearly all cases before matching on a const. |
| /// Not doing so is likely to be incomplete and therefore unsound during |
| /// coherence. |
| #[instrument(level = "trace", skip(self, param_env), ret)] |
| fn structurally_normalize_const( |
| &mut self, |
| param_env: I::ParamEnv, |
| ct: I::Const, |
| ) -> Result<I::Const, NoSolution> { |
| self.structurally_normalize_term(param_env, ct.into()).map(|term| term.expect_const()) |
| } |
| |
| /// Normalize a term for when it is structurally matched on. |
| /// |
| /// This function is necessary in nearly all cases before matching on a ty/const. |
| /// Not doing so is likely to be incomplete and therefore unsound during coherence. |
| fn structurally_normalize_term( |
| &mut self, |
| param_env: I::ParamEnv, |
| term: I::Term, |
| ) -> Result<I::Term, NoSolution> { |
| if let Some(_) = term.to_alias_term() { |
| let normalized_term = self.next_term_infer_of_kind(term); |
| let alias_relate_goal = Goal::new( |
| self.cx(), |
| param_env, |
| ty::PredicateKind::AliasRelate( |
| term, |
| normalized_term, |
| ty::AliasRelationDirection::Equate, |
| ), |
| ); |
| // We normalize the self type to be able to relate it with |
| // types from candidates. |
| self.add_goal(GoalSource::TypeRelating, alias_relate_goal); |
| self.try_evaluate_added_goals()?; |
| Ok(self.resolve_vars_if_possible(normalized_term)) |
| } else { |
| Ok(term) |
| } |
| } |
| |
| fn opaque_type_is_rigid(&self, def_id: I::DefId) -> bool { |
| match self.typing_mode() { |
| // Opaques are never rigid outside of analysis mode. |
| TypingMode::Coherence | TypingMode::PostAnalysis => false, |
| // During analysis, opaques are rigid unless they may be defined by |
| // the current body. |
| TypingMode::Analysis { defining_opaque_types_and_generators: non_rigid_opaques } |
| | TypingMode::Borrowck { defining_opaque_types: non_rigid_opaques } |
| | TypingMode::PostBorrowckAnalysis { defined_opaque_types: non_rigid_opaques } => { |
| !def_id.as_local().is_some_and(|def_id| non_rigid_opaques.contains(&def_id)) |
| } |
| } |
| } |
| } |
| |
| fn response_no_constraints_raw<I: Interner>( |
| cx: I, |
| max_universe: ty::UniverseIndex, |
| variables: I::CanonicalVarKinds, |
| certainty: Certainty, |
| ) -> CanonicalResponse<I> { |
| ty::Canonical { |
| max_universe, |
| variables, |
| value: Response { |
| var_values: ty::CanonicalVarValues::make_identity(cx, variables), |
| // FIXME: maybe we should store the "no response" version in cx, like |
| // we do for cx.types and stuff. |
| external_constraints: cx.mk_external_constraints(ExternalConstraintsData::default()), |
| certainty, |
| }, |
| } |
| } |
| |
| /// The result of evaluating a goal. |
| pub struct GoalEvaluation<I: Interner> { |
| /// The goal we've evaluated. This is the input goal, but potentially with its |
| /// inference variables resolved. This never applies any inference constraints |
| /// from evaluating the goal. |
| /// |
| /// We rely on this to check whether root goals in HIR typeck had an unresolved |
| /// type inference variable in the input. We must not resolve this after evaluating |
| /// the goal as even if the inference variable has been resolved by evaluating the |
| /// goal itself, this goal may still end up failing due to region uniquification |
| /// later on. |
| /// |
| /// This is used as a minor optimization to avoid re-resolving inference variables |
| /// when reevaluating ambiguous goals. E.g. if we've got a goal `?x: Trait` with `?x` |
| /// already being constrained to `Vec<?y>`, then the first evaluation resolves it to |
| /// `Vec<?y>: Trait`. If this goal is still ambiguous and we later resolve `?y` to `u32`, |
| /// then reevaluating this goal now only needs to resolve `?y` while it would otherwise |
| /// have to resolve both `?x` and `?y`, |
| pub goal: Goal<I, I::Predicate>, |
| pub certainty: Certainty, |
| pub has_changed: HasChanged, |
| /// If the [`Certainty`] was `Maybe`, then keep track of whether the goal has changed |
| /// before rerunning it. |
| pub stalled_on: Option<GoalStalledOn<I>>, |
| } |
| |
| /// The conditions that must change for a goal to warrant |
| #[derive_where(Clone, Debug; I: Interner)] |
| pub struct GoalStalledOn<I: Interner> { |
| pub num_opaques: usize, |
| pub stalled_vars: Vec<I::GenericArg>, |
| /// The cause that will be returned on subsequent evaluations if this goal remains stalled. |
| pub stalled_cause: MaybeCause, |
| } |