orx-tree 2.2.0

A beautiful tree 🌳 with convenient, efficient, parallelizable growth, mutation and traversal features.
Documentation
use super::Dary;
use crate::aliases::N;
use orx_pinned_vec::PinnedVec;
use orx_selfref_col::{CoreCol, MemoryReclaimer, NodePtr};

#[derive(Clone, Default)]
pub struct DaryReclaimer;

impl<const D: usize, T> MemoryReclaimer<Dary<D, T>> for DaryReclaimer {
    fn reclaim_nodes<P>(col: &mut CoreCol<Dary<D, T>, P>) -> bool
    where
        P: PinnedVec<N<Dary<D, T>>>,
    {
        let mut any_swapped = false;

        // SAFETY: lifetimes of `forward` and `backward` iterators are limited to this method
        // which is shorter than the lifetime of the `col`
        let forward = unsafe { col.nodes().iter_ptr() };
        let mut backward = unsafe { col.nodes().iter_ptr_rev() };
        let mut o = col.nodes().len();

        for (v, vacant_ptr) in forward.enumerate() {
            if v >= o {
                break;
            }

            if unsafe { &*vacant_ptr }.is_closed() {
                while o > v {
                    o -= 1;
                    let occupied_ptr = backward.next().expect("cannot be consumed before forward");

                    if unsafe { &*occupied_ptr }.is_active() {
                        any_swapped = true;
                        swap(col, vacant_ptr, occupied_ptr);
                        break;
                    }
                }
            }
        }

        any_swapped
    }
}

fn swap<const D: usize, P, T>(
    col: &mut CoreCol<Dary<D, T>, P>,
    vacant: *const N<Dary<D, T>>,
    occupied: *const N<Dary<D, T>>,
) where
    P: PinnedVec<N<Dary<D, T>>>,
{
    // occupied.parent.child => vacant
    if let Some(parent) = (unsafe { &*occupied }).prev().get() {
        let parent = col.node_mut(parent);
        let children = parent.next_mut();

        let child = children
            .iter_mut()
            // SAFETY: we have a mutual &mut reference to the underlying collection
            // which is guaranteed to be in the same memory state as occupied
            .find(|x| unsafe { x.ptr() == occupied })
            .expect("valid tree state ensures this");

        *child = NodePtr::new(vacant);
    }

    // occupied.children.parent => vacant
    for child in (unsafe { &*occupied }).next().iter() {
        let into_child_mut = unsafe { child.node_mut() };
        let parent = into_child_mut.prev_mut();
        parent.set_some(NodePtr::new(vacant));
    }

    // root => vacant

    // SAFETY: we have a mutual &mut reference to the underlying collection
    // which is guaranteed to be in the same memory state as occupied
    if occupied == unsafe { col.ends().get().expect("nonempty tree").ptr() } {
        col.ends_mut().set_some(NodePtr::new(vacant));
    }

    core::mem::swap(unsafe { &mut *(vacant as *mut N<Dary<D, T>>) }, unsafe {
        &mut *(occupied as *mut N<Dary<D, T>>)
    });
}