Skip to content

Commit eec6bd9

Browse files
committed
Auto merge of #146516 - cjgillot:dest-prop-aggregate, r=Amanieu
DestinationPropagation: avoid creating overlapping assignments. r? `@Amanieu` Fixes #146383
2 parents 8a1b399 + d58061e commit eec6bd9

6 files changed

+164
-12
lines changed

compiler/rustc_mir_transform/src/dest_prop.rs

Lines changed: 31 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -567,13 +567,15 @@ fn save_as_intervals<'tcx>(
567567
// the written-to locals as live in the second half of the statement.
568568
// We also ensure that operands read by terminators conflict with writes by that terminator.
569569
// For instance a function call may read args after having written to the destination.
570-
VisitPlacesWith(|place, ctxt| match DefUse::for_place(place, ctxt) {
571-
DefUse::Def | DefUse::Use | DefUse::PartialWrite => {
572-
if let Some(relevant) = relevant.shrink[place.local] {
573-
values.insert(relevant, twostep);
570+
VisitPlacesWith(|place: Place<'tcx>, ctxt| {
571+
if let Some(relevant) = relevant.shrink[place.local] {
572+
match DefUse::for_place(place, ctxt) {
573+
DefUse::Def | DefUse::Use | DefUse::PartialWrite => {
574+
values.insert(relevant, twostep);
575+
}
576+
DefUse::NonUse => {}
574577
}
575578
}
576-
DefUse::NonUse => {}
577579
})
578580
.visit_terminator(term, loc);
579581

@@ -588,15 +590,32 @@ fn save_as_intervals<'tcx>(
588590
twostep = TwoStepIndex::from_u32(twostep.as_u32() + 1);
589591
debug_assert_eq!(twostep, two_step_loc(loc, Effect::After));
590592
append_at(&mut values, &state, twostep);
591-
// Ensure we have a non-zero live range even for dead stores. This is done by marking
592-
// all the written-to locals as live in the second half of the statement.
593-
VisitPlacesWith(|place, ctxt| match DefUse::for_place(place, ctxt) {
594-
DefUse::Def | DefUse::PartialWrite => {
595-
if let Some(relevant) = relevant.shrink[place.local] {
596-
values.insert(relevant, twostep);
593+
// Like terminators, ensure we have a non-zero live range even for dead stores.
594+
// Some rvalues interleave reads and writes, for instance `Rvalue::Aggregate`, see
595+
// https://github.com/rust-lang/rust/issues/146383. By precaution, treat statements
596+
// as behaving so by default.
597+
// We make an exception for simple assignments `_a.stuff = {copy|move} _b.stuff`,
598+
// as marking `_b` live here would prevent unification.
599+
let is_simple_assignment = match stmt.kind {
600+
StatementKind::Assign(box (
601+
lhs,
602+
Rvalue::CopyForDeref(rhs)
603+
| Rvalue::Use(Operand::Copy(rhs) | Operand::Move(rhs)),
604+
)) => lhs.projection == rhs.projection,
605+
_ => false,
606+
};
607+
VisitPlacesWith(|place: Place<'tcx>, ctxt| {
608+
if let Some(relevant) = relevant.shrink[place.local] {
609+
match DefUse::for_place(place, ctxt) {
610+
DefUse::Def | DefUse::PartialWrite => {
611+
values.insert(relevant, twostep);
612+
}
613+
DefUse::Use if !is_simple_assignment => {
614+
values.insert(relevant, twostep);
615+
}
616+
DefUse::Use | DefUse::NonUse => {}
597617
}
598618
}
599-
DefUse::Use | DefUse::NonUse => {}
600619
})
601620
.visit_statement(stmt, loc);
602621

Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
- // MIR for `rewrap` before DestinationPropagation
2+
+ // MIR for `rewrap` after DestinationPropagation
3+
4+
fn rewrap() -> (u8,) {
5+
let mut _0: (u8,);
6+
let mut _1: (u8,);
7+
let mut _2: (u8,);
8+
9+
bb0: {
10+
- (_1.0: u8) = const 0_u8;
11+
- _0 = copy _1;
12+
+ (_0.0: u8) = const 0_u8;
13+
+ nop;
14+
_2 = (copy (_0.0: u8),);
15+
_0 = copy _2;
16+
return;
17+
}
18+
}
19+
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
- // MIR for `rewrap` before DestinationPropagation
2+
+ // MIR for `rewrap` after DestinationPropagation
3+
4+
fn rewrap() -> (u8,) {
5+
let mut _0: (u8,);
6+
let mut _1: (u8,);
7+
let mut _2: (u8,);
8+
9+
bb0: {
10+
- (_1.0: u8) = const 0_u8;
11+
- _0 = copy _1;
12+
+ (_0.0: u8) = const 0_u8;
13+
+ nop;
14+
_2 = (copy (_0.0: u8),);
15+
_0 = copy _2;
16+
return;
17+
}
18+
}
19+
Lines changed: 51 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
//@ test-mir-pass: DestinationPropagation
2+
// EMIT_MIR_FOR_EACH_PANIC_STRATEGY
3+
4+
#![feature(custom_mir, core_intrinsics)]
5+
#![allow(internal_features)]
6+
7+
use std::intrinsics::mir::*;
8+
use std::mem::MaybeUninit;
9+
10+
fn dump_var<T>(_: T) {}
11+
12+
// EMIT_MIR aggregate.rewrap.DestinationPropagation.diff
13+
#[custom_mir(dialect = "runtime")]
14+
fn rewrap() -> (u8,) {
15+
// CHECK-LABEL: fn rewrap(
16+
// CHECK: (_0.0: u8) = const 0_u8;
17+
// CHECK: _2 = (copy (_0.0: u8),);
18+
// CHECK: _0 = copy _2;
19+
mir! {
20+
let _1: (u8,);
21+
let _2: (u8,);
22+
{
23+
_1.0 = 0;
24+
RET = _1;
25+
_2 = (RET.0, );
26+
RET = _2;
27+
Return()
28+
}
29+
}
30+
}
31+
32+
// EMIT_MIR aggregate.swap.DestinationPropagation.diff
33+
#[custom_mir(dialect = "runtime")]
34+
fn swap() -> (MaybeUninit<[u8; 10]>, MaybeUninit<[u8; 10]>) {
35+
// CHECK-LABEL: fn swap(
36+
// CHECK: _0 = const
37+
// CHECK: _2 = copy _0;
38+
// CHECK: _0 = (copy (_2.1: {{.*}}), copy (_2.0: {{.*}}));
39+
mir! {
40+
let _1: (MaybeUninit<[u8; 10]>, MaybeUninit<[u8; 10]>);
41+
let _2: (MaybeUninit<[u8; 10]>, MaybeUninit<[u8; 10]>);
42+
let _3: ();
43+
{
44+
_1 = const { (MaybeUninit::new([0; 10]), MaybeUninit::new([1; 10])) };
45+
_2 = _1;
46+
_1 = (_2.1, _2.0);
47+
RET = _1;
48+
Return()
49+
}
50+
}
51+
}
Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
- // MIR for `swap` before DestinationPropagation
2+
+ // MIR for `swap` after DestinationPropagation
3+
4+
fn swap() -> (MaybeUninit<[u8; 10]>, MaybeUninit<[u8; 10]>) {
5+
let mut _0: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
6+
let mut _1: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
7+
let mut _2: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
8+
let mut _3: ();
9+
10+
bb0: {
11+
- _1 = const swap::{constant#6};
12+
- _2 = copy _1;
13+
- _1 = (copy (_2.1: std::mem::MaybeUninit<[u8; 10]>), copy (_2.0: std::mem::MaybeUninit<[u8; 10]>));
14+
- _0 = copy _1;
15+
+ _0 = const swap::{constant#6};
16+
+ _2 = copy _0;
17+
+ _0 = (copy (_2.1: std::mem::MaybeUninit<[u8; 10]>), copy (_2.0: std::mem::MaybeUninit<[u8; 10]>));
18+
+ nop;
19+
return;
20+
}
21+
}
22+
Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
- // MIR for `swap` before DestinationPropagation
2+
+ // MIR for `swap` after DestinationPropagation
3+
4+
fn swap() -> (MaybeUninit<[u8; 10]>, MaybeUninit<[u8; 10]>) {
5+
let mut _0: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
6+
let mut _1: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
7+
let mut _2: (std::mem::MaybeUninit<[u8; 10]>, std::mem::MaybeUninit<[u8; 10]>);
8+
let mut _3: ();
9+
10+
bb0: {
11+
- _1 = const swap::{constant#6};
12+
- _2 = copy _1;
13+
- _1 = (copy (_2.1: std::mem::MaybeUninit<[u8; 10]>), copy (_2.0: std::mem::MaybeUninit<[u8; 10]>));
14+
- _0 = copy _1;
15+
+ _0 = const swap::{constant#6};
16+
+ _2 = copy _0;
17+
+ _0 = (copy (_2.1: std::mem::MaybeUninit<[u8; 10]>), copy (_2.0: std::mem::MaybeUninit<[u8; 10]>));
18+
+ nop;
19+
return;
20+
}
21+
}
22+

0 commit comments

Comments
 (0)