Struct Decycler

Source
pub(crate) struct Decycler<T, const D: usize> {
    node_ids: [T; D],
    depth: usize,
}
Expand description

Cycle detector for DFS traversal of a graph.

The graph is expected to have unique node identifiers of type T and traversal depth is limited to the constant D.

This is based on hb_decycler_t in HarfBuzz (https://github.com/harfbuzz/harfbuzz/blob/a2ea5d28cb5387f4de2049802474b817be15ad5b/src/hb-decycler.hh) which is an extension of Floyd’s tortoise and hare algorithm (https://en.wikipedia.org/wiki/Cycle_detection#Floyd's_tortoise_and_hare) to DFS traversals.

Unlike the implementation in HB which supports traversals of arbitrary depth, this is limited to a constant value. Instead of building a forward-linked list of nodes (on the stack) to track the traversal chain, we simply use a fixed size array, indexed by depth, which imposes the limit.

It might be possible to implement the HB algorithm in safe Rust but satisfying the borrow checker would be a challenge and we require a depth limit to prevent stack overflows anyway. Improvements welcome!

Fields§

§node_ids: [T; D]§depth: usize

Implementations§

Source§

impl<T, const D: usize> Decycler<T, D>
where T: Copy + PartialEq + Default,

Source

pub fn new() -> Self

Source

pub fn enter( &mut self, node_id: T, ) -> Result<DecyclerGuard<'_, T, D>, DecyclerError>

Enters a new graph node with the given value that uniquely identifies the current node.

Returns an error when a cycle is detected or the max depth of the traversal is exceeded. Otherwise, increases the current depth and returns a guard object that will decrease the depth when dropped.

The guard object derefs to the decycler, so it can be passed to a recursive traversal function to check for cycles in descendent nodes in a graph.

Trait Implementations§

Source§

impl<T, const D: usize> Default for Decycler<T, D>
where T: Copy + PartialEq + Default,

Source§

fn default() -> Self

Returns the “default value” for a type. Read more

Auto Trait Implementations§

§

impl<T, const D: usize> Freeze for Decycler<T, D>
where T: Freeze,

§

impl<T, const D: usize> RefUnwindSafe for Decycler<T, D>
where T: RefUnwindSafe,

§

impl<T, const D: usize> Send for Decycler<T, D>
where T: Send,

§

impl<T, const D: usize> Sync for Decycler<T, D>
where T: Sync,

§

impl<T, const D: usize> Unpin for Decycler<T, D>
where T: Unpin,

§

impl<T, const D: usize> UnwindSafe for Decycler<T, D>
where T: UnwindSafe,

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> 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, 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.