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:
Matthias Krüger 2024-02-08 09:06:34 +01:00 committed by GitHub
commit 65aa9eae73
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 78 additions and 5 deletions

View file

@ -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);
}

View 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;
}
}

View 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

View file

@ -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;
}