Rollup merge of #120688 - cjgillot:gvn-partial-move, r=oli-obk
GVN: also turn moves into copies with projections Fixes https://github.com/rust-lang/rust/issues/120613
This commit is contained in:
commit
65aa9eae73
4 changed files with 78 additions and 5 deletions
|
@ -1231,8 +1231,8 @@ impl<'tcx> MutVisitor<'tcx> for StorageRemover<'tcx> {
|
|||
|
||||
fn visit_operand(&mut self, operand: &mut Operand<'tcx>, _: Location) {
|
||||
if let Operand::Move(place) = *operand
|
||||
&& let Some(local) = place.as_local()
|
||||
&& self.reused_locals.contains(local)
|
||||
&& !place.is_indirect_first_projection()
|
||||
&& self.reused_locals.contains(place.local)
|
||||
{
|
||||
*operand = Operand::Copy(place);
|
||||
}
|
||||
|
|
27
tests/mir-opt/gvn_copy_moves.fn0.GVN.diff
Normal file
27
tests/mir-opt/gvn_copy_moves.fn0.GVN.diff
Normal file
|
@ -0,0 +1,27 @@
|
|||
- // MIR for `fn0` before GVN
|
||||
+ // MIR for `fn0` after GVN
|
||||
|
||||
fn fn0() -> () {
|
||||
let mut _0: ();
|
||||
let mut _1: usize;
|
||||
let mut _2: [u128; 6];
|
||||
let mut _3: ([u128; 6],);
|
||||
let mut _4: ([u128; 6],);
|
||||
let mut _5: ();
|
||||
|
||||
bb0: {
|
||||
_1 = const 1_usize;
|
||||
_2 = [const 42_u128; 6];
|
||||
- _2[_1] = const 1_u128;
|
||||
+ _2[1 of 2] = const 1_u128;
|
||||
_3 = (_2,);
|
||||
_4 = _3;
|
||||
- _5 = fn1(move (_3.0: [u128; 6]), _4) -> [return: bb1, unwind unreachable];
|
||||
+ _5 = fn1((_3.0: [u128; 6]), _3) -> [return: bb1, unwind unreachable];
|
||||
}
|
||||
|
||||
bb1: {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
46
tests/mir-opt/gvn_copy_moves.rs
Normal file
46
tests/mir-opt/gvn_copy_moves.rs
Normal file
|
@ -0,0 +1,46 @@
|
|||
// unit-test: GVN
|
||||
|
||||
#![feature(custom_mir, core_intrinsics)]
|
||||
extern crate core;
|
||||
use core::intrinsics::mir::*;
|
||||
|
||||
#[custom_mir(dialect = "runtime", phase = "initial")]
|
||||
fn fn0() {
|
||||
// CHECK-LABEL: fn fn0(
|
||||
mir! {
|
||||
let a: usize;
|
||||
let b: [u128; 6];
|
||||
let c: ([u128; 6],);
|
||||
let d: ([u128; 6],);
|
||||
let x: ();
|
||||
{
|
||||
// CHECK: bb0: {
|
||||
// CHECK-NEXT: _1 = const 1_usize;
|
||||
// CHECK-NEXT: _2 = [const 42_u128; 6];
|
||||
// CHECK-NEXT: _2[1 of 2] = const 1_u128;
|
||||
// CHECK-NEXT: _3 = (_2,);
|
||||
// CHECK-NEXT: _4 = _3;
|
||||
// CHECK-NEXT: _5 = fn1((_3.0: [u128; 6]), _3)
|
||||
a = 1_usize;
|
||||
b = [42; 6];
|
||||
b[a] = 1;
|
||||
c = (b,);
|
||||
d = c;
|
||||
Call(x = fn1(Move(c.0), d), ReturnTo(bb1), UnwindUnreachable())
|
||||
}
|
||||
bb1 = {
|
||||
Return()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[inline(never)]
|
||||
fn fn1(a: [u128; 6], mut b: ([u128; 6],)) {
|
||||
b.0 = [0; 6];
|
||||
}
|
||||
|
||||
fn main() {
|
||||
fn0();
|
||||
}
|
||||
|
||||
// EMIT_MIR gvn_copy_moves.fn0.GVN.diff
|
|
@ -66,7 +66,7 @@
|
|||
_5 = ((_2 as Break).0: std::result::Result<std::convert::Infallible, i32>);
|
||||
StorageLive(_6);
|
||||
_6 = _5;
|
||||
_12 = move ((_5 as Err).0: i32);
|
||||
_12 = ((_5 as Err).0: i32);
|
||||
_0 = Result::<i32, i32>::Err(_12);
|
||||
StorageDead(_6);
|
||||
StorageDead(_2);
|
||||
|
@ -83,7 +83,7 @@
|
|||
}
|
||||
|
||||
bb4: {
|
||||
_10 = move ((_1 as Err).0: i32);
|
||||
_10 = ((_1 as Err).0: i32);
|
||||
StorageLive(_11);
|
||||
_11 = Result::<Infallible, i32>::Err(_10);
|
||||
_2 = ControlFlow::<Result<Infallible, i32>, i32>::Break(move _11);
|
||||
|
@ -92,7 +92,7 @@
|
|||
}
|
||||
|
||||
bb5: {
|
||||
_9 = move ((_1 as Ok).0: i32);
|
||||
_9 = ((_1 as Ok).0: i32);
|
||||
_2 = ControlFlow::<Result<Infallible, i32>, i32>::Continue(_9);
|
||||
goto -> bb3;
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue