rustc_next_trait_solver/solve/eval_ctxt/
probe.rs1use std::marker::PhantomData;
2
3use rustc_type_ir::search_graph::CandidateHeadUsages;
4use rustc_type_ir::solve::{
5 AccessedOpaques, CanonicalResponse, NoSolutionOrRerunNonErased, RerunResultExt,
6};
7use rustc_type_ir::{InferCtxtLike, Interner};
8use tracing::{instrument, warn};
9
10use crate::delegate::SolverDelegate;
11use crate::solve::assembly::Candidate;
12use crate::solve::{
13 BuiltinImplSource, CandidateSource, EvalCtxt, Goal, GoalSource, GoalStalledOn, NoSolution,
14 QueryResult, inspect,
15};
16
17pub(in crate::solve) struct ProbeCtxt<'me, 'a, D, I, F, T>
18where
19 D: SolverDelegate<Interner = I>,
20 I: Interner,
21{
22 ecx: &'me mut EvalCtxt<'a, D, I>,
23 probe_kind: F,
24 _result: PhantomData<T>,
25}
26
27impl<D, I, F, T> ProbeCtxt<'_, '_, D, I, F, T>
28where
29 F: FnOnce(&Result<T, NoSolution>) -> inspect::ProbeKind<I>,
30 D: SolverDelegate<Interner = I>,
31 I: Interner,
32{
33 pub(in crate::solve) fn enter_single_candidate(
34 self,
35 f: impl FnOnce(&mut EvalCtxt<'_, D>) -> Result<T, NoSolutionOrRerunNonErased>,
36 ) -> (Result<T, NoSolutionOrRerunNonErased>, CandidateHeadUsages) {
37 let mut candidate_usages = CandidateHeadUsages::default();
38
39 if let Err(e) = self.ecx.opaque_accesses.should_bail() {
40 return (Err(e.into()), candidate_usages);
41 }
42
43 self.ecx.search_graph.enter_single_candidate();
44 let result = self.enter(|ecx| {
45 let result = f(ecx);
46 candidate_usages = ecx.search_graph.finish_single_candidate();
47 result
48 });
49 (result, candidate_usages)
50 }
51
52 pub(in crate::solve) fn enter(
53 self,
54 f: impl FnOnce(&mut EvalCtxt<'_, D>) -> Result<T, NoSolutionOrRerunNonErased>,
55 ) -> Result<T, NoSolutionOrRerunNonErased> {
56 let nested_goals = self.ecx.nested_goals.clone();
57 self.enter_inner(f, nested_goals)
58 }
59
60 pub(in crate::solve) fn enter_without_propagated_nested_goals(
61 self,
62 f: impl FnOnce(&mut EvalCtxt<'_, D>) -> Result<T, NoSolutionOrRerunNonErased>,
63 ) -> Result<T, NoSolutionOrRerunNonErased> {
64 self.enter_inner(f, Default::default())
65 }
66
67 pub(in crate::solve) fn enter_inner(
68 self,
69 f: impl FnOnce(&mut EvalCtxt<'_, D>) -> Result<T, NoSolutionOrRerunNonErased>,
70 propagated_nested_goals: Vec<(GoalSource, Goal<I, I::Predicate>, Option<GoalStalledOn<I>>)>,
71 ) -> Result<T, NoSolutionOrRerunNonErased> {
72 let ProbeCtxt { ecx: outer, probe_kind, _result } = self;
73
74 outer.opaque_accesses.should_bail()?;
75
76 let delegate = outer.delegate;
77 let max_input_universe = outer.max_input_universe;
78 let mut nested = EvalCtxt {
79 delegate,
80 var_kinds: outer.var_kinds,
81 var_values: outer.var_values,
82 current_goal_kind: outer.current_goal_kind,
83 max_input_universe,
84 initial_opaque_types_storage_num_entries: outer
85 .initial_opaque_types_storage_num_entries,
86 search_graph: outer.search_graph,
87 nested_goals: propagated_nested_goals,
88 origin_span: outer.origin_span,
89 tainted: outer.tainted,
90 inspect: outer.inspect.take_and_enter_probe(),
91 opaque_accesses: AccessedOpaques::default(),
92 };
93 let r = nested.delegate.probe(|| {
94 let r = f(&mut nested);
95 nested.inspect.probe_final_state(delegate, max_input_universe);
96 r
97 });
98
99 outer.opaque_accesses.update(nested.opaque_accesses)?;
100
101 let r = match r.map_err_to_rerun()? {
102 Ok(i) => Ok(i),
103 Err(NoSolution) => Err(NoSolution),
104 };
105
106 if !nested.inspect.is_noop() {
107 let probe_kind = probe_kind(&r);
108 nested.inspect.probe_kind(probe_kind);
109 outer.inspect = nested.inspect.finish_probe();
110 }
111
112 r.map_err(Into::into)
113 }
114}
115
116pub(in crate::solve) struct TraitProbeCtxt<'me, 'a, D, I, F>
117where
118 D: SolverDelegate<Interner = I>,
119 I: Interner,
120{
121 cx: ProbeCtxt<'me, 'a, D, I, F, CanonicalResponse<I>>,
122 source: CandidateSource<I>,
123}
124
125impl<D, I, F> TraitProbeCtxt<'_, '_, D, I, F>
126where
127 D: SolverDelegate<Interner = I>,
128 I: Interner,
129 F: FnOnce(&QueryResult<I>) -> inspect::ProbeKind<I>,
130{
131 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("enter",
"rustc_next_trait_solver::solve::eval_ctxt::probe",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/eval_ctxt/probe.rs"),
::tracing_core::__macro_support::Option::Some(131u32),
::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::eval_ctxt::probe"),
::tracing_core::field::FieldSet::new(&["source"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&self.source)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return:
Result<Candidate<I>, NoSolutionOrRerunNonErased> = loop {};
return __tracing_attr_fake_return;
}
{
let (result, head_usages) = self.cx.enter_single_candidate(f);
Ok(Candidate {
source: self.source,
result: result?,
head_usages,
})
}
}
}#[instrument(level = "debug", skip_all, fields(source = ?self.source))]
132 pub(in crate::solve) fn enter(
133 self,
134 f: impl FnOnce(&mut EvalCtxt<'_, D>) -> Result<CanonicalResponse<I>, NoSolutionOrRerunNonErased>,
135 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
136 let (result, head_usages) = self.cx.enter_single_candidate(f);
137 Ok(Candidate { source: self.source, result: result?, head_usages })
138 }
139}
140
141impl<'a, D, I> EvalCtxt<'a, D, I>
142where
143 D: SolverDelegate<Interner = I>,
144 I: Interner,
145{
146 pub(in crate::solve) fn probe<F, T>(&mut self, probe_kind: F) -> ProbeCtxt<'_, 'a, D, I, F, T>
149 where
150 F: FnOnce(&Result<T, NoSolution>) -> inspect::ProbeKind<I>,
151 {
152 ProbeCtxt { ecx: self, probe_kind, _result: PhantomData }
153 }
154
155 pub(in crate::solve) fn probe_builtin_trait_candidate(
156 &mut self,
157 source: BuiltinImplSource,
158 ) -> TraitProbeCtxt<'_, 'a, D, I, impl FnOnce(&QueryResult<I>) -> inspect::ProbeKind<I>> {
159 self.probe_trait_candidate(CandidateSource::BuiltinImpl(source))
160 }
161
162 pub(in crate::solve) fn probe_trait_candidate(
163 &mut self,
164 source: CandidateSource<I>,
165 ) -> TraitProbeCtxt<'_, 'a, D, I, impl FnOnce(&QueryResult<I>) -> inspect::ProbeKind<I>> {
166 TraitProbeCtxt {
167 cx: ProbeCtxt {
168 ecx: self,
169 probe_kind: move |result: &QueryResult<I>| inspect::ProbeKind::TraitCandidate {
170 source,
171 result: *result,
172 },
173 _result: PhantomData,
174 },
175 source,
176 }
177 }
178}