//! 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, TyVid, TypingMode}; use tracing::instrument; pub use self::eval_ctxt::{ EvalCtxt, GenerateProofTree, SolverDelegateEvalExt, evaluate_root_goal_for_proof_tree_raw_provider, }; 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; /// 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( response: ty::Canonical>, ) -> 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(response: ty::Canonical>) -> 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, I: Interner, { #[instrument(level = "trace", skip(self))] fn compute_type_outlives_goal( &mut self, goal: Goal>, ) -> QueryResult { 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>, ) -> QueryResult { 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>) -> QueryResult { 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>) -> QueryResult { match (goal.predicate.a.kind(), goal.predicate.b.kind()) { (ty::Infer(ty::TyVar(a_vid)), ty::Infer(ty::TyVar(b_vid))) => { self.sub_unify_ty_vids_raw(a_vid, b_vid); self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS) } _ => { 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 { 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) -> QueryResult { 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: ::ParamEnv, symbol: ::Symbol, ) -> QueryResult { 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 { cause: MaybeCause::Ambiguity, opaque_types_jank: OpaqueTypesJank::AllGood, }) } } #[instrument(level = "trace", skip(self))] fn compute_const_evaluatable_goal( &mut self, Goal { param_env, predicate: ct }: Goal, ) -> QueryResult { 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, ) -> QueryResult { 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 EvalCtxt<'_, D> where D: SolverDelegate, 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], ) -> Option<(CanonicalResponse, 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 = 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]) -> CanonicalResponse { debug_assert!(candidates.len() > 1); let (cause, opaque_types_jank) = candidates.iter().fold( (MaybeCause::Ambiguity, OpaqueTypesJank::AllGood), |(c, jank), candidates| { // We 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. match candidates.result.value.certainty { Certainty::Yes => (c, jank), Certainty::Maybe { cause, opaque_types_jank } => { (c.or(cause), jank.or(opaque_types_jank)) } } }, ); self.make_ambiguous_response_no_constraints(cause, opaque_types_jank) } /// 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]) -> QueryResult { 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 { 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 { 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 { 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( cx: I, max_universe: ty::UniverseIndex, variables: I::CanonicalVarKinds, certainty: Certainty, ) -> CanonicalResponse { 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 { /// 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`, then the first evaluation resolves it to /// `Vec: 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, 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>, } /// The conditions that must change for a goal to warrant #[derive_where(Clone, Debug; I: Interner)] pub struct GoalStalledOn { pub num_opaques: usize, pub stalled_vars: Vec, pub sub_roots: Vec, /// The certainty that will be returned on subsequent evaluations if this /// goal remains stalled. pub stalled_certainty: Certainty, }