rustc_next_trait_solver/canonical/
mod.rs1use std::iter;
13
14use canonicalizer::Canonicalizer;
15use rustc_index::IndexVec;
16use rustc_type_ir::inherent::*;
17use rustc_type_ir::relate::solver_relating::RelateExt;
18use rustc_type_ir::{
19 self as ty, Canonical, CanonicalVarKind, CanonicalVarValues, InferCtxtLike, Interner,
20 TypeFoldable,
21};
22use tracing::instrument;
23
24use crate::delegate::SolverDelegate;
25use crate::resolve::eager_resolve_vars;
26use crate::solve::{
27 CanonicalInput, CanonicalResponse, Certainty, ExternalConstraintsData, Goal,
28 NestedNormalizationGoals, QueryInput, Response, inspect,
29};
30
31pub mod canonicalizer;
32
33trait ResponseT<I: Interner> {
34 fn var_values(&self) -> CanonicalVarValues<I>;
35}
36
37impl<I: Interner> ResponseT<I> for Response<I> {
38 fn var_values(&self) -> CanonicalVarValues<I> {
39 self.var_values
40 }
41}
42
43impl<I: Interner, T> ResponseT<I> for inspect::State<I, T> {
44 fn var_values(&self) -> CanonicalVarValues<I> {
45 self.var_values
46 }
47}
48
49pub(super) fn canonicalize_goal<D, I>(
54 delegate: &D,
55 goal: Goal<I, I::Predicate>,
56 opaque_types: &[(ty::OpaqueTypeKey<I>, I::Ty)],
57) -> (Vec<I::GenericArg>, CanonicalInput<I, I::Predicate>)
58where
59 D: SolverDelegate<Interner = I>,
60 I: Interner,
61{
62 let (orig_values, canonical) = Canonicalizer::canonicalize_input(
63 delegate,
64 QueryInput {
65 goal,
66 predefined_opaques_in_body: delegate.cx().mk_predefined_opaques_in_body(opaque_types),
67 },
68 );
69 let query_input = ty::CanonicalQueryInput { canonical, typing_mode: delegate.typing_mode() };
70 (orig_values, query_input)
71}
72
73pub(super) fn canonicalize_response<D, I, T>(
74 delegate: &D,
75 max_input_universe: ty::UniverseIndex,
76 value: T,
77) -> ty::Canonical<I, T>
78where
79 D: SolverDelegate<Interner = I>,
80 I: Interner,
81 T: TypeFoldable<I>,
82{
83 Canonicalizer::canonicalize_response(delegate, max_input_universe, value)
84}
85
86pub(super) fn instantiate_and_apply_query_response<D, I>(
95 delegate: &D,
96 param_env: I::ParamEnv,
97 original_values: &[I::GenericArg],
98 response: CanonicalResponse<I>,
99 span: I::Span,
100) -> (NestedNormalizationGoals<I>, Certainty)
101where
102 D: SolverDelegate<Interner = I>,
103 I: Interner,
104{
105 let instantiation =
106 compute_query_response_instantiation_values(delegate, &original_values, &response, span);
107
108 let Response { var_values, external_constraints, certainty } =
109 delegate.instantiate_canonical(response, instantiation);
110
111 unify_query_var_values(delegate, param_env, &original_values, var_values, span);
112
113 let ExternalConstraintsData { region_constraints, opaque_types, normalization_nested_goals } =
114 &*external_constraints;
115
116 register_region_constraints(delegate, region_constraints, span);
117 register_new_opaque_types(delegate, opaque_types, span);
118
119 (normalization_nested_goals.clone(), certainty)
120}
121
122fn compute_query_response_instantiation_values<D, I, T>(
126 delegate: &D,
127 original_values: &[I::GenericArg],
128 response: &Canonical<I, T>,
129 span: I::Span,
130) -> CanonicalVarValues<I>
131where
132 D: SolverDelegate<Interner = I>,
133 I: Interner,
134 T: ResponseT<I>,
135{
136 let prev_universe = delegate.universe();
140 let universes_created_in_query = response.max_universe.index();
141 for _ in 0..universes_created_in_query {
142 delegate.create_next_universe();
143 }
144
145 let var_values = response.value.var_values();
146 assert_eq!(original_values.len(), var_values.len());
147
148 let mut opt_values = IndexVec::from_elem_n(None, response.var_kinds.len());
156 for (original_value, result_value) in iter::zip(original_values, var_values.var_values.iter()) {
157 match result_value.kind() {
158 ty::GenericArgKind::Type(t) => {
159 if let ty::Bound(index_kind, b) = t.kind()
164 && !matches!(
165 response.var_kinds.get(b.var().as_usize()).unwrap(),
166 CanonicalVarKind::Ty { .. }
167 )
168 {
169 assert!(matches!(index_kind, ty::BoundVarIndexKind::Canonical));
170 opt_values[b.var()] = Some(*original_value);
171 }
172 }
173 ty::GenericArgKind::Lifetime(r) => {
174 if let ty::ReBound(index_kind, br) = r.kind() {
175 assert!(matches!(index_kind, ty::BoundVarIndexKind::Canonical));
176 opt_values[br.var()] = Some(*original_value);
177 }
178 }
179 ty::GenericArgKind::Const(c) => {
180 if let ty::ConstKind::Bound(index_kind, bv) = c.kind() {
181 assert!(matches!(index_kind, ty::BoundVarIndexKind::Canonical));
182 opt_values[bv.var()] = Some(*original_value);
183 }
184 }
185 }
186 }
187 CanonicalVarValues::instantiate(delegate.cx(), response.var_kinds, |var_values, kind| {
188 if kind.universe() != ty::UniverseIndex::ROOT {
189 delegate.instantiate_canonical_var(kind, span, &var_values, |idx| {
193 prev_universe + idx.index()
194 })
195 } else if kind.is_existential() {
196 if let Some(v) = opt_values[ty::BoundVar::from_usize(var_values.len())] {
204 v
205 } else {
206 delegate.instantiate_canonical_var(kind, span, &var_values, |_| prev_universe)
207 }
208 } else {
209 original_values[kind.expect_placeholder_index()]
212 }
213 })
214}
215
216#[instrument(level = "trace", skip(delegate))]
229fn unify_query_var_values<D, I>(
230 delegate: &D,
231 param_env: I::ParamEnv,
232 original_values: &[I::GenericArg],
233 var_values: CanonicalVarValues<I>,
234 span: I::Span,
235) where
236 D: SolverDelegate<Interner = I>,
237 I: Interner,
238{
239 assert_eq!(original_values.len(), var_values.len());
240
241 for (&orig, response) in iter::zip(original_values, var_values.var_values.iter()) {
242 let goals =
243 delegate.eq_structurally_relating_aliases(param_env, orig, response, span).unwrap();
244 assert!(goals.is_empty());
245 }
246}
247
248fn register_region_constraints<D, I>(
249 delegate: &D,
250 outlives: &[ty::OutlivesPredicate<I, I::GenericArg>],
251 span: I::Span,
252) where
253 D: SolverDelegate<Interner = I>,
254 I: Interner,
255{
256 for &ty::OutlivesPredicate(lhs, rhs) in outlives {
257 match lhs.kind() {
258 ty::GenericArgKind::Lifetime(lhs) => delegate.sub_regions(rhs, lhs, span),
259 ty::GenericArgKind::Type(lhs) => delegate.register_ty_outlives(lhs, rhs, span),
260 ty::GenericArgKind::Const(_) => panic!("const outlives: {lhs:?}: {rhs:?}"),
261 }
262 }
263}
264
265fn register_new_opaque_types<D, I>(
266 delegate: &D,
267 opaque_types: &[(ty::OpaqueTypeKey<I>, I::Ty)],
268 span: I::Span,
269) where
270 D: SolverDelegate<Interner = I>,
271 I: Interner,
272{
273 for &(key, ty) in opaque_types {
274 let prev = delegate.register_hidden_type_in_storage(key, ty, span);
275 if let Some(prev) = prev {
283 delegate.add_duplicate_opaque_type(key, prev, span);
284 }
285 }
286}
287
288pub fn make_canonical_state<D, I, T>(
293 delegate: &D,
294 var_values: &[I::GenericArg],
295 max_input_universe: ty::UniverseIndex,
296 data: T,
297) -> inspect::CanonicalState<I, T>
298where
299 D: SolverDelegate<Interner = I>,
300 I: Interner,
301 T: TypeFoldable<I>,
302{
303 let var_values = CanonicalVarValues { var_values: delegate.cx().mk_args(var_values) };
304 let state = inspect::State { var_values, data };
305 let state = eager_resolve_vars(delegate, state);
306 Canonicalizer::canonicalize_response(delegate, max_input_universe, state)
307}
308
309pub fn instantiate_canonical_state<D, I, T>(
312 delegate: &D,
313 span: I::Span,
314 param_env: I::ParamEnv,
315 orig_values: &mut Vec<I::GenericArg>,
316 state: inspect::CanonicalState<I, T>,
317) -> T
318where
319 D: SolverDelegate<Interner = I>,
320 I: Interner,
321 T: TypeFoldable<I>,
322{
323 orig_values.extend(
326 state.value.var_values.var_values.as_slice()[orig_values.len()..]
327 .iter()
328 .map(|&arg| delegate.fresh_var_for_kind_with_span(arg, span)),
329 );
330
331 let instantiation =
332 compute_query_response_instantiation_values(delegate, orig_values, &state, span);
333
334 let inspect::State { var_values, data } = delegate.instantiate_canonical(state, instantiation);
335
336 unify_query_var_values(delegate, param_env, orig_values, var_values, span);
337 data
338}
339
340pub fn response_no_constraints_raw<I: Interner>(
341 cx: I,
342 max_universe: ty::UniverseIndex,
343 var_kinds: I::CanonicalVarKinds,
344 certainty: Certainty,
345) -> CanonicalResponse<I> {
346 ty::Canonical {
347 max_universe,
348 var_kinds,
349 value: Response {
350 var_values: ty::CanonicalVarValues::make_identity(cx, var_kinds),
351 external_constraints: cx.mk_external_constraints(ExternalConstraintsData::default()),
354 certainty,
355 },
356 }
357}