mirror of
https://github.com/rust-lang/rust.git
synced 2025-05-05 22:48:02 +00:00
57 lines
2.3 KiB
Rust
57 lines
2.3 KiB
Rust
![]() |
//! 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. `<T as Foo>`
|
||
|
//! 2. We avoid trying to normalize predicates involving generic
|
||
|
//! parameters (e.g. `<T as Foo>::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);
|
||
|
}
|
||
|
}
|
||
|
}
|