From 853ac8fd875d11a54ee445d209b5e5f78aa8c54f Mon Sep 17 00:00:00 2001 From: Camille GILLOT Date: Sun, 13 Aug 2023 09:18:35 +0000 Subject: [PATCH] Check dominance for the full edge. --- compiler/rustc_mir_transform/src/ssa.rs | 12 +++++- .../calls.multiple_edges.CopyProp.diff | 21 +++++++++ .../copy-prop/calls.nrvo.CopyProp.diff | 24 +++++++++++ tests/mir-opt/copy-prop/calls.rs | 43 +++++++++++++++++++ 4 files changed, 98 insertions(+), 2 deletions(-) create mode 100644 tests/mir-opt/copy-prop/calls.multiple_edges.CopyProp.diff create mode 100644 tests/mir-opt/copy-prop/calls.nrvo.CopyProp.diff create mode 100644 tests/mir-opt/copy-prop/calls.rs diff --git a/compiler/rustc_mir_transform/src/ssa.rs b/compiler/rustc_mir_transform/src/ssa.rs index badb0689db0cf..d632902671868 100644 --- a/compiler/rustc_mir_transform/src/ssa.rs +++ b/compiler/rustc_mir_transform/src/ssa.rs @@ -246,8 +246,16 @@ impl<'tcx> Visitor<'tcx> for SsaVisitor<'_, 'tcx> { Some(LocationExtended::Assign(loc)) } PlaceContext::MutatingUse(MutatingUseContext::Call | MutatingUseContext::Yield) => { - let target = self.body.basic_blocks[loc.block].terminator().successors().next(); - target.map(|target| LocationExtended::Terminator(loc.block, target)) + // The assignment happens on the `loc -> target` edge. We need to ensure that + // this *edge* dominates all uses of the local. This is true if + // `loc` dominates `target` *and* `target` dominates all uses. + if let Some(target) = self.body.basic_blocks[loc.block].terminator().successors().next() + && self.dominators.dominates(loc, Location { block: target, statement_index: 0 }) + { + Some(LocationExtended::Terminator(loc.block, target)) + } else { + None + } } _ => None, }; diff --git a/tests/mir-opt/copy-prop/calls.multiple_edges.CopyProp.diff b/tests/mir-opt/copy-prop/calls.multiple_edges.CopyProp.diff new file mode 100644 index 0000000000000..4d56a8b25d360 --- /dev/null +++ b/tests/mir-opt/copy-prop/calls.multiple_edges.CopyProp.diff @@ -0,0 +1,21 @@ +- // MIR for `multiple_edges` before CopyProp ++ // MIR for `multiple_edges` after CopyProp + + fn multiple_edges(_1: bool) -> u8 { + let mut _0: u8; + let mut _2: u8; + + bb0: { + switchInt(_1) -> [1: bb1, otherwise: bb2]; + } + + bb1: { + _2 = dummy(const 13_u8) -> [return: bb2, unwind continue]; + } + + bb2: { + _0 = _2; + return; + } + } + diff --git a/tests/mir-opt/copy-prop/calls.nrvo.CopyProp.diff b/tests/mir-opt/copy-prop/calls.nrvo.CopyProp.diff new file mode 100644 index 0000000000000..b5d56909b0ddb --- /dev/null +++ b/tests/mir-opt/copy-prop/calls.nrvo.CopyProp.diff @@ -0,0 +1,24 @@ +- // MIR for `nrvo` before CopyProp ++ // MIR for `nrvo` after CopyProp + + fn nrvo() -> u8 { + let mut _0: u8; + let _1: u8; + scope 1 { +- debug y => _1; ++ debug y => _0; + } + + bb0: { +- StorageLive(_1); +- _1 = dummy(const 5_u8) -> [return: bb1, unwind continue]; ++ _0 = dummy(const 5_u8) -> [return: bb1, unwind continue]; + } + + bb1: { +- _0 = _1; +- StorageDead(_1); + return; + } + } + diff --git a/tests/mir-opt/copy-prop/calls.rs b/tests/mir-opt/copy-prop/calls.rs new file mode 100644 index 0000000000000..c1f86f1f4128e --- /dev/null +++ b/tests/mir-opt/copy-prop/calls.rs @@ -0,0 +1,43 @@ +// Check that CopyProp does propagate return values of call terminators. +// unit-test: CopyProp +// needs-unwind + +#![feature(custom_mir, core_intrinsics)] +use std::intrinsics::mir::*; + +#[inline(never)] +fn dummy(x: u8) -> u8 { + x +} + +// EMIT_MIR calls.nrvo.CopyProp.diff +fn nrvo() -> u8 { + let y = dummy(5); // this should get NRVO + y +} + +// EMIT_MIR calls.multiple_edges.CopyProp.diff +#[custom_mir(dialect = "runtime", phase = "initial")] +fn multiple_edges(t: bool) -> u8 { + mir! { + let x: u8; + { + match t { true => bbt, _ => ret } + } + bbt = { + Call(x = dummy(13), ret) + } + ret = { + // `x` is not assigned on the `bb0 -> ret` edge, + // so should not be marked as SSA for merging with `_0`. + RET = x; + Return() + } + } +} + +fn main() { + // Make sure the function actually gets instantiated. + nrvo(); + multiple_edges(false); +}