Skip to content

Commit 76aca66

Browse files
committed
Auto merge of rust-lang#78454 - bugadani:cyclic, r=oli-obk
MIR Body: Cache result of `is_cyclic` call
2 parents aef92d4 + 119879c commit 76aca66

File tree

2 files changed

+73
-3
lines changed

2 files changed

+73
-3
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
use rustc_data_structures::graph::{
2+
self, DirectedGraph, WithNumNodes, WithStartNode, WithSuccessors,
3+
};
4+
use rustc_data_structures::stable_hasher::{HashStable, StableHasher};
5+
use rustc_data_structures::sync::OnceCell;
6+
use rustc_serialize as serialize;
7+
8+
/// Helper type to cache the result of `graph::is_cyclic`.
9+
#[derive(Clone, Debug)]
10+
pub(super) struct GraphIsCyclicCache {
11+
cache: OnceCell<bool>,
12+
}
13+
14+
impl GraphIsCyclicCache {
15+
#[inline]
16+
pub(super) fn new() -> Self {
17+
GraphIsCyclicCache { cache: OnceCell::new() }
18+
}
19+
20+
pub(super) fn is_cyclic<G>(&self, graph: &G) -> bool
21+
where
22+
G: ?Sized + DirectedGraph + WithStartNode + WithSuccessors + WithNumNodes,
23+
{
24+
*self.cache.get_or_init(|| graph::is_cyclic(graph))
25+
}
26+
27+
/// Invalidates the cache.
28+
#[inline]
29+
pub(super) fn invalidate(&mut self) {
30+
// Invalidating the cache requires mutating the MIR, which in turn requires a unique
31+
// reference (`&mut`) to the `mir::Body`. Because of this, we can assume that all
32+
// callers of `invalidate` have a unique reference to the MIR and thus to the
33+
// cache. This means we never need to do synchronization when `invalidate` is called,
34+
// we can simply reinitialize the `OnceCell`.
35+
self.cache = OnceCell::new();
36+
}
37+
}
38+
39+
impl<S: serialize::Encoder> serialize::Encodable<S> for GraphIsCyclicCache {
40+
#[inline]
41+
fn encode(&self, s: &mut S) -> Result<(), S::Error> {
42+
serialize::Encodable::encode(&(), s)
43+
}
44+
}
45+
46+
impl<D: serialize::Decoder> serialize::Decodable<D> for GraphIsCyclicCache {
47+
#[inline]
48+
fn decode(d: &mut D) -> Result<Self, D::Error> {
49+
serialize::Decodable::decode(d).map(|_v: ()| Self::new())
50+
}
51+
}
52+
53+
impl<CTX> HashStable<CTX> for GraphIsCyclicCache {
54+
#[inline]
55+
fn hash_stable(&self, _: &mut CTX, _: &mut StableHasher) {
56+
// do nothing
57+
}
58+
}
59+
60+
TrivialTypeFoldableAndLiftImpls! {
61+
GraphIsCyclicCache,
62+
}

compiler/rustc_middle/src/mir/mod.rs

+11-3
Original file line numberDiff line numberDiff line change
@@ -35,11 +35,13 @@ use std::ops::{ControlFlow, Index, IndexMut};
3535
use std::slice;
3636
use std::{iter, mem, option};
3737

38+
use self::graph_cyclic_cache::GraphIsCyclicCache;
3839
use self::predecessors::{PredecessorCache, Predecessors};
3940
pub use self::query::*;
4041

4142
pub mod abstract_const;
4243
pub mod coverage;
44+
mod graph_cyclic_cache;
4345
pub mod interpret;
4446
pub mod mono;
4547
mod predecessors;
@@ -227,6 +229,7 @@ pub struct Body<'tcx> {
227229
pub is_polymorphic: bool,
228230

229231
predecessor_cache: PredecessorCache,
232+
is_cyclic: GraphIsCyclicCache,
230233
}
231234

232235
impl<'tcx> Body<'tcx> {
@@ -267,6 +270,7 @@ impl<'tcx> Body<'tcx> {
267270
required_consts: Vec::new(),
268271
is_polymorphic: false,
269272
predecessor_cache: PredecessorCache::new(),
273+
is_cyclic: GraphIsCyclicCache::new(),
270274
};
271275
body.is_polymorphic = body.has_param_types_or_consts();
272276
body
@@ -296,6 +300,7 @@ impl<'tcx> Body<'tcx> {
296300
var_debug_info: Vec::new(),
297301
is_polymorphic: false,
298302
predecessor_cache: PredecessorCache::new(),
303+
is_cyclic: GraphIsCyclicCache::new(),
299304
};
300305
body.is_polymorphic = body.has_param_types_or_consts();
301306
body
@@ -309,11 +314,12 @@ impl<'tcx> Body<'tcx> {
309314
#[inline]
310315
pub fn basic_blocks_mut(&mut self) -> &mut IndexVec<BasicBlock, BasicBlockData<'tcx>> {
311316
// Because the user could mutate basic block terminators via this reference, we need to
312-
// invalidate the predecessor cache.
317+
// invalidate the caches.
313318
//
314319
// FIXME: Use a finer-grained API for this, so only transformations that alter terminators
315-
// invalidate the predecessor cache.
320+
// invalidate the caches.
316321
self.predecessor_cache.invalidate();
322+
self.is_cyclic.invalidate();
317323
&mut self.basic_blocks
318324
}
319325

@@ -322,6 +328,7 @@ impl<'tcx> Body<'tcx> {
322328
&mut self,
323329
) -> (&mut IndexVec<BasicBlock, BasicBlockData<'tcx>>, &mut LocalDecls<'tcx>) {
324330
self.predecessor_cache.invalidate();
331+
self.is_cyclic.invalidate();
325332
(&mut self.basic_blocks, &mut self.local_decls)
326333
}
327334

@@ -334,13 +341,14 @@ impl<'tcx> Body<'tcx> {
334341
&mut Vec<VarDebugInfo<'tcx>>,
335342
) {
336343
self.predecessor_cache.invalidate();
344+
self.is_cyclic.invalidate();
337345
(&mut self.basic_blocks, &mut self.local_decls, &mut self.var_debug_info)
338346
}
339347

340348
/// Returns `true` if a cycle exists in the control-flow graph that is reachable from the
341349
/// `START_BLOCK`.
342350
pub fn is_cfg_cyclic(&self) -> bool {
343-
graph::is_cyclic(self)
351+
self.is_cyclic.is_cyclic(self)
344352
}
345353

346354
#[inline]

0 commit comments

Comments
 (0)