kernel::rbtree

Struct Cursor

Source
pub struct Cursor<'a, K, V> { /* private fields */ }
Expand description

A bidirectional cursor over the tree nodes, sorted by key.

§Examples

In the following example, we obtain a cursor to the first element in the tree. The cursor allows us to iterate bidirectionally over key/value pairs in the tree.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert three elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;

// Get a cursor to the first element.
let mut cursor = tree.cursor_front().unwrap();
let mut current = cursor.current();
assert_eq!(current, (&10, &100));

// Move the cursor, updating it to the 2nd element.
cursor = cursor.move_next().unwrap();
current = cursor.current();
assert_eq!(current, (&20, &200));

// Peek at the next element without impacting the cursor.
let next = cursor.peek_next().unwrap();
assert_eq!(next, (&30, &300));
current = cursor.current();
assert_eq!(current, (&20, &200));

// Moving past the last element causes the cursor to return [`None`].
cursor = cursor.move_next().unwrap();
current = cursor.current();
assert_eq!(current, (&30, &300));
let cursor = cursor.move_next();
assert!(cursor.is_none());

A cursor can also be obtained at the last element in the tree.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert three elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;

let mut cursor = tree.cursor_back().unwrap();
let current = cursor.current();
assert_eq!(current, (&30, &300));

Obtaining a cursor returns None if the tree is empty.

use kernel::rbtree::RBTree;

let mut tree: RBTree<u16, u16> = RBTree::new();
assert!(tree.cursor_front().is_none());

RBTree::cursor_lower_bound can be used to start at an arbitrary node in the tree.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert five elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;
tree.try_create_and_insert(40, 400, flags::GFP_KERNEL)?;
tree.try_create_and_insert(50, 500, flags::GFP_KERNEL)?;

// If the provided key exists, a cursor to that key is returned.
let cursor = tree.cursor_lower_bound(&20).unwrap();
let current = cursor.current();
assert_eq!(current, (&20, &200));

// If the provided key doesn't exist, a cursor to the first larger element in sort order is returned.
let cursor = tree.cursor_lower_bound(&25).unwrap();
let current = cursor.current();
assert_eq!(current, (&30, &300));

// If there is no larger key, [`None`] is returned.
let cursor = tree.cursor_lower_bound(&55);
assert!(cursor.is_none());

The cursor allows mutation of values in the tree.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert three elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;

// Retrieve a cursor.
let mut cursor = tree.cursor_front().unwrap();

// Get a mutable reference to the current value.
let (k, v) = cursor.current_mut();
*v = 1000;

// The updated value is reflected in the tree.
let updated = tree.get(&10).unwrap();
assert_eq!(updated, &1000);

It also allows node removal. The following examples demonstrate the behavior of removing the current node.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert three elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;

// Remove the first element.
let mut cursor = tree.cursor_front().unwrap();
let mut current = cursor.current();
assert_eq!(current, (&10, &100));
cursor = cursor.remove_current().0.unwrap();

// If a node exists after the current element, it is returned.
current = cursor.current();
assert_eq!(current, (&20, &200));

// Get a cursor to the last element, and remove it.
cursor = tree.cursor_back().unwrap();
current = cursor.current();
assert_eq!(current, (&30, &300));

// Since there is no next node, the previous node is returned.
cursor = cursor.remove_current().0.unwrap();
current = cursor.current();
assert_eq!(current, (&20, &200));

// Removing the last element in the tree returns [`None`].
assert!(cursor.remove_current().0.is_none());

Nodes adjacent to the current node can also be removed.

use kernel::{alloc::flags, rbtree::RBTree};

// Create a new tree.
let mut tree = RBTree::new();

// Insert three elements.
tree.try_create_and_insert(10, 100, flags::GFP_KERNEL)?;
tree.try_create_and_insert(20, 200, flags::GFP_KERNEL)?;
tree.try_create_and_insert(30, 300, flags::GFP_KERNEL)?;

// Get a cursor to the first element.
let mut cursor = tree.cursor_front().unwrap();
let mut current = cursor.current();
assert_eq!(current, (&10, &100));

// Calling `remove_prev` from the first element returns [`None`].
assert!(cursor.remove_prev().is_none());

// Get a cursor to the last element.
cursor = tree.cursor_back().unwrap();
current = cursor.current();
assert_eq!(current, (&30, &300));

// Calling `remove_prev` removes and returns the middle element.
assert_eq!(cursor.remove_prev().unwrap().to_key_value(), (20, 200));

// Calling `remove_next` from the last element returns [`None`].
assert!(cursor.remove_next().is_none());

// Move to the first element
cursor = cursor.move_prev().unwrap();
current = cursor.current();
assert_eq!(current, (&10, &100));

// Calling `remove_next` removes and returns the last element.
assert_eq!(cursor.remove_next().unwrap().to_key_value(), (30, 300));

§Invariants

  • current points to a node that is in the same RBTree as tree.

Implementations§

Source§

impl<'a, K, V> Cursor<'a, K, V>

Source

pub fn current(&self) -> (&K, &V)

The current node

Source

pub fn current_mut(&mut self) -> (&K, &mut V)

The current node, with a mutable value

Source

pub fn remove_current(self) -> (Option<Self>, RBTreeNode<K, V>)

Remove the current node from the tree.

Returns a tuple where the first element is a cursor to the next node, if it exists, else the previous node, else None (if the tree becomes empty). The second element is the removed node.

Source

pub fn remove_prev(&mut self) -> Option<RBTreeNode<K, V>>

Remove the previous node, returning it if it exists.

Source

pub fn remove_next(&mut self) -> Option<RBTreeNode<K, V>>

Remove the next node, returning it if it exists.

Source

pub fn move_prev(self) -> Option<Self>

Move the cursor to the previous node, returning None if it doesn’t exist.

Source

pub fn move_next(self) -> Option<Self>

Move the cursor to the next node, returning None if it doesn’t exist.

Source

pub fn peek_prev(&self) -> Option<(&K, &V)>

Access the previous node without moving the cursor.

Source

pub fn peek_next(&self) -> Option<(&K, &V)>

Access the previous node without moving the cursor.

Source

pub fn peek_prev_mut(&mut self) -> Option<(&K, &mut V)>

Access the previous node mutably without moving the cursor.

Source

pub fn peek_next_mut(&mut self) -> Option<(&K, &mut V)>

Access the next node mutably without moving the cursor.

Trait Implementations§

Source§

impl<'a, K: Send, V: Send> Send for Cursor<'a, K, V>

Source§

impl<'a, K: Sync, V: Sync> Sync for Cursor<'a, K, V>

Auto Trait Implementations§

§

impl<'a, K, V> Freeze for Cursor<'a, K, V>

§

impl<'a, K, V> RefUnwindSafe for Cursor<'a, K, V>

§

impl<'a, K, V> Unpin for Cursor<'a, K, V>

§

impl<'a, K, V> !UnwindSafe for Cursor<'a, K, V>

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, E> Init<T, E> for T

Source§

unsafe fn __init(self, slot: *mut T) -> Result<(), E>

Initializes slot. Read more
Source§

fn chain<F>(self, f: F) -> ChainInit<Self, F, T, E>
where F: FnOnce(&mut T) -> Result<(), E>,

First initializes the value using self then calls the function f with the initialized value. Read more
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, E> PinInit<T, E> for T

Source§

unsafe fn __pinned_init(self, slot: *mut T) -> Result<(), E>

Initializes slot. Read more
Source§

fn pin_chain<F>(self, f: F) -> ChainPinInit<Self, F, T, E>
where F: FnOnce(Pin<&mut T>) -> Result<(), E>,

First initializes the value using self then calls the function f with the initialized value. 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.