//! A pass that inserts the `ConstEvalCounter` instruction into any blocks that have a back edge //! (thus indicating there is a loop in the CFG), or whose terminator is a function call. use crate::MirPass; use rustc_data_structures::graph::dominators::Dominators; use rustc_middle::mir::{ BasicBlock, BasicBlockData, Body, Statement, StatementKind, TerminatorKind, }; use rustc_middle::ty::TyCtxt; pub struct CtfeLimit; impl<'tcx> MirPass<'tcx> for CtfeLimit { #[instrument(skip(self, _tcx, body))] fn run_pass(&self, _tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) { let doms = body.basic_blocks.dominators(); let indices: Vec = body .basic_blocks .iter_enumerated() .filter_map(|(node, node_data)| { if matches!(node_data.terminator().kind, TerminatorKind::Call { .. }) // Back edges in a CFG indicate loops || has_back_edge(&doms, node, &node_data) { Some(node) } else { None } }) .collect(); for index in indices { insert_counter( body.basic_blocks_mut() .get_mut(index) .expect("basic_blocks index {index} should exist"), ); } } } fn has_back_edge( doms: &Dominators, node: BasicBlock, node_data: &BasicBlockData<'_>, ) -> bool { if !doms.is_reachable(node) { return false; } // Check if any of the dominators of the node are also the node's successor. doms.dominators(node) .any(|dom| node_data.terminator().successors().into_iter().any(|succ| succ == dom)) } fn insert_counter(basic_block_data: &mut BasicBlockData<'_>) { basic_block_data.statements.push(Statement { source_info: basic_block_data.terminator().source_info, kind: StatementKind::ConstEvalCounter, }); }