mirror of
https://github.com/torvalds/linux.git
synced 2026-05-29 17:43:52 +02:00
rust: maple_tree: add lock guard for maple tree
To load a value, one must be careful to hold the lock while accessing it. To enable this, we add a lock() method so that you can perform operations on the value before the spinlock is released. This adds a MapleGuard type without using the existing SpinLock type. This ensures that the MapleGuard type is not unnecessarily large, and that it is easy to swap out the type of lock in case the C maple tree is changed to use a different kind of lock. There are two ways of using the lock guard: You can call load() directly to load a value under the lock, or you can create an MaState to iterate the tree with find(). The find() method does not have the mas_ prefix since it's a method on MaState, and being a method on that struct serves a similar purpose to the mas_ prefix in C. Link: https://lkml.kernel.org/r/20250902-maple-tree-v3-2-fb5c8958fb1e@google.com Co-developed-by: Andrew Ballance <andrewjballance@gmail.com> Signed-off-by: Andrew Ballance <andrewjballance@gmail.com> Reviewed-by: Andrew Ballance <andrewjballance@gmail.com> Reviewed-by: Danilo Krummrich <dakr@kernel.org> Signed-off-by: Alice Ryhl <aliceryhl@google.com> Cc: Andreas Hindborg <a.hindborg@kernel.org> Cc: Björn Roy Baron <bjorn3_gh@protonmail.com> Cc: Boqun Feng <boqun.feng@gmail.com> Cc: Daniel Almeida <daniel.almeida@collabora.com> Cc: Gary Guo <gary@garyguo.net> Cc: Liam Howlett <liam.howlett@oracle.com> Cc: Lorenzo Stoakes <lorenzo.stoakes@oracle.com> Cc: Miguel Ojeda <ojeda@kernel.org> Cc: Trevor Gross <tmgross@umich.edu> Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
This commit is contained in:
parent
da939ef4c4
commit
01422da19c
|
|
@ -213,6 +213,23 @@ pub fn erase(&self, index: usize) -> Option<T> {
|
|||
unsafe { T::try_from_foreign(ret) }
|
||||
}
|
||||
|
||||
/// Lock the internal spinlock.
|
||||
#[inline]
|
||||
pub fn lock(&self) -> MapleGuard<'_, T> {
|
||||
// SAFETY: It's safe to lock the spinlock in a maple tree.
|
||||
unsafe { bindings::spin_lock(self.ma_lock()) };
|
||||
|
||||
// INVARIANT: We just took the spinlock.
|
||||
MapleGuard(self)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn ma_lock(&self) -> *mut bindings::spinlock_t {
|
||||
// SAFETY: This pointer offset operation stays in-bounds.
|
||||
let lock_ptr = unsafe { &raw mut (*self.tree.get()).__bindgen_anon_1.ma_lock };
|
||||
lock_ptr.cast()
|
||||
}
|
||||
|
||||
/// Free all `T` instances in this tree.
|
||||
///
|
||||
/// # Safety
|
||||
|
|
@ -256,6 +273,91 @@ fn drop(mut self: Pin<&mut Self>) {
|
|||
}
|
||||
}
|
||||
|
||||
/// A reference to a [`MapleTree`] that owns the inner lock.
|
||||
///
|
||||
/// # Invariants
|
||||
///
|
||||
/// This guard owns the inner spinlock.
|
||||
#[must_use = "if unused, the lock will be immediately unlocked"]
|
||||
pub struct MapleGuard<'tree, T: ForeignOwnable>(&'tree MapleTree<T>);
|
||||
|
||||
impl<'tree, T: ForeignOwnable> Drop for MapleGuard<'tree, T> {
|
||||
#[inline]
|
||||
fn drop(&mut self) {
|
||||
// SAFETY: By the type invariants, we hold this spinlock.
|
||||
unsafe { bindings::spin_unlock(self.0.ma_lock()) };
|
||||
}
|
||||
}
|
||||
|
||||
impl<'tree, T: ForeignOwnable> MapleGuard<'tree, T> {
|
||||
/// Create a [`MaState`] protected by this lock guard.
|
||||
pub fn ma_state(&mut self, first: usize, end: usize) -> MaState<'_, T> {
|
||||
// SAFETY: The `MaState` borrows this `MapleGuard`, so it can also borrow the `MapleGuard`s
|
||||
// read/write permissions to the maple tree.
|
||||
unsafe { MaState::new_raw(self.0, first, end) }
|
||||
}
|
||||
|
||||
/// Load the value at the given index.
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Read the value while holding the spinlock.
|
||||
///
|
||||
/// ```
|
||||
/// use kernel::maple_tree::MapleTree;
|
||||
///
|
||||
/// let tree = KBox::pin_init(MapleTree::<KBox<i32>>::new(), GFP_KERNEL)?;
|
||||
///
|
||||
/// let ten = KBox::new(10, GFP_KERNEL)?;
|
||||
/// let twenty = KBox::new(20, GFP_KERNEL)?;
|
||||
/// tree.insert(100, ten, GFP_KERNEL)?;
|
||||
/// tree.insert(200, twenty, GFP_KERNEL)?;
|
||||
///
|
||||
/// let mut lock = tree.lock();
|
||||
/// assert_eq!(lock.load(100).map(|v| *v), Some(10));
|
||||
/// assert_eq!(lock.load(200).map(|v| *v), Some(20));
|
||||
/// assert_eq!(lock.load(300).map(|v| *v), None);
|
||||
/// # Ok::<_, Error>(())
|
||||
/// ```
|
||||
///
|
||||
/// Increment refcount under the lock, to keep value alive afterwards.
|
||||
///
|
||||
/// ```
|
||||
/// use kernel::maple_tree::MapleTree;
|
||||
/// use kernel::sync::Arc;
|
||||
///
|
||||
/// let tree = KBox::pin_init(MapleTree::<Arc<i32>>::new(), GFP_KERNEL)?;
|
||||
///
|
||||
/// let ten = Arc::new(10, GFP_KERNEL)?;
|
||||
/// let twenty = Arc::new(20, GFP_KERNEL)?;
|
||||
/// tree.insert(100, ten, GFP_KERNEL)?;
|
||||
/// tree.insert(200, twenty, GFP_KERNEL)?;
|
||||
///
|
||||
/// // Briefly take the lock to increment the refcount.
|
||||
/// let value = tree.lock().load(100).map(Arc::from);
|
||||
///
|
||||
/// // At this point, another thread might remove the value.
|
||||
/// tree.erase(100);
|
||||
///
|
||||
/// // But we can still access it because we took a refcount.
|
||||
/// assert_eq!(value.map(|v| *v), Some(10));
|
||||
/// # Ok::<_, Error>(())
|
||||
/// ```
|
||||
#[inline]
|
||||
pub fn load(&mut self, index: usize) -> Option<T::BorrowedMut<'_>> {
|
||||
// SAFETY: `self.tree` contains a valid maple tree.
|
||||
let ret = unsafe { bindings::mtree_load(self.0.tree.get(), index) };
|
||||
if ret.is_null() {
|
||||
return None;
|
||||
}
|
||||
|
||||
// SAFETY: If the pointer is not null, then it references a valid instance of `T`. It is
|
||||
// safe to borrow the instance mutably because the signature of this function enforces that
|
||||
// the mutable borrow is not used after the spinlock is dropped.
|
||||
Some(unsafe { T::borrow_mut(ret) })
|
||||
}
|
||||
}
|
||||
|
||||
/// A helper type used for navigating a [`MapleTree`].
|
||||
///
|
||||
/// # Invariants
|
||||
|
|
@ -309,6 +411,44 @@ fn mas_find_raw(&mut self, max: usize) -> *mut c_void {
|
|||
// to the tree.
|
||||
unsafe { bindings::mas_find(self.as_raw(), max) }
|
||||
}
|
||||
|
||||
/// Find the next entry in the maple tree.
|
||||
///
|
||||
/// # Examples
|
||||
///
|
||||
/// Iterate the maple tree.
|
||||
///
|
||||
/// ```
|
||||
/// use kernel::maple_tree::MapleTree;
|
||||
/// use kernel::sync::Arc;
|
||||
///
|
||||
/// let tree = KBox::pin_init(MapleTree::<Arc<i32>>::new(), GFP_KERNEL)?;
|
||||
///
|
||||
/// let ten = Arc::new(10, GFP_KERNEL)?;
|
||||
/// let twenty = Arc::new(20, GFP_KERNEL)?;
|
||||
/// tree.insert(100, ten, GFP_KERNEL)?;
|
||||
/// tree.insert(200, twenty, GFP_KERNEL)?;
|
||||
///
|
||||
/// let mut ma_lock = tree.lock();
|
||||
/// let mut iter = ma_lock.ma_state(0, usize::MAX);
|
||||
///
|
||||
/// assert_eq!(iter.find(usize::MAX).map(|v| *v), Some(10));
|
||||
/// assert_eq!(iter.find(usize::MAX).map(|v| *v), Some(20));
|
||||
/// assert!(iter.find(usize::MAX).is_none());
|
||||
/// # Ok::<_, Error>(())
|
||||
/// ```
|
||||
#[inline]
|
||||
pub fn find(&mut self, max: usize) -> Option<T::BorrowedMut<'_>> {
|
||||
let ret = self.mas_find_raw(max);
|
||||
if ret.is_null() {
|
||||
return None;
|
||||
}
|
||||
|
||||
// SAFETY: If the pointer is not null, then it references a valid instance of `T`. It's
|
||||
// safe to access it mutably as the returned reference borrows this `MaState`, and the
|
||||
// `MaState` has read/write access to the maple tree.
|
||||
Some(unsafe { T::borrow_mut(ret) })
|
||||
}
|
||||
}
|
||||
|
||||
/// Error type for failure to insert a new value.
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user