//! Check if it's even possible to satisfy the 'where' clauses //! for this item. //! //! It's possible to `#!feature(trivial_bounds)]` to write //! a function with impossible to satisfy clauses, e.g.: //! `fn foo() where String: Copy {}`. //! //! We don't usually need to worry about this kind of case, //! since we would get a compilation error if the user tried //! to call it. However, since we optimize even without any //! calls to the function, we need to make sure that it even //! makes sense to try to evaluate the body. //! //! If there are unsatisfiable where clauses, then all bets are //! off, and we just give up. //! //! We manually filter the predicates, skipping anything that's not //! "global". We are in a potentially generic context //! (e.g. we are evaluating a function without instantiating generic //! parameters, so this filtering serves two purposes: //! //! 1. We skip evaluating any predicates that we would //! never be able prove are unsatisfiable (e.g. `` //! 2. We avoid trying to normalize predicates involving generic //! parameters (e.g. `::MyItem`). This can confuse //! the normalization code (leading to cycle errors), since //! it's usually never invoked in this way. use rustc_middle::mir::{Body, START_BLOCK, TerminatorKind}; use rustc_middle::ty::{TyCtxt, TypeVisitableExt}; use rustc_trait_selection::traits; use tracing::trace; use crate::pass_manager::MirPass; pub(crate) struct ImpossiblePredicates; impl<'tcx> MirPass<'tcx> for ImpossiblePredicates { fn run_pass(&self, tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) { let predicates = tcx .predicates_of(body.source.def_id()) .predicates .iter() .filter_map(|(p, _)| if p.is_global() { Some(*p) } else { None }); if traits::impossible_predicates(tcx, traits::elaborate(tcx, predicates).collect()) { trace!("found unsatisfiable predicates for {:?}", body.source); // Clear the body to only contain a single `unreachable` statement. let bbs = body.basic_blocks.as_mut(); bbs.raw.truncate(1); bbs[START_BLOCK].statements.clear(); bbs[START_BLOCK].terminator_mut().kind = TerminatorKind::Unreachable; body.var_debug_info.clear(); body.local_decls.raw.truncate(body.arg_count + 1); } } fn is_required(&self) -> bool { true } }