@@ -35,11 +35,13 @@ use std::ops::{ControlFlow, Index, IndexMut};
35
35
use std:: slice;
36
36
use std:: { iter, mem, option} ;
37
37
38
+ use self :: graph_cyclic_cache:: GraphIsCyclicCache ;
38
39
use self :: predecessors:: { PredecessorCache , Predecessors } ;
39
40
pub use self :: query:: * ;
40
41
41
42
pub mod abstract_const;
42
43
pub mod coverage;
44
+ mod graph_cyclic_cache;
43
45
pub mod interpret;
44
46
pub mod mono;
45
47
mod predecessors;
@@ -227,6 +229,7 @@ pub struct Body<'tcx> {
227
229
pub is_polymorphic : bool ,
228
230
229
231
predecessor_cache : PredecessorCache ,
232
+ is_cyclic : GraphIsCyclicCache ,
230
233
}
231
234
232
235
impl < ' tcx > Body < ' tcx > {
@@ -267,6 +270,7 @@ impl<'tcx> Body<'tcx> {
267
270
required_consts : Vec :: new ( ) ,
268
271
is_polymorphic : false ,
269
272
predecessor_cache : PredecessorCache :: new ( ) ,
273
+ is_cyclic : GraphIsCyclicCache :: new ( ) ,
270
274
} ;
271
275
body. is_polymorphic = body. has_param_types_or_consts ( ) ;
272
276
body
@@ -296,6 +300,7 @@ impl<'tcx> Body<'tcx> {
296
300
var_debug_info : Vec :: new ( ) ,
297
301
is_polymorphic : false ,
298
302
predecessor_cache : PredecessorCache :: new ( ) ,
303
+ is_cyclic : GraphIsCyclicCache :: new ( ) ,
299
304
} ;
300
305
body. is_polymorphic = body. has_param_types_or_consts ( ) ;
301
306
body
@@ -309,11 +314,12 @@ impl<'tcx> Body<'tcx> {
309
314
#[ inline]
310
315
pub fn basic_blocks_mut ( & mut self ) -> & mut IndexVec < BasicBlock , BasicBlockData < ' tcx > > {
311
316
// Because the user could mutate basic block terminators via this reference, we need to
312
- // invalidate the predecessor cache .
317
+ // invalidate the caches .
313
318
//
314
319
// FIXME: Use a finer-grained API for this, so only transformations that alter terminators
315
- // invalidate the predecessor cache .
320
+ // invalidate the caches .
316
321
self . predecessor_cache . invalidate ( ) ;
322
+ self . is_cyclic . invalidate ( ) ;
317
323
& mut self . basic_blocks
318
324
}
319
325
@@ -322,6 +328,7 @@ impl<'tcx> Body<'tcx> {
322
328
& mut self ,
323
329
) -> ( & mut IndexVec < BasicBlock , BasicBlockData < ' tcx > > , & mut LocalDecls < ' tcx > ) {
324
330
self . predecessor_cache . invalidate ( ) ;
331
+ self . is_cyclic . invalidate ( ) ;
325
332
( & mut self . basic_blocks , & mut self . local_decls )
326
333
}
327
334
@@ -334,13 +341,14 @@ impl<'tcx> Body<'tcx> {
334
341
& mut Vec < VarDebugInfo < ' tcx > > ,
335
342
) {
336
343
self . predecessor_cache . invalidate ( ) ;
344
+ self . is_cyclic . invalidate ( ) ;
337
345
( & mut self . basic_blocks , & mut self . local_decls , & mut self . var_debug_info )
338
346
}
339
347
340
348
/// Returns `true` if a cycle exists in the control-flow graph that is reachable from the
341
349
/// `START_BLOCK`.
342
350
pub fn is_cfg_cyclic ( & self ) -> bool {
343
- graph :: is_cyclic ( self )
351
+ self . is_cyclic . is_cyclic ( self )
344
352
}
345
353
346
354
#[ inline]
0 commit comments