use rustc_hir::{self as hir, LangItem}; use rustc_infer::infer::{BoundRegionConversionTime, DefineOpaqueTypes}; use rustc_infer::traits::{ ImplDerivedHostCause, ImplSource, Obligation, ObligationCauseCode, PredicateObligation, }; use rustc_middle::span_bug; use rustc_middle::ty::fast_reject::DeepRejectCtxt; use rustc_middle::ty::{self, TypingMode}; use rustc_type_ir::elaborate::elaborate; use rustc_type_ir::solve::NoSolution; use thin_vec::{ThinVec, thin_vec}; use super::SelectionContext; use super::normalize::normalize_with_depth_to; pub type HostEffectObligation<'tcx> = Obligation<'tcx, ty::HostEffectPredicate<'tcx>>; pub enum EvaluationFailure { Ambiguous, NoSolution, } pub fn evaluate_host_effect_obligation<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { if matches!(selcx.infcx.typing_mode(), TypingMode::Coherence) { span_bug!( obligation.cause.span, "should not select host obligation in old solver in intercrate mode" ); } let ref obligation = selcx.infcx.resolve_vars_if_possible(obligation.clone()); // Force ambiguity for infer self ty. if obligation.predicate.self_ty().is_ty_var() { return Err(EvaluationFailure::Ambiguous); } match evaluate_host_effect_from_bounds(selcx, obligation) { Ok(result) => return Ok(result), Err(EvaluationFailure::Ambiguous) => return Err(EvaluationFailure::Ambiguous), Err(EvaluationFailure::NoSolution) => {} } match evaluate_host_effect_from_item_bounds(selcx, obligation) { Ok(result) => return Ok(result), Err(EvaluationFailure::Ambiguous) => return Err(EvaluationFailure::Ambiguous), Err(EvaluationFailure::NoSolution) => {} } match evaluate_host_effect_from_builtin_impls(selcx, obligation) { Ok(result) => return Ok(result), Err(EvaluationFailure::Ambiguous) => return Err(EvaluationFailure::Ambiguous), Err(EvaluationFailure::NoSolution) => {} } match evaluate_host_effect_from_selection_candiate(selcx, obligation) { Ok(result) => return Ok(result), Err(EvaluationFailure::Ambiguous) => return Err(EvaluationFailure::Ambiguous), Err(EvaluationFailure::NoSolution) => {} } Err(EvaluationFailure::NoSolution) } fn match_candidate<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, candidate: ty::Binder<'tcx, ty::HostEffectPredicate<'tcx>>, candidate_is_unnormalized: bool, more_nested: impl FnOnce(&mut SelectionContext<'_, 'tcx>, &mut ThinVec>), ) -> Result>, NoSolution> { if !candidate.skip_binder().constness.satisfies(obligation.predicate.constness) { return Err(NoSolution); } let mut candidate = selcx.infcx.instantiate_binder_with_fresh_vars( obligation.cause.span, BoundRegionConversionTime::HigherRankedType, candidate, ); let mut nested = thin_vec![]; // Unlike param-env bounds, item bounds may not be normalized. if candidate_is_unnormalized { candidate = normalize_with_depth_to( selcx, obligation.param_env, obligation.cause.clone(), obligation.recursion_depth, candidate, &mut nested, ); } nested.extend( selcx .infcx .at(&obligation.cause, obligation.param_env) .eq(DefineOpaqueTypes::Yes, obligation.predicate.trait_ref, candidate.trait_ref)? .into_obligations(), ); more_nested(selcx, &mut nested); for nested in &mut nested { nested.set_depth_from_parent(obligation.recursion_depth); } Ok(nested) } fn evaluate_host_effect_from_bounds<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { let infcx = selcx.infcx; let drcx = DeepRejectCtxt::relate_rigid_rigid(selcx.tcx()); let mut candidate = None; for clause in obligation.param_env.caller_bounds() { let bound_clause = clause.kind(); let ty::ClauseKind::HostEffect(data) = bound_clause.skip_binder() else { continue; }; let data = bound_clause.rebind(data); if data.skip_binder().trait_ref.def_id != obligation.predicate.trait_ref.def_id { continue; } if !drcx .args_may_unify(obligation.predicate.trait_ref.args, data.skip_binder().trait_ref.args) { continue; } let is_match = infcx.probe(|_| match_candidate(selcx, obligation, data, false, |_, _| {}).is_ok()); if is_match { if candidate.is_some() { return Err(EvaluationFailure::Ambiguous); } else { candidate = Some(data); } } } if let Some(data) = candidate { Ok(match_candidate(selcx, obligation, data, false, |_, _| {}) .expect("candidate matched before, so it should match again")) } else { Err(EvaluationFailure::NoSolution) } } fn evaluate_host_effect_from_item_bounds<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { let infcx = selcx.infcx; let tcx = infcx.tcx; let drcx = DeepRejectCtxt::relate_rigid_rigid(selcx.tcx()); let mut candidate = None; let mut consider_ty = obligation.predicate.self_ty(); while let ty::Alias(kind @ (ty::Projection | ty::Opaque), alias_ty) = *consider_ty.kind() { if tcx.is_conditionally_const(alias_ty.def_id) { for clause in elaborate( tcx, tcx.explicit_implied_const_bounds(alias_ty.def_id) .iter_instantiated_copied(tcx, alias_ty.args) .map(|(trait_ref, _)| { trait_ref.to_host_effect_clause(tcx, obligation.predicate.constness) }), ) { let bound_clause = clause.kind(); let ty::ClauseKind::HostEffect(data) = bound_clause.skip_binder() else { unreachable!("should not elaborate non-HostEffect from HostEffect") }; let data = bound_clause.rebind(data); if data.skip_binder().trait_ref.def_id != obligation.predicate.trait_ref.def_id { continue; } if !drcx.args_may_unify( obligation.predicate.trait_ref.args, data.skip_binder().trait_ref.args, ) { continue; } let is_match = infcx .probe(|_| match_candidate(selcx, obligation, data, true, |_, _| {}).is_ok()); if is_match { if candidate.is_some() { return Err(EvaluationFailure::Ambiguous); } else { candidate = Some((data, alias_ty)); } } } } if kind != ty::Projection { break; } consider_ty = alias_ty.self_ty(); } if let Some((data, alias_ty)) = candidate { Ok(match_candidate(selcx, obligation, data, true, |selcx, nested| { // An alias bound only holds if we also check the const conditions // of the alias, so we need to register those, too. let const_conditions = normalize_with_depth_to( selcx, obligation.param_env, obligation.cause.clone(), obligation.recursion_depth, tcx.const_conditions(alias_ty.def_id).instantiate(tcx, alias_ty.args), nested, ); nested.extend(const_conditions.into_iter().map(|(trait_ref, _)| { obligation .with(tcx, trait_ref.to_host_effect_clause(tcx, obligation.predicate.constness)) })); }) .expect("candidate matched before, so it should match again")) } else { Err(EvaluationFailure::NoSolution) } } fn evaluate_host_effect_from_builtin_impls<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { match selcx.tcx().as_lang_item(obligation.predicate.def_id()) { Some(LangItem::Destruct) => evaluate_host_effect_for_destruct_goal(selcx, obligation), _ => Err(EvaluationFailure::NoSolution), } } // NOTE: Keep this in sync with `const_conditions_for_destruct` in the new solver. fn evaluate_host_effect_for_destruct_goal<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { let tcx = selcx.tcx(); let destruct_def_id = tcx.require_lang_item(LangItem::Destruct, None); let self_ty = obligation.predicate.self_ty(); let const_conditions = match *self_ty.kind() { // An ADT is `~const Destruct` only if all of the fields are, // *and* if there is a `Drop` impl, that `Drop` impl is also `~const`. ty::Adt(adt_def, args) => { let mut const_conditions: ThinVec<_> = adt_def .all_fields() .map(|field| ty::TraitRef::new(tcx, destruct_def_id, [field.ty(tcx, args)])) .collect(); match adt_def.destructor(tcx).map(|dtor| dtor.constness) { // `Drop` impl exists, but it's not const. Type cannot be `~const Destruct`. Some(hir::Constness::NotConst) => return Err(EvaluationFailure::NoSolution), // `Drop` impl exists, and it's const. Require `Ty: ~const Drop` to hold. Some(hir::Constness::Const) => { let drop_def_id = tcx.require_lang_item(LangItem::Drop, None); let drop_trait_ref = ty::TraitRef::new(tcx, drop_def_id, [self_ty]); const_conditions.push(drop_trait_ref); } // No `Drop` impl, no need to require anything else. None => {} } const_conditions } ty::Array(ty, _) | ty::Pat(ty, _) | ty::Slice(ty) => { thin_vec![ty::TraitRef::new(tcx, destruct_def_id, [ty])] } ty::Tuple(tys) => { tys.iter().map(|field_ty| ty::TraitRef::new(tcx, destruct_def_id, [field_ty])).collect() } // Trivially implement `~const Destruct` ty::Bool | ty::Char | ty::Int(..) | ty::Uint(..) | ty::Float(..) | ty::Str | ty::RawPtr(..) | ty::Ref(..) | ty::FnDef(..) | ty::FnPtr(..) | ty::Never | ty::Infer(ty::InferTy::FloatVar(_) | ty::InferTy::IntVar(_)) | ty::Error(_) => thin_vec![], // Coroutines and closures could implement `~const Drop`, // but they don't really need to right now. ty::Closure(_, _) | ty::CoroutineClosure(_, _) | ty::Coroutine(_, _) | ty::CoroutineWitness(_, _) => return Err(EvaluationFailure::NoSolution), // FIXME(unsafe_binders): Unsafe binders could implement `~const Drop` // if their inner type implements it. ty::UnsafeBinder(_) => return Err(EvaluationFailure::NoSolution), ty::Dynamic(..) | ty::Param(_) | ty::Alias(..) | ty::Placeholder(_) | ty::Foreign(_) => { return Err(EvaluationFailure::NoSolution); } ty::Bound(..) | ty::Infer(ty::TyVar(_) | ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_)) => { panic!("unexpected type `{self_ty:?}`") } }; Ok(const_conditions .into_iter() .map(|trait_ref| { obligation.with( tcx, ty::Binder::dummy(trait_ref) .to_host_effect_clause(tcx, obligation.predicate.constness), ) }) .collect()) } fn evaluate_host_effect_from_selection_candiate<'tcx>( selcx: &mut SelectionContext<'_, 'tcx>, obligation: &HostEffectObligation<'tcx>, ) -> Result>, EvaluationFailure> { let tcx = selcx.tcx(); selcx.infcx.commit_if_ok(|_| { match selcx.select(&obligation.with(tcx, obligation.predicate.trait_ref)) { Ok(None) => Err(EvaluationFailure::Ambiguous), Err(_) => Err(EvaluationFailure::NoSolution), Ok(Some(source)) => match source { ImplSource::UserDefined(impl_) => { if tcx.impl_trait_header(impl_.impl_def_id).unwrap().constness != hir::Constness::Const { return Err(EvaluationFailure::NoSolution); } let mut nested = impl_.nested; nested.extend( tcx.const_conditions(impl_.impl_def_id) .instantiate(tcx, impl_.args) .into_iter() .map(|(trait_ref, span)| { Obligation::new( tcx, obligation.cause.clone().derived_host_cause( ty::Binder::dummy(obligation.predicate), |derived| { ObligationCauseCode::ImplDerivedHost(Box::new( ImplDerivedHostCause { derived, impl_def_id: impl_.impl_def_id, span, }, )) }, ), obligation.param_env, trait_ref .to_host_effect_clause(tcx, obligation.predicate.constness), ) }), ); for nested in &mut nested { nested.set_depth_from_parent(obligation.recursion_depth); } Ok(nested) } _ => Err(EvaluationFailure::NoSolution), }, } }) }