Simplify some mir passes by using let chains
This commit is contained in:
parent
bf9a1c8a19
commit
47ebffabb8
@ -2,9 +2,8 @@
|
||||
|
||||
use crate::MirPass;
|
||||
use rustc_middle::mir::*;
|
||||
use rustc_middle::ty::GenericArgsRef;
|
||||
use rustc_middle::ty::{self, Ty, TyCtxt};
|
||||
use rustc_span::symbol::{sym, Symbol};
|
||||
use rustc_middle::ty::{self, TyCtxt};
|
||||
use rustc_span::symbol::sym;
|
||||
use rustc_target::abi::{FieldIdx, VariantIdx};
|
||||
|
||||
pub struct LowerIntrinsics;
|
||||
@ -16,12 +15,10 @@ impl<'tcx> MirPass<'tcx> for LowerIntrinsics {
|
||||
let terminator = block.terminator.as_mut().unwrap();
|
||||
if let TerminatorKind::Call { func, args, destination, target, .. } =
|
||||
&mut terminator.kind
|
||||
&& let ty::FnDef(def_id, generic_args) = *func.ty(local_decls, tcx).kind()
|
||||
&& tcx.is_intrinsic(def_id)
|
||||
{
|
||||
let func_ty = func.ty(local_decls, tcx);
|
||||
let Some((intrinsic_name, generic_args)) = resolve_rust_intrinsic(tcx, func_ty)
|
||||
else {
|
||||
continue;
|
||||
};
|
||||
let intrinsic_name = tcx.item_name(def_id);
|
||||
match intrinsic_name {
|
||||
sym::unreachable => {
|
||||
terminator.kind = TerminatorKind::Unreachable;
|
||||
@ -309,15 +306,3 @@ impl<'tcx> MirPass<'tcx> for LowerIntrinsics {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn resolve_rust_intrinsic<'tcx>(
|
||||
tcx: TyCtxt<'tcx>,
|
||||
func_ty: Ty<'tcx>,
|
||||
) -> Option<(Symbol, GenericArgsRef<'tcx>)> {
|
||||
if let ty::FnDef(def_id, args) = *func_ty.kind() {
|
||||
if tcx.is_intrinsic(def_id) {
|
||||
return Some((tcx.item_name(def_id), args));
|
||||
}
|
||||
}
|
||||
None
|
||||
}
|
||||
|
@ -34,67 +34,44 @@ pub fn lower_slice_len_calls<'tcx>(tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) {
|
||||
}
|
||||
}
|
||||
|
||||
struct SliceLenPatchInformation<'tcx> {
|
||||
add_statement: Statement<'tcx>,
|
||||
new_terminator_kind: TerminatorKind<'tcx>,
|
||||
}
|
||||
|
||||
fn lower_slice_len_call<'tcx>(
|
||||
tcx: TyCtxt<'tcx>,
|
||||
block: &mut BasicBlockData<'tcx>,
|
||||
local_decls: &IndexSlice<Local, LocalDecl<'tcx>>,
|
||||
slice_len_fn_item_def_id: DefId,
|
||||
) {
|
||||
let mut patch_found: Option<SliceLenPatchInformation<'_>> = None;
|
||||
|
||||
let terminator = block.terminator();
|
||||
match &terminator.kind {
|
||||
TerminatorKind::Call {
|
||||
func,
|
||||
args,
|
||||
destination,
|
||||
target: Some(bb),
|
||||
call_source: CallSource::Normal,
|
||||
..
|
||||
} => {
|
||||
// some heuristics for fast rejection
|
||||
if args.len() != 1 {
|
||||
return;
|
||||
}
|
||||
let Some(arg) = args[0].place() else { return };
|
||||
let func_ty = func.ty(local_decls, tcx);
|
||||
match func_ty.kind() {
|
||||
ty::FnDef(fn_def_id, _) if fn_def_id == &slice_len_fn_item_def_id => {
|
||||
// perform modifications
|
||||
// from something like `_5 = core::slice::<impl [u8]>::len(move _6) -> bb1`
|
||||
// into:
|
||||
// ```
|
||||
// _5 = Len(*_6)
|
||||
// goto bb1
|
||||
// ```
|
||||
if let TerminatorKind::Call {
|
||||
func,
|
||||
args,
|
||||
destination,
|
||||
target: Some(bb),
|
||||
call_source: CallSource::Normal,
|
||||
..
|
||||
} = &terminator.kind
|
||||
// some heuristics for fast rejection
|
||||
&& let [arg] = &args[..]
|
||||
&& let Some(arg) = arg.place()
|
||||
&& let ty::FnDef(fn_def_id, _) = func.ty(local_decls, tcx).kind()
|
||||
&& *fn_def_id == slice_len_fn_item_def_id
|
||||
{
|
||||
// perform modifications from something like:
|
||||
// _5 = core::slice::<impl [u8]>::len(move _6) -> bb1
|
||||
// into:
|
||||
// _5 = Len(*_6)
|
||||
// goto bb1
|
||||
|
||||
// make new RValue for Len
|
||||
let deref_arg = tcx.mk_place_deref(arg);
|
||||
let r_value = Rvalue::Len(deref_arg);
|
||||
let len_statement_kind =
|
||||
StatementKind::Assign(Box::new((*destination, r_value)));
|
||||
let add_statement =
|
||||
Statement { kind: len_statement_kind, source_info: terminator.source_info };
|
||||
// make new RValue for Len
|
||||
let deref_arg = tcx.mk_place_deref(arg);
|
||||
let r_value = Rvalue::Len(deref_arg);
|
||||
let len_statement_kind =
|
||||
StatementKind::Assign(Box::new((*destination, r_value)));
|
||||
let add_statement =
|
||||
Statement { kind: len_statement_kind, source_info: terminator.source_info };
|
||||
|
||||
// modify terminator into simple Goto
|
||||
let new_terminator_kind = TerminatorKind::Goto { target: *bb };
|
||||
// modify terminator into simple Goto
|
||||
let new_terminator_kind = TerminatorKind::Goto { target: *bb };
|
||||
|
||||
let patch = SliceLenPatchInformation { add_statement, new_terminator_kind };
|
||||
|
||||
patch_found = Some(patch);
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
|
||||
if let Some(SliceLenPatchInformation { add_statement, new_terminator_kind }) = patch_found {
|
||||
block.statements.push(add_statement);
|
||||
block.terminator_mut().kind = new_terminator_kind;
|
||||
}
|
||||
|
@ -30,22 +30,17 @@ fn get_switched_on_type<'tcx>(
|
||||
let terminator = block_data.terminator();
|
||||
|
||||
// Only bother checking blocks which terminate by switching on a local.
|
||||
if let Some(local) = get_discriminant_local(&terminator.kind) {
|
||||
let stmt_before_term = (!block_data.statements.is_empty())
|
||||
.then(|| &block_data.statements[block_data.statements.len() - 1].kind);
|
||||
|
||||
if let Some(StatementKind::Assign(box (l, Rvalue::Discriminant(place)))) = stmt_before_term
|
||||
{
|
||||
if l.as_local() == Some(local) {
|
||||
let ty = place.ty(body, tcx).ty;
|
||||
if ty.is_enum() {
|
||||
return Some(ty);
|
||||
}
|
||||
}
|
||||
}
|
||||
if let Some(local) = get_discriminant_local(&terminator.kind)
|
||||
&& let [.., stmt_before_term] = &block_data.statements[..]
|
||||
&& let StatementKind::Assign(box (l, Rvalue::Discriminant(place))) = stmt_before_term.kind
|
||||
&& l.as_local() == Some(local)
|
||||
&& let ty = place.ty(body, tcx).ty
|
||||
&& ty.is_enum()
|
||||
{
|
||||
Some(ty)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
|
||||
None
|
||||
}
|
||||
|
||||
fn variant_discriminants<'tcx>(
|
||||
|
Loading…
x
Reference in New Issue
Block a user