Skip to main content

ControlFlowGraph

Struct ControlFlowGraph 

Source
pub struct ControlFlowGraph {
    pub blocks: HashMap<String, BasicBlock>,
    pub edges: HashMap<String, Vec<String>>,
    pub entry_block: String,
    pub exit_blocks: Vec<String>,
}
Expand description

Control Flow Graph extracted from MIR basic blocks

Fields§

§blocks: HashMap<String, BasicBlock>

Map from block ID to BasicBlock

§edges: HashMap<String, Vec<String>>

Map from block ID to successor block IDs

§entry_block: String

Entry block (usually “bb0”)

§exit_blocks: Vec<String>

Exit blocks (blocks with Return terminator)

Implementations§

Source§

impl ControlFlowGraph

Source

pub fn from_mir_function(function: &MirFunction) -> Self

Extract CFG from a MIR function’s body

Source

pub fn block_count(&self) -> usize

Get the number of basic blocks in the CFG

Source

pub fn branch_count(&self) -> usize

Count branch points in the CFG (blocks with multiple successors)

Source

pub fn is_too_complex_for_path_enumeration(&self) -> bool

Check if this CFG is too complex for full path enumeration. Returns true if enumeration should be skipped to prevent memory explosion.

The heuristic is based on empirical observation:

  • influxdb3_id: 78 blocks, 20 branches → works fine (~100 MB)
  • influxdb3 serve: 2442 blocks, 381 branches → 58 GB explosion

A CFG with B branches can have up to 2^B paths. With MAX_DEPTH=50, we limit path length, but the DFS exploration tree still grows exponentially.

Safe thresholds (empirically derived):

  • MAX_BLOCKS: 500 (well above typical functions, catches mega-closures)
  • MAX_BRANCHES: 100 (limits exponential exploration)
Source

pub fn get_all_paths(&self) -> (Vec<Vec<String>>, bool)

Enumerate all paths from entry to exit blocks Returns paths as sequences of block IDs

For extremely complex CFGs (large async closures, state machines), returns an empty vec to prevent memory explosion.

Returns (paths, was_skipped_due_to_complexity)

Source

pub fn get_block(&self, block_id: &str) -> Option<&BasicBlock>

Get the basic block for a given ID

Source

pub fn has_branching(&self) -> bool

Check if the CFG has any branching (multiple paths)

Trait Implementations§

Source§

impl Clone for ControlFlowGraph

Source§

fn clone(&self) -> ControlFlowGraph

Returns a duplicate of the value. Read more
1.0.0 · Source§

fn clone_from(&mut self, source: &Self)

Performs copy-assignment from source. Read more
Source§

impl Debug for ControlFlowGraph

Source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more

Auto Trait Implementations§

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> CloneToUninit for T
where T: Clone,

Source§

unsafe fn clone_to_uninit(&self, dest: *mut u8)

🔬This is a nightly-only experimental API. (clone_to_uninit)
Performs copy-assignment from self to dest. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T> IntoEither for T

Source§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

impl<T> Pointable for T

Source§

const ALIGN: usize

The alignment of pointer.
Source§

type Init = T

The type for initializers.
Source§

unsafe fn init(init: <T as Pointable>::Init) -> usize

Initializes a with the given initializer. Read more
Source§

unsafe fn deref<'a>(ptr: usize) -> &'a T

Dereferences the given pointer. Read more
Source§

unsafe fn deref_mut<'a>(ptr: usize) -> &'a mut T

Mutably dereferences the given pointer. Read more
Source§

unsafe fn drop(ptr: usize)

Drops the object pointed to by the given pointer. Read more
Source§

impl<T> Same for T

Source§

type Output = T

Should always be Self
Source§

impl<T> ToOwned for T
where T: Clone,

Source§

type Owned = T

The resulting type after obtaining ownership.
Source§

fn to_owned(&self) -> T

Creates owned data from borrowed data, usually by cloning. Read more
Source§

fn clone_into(&self, target: &mut T)

Uses borrowed data to replace owned data, usually by cloning. Read more
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.