Skip to content

Commit abd4ad1

Browse files
committed
...
1 parent 093b9d5 commit abd4ad1

File tree

3 files changed

+34
-12
lines changed

3 files changed

+34
-12
lines changed

compiler/rustc_codegen_ssa/src/mir/analyze.rs

+13-6
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ pub fn non_ssa_locals<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
3636

3737
// Arguments get assigned to by means of the function being called
3838
for arg in mir.args_iter() {
39-
analyzer.assign(arg, DefLocation::Argument);
39+
analyzer.define(arg, DefLocation::Argument);
4040
}
4141

4242
// If there exists a local definition that dominates all uses of that local,
@@ -74,7 +74,7 @@ struct LocalAnalyzer<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> {
7474
}
7575

7676
impl<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> LocalAnalyzer<'mir, 'a, 'tcx, Bx> {
77-
fn assign(&mut self, local: mir::Local, location: DefLocation) {
77+
fn define(&mut self, local: mir::Local, location: DefLocation) {
7878
let kind = &mut self.locals[local];
7979
match *kind {
8080
LocalKind::ZST => {}
@@ -162,7 +162,7 @@ impl<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> Visitor<'tcx>
162162
debug!("visit_assign(place={:?}, rvalue={:?})", place, rvalue);
163163

164164
if let Some(local) = place.as_local() {
165-
self.assign(local, DefLocation::Body(location));
165+
self.define(local, DefLocation::Assignment(location));
166166
if self.locals[local] != LocalKind::Memory {
167167
let decl_span = self.fx.mir.local_decls[local].source_info.span;
168168
if !self.fx.rvalue_creates_operand(rvalue, decl_span) {
@@ -183,9 +183,14 @@ impl<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> Visitor<'tcx>
183183

184184
fn visit_local(&mut self, local: mir::Local, context: PlaceContext, location: Location) {
185185
match context {
186-
PlaceContext::MutatingUse(MutatingUseContext::Call)
187-
| PlaceContext::MutatingUse(MutatingUseContext::Yield) => {
188-
self.assign(local, DefLocation::Body(location));
186+
PlaceContext::MutatingUse(MutatingUseContext::Call) => {
187+
let call = location.block;
188+
let TerminatorKind::Call { target, .. } =
189+
self.fx.mir.basic_blocks[call].terminator().kind
190+
else {
191+
unreachable!()
192+
};
193+
self.define(local, DefLocation::CallReturn { call, target });
189194
}
190195

191196
PlaceContext::NonUse(_)
@@ -237,6 +242,8 @@ impl<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> Visitor<'tcx>
237242
}
238243
}
239244
}
245+
246+
PlaceContext::MutatingUse(MutatingUseContext::Yield) => unreachable!(),
240247
}
241248
}
242249
}

compiler/rustc_middle/src/mir/mod.rs

+17-2
Original file line numberDiff line numberDiff line change
@@ -1567,14 +1567,29 @@ impl Location {
15671567
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
15681568
pub enum DefLocation {
15691569
Argument,
1570-
Body(Location),
1570+
Assignment(Location),
1571+
CallReturn { call: BasicBlock, target: Option<BasicBlock> },
15711572
}
15721573

15731574
impl DefLocation {
15741575
pub fn dominates(self, location: Location, dominators: &Dominators<BasicBlock>) -> bool {
15751576
match self {
15761577
DefLocation::Argument => true,
1577-
DefLocation::Body(def) => def.successor_within_block().dominates(location, dominators),
1578+
DefLocation::Assignment(def) => {
1579+
def.successor_within_block().dominates(location, dominators)
1580+
}
1581+
DefLocation::CallReturn { target: None, .. } => false,
1582+
DefLocation::CallReturn { call, target: Some(target) } => {
1583+
// The definition occurs on the call -> target edge. The definition dominates a use
1584+
// if and only if the edge is on all paths from the entry to the use.
1585+
//
1586+
// Note that a call terminator has only one edge that can reach the target, so when
1587+
// the call strongly dominates the target, all paths from the entry to the target
1588+
// go through the call -> target edge.
1589+
call != target
1590+
&& dominators.dominates(call, target)
1591+
&& dominators.dominates(target, location.block)
1592+
}
15781593
}
15791594
}
15801595
}

compiler/rustc_mir_transform/src/ssa.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ impl SsaLocals {
104104
body: &'a Body<'tcx>,
105105
) -> impl Iterator<Item = (Local, &'a Rvalue<'tcx>, Location)> + 'a {
106106
self.assignment_order.iter().filter_map(|&local| {
107-
if let Set1::One(DefLocation::Body(loc)) = self.assignments[local] {
107+
if let Set1::One(DefLocation::Assignment(loc)) = self.assignments[local] {
108108
// `loc` must point to a direct assignment to `local`.
109109
let Either::Left(stmt) = body.stmt_at(loc) else { bug!() };
110110
let Some((target, rvalue)) = stmt.kind.as_assign() else { bug!() };
@@ -122,7 +122,7 @@ impl SsaLocals {
122122
mut f: impl FnMut(Local, &mut Rvalue<'tcx>, Location),
123123
) {
124124
for &local in &self.assignment_order {
125-
if let Set1::One(DefLocation::Body(loc)) = self.assignments[local] {
125+
if let Set1::One(DefLocation::Assignment(loc)) = self.assignments[local] {
126126
// `loc` must point to a direct assignment to `local`.
127127
let bbs = basic_blocks.as_mut_preserves_cfg();
128128
let bb = &mut bbs[loc.block];
@@ -246,7 +246,7 @@ impl<'tcx> Visitor<'tcx> for SsaVisitor<'_> {
246246

247247
fn visit_assign(&mut self, place: &Place<'tcx>, rvalue: &Rvalue<'tcx>, loc: Location) {
248248
if let Some(local) = place.as_local() {
249-
self.assignments[local].insert(DefLocation::Body(loc));
249+
self.assignments[local].insert(DefLocation::Assignment(loc));
250250
if let Set1::One(_) = self.assignments[local] {
251251
// Only record if SSA-like, to avoid growing the vector needlessly.
252252
self.assignment_order.push(local);
@@ -338,7 +338,7 @@ impl StorageLiveLocals {
338338
for (statement_index, statement) in bbdata.statements.iter().enumerate() {
339339
if let StatementKind::StorageLive(local) = statement.kind {
340340
storage_live[local]
341-
.insert(DefLocation::Body(Location { block, statement_index }));
341+
.insert(DefLocation::Assignment(Location { block, statement_index }));
342342
}
343343
}
344344
}

0 commit comments

Comments
 (0)