rustc_next_trait_solver/solve/
effect_goals.rs1use rustc_type_ir::fast_reject::DeepRejectCtxt;
5use rustc_type_ir::inherent::*;
6use rustc_type_ir::lang_items::SolverTraitLangItem;
7use rustc_type_ir::solve::SizedTraitKind;
8use rustc_type_ir::solve::inspect::ProbeKind;
9use rustc_type_ir::{self as ty, Interner, TypingMode, elaborate};
10use tracing::instrument;
11
12use super::assembly::{Candidate, structural_traits};
13use crate::delegate::SolverDelegate;
14use crate::solve::{
15 BuiltinImplSource, CandidateSource, Certainty, EvalCtxt, Goal, GoalSource, NoSolution,
16 QueryResult, assembly,
17};
18
19impl<D, I> assembly::GoalKind<D> for ty::HostEffectPredicate<I>
20where
21 D: SolverDelegate<Interner = I>,
22 I: Interner,
23{
24 fn self_ty(self) -> I::Ty {
25 self.self_ty()
26 }
27
28 fn trait_ref(self, _: I) -> ty::TraitRef<I> {
29 self.trait_ref
30 }
31
32 fn with_replaced_self_ty(self, cx: I, self_ty: I::Ty) -> Self {
33 self.with_replaced_self_ty(cx, self_ty)
34 }
35
36 fn trait_def_id(self, _: I) -> I::TraitId {
37 self.def_id()
38 }
39
40 fn fast_reject_assumption(
41 ecx: &mut EvalCtxt<'_, D>,
42 goal: Goal<I, Self>,
43 assumption: I::Clause,
44 ) -> Result<(), NoSolution> {
45 if let Some(host_clause) = assumption.as_host_effect_clause()
46 && host_clause.def_id() == goal.predicate.def_id()
47 && host_clause.constness().satisfies(goal.predicate.constness)
48 && DeepRejectCtxt::relate_rigid_rigid(ecx.cx()).args_may_unify(
49 goal.predicate.trait_ref.args,
50 host_clause.skip_binder().trait_ref.args,
51 )
52 {
53 Ok(())
54 } else {
55 Err(NoSolution)
56 }
57 }
58
59 fn match_assumption(
60 ecx: &mut EvalCtxt<'_, D>,
61 goal: Goal<I, Self>,
62 assumption: I::Clause,
63 then: impl FnOnce(&mut EvalCtxt<'_, D>) -> QueryResult<I>,
64 ) -> QueryResult<I> {
65 let host_clause = assumption.as_host_effect_clause().unwrap();
66
67 let assumption_trait_pred = ecx.instantiate_binder_with_infer(host_clause);
68 ecx.eq(goal.param_env, goal.predicate.trait_ref, assumption_trait_pred.trait_ref)?;
69
70 then(ecx)
71 }
72
73 fn consider_additional_alias_assumptions(
80 ecx: &mut EvalCtxt<'_, D>,
81 goal: Goal<I, Self>,
82 alias_ty: ty::AliasTy<I>,
83 ) -> Vec<Candidate<I>> {
84 let cx = ecx.cx();
85 let mut candidates = vec![];
86
87 if !ecx.cx().alias_has_const_conditions(alias_ty.def_id) {
88 return vec![];
89 }
90
91 for clause in elaborate::elaborate(
92 cx,
93 cx.explicit_implied_const_bounds(alias_ty.def_id)
94 .iter_instantiated(cx, alias_ty.args)
95 .map(|trait_ref| trait_ref.to_host_effect_clause(cx, goal.predicate.constness)),
96 ) {
97 candidates.extend(Self::probe_and_match_goal_against_assumption(
98 ecx,
99 CandidateSource::AliasBound,
100 goal,
101 clause,
102 |ecx| {
103 ecx.add_goals(
105 GoalSource::AliasBoundConstCondition,
106 cx.const_conditions(alias_ty.def_id)
107 .iter_instantiated(cx, alias_ty.args)
108 .map(|trait_ref| {
109 goal.with(
110 cx,
111 trait_ref.to_host_effect_clause(cx, goal.predicate.constness),
112 )
113 }),
114 );
115 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
116 },
117 ));
118 }
119
120 candidates
121 }
122
123 fn consider_impl_candidate(
124 ecx: &mut EvalCtxt<'_, D>,
125 goal: Goal<I, Self>,
126 impl_def_id: I::ImplId,
127 then: impl FnOnce(&mut EvalCtxt<'_, D>, Certainty) -> QueryResult<I>,
128 ) -> Result<Candidate<I>, NoSolution> {
129 let cx = ecx.cx();
130
131 let impl_trait_ref = cx.impl_trait_ref(impl_def_id);
132 if !DeepRejectCtxt::relate_rigid_infer(ecx.cx())
133 .args_may_unify(goal.predicate.trait_ref.args, impl_trait_ref.skip_binder().args)
134 {
135 return Err(NoSolution);
136 }
137
138 let impl_polarity = cx.impl_polarity(impl_def_id);
139 let certainty = match impl_polarity {
140 ty::ImplPolarity::Negative => return Err(NoSolution),
141 ty::ImplPolarity::Reservation => match ecx.typing_mode() {
142 TypingMode::Coherence => Certainty::AMBIGUOUS,
143 TypingMode::Analysis { .. }
144 | TypingMode::Borrowck { .. }
145 | TypingMode::PostBorrowckAnalysis { .. }
146 | TypingMode::PostAnalysis => return Err(NoSolution),
147 },
148 ty::ImplPolarity::Positive => Certainty::Yes,
149 };
150
151 if !cx.impl_is_const(impl_def_id) {
152 return Err(NoSolution);
153 }
154
155 ecx.probe_trait_candidate(CandidateSource::Impl(impl_def_id)).enter(|ecx| {
156 let impl_args = ecx.fresh_args_for_item(impl_def_id.into());
157 ecx.record_impl_args(impl_args);
158 let impl_trait_ref = impl_trait_ref.instantiate(cx, impl_args);
159
160 ecx.eq(goal.param_env, goal.predicate.trait_ref, impl_trait_ref)?;
161 let where_clause_bounds = cx
162 .predicates_of(impl_def_id.into())
163 .iter_instantiated(cx, impl_args)
164 .map(|pred| goal.with(cx, pred));
165 ecx.add_goals(GoalSource::ImplWhereBound, where_clause_bounds);
166
167 let const_conditions = cx
169 .const_conditions(impl_def_id.into())
170 .iter_instantiated(cx, impl_args)
171 .map(|bound_trait_ref| {
172 goal.with(
173 cx,
174 bound_trait_ref.to_host_effect_clause(cx, goal.predicate.constness),
175 )
176 });
177 ecx.add_goals(GoalSource::ImplWhereBound, const_conditions);
178
179 then(ecx, certainty)
180 })
181 }
182
183 fn consider_error_guaranteed_candidate(
184 ecx: &mut EvalCtxt<'_, D>,
185 _guar: I::ErrorGuaranteed,
186 ) -> Result<Candidate<I>, NoSolution> {
187 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
188 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
189 }
190
191 fn consider_auto_trait_candidate(
192 _ecx: &mut EvalCtxt<'_, D>,
193 _goal: Goal<I, Self>,
194 ) -> Result<Candidate<I>, NoSolution> {
195 unreachable!("auto traits are never const")
196 }
197
198 fn consider_trait_alias_candidate(
199 _ecx: &mut EvalCtxt<'_, D>,
200 _goal: Goal<I, Self>,
201 ) -> Result<Candidate<I>, NoSolution> {
202 unreachable!("trait aliases are never const")
203 }
204
205 fn consider_builtin_sizedness_candidates(
206 _ecx: &mut EvalCtxt<'_, D>,
207 _goal: Goal<I, Self>,
208 _sizedness: SizedTraitKind,
209 ) -> Result<Candidate<I>, NoSolution> {
210 unreachable!("Sized/MetaSized is never const")
211 }
212
213 fn consider_builtin_copy_clone_candidate(
214 _ecx: &mut EvalCtxt<'_, D>,
215 _goal: Goal<I, Self>,
216 ) -> Result<Candidate<I>, NoSolution> {
217 Err(NoSolution)
218 }
219
220 fn consider_builtin_fn_ptr_trait_candidate(
221 _ecx: &mut EvalCtxt<'_, D>,
222 _goal: Goal<I, Self>,
223 ) -> Result<Candidate<I>, NoSolution> {
224 todo!("Fn* are not yet const")
225 }
226
227 fn consider_builtin_fn_trait_candidates(
228 ecx: &mut EvalCtxt<'_, D>,
229 goal: Goal<I, Self>,
230 _kind: rustc_type_ir::ClosureKind,
231 ) -> Result<Candidate<I>, NoSolution> {
232 let cx = ecx.cx();
233
234 let self_ty = goal.predicate.self_ty();
235 let (inputs_and_output, def_id, args) =
236 structural_traits::extract_fn_def_from_const_callable(cx, self_ty)?;
237
238 let output_is_sized_pred = inputs_and_output.map_bound(|(_, output)| {
241 ty::TraitRef::new(cx, cx.require_trait_lang_item(SolverTraitLangItem::Sized), [output])
242 });
243 let requirements = cx
244 .const_conditions(def_id.into())
245 .iter_instantiated(cx, args)
246 .map(|trait_ref| {
247 (
248 GoalSource::ImplWhereBound,
249 goal.with(cx, trait_ref.to_host_effect_clause(cx, goal.predicate.constness)),
250 )
251 })
252 .chain([(GoalSource::ImplWhereBound, goal.with(cx, output_is_sized_pred))]);
253
254 let pred = inputs_and_output
255 .map_bound(|(inputs, _)| {
256 ty::TraitRef::new(
257 cx,
258 goal.predicate.def_id(),
259 [goal.predicate.self_ty(), Ty::new_tup(cx, inputs.as_slice())],
260 )
261 })
262 .to_host_effect_clause(cx, goal.predicate.constness);
263
264 Self::probe_and_consider_implied_clause(
265 ecx,
266 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
267 goal,
268 pred,
269 requirements,
270 )
271 }
272
273 fn consider_builtin_async_fn_trait_candidates(
274 _ecx: &mut EvalCtxt<'_, D>,
275 _goal: Goal<I, Self>,
276 _kind: rustc_type_ir::ClosureKind,
277 ) -> Result<Candidate<I>, NoSolution> {
278 todo!("AsyncFn* are not yet const")
279 }
280
281 fn consider_builtin_async_fn_kind_helper_candidate(
282 _ecx: &mut EvalCtxt<'_, D>,
283 _goal: Goal<I, Self>,
284 ) -> Result<Candidate<I>, NoSolution> {
285 unreachable!("AsyncFnKindHelper is not const")
286 }
287
288 fn consider_builtin_tuple_candidate(
289 _ecx: &mut EvalCtxt<'_, D>,
290 _goal: Goal<I, Self>,
291 ) -> Result<Candidate<I>, NoSolution> {
292 unreachable!("Tuple trait is not const")
293 }
294
295 fn consider_builtin_pointee_candidate(
296 _ecx: &mut EvalCtxt<'_, D>,
297 _goal: Goal<I, Self>,
298 ) -> Result<Candidate<I>, NoSolution> {
299 unreachable!("Pointee is not const")
300 }
301
302 fn consider_builtin_future_candidate(
303 _ecx: &mut EvalCtxt<'_, D>,
304 _goal: Goal<I, Self>,
305 ) -> Result<Candidate<I>, NoSolution> {
306 unreachable!("Future is not const")
307 }
308
309 fn consider_builtin_iterator_candidate(
310 _ecx: &mut EvalCtxt<'_, D>,
311 _goal: Goal<I, Self>,
312 ) -> Result<Candidate<I>, NoSolution> {
313 todo!("Iterator is not yet const")
314 }
315
316 fn consider_builtin_fused_iterator_candidate(
317 _ecx: &mut EvalCtxt<'_, D>,
318 _goal: Goal<I, Self>,
319 ) -> Result<Candidate<I>, NoSolution> {
320 unreachable!("FusedIterator is not const")
321 }
322
323 fn consider_builtin_async_iterator_candidate(
324 _ecx: &mut EvalCtxt<'_, D>,
325 _goal: Goal<I, Self>,
326 ) -> Result<Candidate<I>, NoSolution> {
327 unreachable!("AsyncIterator is not const")
328 }
329
330 fn consider_builtin_coroutine_candidate(
331 _ecx: &mut EvalCtxt<'_, D>,
332 _goal: Goal<I, Self>,
333 ) -> Result<Candidate<I>, NoSolution> {
334 unreachable!("Coroutine is not const")
335 }
336
337 fn consider_builtin_discriminant_kind_candidate(
338 _ecx: &mut EvalCtxt<'_, D>,
339 _goal: Goal<I, Self>,
340 ) -> Result<Candidate<I>, NoSolution> {
341 unreachable!("DiscriminantKind is not const")
342 }
343
344 fn consider_builtin_destruct_candidate(
345 ecx: &mut EvalCtxt<'_, D>,
346 goal: Goal<I, Self>,
347 ) -> Result<Candidate<I>, NoSolution> {
348 let cx = ecx.cx();
349
350 let self_ty = goal.predicate.self_ty();
351 let const_conditions = structural_traits::const_conditions_for_destruct(cx, self_ty)?;
352
353 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
354 ecx.add_goals(
355 GoalSource::AliasBoundConstCondition,
356 const_conditions.into_iter().map(|trait_ref| {
357 goal.with(
358 cx,
359 ty::Binder::dummy(trait_ref)
360 .to_host_effect_clause(cx, goal.predicate.constness),
361 )
362 }),
363 );
364 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
365 })
366 }
367
368 fn consider_builtin_transmute_candidate(
369 _ecx: &mut EvalCtxt<'_, D>,
370 _goal: Goal<I, Self>,
371 ) -> Result<Candidate<I>, NoSolution> {
372 unreachable!("TransmuteFrom is not const")
373 }
374
375 fn consider_builtin_bikeshed_guaranteed_no_drop_candidate(
376 _ecx: &mut EvalCtxt<'_, D>,
377 _goal: Goal<I, Self>,
378 ) -> Result<Candidate<I>, NoSolution> {
379 unreachable!("BikeshedGuaranteedNoDrop is not const");
380 }
381
382 fn consider_structural_builtin_unsize_candidates(
383 _ecx: &mut EvalCtxt<'_, D>,
384 _goal: Goal<I, Self>,
385 ) -> Vec<Candidate<I>> {
386 unreachable!("Unsize is not const")
387 }
388}
389
390impl<D, I> EvalCtxt<'_, D>
391where
392 D: SolverDelegate<Interner = I>,
393 I: Interner,
394{
395 #[instrument(level = "trace", skip(self))]
396 pub(super) fn compute_host_effect_goal(
397 &mut self,
398 goal: Goal<I, ty::HostEffectPredicate<I>>,
399 ) -> QueryResult<I> {
400 let (_, proven_via) = self.probe(|_| ProbeKind::ShadowedEnvProbing).enter(|ecx| {
401 let trait_goal: Goal<I, ty::TraitPredicate<I>> =
402 goal.with(ecx.cx(), goal.predicate.trait_ref);
403 ecx.compute_trait_goal(trait_goal)
404 })?;
405 self.assemble_and_merge_candidates(proven_via, goal, |_ecx| Err(NoSolution))
406 }
407}