Skip to content

Commit 828aaab

Browse files
committed
Auto merge of rust-lang#116454 - tmiasko:small-dominators, r=<try>
Generalize small dominators optimization * Use small dominators optimization from 640ede7 more generally. * Merge `DefLocation` and `LocationExtended` since they serve the same purpose.
2 parents cf9fd95 + 827dc2a commit 828aaab

File tree

5 files changed

+138
-109
lines changed

5 files changed

+138
-109
lines changed

compiler/rustc_codegen_ssa/src/mir/analyze.rs

+1-16
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use rustc_index::bit_set::BitSet;
88
use rustc_index::{IndexSlice, IndexVec};
99
use rustc_middle::mir::traversal;
1010
use rustc_middle::mir::visit::{MutatingUseContext, NonMutatingUseContext, PlaceContext, Visitor};
11-
use rustc_middle::mir::{self, Location, TerminatorKind};
11+
use rustc_middle::mir::{self, DefLocation, Location, TerminatorKind};
1212
use rustc_middle::ty::layout::{HasTyCtxt, LayoutOf};
1313

1414
pub fn non_ssa_locals<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
@@ -67,21 +67,6 @@ enum LocalKind {
6767
SSA(DefLocation),
6868
}
6969

70-
#[derive(Copy, Clone, PartialEq, Eq)]
71-
enum DefLocation {
72-
Argument,
73-
Body(Location),
74-
}
75-
76-
impl DefLocation {
77-
fn dominates(self, location: Location, dominators: &Dominators<mir::BasicBlock>) -> bool {
78-
match self {
79-
DefLocation::Argument => true,
80-
DefLocation::Body(def) => def.successor_within_block().dominates(location, dominators),
81-
}
82-
}
83-
}
84-
8570
struct LocalAnalyzer<'mir, 'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> {
8671
fx: &'mir FunctionCx<'a, 'tcx, Bx>,
8772
dominators: &'mir Dominators<mir::BasicBlock>,

compiler/rustc_data_structures/src/graph/dominators/mod.rs

+65-11
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,42 @@ rustc_index::newtype_index! {
2626
struct PreorderIndex {}
2727
}
2828

29-
pub fn dominators<G: ControlFlowGraph>(graph: &G) -> Dominators<G::Node> {
29+
#[derive(Clone, Debug)]
30+
pub struct Dominators<Node: Idx> {
31+
kind: Kind<Node>,
32+
}
33+
34+
#[derive(Clone, Debug)]
35+
enum Kind<Node: Idx> {
36+
/// A representation optimized for a small path graphs.
37+
Path(usize),
38+
General(Inner<Node>),
39+
}
40+
41+
pub fn dominators<G: ControlFlowGraph>(g: &G) -> Dominators<G::Node> {
42+
// We often encounter MIR bodies with 1 or 2 basic blocks. Special case the dominators
43+
// computation and representation for those cases.
44+
if is_small_path_graph(g) {
45+
Dominators { kind: Kind::Path(g.num_nodes()) }
46+
} else {
47+
Dominators { kind: Kind::General(dominators_impl(g)) }
48+
}
49+
}
50+
51+
fn is_small_path_graph<G: ControlFlowGraph>(g: &G) -> bool {
52+
if g.start_node().index() != 0 {
53+
return false;
54+
}
55+
if g.num_nodes() == 1 {
56+
return true;
57+
}
58+
if g.num_nodes() == 2 {
59+
return g.successors(g.start_node()).any(|n| n.index() == 1);
60+
}
61+
false
62+
}
63+
64+
fn dominators_impl<G: ControlFlowGraph>(graph: &G) -> Inner<G::Node> {
3065
// compute the post order index (rank) for each node
3166
let mut post_order_rank = IndexVec::from_elem_n(0, graph.num_nodes());
3267

@@ -245,7 +280,7 @@ pub fn dominators<G: ControlFlowGraph>(graph: &G) -> Dominators<G::Node> {
245280

246281
let time = compute_access_time(start_node, &immediate_dominators);
247282

248-
Dominators { start_node, post_order_rank, immediate_dominators, time }
283+
Inner { post_order_rank, immediate_dominators, time }
249284
}
250285

251286
/// Evaluate the link-eval virtual forest, providing the currently minimum semi
@@ -310,8 +345,7 @@ fn compress(
310345

311346
/// Tracks the list of dominators for each node.
312347
#[derive(Clone, Debug)]
313-
pub struct Dominators<N: Idx> {
314-
start_node: N,
348+
struct Inner<N: Idx> {
315349
post_order_rank: IndexVec<N, usize>,
316350
// Even though we track only the immediate dominator of each node, it's
317351
// possible to get its full list of dominators by looking up the dominator
@@ -323,12 +357,24 @@ pub struct Dominators<N: Idx> {
323357
impl<Node: Idx> Dominators<Node> {
324358
/// Returns true if node is reachable from the start node.
325359
pub fn is_reachable(&self, node: Node) -> bool {
326-
node == self.start_node || self.immediate_dominators[node].is_some()
360+
match &self.kind {
361+
Kind::Path(_) => true,
362+
Kind::General(g) => g.time[node].start != 0,
363+
}
327364
}
328365

329366
/// Returns the immediate dominator of node, if any.
330367
pub fn immediate_dominator(&self, node: Node) -> Option<Node> {
331-
self.immediate_dominators[node]
368+
match &self.kind {
369+
Kind::Path(n) => {
370+
if 0 < node.index() && node.index() < *n {
371+
Some(Node::new(node.index() - 1))
372+
} else {
373+
None
374+
}
375+
}
376+
Kind::General(g) => g.immediate_dominators[node],
377+
}
332378
}
333379

334380
/// Provides an iterator over each dominator up the CFG, for the given Node.
@@ -343,7 +389,10 @@ impl<Node: Idx> Dominators<Node> {
343389
/// of two unrelated nodes will also be consistent, but otherwise the order has no
344390
/// meaning.) This method cannot be used to determine if either Node dominates the other.
345391
pub fn cmp_in_dominator_order(&self, lhs: Node, rhs: Node) -> Ordering {
346-
self.post_order_rank[rhs].cmp(&self.post_order_rank[lhs])
392+
match &self.kind {
393+
Kind::Path(_) => lhs.index().cmp(&rhs.index()),
394+
Kind::General(g) => g.post_order_rank[rhs].cmp(&g.post_order_rank[lhs]),
395+
}
347396
}
348397

349398
/// Returns true if `a` dominates `b`.
@@ -352,10 +401,15 @@ impl<Node: Idx> Dominators<Node> {
352401
///
353402
/// Panics if `b` is unreachable.
354403
pub fn dominates(&self, a: Node, b: Node) -> bool {
355-
let a = self.time[a];
356-
let b = self.time[b];
357-
assert!(b.start != 0, "node {b:?} is not reachable");
358-
a.start <= b.start && b.finish <= a.finish
404+
match &self.kind {
405+
Kind::Path(_) => a.index() <= b.index(),
406+
Kind::General(g) => {
407+
let a = g.time[a];
408+
let b = g.time[b];
409+
assert!(b.start != 0, "node {b:?} is not reachable");
410+
a.start <= b.start && b.finish <= a.finish
411+
}
412+
}
359413
}
360414
}
361415

compiler/rustc_data_structures/src/graph/dominators/tests.rs

+21-24
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,11 @@ use super::super::tests::TestGraph;
66
fn diamond() {
77
let graph = TestGraph::new(0, &[(0, 1), (0, 2), (1, 3), (2, 3)]);
88

9-
let dominators = dominators(&graph);
10-
let immediate_dominators = &dominators.immediate_dominators;
11-
assert_eq!(immediate_dominators[0], None);
12-
assert_eq!(immediate_dominators[1], Some(0));
13-
assert_eq!(immediate_dominators[2], Some(0));
14-
assert_eq!(immediate_dominators[3], Some(0));
9+
let d = dominators(&graph);
10+
assert_eq!(d.immediate_dominator(0), None);
11+
assert_eq!(d.immediate_dominator(1), Some(0));
12+
assert_eq!(d.immediate_dominator(2), Some(0));
13+
assert_eq!(d.immediate_dominator(3), Some(0));
1514
}
1615

1716
#[test]
@@ -22,15 +21,14 @@ fn paper() {
2221
&[(6, 5), (6, 4), (5, 1), (4, 2), (4, 3), (1, 2), (2, 3), (3, 2), (2, 1)],
2322
);
2423

25-
let dominators = dominators(&graph);
26-
let immediate_dominators = &dominators.immediate_dominators;
27-
assert_eq!(immediate_dominators[0], None); // <-- note that 0 is not in graph
28-
assert_eq!(immediate_dominators[1], Some(6));
29-
assert_eq!(immediate_dominators[2], Some(6));
30-
assert_eq!(immediate_dominators[3], Some(6));
31-
assert_eq!(immediate_dominators[4], Some(6));
32-
assert_eq!(immediate_dominators[5], Some(6));
33-
assert_eq!(immediate_dominators[6], None);
24+
let d = dominators(&graph);
25+
assert_eq!(d.immediate_dominator(0), None); // <-- note that 0 is not in graph
26+
assert_eq!(d.immediate_dominator(1), Some(6));
27+
assert_eq!(d.immediate_dominator(2), Some(6));
28+
assert_eq!(d.immediate_dominator(3), Some(6));
29+
assert_eq!(d.immediate_dominator(4), Some(6));
30+
assert_eq!(d.immediate_dominator(5), Some(6));
31+
assert_eq!(d.immediate_dominator(6), None);
3432
}
3533

3634
#[test]
@@ -47,11 +45,11 @@ fn paper_slt() {
4745
#[test]
4846
fn immediate_dominator() {
4947
let graph = TestGraph::new(1, &[(1, 2), (2, 3)]);
50-
let dominators = dominators(&graph);
51-
assert_eq!(dominators.immediate_dominator(0), None);
52-
assert_eq!(dominators.immediate_dominator(1), None);
53-
assert_eq!(dominators.immediate_dominator(2), Some(1));
54-
assert_eq!(dominators.immediate_dominator(3), Some(2));
48+
let d = dominators(&graph);
49+
assert_eq!(d.immediate_dominator(0), None);
50+
assert_eq!(d.immediate_dominator(1), None);
51+
assert_eq!(d.immediate_dominator(2), Some(1));
52+
assert_eq!(d.immediate_dominator(3), Some(2));
5553
}
5654

5755
#[test]
@@ -75,8 +73,7 @@ fn transitive_dominator() {
7573
],
7674
);
7775

78-
let dom_tree = dominators(&graph);
79-
let immediate_dominators = &dom_tree.immediate_dominators;
80-
assert_eq!(immediate_dominators[2], Some(0));
81-
assert_eq!(immediate_dominators[3], Some(0)); // This used to return Some(1).
76+
let d = dominators(&graph);
77+
assert_eq!(d.immediate_dominator(2), Some(0));
78+
assert_eq!(d.immediate_dominator(3), Some(0)); // This used to return Some(1).
8279
}

compiler/rustc_middle/src/mir/mod.rs

+17
Original file line numberDiff line numberDiff line change
@@ -1590,6 +1590,23 @@ impl Location {
15901590
}
15911591
}
15921592

1593+
/// `DefLocation` represents the location of a definition - either an argument or an assignment
1594+
/// within MIR body.
1595+
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
1596+
pub enum DefLocation {
1597+
Argument,
1598+
Body(Location),
1599+
}
1600+
1601+
impl DefLocation {
1602+
pub fn dominates(self, location: Location, dominators: &Dominators<BasicBlock>) -> bool {
1603+
match self {
1604+
DefLocation::Argument => true,
1605+
DefLocation::Body(def) => def.successor_within_block().dominates(location, dominators),
1606+
}
1607+
}
1608+
}
1609+
15931610
// Some nodes are used a lot. Make sure they don't unintentionally get bigger.
15941611
#[cfg(all(target_arch = "x86_64", target_pointer_width = "64"))]
15951612
mod size_asserts {

0 commit comments

Comments
 (0)