- 
                Notifications
    You must be signed in to change notification settings 
- Fork 13.9k
          Add MIR pass to lower call to core::slice::len into Len operand
          #86383
        
          New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,100 @@ | ||
| //! This pass lowers calls to core::slice::len to just Len op. | ||
| //! It should run before inlining! | ||
|  | ||
| use crate::transform::MirPass; | ||
| use rustc_hir::def_id::DefId; | ||
| use rustc_index::vec::IndexVec; | ||
| use rustc_middle::mir::*; | ||
| use rustc_middle::ty::{self, TyCtxt}; | ||
|  | ||
| pub struct LowerSliceLenCalls; | ||
|  | ||
| impl<'tcx> MirPass<'tcx> for LowerSliceLenCalls { | ||
| fn run_pass(&self, tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) { | ||
| lower_slice_len_calls(tcx, body) | ||
| } | ||
| } | ||
|  | ||
| pub fn lower_slice_len_calls<'tcx>(tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) { | ||
| let language_items = tcx.lang_items(); | ||
| let slice_len_fn_item_def_id = if let Some(slice_len_fn_item) = language_items.slice_len_fn() { | ||
| slice_len_fn_item | ||
| } else { | ||
| // there is no language item to compare to :) | ||
| return; | ||
| }; | ||
|  | ||
| let (basic_blocks, local_decls) = body.basic_blocks_and_local_decls_mut(); | ||
|  | ||
| for block in basic_blocks { | ||
| // lower `<[_]>::len` calls | ||
| lower_slice_len_call(tcx, block, &*local_decls, slice_len_fn_item_def_id); | ||
| } | ||
| } | ||
|  | ||
| 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: &IndexVec<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: Some((dest, bb)), | ||
| cleanup: None, | ||
| from_hir_call: true, | ||
| .. | ||
| } => { | ||
| // some heuristics for fast rejection | ||
| if args.len() != 1 { | ||
| return; | ||
| } | ||
| let arg = match args[0].place() { | ||
| Some(arg) => arg, | ||
| None => 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 | ||
|  | ||
| // 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((*dest, r_value))); | ||
| let add_statement = Statement { | ||
| kind: len_statement_kind, | ||
| source_info: terminator.source_info.clone(), | ||
| }; | ||
|  | ||
| // modify terminator into simple Goto | ||
| let new_terminator_kind = TerminatorKind::Goto { target: bb.clone() }; | ||
|  | ||
| 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; | ||
| } | ||
| } | ||
| Original file line number | Diff line number | Diff line change | ||
|---|---|---|---|---|
|  | @@ -681,6 +681,7 @@ symbols! { | |||
| lateout, | ||||
| lazy_normalization_consts, | ||||
| le, | ||||
| len, | ||||
|          | ||||
| len, | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe it was a source of a performance gain in the first performance CI test: I was using a combination of slice_impl + len to find out a function, that is not optimal, but in some tests performance gains were substantial and I believe it may have helped. In any case, it's quite common expression and it may be good to intern in any case
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,63 @@ | ||
| - // MIR for `bound` before LowerSliceLenCalls | ||
| + // MIR for `bound` after LowerSliceLenCalls | ||
|  | ||
| fn bound(_1: usize, _2: &[u8]) -> u8 { | ||
| debug index => _1; // in scope 0 at $DIR/lower_slice_len.rs:4:14: 4:19 | ||
| debug slice => _2; // in scope 0 at $DIR/lower_slice_len.rs:4:28: 4:33 | ||
| let mut _0: u8; // return place in scope 0 at $DIR/lower_slice_len.rs:4:45: 4:47 | ||
| let mut _3: bool; // in scope 0 at $DIR/lower_slice_len.rs:5:8: 5:27 | ||
| let mut _4: usize; // in scope 0 at $DIR/lower_slice_len.rs:5:8: 5:13 | ||
| let mut _5: usize; // in scope 0 at $DIR/lower_slice_len.rs:5:16: 5:27 | ||
| let mut _6: &[u8]; // in scope 0 at $DIR/lower_slice_len.rs:5:16: 5:21 | ||
| let _7: usize; // in scope 0 at $DIR/lower_slice_len.rs:6:15: 6:20 | ||
| let mut _8: usize; // in scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
| let mut _9: bool; // in scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
|  | ||
| bb0: { | ||
| StorageLive(_3); // scope 0 at $DIR/lower_slice_len.rs:5:8: 5:27 | ||
| StorageLive(_4); // scope 0 at $DIR/lower_slice_len.rs:5:8: 5:13 | ||
| _4 = _1; // scope 0 at $DIR/lower_slice_len.rs:5:8: 5:13 | ||
| StorageLive(_5); // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:27 | ||
| StorageLive(_6); // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:21 | ||
| _6 = &(*_2); // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:21 | ||
| - _5 = core::slice::<impl [u8]>::len(move _6) -> bb1; // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:27 | ||
| - // mir::Constant | ||
| - // + span: $DIR/lower_slice_len.rs:5:22: 5:25 | ||
| - // + literal: Const { ty: for<'r> fn(&'r [u8]) -> usize {core::slice::<impl [u8]>::len}, val: Value(Scalar(<ZST>)) } | ||
| + _5 = Len((*_6)); // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:27 | ||
| + goto -> bb1; // scope 0 at $DIR/lower_slice_len.rs:5:16: 5:27 | ||
| } | ||
|  | ||
| bb1: { | ||
| StorageDead(_6); // scope 0 at $DIR/lower_slice_len.rs:5:26: 5:27 | ||
| _3 = Lt(move _4, move _5); // scope 0 at $DIR/lower_slice_len.rs:5:8: 5:27 | ||
| StorageDead(_5); // scope 0 at $DIR/lower_slice_len.rs:5:26: 5:27 | ||
| StorageDead(_4); // scope 0 at $DIR/lower_slice_len.rs:5:26: 5:27 | ||
| switchInt(move _3) -> [false: bb3, otherwise: bb2]; // scope 0 at $DIR/lower_slice_len.rs:5:5: 9:6 | ||
| } | ||
|  | ||
| bb2: { | ||
| StorageLive(_7); // scope 0 at $DIR/lower_slice_len.rs:6:15: 6:20 | ||
| _7 = _1; // scope 0 at $DIR/lower_slice_len.rs:6:15: 6:20 | ||
| _8 = Len((*_2)); // scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
| _9 = Lt(_7, _8); // scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
| assert(move _9, "index out of bounds: the length is {} but the index is {}", move _8, _7) -> bb4; // scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
| } | ||
|  | ||
| bb3: { | ||
| _0 = const 42_u8; // scope 0 at $DIR/lower_slice_len.rs:8:9: 8:11 | ||
| goto -> bb5; // scope 0 at $DIR/lower_slice_len.rs:5:5: 9:6 | ||
| } | ||
|  | ||
| bb4: { | ||
| _0 = (*_2)[_7]; // scope 0 at $DIR/lower_slice_len.rs:6:9: 6:21 | ||
| StorageDead(_7); // scope 0 at $DIR/lower_slice_len.rs:7:5: 7:6 | ||
| goto -> bb5; // scope 0 at $DIR/lower_slice_len.rs:5:5: 9:6 | ||
| } | ||
|  | ||
| bb5: { | ||
| StorageDead(_3); // scope 0 at $DIR/lower_slice_len.rs:9:5: 9:6 | ||
| return; // scope 0 at $DIR/lower_slice_len.rs:10:2: 10:2 | ||
| } | ||
| } | ||
|  | 
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,14 @@ | ||
| // compile-flags: -Z mir-opt-level=3 | ||
|  | ||
| // EMIT_MIR lower_slice_len.bound.LowerSliceLenCalls.diff | ||
| pub fn bound(index: usize, slice: &[u8]) -> u8 { | ||
| if index < slice.len() { | ||
| slice[index] | ||
| } else { | ||
| 42 | ||
| } | ||
| } | ||
|  | ||
| fn main() { | ||
| let _ = bound(1, &[1, 2, 3]); | ||
| } | 
Uh oh!
There was an error while loading. Please reload this page.