use rustc_index::vec::Idx; pub mod dominators; pub mod implementation; pub mod iterate; mod reference; pub mod scc; pub mod vec_graph; #[cfg(test)] mod tests; pub trait DirectedGraph { type Node: Idx; } pub trait WithNumNodes: DirectedGraph { fn num_nodes(&self) -> usize; } pub trait WithNumEdges: DirectedGraph { fn num_edges(&self) -> usize; } pub trait WithSuccessors: DirectedGraph where Self: for<'graph> GraphSuccessors<'graph, Item = ::Node>, { fn successors(&self, node: Self::Node) -> >::Iter; fn depth_first_search(&self, from: Self::Node) -> iterate::DepthFirstSearch<'_, Self> where Self: WithNumNodes, { iterate::DepthFirstSearch::new(self).with_start_node(from) } } #[allow(unused_lifetimes)] pub trait GraphSuccessors<'graph> { type Item; type Iter: Iterator; } pub trait WithPredecessors: DirectedGraph where Self: for<'graph> GraphPredecessors<'graph, Item = ::Node>, { fn predecessors(&self, node: Self::Node) -> >::Iter; } #[allow(unused_lifetimes)] pub trait GraphPredecessors<'graph> { type Item; type Iter: Iterator; } pub trait WithStartNode: DirectedGraph { fn start_node(&self) -> Self::Node; } pub trait ControlFlowGraph: DirectedGraph + WithStartNode + WithPredecessors + WithSuccessors + WithNumNodes { // convenient trait } impl ControlFlowGraph for T where T: DirectedGraph + WithStartNode + WithPredecessors + WithSuccessors + WithNumNodes { } /// Returns `true` if the graph has a cycle that is reachable from the start node. pub fn is_cyclic(graph: &G) -> bool where G: ?Sized + DirectedGraph + WithStartNode + WithSuccessors + WithNumNodes, { iterate::TriColorDepthFirstSearch::new(graph) .run_from_start(&mut iterate::CycleDetector) .is_some() }