Skip to content

Commit 71b3fbd

Browse files
authored
Rollup merge of #98930 - tmiasko:pub-basic-blocks, r=oli-obk
Make MIR basic blocks field public This makes it possible to mutably borrow different fields of the MIR body without resorting to methods like `basic_blocks_local_decls_mut_and_var_debug_info`. To preserve validity of control flow graph caches in the presence of modifications, a new struct `BasicBlocks` wraps together basic blocks and control flow graph caches. The `BasicBlocks` dereferences to `IndexVec<BasicBlock, BasicBlockData>`. On the other hand a mutable access requires explicit `as_mut()` call.
2 parents ade6d2c + 17adfeb commit 71b3fbd

32 files changed

+229
-230
lines changed

compiler/rustc_borrowck/src/diagnostics/conflict_errors.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1628,7 +1628,7 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
16281628
location: Location,
16291629
) -> impl Iterator<Item = Location> + Captures<'tcx> + 'a {
16301630
if location.statement_index == 0 {
1631-
let predecessors = body.predecessors()[location.block].to_vec();
1631+
let predecessors = body.basic_blocks.predecessors()[location.block].to_vec();
16321632
Either::Left(predecessors.into_iter().map(move |bb| body.terminator_loc(bb)))
16331633
} else {
16341634
Either::Right(std::iter::once(Location {

compiler/rustc_borrowck/src/invalidation.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ pub(super) fn generate_invalidates<'tcx>(
2626

2727
if let Some(all_facts) = all_facts {
2828
let _prof_timer = tcx.prof.generic_activity("polonius_fact_generation");
29-
let dominators = body.dominators();
29+
let dominators = body.basic_blocks.dominators();
3030
let mut ig = InvalidationGenerator {
3131
all_facts,
3232
borrow_set,

compiler/rustc_borrowck/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -334,7 +334,7 @@ fn do_mir_borrowck<'a, 'tcx>(
334334
};
335335
}
336336

337-
let dominators = body.dominators();
337+
let dominators = body.basic_blocks.dominators();
338338

339339
let mut mbcx = MirBorrowckCtxt {
340340
infcx,

compiler/rustc_borrowck/src/type_check/liveness/trace.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -258,7 +258,7 @@ impl<'me, 'typeck, 'flow, 'tcx> LivenessResults<'me, 'typeck, 'flow, 'tcx> {
258258

259259
let block = self.cx.elements.to_location(block_start).block;
260260
self.stack.extend(
261-
self.cx.body.predecessors()[block]
261+
self.cx.body.basic_blocks.predecessors()[block]
262262
.iter()
263263
.map(|&pred_bb| self.cx.body.terminator_loc(pred_bb))
264264
.map(|pred_loc| self.cx.elements.point_from_location(pred_loc)),
@@ -354,7 +354,7 @@ impl<'me, 'typeck, 'flow, 'tcx> LivenessResults<'me, 'typeck, 'flow, 'tcx> {
354354
}
355355

356356
let body = self.cx.body;
357-
for &pred_block in body.predecessors()[block].iter() {
357+
for &pred_block in body.basic_blocks.predecessors()[block].iter() {
358358
debug!("compute_drop_live_points_for_block: pred_block = {:?}", pred_block,);
359359

360360
// Check whether the variable is (at least partially)

compiler/rustc_codegen_ssa/src/mir/analyze.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ pub fn non_ssa_locals<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
1515
fx: &FunctionCx<'a, 'tcx, Bx>,
1616
) -> BitSet<mir::Local> {
1717
let mir = fx.mir;
18-
let dominators = mir.dominators();
18+
let dominators = mir.basic_blocks.dominators();
1919
let locals = mir
2020
.local_decls
2121
.iter()

compiler/rustc_const_eval/src/transform/promote_consts.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -856,7 +856,8 @@ impl<'a, 'tcx> Promoter<'a, 'tcx> {
856856
literal: ConstantKind::from_const(_const, tcx),
857857
}))
858858
};
859-
let (blocks, local_decls) = self.source.basic_blocks_and_local_decls_mut();
859+
let blocks = self.source.basic_blocks.as_mut();
860+
let local_decls = &mut self.source.local_decls;
860861
let loc = candidate.location;
861862
let statement = &mut blocks[loc.block].statements[loc.statement_index];
862863
match statement.kind {
@@ -865,7 +866,7 @@ impl<'a, 'tcx> Promoter<'a, 'tcx> {
865866
Rvalue::Ref(ref mut region, borrow_kind, ref mut place),
866867
)) => {
867868
// Use the underlying local for this (necessarily interior) borrow.
868-
let ty = local_decls.local_decls()[place.local].ty;
869+
let ty = local_decls[place.local].ty;
869870
let span = statement.source_info.span;
870871

871872
let ref_ty = tcx.mk_ref(
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,147 @@
1+
use crate::mir::graph_cyclic_cache::GraphIsCyclicCache;
2+
use crate::mir::predecessors::{PredecessorCache, Predecessors};
3+
use crate::mir::switch_sources::{SwitchSourceCache, SwitchSources};
4+
use crate::mir::traversal::PostorderCache;
5+
use crate::mir::{BasicBlock, BasicBlockData, Successors, START_BLOCK};
6+
7+
use rustc_data_structures::graph;
8+
use rustc_data_structures::graph::dominators::{dominators, Dominators};
9+
use rustc_index::vec::IndexVec;
10+
11+
#[derive(Clone, TyEncodable, TyDecodable, Debug, HashStable, TypeFoldable, TypeVisitable)]
12+
pub struct BasicBlocks<'tcx> {
13+
basic_blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>,
14+
predecessor_cache: PredecessorCache,
15+
switch_source_cache: SwitchSourceCache,
16+
is_cyclic: GraphIsCyclicCache,
17+
postorder_cache: PostorderCache,
18+
}
19+
20+
impl<'tcx> BasicBlocks<'tcx> {
21+
#[inline]
22+
pub fn new(basic_blocks: IndexVec<BasicBlock, BasicBlockData<'tcx>>) -> Self {
23+
BasicBlocks {
24+
basic_blocks,
25+
predecessor_cache: PredecessorCache::new(),
26+
switch_source_cache: SwitchSourceCache::new(),
27+
is_cyclic: GraphIsCyclicCache::new(),
28+
postorder_cache: PostorderCache::new(),
29+
}
30+
}
31+
32+
/// Returns true if control-flow graph contains a cycle reachable from the `START_BLOCK`.
33+
#[inline]
34+
pub fn is_cfg_cyclic(&self) -> bool {
35+
self.is_cyclic.is_cyclic(self)
36+
}
37+
38+
#[inline]
39+
pub fn dominators(&self) -> Dominators<BasicBlock> {
40+
dominators(&self)
41+
}
42+
43+
/// Returns predecessors for each basic block.
44+
#[inline]
45+
pub fn predecessors(&self) -> &Predecessors {
46+
self.predecessor_cache.compute(&self.basic_blocks)
47+
}
48+
49+
/// Returns basic blocks in a postorder.
50+
#[inline]
51+
pub fn postorder(&self) -> &[BasicBlock] {
52+
self.postorder_cache.compute(&self.basic_blocks)
53+
}
54+
55+
/// `switch_sources()[&(target, switch)]` returns a list of switch
56+
/// values that lead to a `target` block from a `switch` block.
57+
#[inline]
58+
pub fn switch_sources(&self) -> &SwitchSources {
59+
self.switch_source_cache.compute(&self.basic_blocks)
60+
}
61+
62+
/// Returns mutable reference to basic blocks. Invalidates CFG cache.
63+
#[inline]
64+
pub fn as_mut(&mut self) -> &mut IndexVec<BasicBlock, BasicBlockData<'tcx>> {
65+
self.invalidate_cfg_cache();
66+
&mut self.basic_blocks
67+
}
68+
69+
/// Get mutable access to basic blocks without invalidating the CFG cache.
70+
///
71+
/// By calling this method instead of e.g. [`BasicBlocks::as_mut`] you promise not to change
72+
/// the CFG. This means that
73+
///
74+
/// 1) The number of basic blocks remains unchanged
75+
/// 2) The set of successors of each terminator remains unchanged.
76+
/// 3) For each `TerminatorKind::SwitchInt`, the `targets` remains the same and the terminator
77+
/// kind is not changed.
78+
///
79+
/// If any of these conditions cannot be upheld, you should call [`BasicBlocks::invalidate_cfg_cache`].
80+
#[inline]
81+
pub fn as_mut_preserves_cfg(&mut self) -> &mut IndexVec<BasicBlock, BasicBlockData<'tcx>> {
82+
&mut self.basic_blocks
83+
}
84+
85+
/// Invalidates cached information about the CFG.
86+
///
87+
/// You will only ever need this if you have also called [`BasicBlocks::as_mut_preserves_cfg`].
88+
/// All other methods that allow you to mutate the basic blocks also call this method
89+
/// themselves, thereby avoiding any risk of accidentaly cache invalidation.
90+
pub fn invalidate_cfg_cache(&mut self) {
91+
self.predecessor_cache.invalidate();
92+
self.switch_source_cache.invalidate();
93+
self.is_cyclic.invalidate();
94+
self.postorder_cache.invalidate();
95+
}
96+
}
97+
98+
impl<'tcx> std::ops::Deref for BasicBlocks<'tcx> {
99+
type Target = IndexVec<BasicBlock, BasicBlockData<'tcx>>;
100+
101+
#[inline]
102+
fn deref(&self) -> &IndexVec<BasicBlock, BasicBlockData<'tcx>> {
103+
&self.basic_blocks
104+
}
105+
}
106+
107+
impl<'tcx> graph::DirectedGraph for BasicBlocks<'tcx> {
108+
type Node = BasicBlock;
109+
}
110+
111+
impl<'tcx> graph::WithNumNodes for BasicBlocks<'tcx> {
112+
#[inline]
113+
fn num_nodes(&self) -> usize {
114+
self.basic_blocks.len()
115+
}
116+
}
117+
118+
impl<'tcx> graph::WithStartNode for BasicBlocks<'tcx> {
119+
#[inline]
120+
fn start_node(&self) -> Self::Node {
121+
START_BLOCK
122+
}
123+
}
124+
125+
impl<'tcx> graph::WithSuccessors for BasicBlocks<'tcx> {
126+
#[inline]
127+
fn successors(&self, node: Self::Node) -> <Self as graph::GraphSuccessors<'_>>::Iter {
128+
self.basic_blocks[node].terminator().successors()
129+
}
130+
}
131+
132+
impl<'a, 'b> graph::GraphSuccessors<'b> for BasicBlocks<'a> {
133+
type Item = BasicBlock;
134+
type Iter = Successors<'b>;
135+
}
136+
137+
impl<'tcx, 'graph> graph::GraphPredecessors<'graph> for BasicBlocks<'tcx> {
138+
type Item = BasicBlock;
139+
type Iter = std::iter::Copied<std::slice::Iter<'graph, BasicBlock>>;
140+
}
141+
142+
impl<'tcx> graph::WithPredecessors for BasicBlocks<'tcx> {
143+
#[inline]
144+
fn predecessors(&self, node: Self::Node) -> <Self as graph::GraphPredecessors<'_>>::Iter {
145+
self.predecessors()[node].iter().copied()
146+
}
147+
}

0 commit comments

Comments
 (0)