rustc_traits/
evaluate_obligation.rs1use rustc_infer::infer::TyCtxtInferExt;
2use rustc_middle::query::Providers;
3use rustc_middle::ty::{ParamEnvAnd, TyCtxt};
4use rustc_span::DUMMY_SP;
5use rustc_trait_selection::traits::query::CanonicalPredicateGoal;
6use rustc_trait_selection::traits::{
7 EvaluationResult, Obligation, ObligationCause, OverflowError, SelectionContext, TraitQueryMode,
8 sizedness_fast_path,
9};
10use tracing::debug;
11
12pub(crate) fn provide(p: &mut Providers) {
13 *p = Providers { evaluate_obligation, ..*p };
14}
15
16fn evaluate_obligation<'tcx>(
17 tcx: TyCtxt<'tcx>,
18 canonical_goal: CanonicalPredicateGoal<'tcx>,
19) -> Result<EvaluationResult, OverflowError> {
20 assert!(!tcx.next_trait_solver_globally());
21 debug!("evaluate_obligation(canonical_goal={:#?})", canonical_goal);
22 let (ref infcx, goal, _canonical_inference_vars) =
23 tcx.infer_ctxt().build_with_canonical(DUMMY_SP, &canonical_goal);
24 debug!("evaluate_obligation: goal={:#?}", goal);
25 let ParamEnvAnd { param_env, value: predicate } = goal;
26
27 if sizedness_fast_path(tcx, predicate) {
28 return Ok(EvaluationResult::EvaluatedToOk);
29 }
30
31 let mut selcx = SelectionContext::with_query_mode(infcx, TraitQueryMode::Canonical);
32 let obligation = Obligation::new(tcx, ObligationCause::dummy(), param_env, predicate);
33
34 selcx.evaluate_root_obligation(&obligation)
35}