|
@@ -1,61 +1,176 @@
|
|
|
use core::marker::PhantomData;
|
|
|
-use core::ops::Deref;
|
|
|
|
|
|
use smallvec::SmallVec;
|
|
|
|
|
|
-use crate::entry::{ItemEntry, ItemRef, XEntry};
|
|
|
-use crate::lock::XLock;
|
|
|
+use crate::borrow::DormantMutRef;
|
|
|
+use crate::entry::{ItemEntry, ItemRef, NodeMaybeMut, XEntry};
|
|
|
use crate::mark::XMark;
|
|
|
-use crate::node::{Height, ReadOnly, ReadWrite, XNode};
|
|
|
+use crate::node::XNode;
|
|
|
use crate::xarray::{XArray, MAX_HEIGHT, SLOT_SIZE};
|
|
|
|
|
|
+trait Operation {}
|
|
|
+
|
|
|
+struct ReadOnly {}
|
|
|
+struct ReadWrite {}
|
|
|
+
|
|
|
+impl Operation for ReadOnly {}
|
|
|
+impl Operation for ReadWrite {}
|
|
|
+
|
|
|
/// CursorState represents the current state of the cursor. Currently, there are two possible states:
|
|
|
/// 1. inactive: the state where the cursor is not positioned on any node.
|
|
|
/// 2. positioned on a node: this state includes information about the node the cursor is on,
|
|
|
/// as well as the offset of the entry that needs to be operated on within the slots of the current node.
|
|
|
-enum CursorState<'a, I, L, Operation>
|
|
|
+enum CursorState<'a, I, O>
|
|
|
where
|
|
|
I: ItemEntry,
|
|
|
- L: XLock,
|
|
|
+ O: Operation,
|
|
|
{
|
|
|
- Inactive,
|
|
|
+ Inactive(PhantomData<O>),
|
|
|
AtNode {
|
|
|
- node: &'a XNode<I, L, Operation>,
|
|
|
+ node: &'a XNode<I>,
|
|
|
+ operation_offset: u8,
|
|
|
+ },
|
|
|
+ AtNodeMut {
|
|
|
+ node: &'a mut XNode<I>,
|
|
|
operation_offset: u8,
|
|
|
},
|
|
|
}
|
|
|
|
|
|
-impl<'a, I: ItemEntry, L: XLock, Operation> CursorState<'a, I, L, Operation> {
|
|
|
+impl<'a, I: ItemEntry, O: Operation> Default for CursorState<'a, I, O> {
|
|
|
fn default() -> Self {
|
|
|
- Self::Inactive
|
|
|
+ Self::Inactive(PhantomData)
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- fn arrive_node(&mut self, node: &'a XNode<I, L, Operation>, operation_offset: u8) {
|
|
|
+impl<'a, I: ItemEntry, O: Operation> CursorState<'a, I, O> {
|
|
|
+ fn move_to(&mut self, node: &'a XNode<I>, index: u64) {
|
|
|
+ let operation_offset = node.entry_offset(index);
|
|
|
*self = Self::AtNode {
|
|
|
node,
|
|
|
operation_offset,
|
|
|
};
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- fn is_at_node(&self) -> bool {
|
|
|
- matches!(
|
|
|
- self,
|
|
|
+impl<'a, I: ItemEntry> CursorState<'a, I, ReadWrite> {
|
|
|
+ fn move_to_mut(&mut self, node: &'a mut XNode<I>, index: u64) {
|
|
|
+ let operation_offset = node.entry_offset(index);
|
|
|
+ *self = Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ };
|
|
|
+ }
|
|
|
+
|
|
|
+ fn move_to_maybe_mut(&mut self, node: NodeMaybeMut<'a, I>, index: u64) {
|
|
|
+ match node {
|
|
|
+ NodeMaybeMut::Shared(node) => self.move_to(node, index),
|
|
|
+ NodeMaybeMut::Exclusive(node) => self.move_to_mut(node, index),
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry, O: Operation> CursorState<'a, I, O> {
|
|
|
+ fn into_node(self) -> Option<(&'a XNode<I>, u8)> {
|
|
|
+ match self {
|
|
|
Self::AtNode {
|
|
|
- node: _,
|
|
|
- operation_offset: _
|
|
|
- }
|
|
|
- )
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((node, operation_offset)),
|
|
|
+ Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((node, operation_offset)),
|
|
|
+ Self::Inactive(..) => None,
|
|
|
+ }
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- fn node_info(&self) -> Option<(&'a XNode<I, L, Operation>, u8)> {
|
|
|
- if let Self::AtNode {
|
|
|
- node,
|
|
|
- operation_offset,
|
|
|
- } = self
|
|
|
- {
|
|
|
- Some((node, *operation_offset))
|
|
|
- } else {
|
|
|
- None
|
|
|
+impl<'a, I: ItemEntry> CursorState<'a, I, ReadWrite> {
|
|
|
+ fn into_node_mut(self) -> Option<(&'a mut XNode<I>, u8)> {
|
|
|
+ match self {
|
|
|
+ Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((node, operation_offset)),
|
|
|
+ Self::Inactive(..) | Self::AtNode { .. } => None,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn into_node_maybe_mut(self) -> Option<(NodeMaybeMut<'a, I>, u8)> {
|
|
|
+ match self {
|
|
|
+ Self::AtNode {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((NodeMaybeMut::Shared(node), operation_offset)),
|
|
|
+ Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((NodeMaybeMut::Exclusive(node), operation_offset)),
|
|
|
+ Self::Inactive(..) => None,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry> CursorState<'a, I, ReadOnly> {
|
|
|
+ fn as_node(&self) -> Option<(&'a XNode<I>, u8)> {
|
|
|
+ match self {
|
|
|
+ Self::AtNode {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((*node, *operation_offset)),
|
|
|
+ Self::Inactive(..) | Self::AtNodeMut { .. } => None,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry> CursorState<'a, I, ReadWrite> {
|
|
|
+ fn as_node(&self) -> Option<(&XNode<I>, u8)> {
|
|
|
+ match self {
|
|
|
+ Self::AtNode {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((*node, *operation_offset)),
|
|
|
+ Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((*node, *operation_offset)),
|
|
|
+ Self::Inactive(..) => None,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn as_node_mut(&mut self) -> Option<(&mut XNode<I>, u8)> {
|
|
|
+ match self {
|
|
|
+ Self::AtNodeMut {
|
|
|
+ node,
|
|
|
+ operation_offset,
|
|
|
+ } => Some((*node, *operation_offset)),
|
|
|
+ Self::Inactive(..) | Self::AtNode { .. } => None,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry, O: Operation> CursorState<'a, I, O> {
|
|
|
+ fn is_at_node(&self) -> bool {
|
|
|
+ match self {
|
|
|
+ Self::AtNode { .. } | Self::AtNodeMut { .. } => true,
|
|
|
+ Self::Inactive(..) => false,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn is_leaf(&self) -> bool {
|
|
|
+ match self {
|
|
|
+ Self::AtNodeMut { node, .. } => node.is_leaf(),
|
|
|
+ Self::AtNode { node, .. } => node.is_leaf(),
|
|
|
+ Self::Inactive(..) => false,
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry> CursorState<'a, I, ReadWrite> {
|
|
|
+ fn is_at_node_mut(&self) -> bool {
|
|
|
+ match self {
|
|
|
+ Self::AtNodeMut { .. } => true,
|
|
|
+ Self::Inactive(..) | Self::AtNode { .. } => false,
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -80,179 +195,164 @@ impl<'a, I: ItemEntry, L: XLock, Operation> CursorState<'a, I, L, Operation> {
|
|
|
/// thereby assisting it in performing some operations that involve searching upwards.
|
|
|
///
|
|
|
/// Multiple Cursors are allowed to operate on a single XArray at the same time.
|
|
|
-pub struct Cursor<'a, I, L, M>
|
|
|
+pub struct Cursor<'a, I, M>
|
|
|
where
|
|
|
I: ItemEntry,
|
|
|
- L: XLock,
|
|
|
M: Into<XMark>,
|
|
|
{
|
|
|
/// The `XArray` the cursor located in.
|
|
|
- xa: &'a XArray<I, L, M>,
|
|
|
+ xa: &'a XArray<I, M>,
|
|
|
/// The target index of the cursor in the belonged `XArray`.
|
|
|
index: u64,
|
|
|
/// Represents the current state of the cursor.
|
|
|
- state: CursorState<'a, I, L, ReadOnly>,
|
|
|
+ state: CursorState<'a, I, ReadOnly>,
|
|
|
/// Record add nodes passed from the head node to the target position.
|
|
|
/// The index is the height of the recorded node.
|
|
|
- ancestors: SmallVec<[&'a XNode<I, L, ReadOnly>; MAX_HEIGHT]>,
|
|
|
-
|
|
|
- _marker: PhantomData<I>,
|
|
|
+ ancestors: SmallVec<[&'a XNode<I>; MAX_HEIGHT]>,
|
|
|
}
|
|
|
|
|
|
-impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> Cursor<'a, I, L, M> {
|
|
|
+impl<'a, I: ItemEntry, M: Into<XMark>> Cursor<'a, I, M> {
|
|
|
/// Create an `Cursor` to perform read related operations on the `XArray`.
|
|
|
- pub(super) fn new(xa: &'a XArray<I, L, M>, index: u64) -> Self {
|
|
|
+ pub(super) fn new(xa: &'a XArray<I, M>, index: u64) -> Self {
|
|
|
let mut cursor = Self {
|
|
|
xa,
|
|
|
index,
|
|
|
state: CursorState::default(),
|
|
|
ancestors: SmallVec::new(),
|
|
|
- _marker: PhantomData,
|
|
|
};
|
|
|
-
|
|
|
cursor.traverse_to_target();
|
|
|
- cursor
|
|
|
- }
|
|
|
-
|
|
|
- /// Get a reference to current operated entry of the Cursor.
|
|
|
- fn ref_operated_entry(&self) -> Option<&'a XEntry<I, L>> {
|
|
|
- // SAFETY: The lifetime of the reference to the operated XEntry is `'a`,
|
|
|
- // during which there will not be another mut reference to the belonged xarray,
|
|
|
- // nor will there be any modification operations on the XNode where it resides.
|
|
|
- self.state
|
|
|
- .node_info()
|
|
|
- .map(|info| unsafe { info.0.ref_node_entry(info.1) })
|
|
|
- }
|
|
|
|
|
|
- /// Move the `Cursor` to the `XNode`, and update the cursor's state based on its target index.
|
|
|
- /// Return a reference to the `XEntry` within the slots of the current XNode that needs to be operated on.
|
|
|
- fn move_to(&mut self, node: &'a XNode<I, L, ReadOnly>) -> &'a XEntry<I, L> {
|
|
|
- let offset = node.entry_offset(self.index);
|
|
|
- self.state.arrive_node(node, offset);
|
|
|
- self.ref_operated_entry().unwrap()
|
|
|
+ cursor
|
|
|
}
|
|
|
|
|
|
/// Reset the target index of the Cursor. Once set, it will immediately attempt to move the Cursor
|
|
|
/// to touch the target XEntry.
|
|
|
pub fn reset_to(&mut self, index: u64) {
|
|
|
- self.init();
|
|
|
+ self.reset();
|
|
|
self.index = index;
|
|
|
+
|
|
|
self.traverse_to_target();
|
|
|
}
|
|
|
|
|
|
+ /// Return the target index of the cursor.
|
|
|
+ pub fn index(&self) -> u64 {
|
|
|
+ self.index
|
|
|
+ }
|
|
|
+
|
|
|
/// Move the target index of the cursor to index + 1.
|
|
|
/// If the target index's corresponding XEntry is not within the current XNode, the cursor will
|
|
|
/// move to touch the target XEntry. If the move fails, the cursor's state will be set to `Inactive`.
|
|
|
pub fn next(&mut self) {
|
|
|
- // TODO: overflow;
|
|
|
- self.index += 1;
|
|
|
- if !self.is_arrived() {
|
|
|
- return;
|
|
|
- }
|
|
|
+ self.index = self.index.checked_add(1).unwrap();
|
|
|
|
|
|
- if self.xa.max_index() < self.index {
|
|
|
- self.init();
|
|
|
+ if !self.state.is_at_node() {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
- let (mut current_node, mut operation_offset) = self.state.node_info().unwrap();
|
|
|
+ let (mut current_node, mut operation_offset) =
|
|
|
+ core::mem::take(&mut self.state).into_node().unwrap();
|
|
|
+
|
|
|
operation_offset += 1;
|
|
|
while operation_offset == SLOT_SIZE as u8 {
|
|
|
- operation_offset = current_node.offset_in_parent() + 1;
|
|
|
- if let Some(node) = self.ancestors.pop() {
|
|
|
- current_node = node;
|
|
|
- continue;
|
|
|
- }
|
|
|
-
|
|
|
- operation_offset = 0;
|
|
|
- break;
|
|
|
- }
|
|
|
- self.state.arrive_node(current_node, operation_offset);
|
|
|
-
|
|
|
- while !current_node.is_leaf() {
|
|
|
- let next_entry = self.ref_operated_entry().unwrap();
|
|
|
- if !next_entry.is_node() {
|
|
|
- self.init();
|
|
|
+ let Some(parent_node) = self.ancestors.pop() else {
|
|
|
+ self.reset();
|
|
|
return;
|
|
|
- }
|
|
|
+ };
|
|
|
|
|
|
- let next_node = next_entry.as_node().unwrap();
|
|
|
- self.ancestors.push(current_node);
|
|
|
- self.move_to(next_node);
|
|
|
- current_node = next_node;
|
|
|
+ operation_offset = current_node.offset_in_parent() + 1;
|
|
|
+ current_node = parent_node;
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- /// Judge if the target item is marked with the input `mark`.
|
|
|
- /// If target item does not exist, the function will return `None`.
|
|
|
- pub fn is_marked(&self, mark: M) -> bool {
|
|
|
- if let Some((current_node, operation_offset)) = self.state.node_info() {
|
|
|
- current_node.is_marked(operation_offset, mark.into().index())
|
|
|
- } else {
|
|
|
- false
|
|
|
- }
|
|
|
+ self.state.move_to(current_node, self.index);
|
|
|
+ self.continue_traverse_to_target();
|
|
|
}
|
|
|
|
|
|
/// Load the `XEntry` at the current cursor index within the `XArray`.
|
|
|
///
|
|
|
/// Returns a reference to the `XEntry` at the target index if succeed.
|
|
|
/// If the cursor cannot reach to the target index, the method will return `None`.
|
|
|
- pub fn load(&self) -> Option<ItemRef<'a, I>> {
|
|
|
- if let Some(entry) = self.ref_operated_entry() {
|
|
|
- entry.as_item_ref()
|
|
|
- } else {
|
|
|
- None
|
|
|
- }
|
|
|
+ pub fn load(&mut self) -> Option<ItemRef<'a, I>> {
|
|
|
+ self.traverse_to_target();
|
|
|
+ self.state
|
|
|
+ .as_node()
|
|
|
+ .and_then(|(node, off)| node.entry(off).as_item_ref())
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Judge if the target item is marked with the input `mark`.
|
|
|
+ /// If target item does not exist, the function will return `None`.
|
|
|
+ pub fn is_marked(&mut self, mark: M) -> bool {
|
|
|
+ self.traverse_to_target();
|
|
|
+ self.state
|
|
|
+ .as_node()
|
|
|
+ .map(|(node, off)| node.is_marked(off, mark.into().index()))
|
|
|
+ .unwrap_or(false)
|
|
|
}
|
|
|
|
|
|
/// Traverse the XArray and move to the node that can operate the target entry.
|
|
|
/// It then returns the reference to the `XEntry` stored in the slot corresponding to the target index.
|
|
|
/// A target operated XEntry must be an item entry.
|
|
|
/// If can not touch the target entry, the function will return `None`.
|
|
|
- fn traverse_to_target(&mut self) -> Option<&'a XEntry<I, L>> {
|
|
|
- if self.is_arrived() {
|
|
|
- return self.ref_operated_entry();
|
|
|
+ fn traverse_to_target(&mut self) {
|
|
|
+ if self.state.is_at_node() {
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
let max_index = self.xa.max_index();
|
|
|
if max_index < self.index || max_index == 0 {
|
|
|
- return None;
|
|
|
+ return;
|
|
|
}
|
|
|
- self.move_to(self.xa.head().as_node().unwrap());
|
|
|
|
|
|
- let (mut current_node, _) = self.state.node_info().unwrap();
|
|
|
- let mut operated_entry = self.ref_operated_entry().unwrap();
|
|
|
- while !current_node.is_leaf() {
|
|
|
+ let current_node = self.xa.head().as_node_ref().unwrap();
|
|
|
+ self.state.move_to(current_node, self.index);
|
|
|
+ self.continue_traverse_to_target();
|
|
|
+ }
|
|
|
+
|
|
|
+ fn continue_traverse_to_target(&mut self) {
|
|
|
+ while !self.state.is_leaf() {
|
|
|
+ let (current_node, operation_offset) =
|
|
|
+ core::mem::take(&mut self.state).into_node().unwrap();
|
|
|
+
|
|
|
+ let operated_entry = current_node.entry(operation_offset);
|
|
|
if !operated_entry.is_node() {
|
|
|
- self.init();
|
|
|
- return None;
|
|
|
+ self.reset();
|
|
|
+ return;
|
|
|
}
|
|
|
+
|
|
|
self.ancestors.push(current_node);
|
|
|
|
|
|
- current_node = operated_entry.as_node().unwrap();
|
|
|
- operated_entry = self.move_to(current_node);
|
|
|
+ let new_node = operated_entry.as_node_ref().unwrap();
|
|
|
+ self.state.move_to(new_node, self.index);
|
|
|
}
|
|
|
- Some(operated_entry)
|
|
|
}
|
|
|
|
|
|
/// Initialize the Cursor to its initial state.
|
|
|
- pub fn init(&mut self) {
|
|
|
+ fn reset(&mut self) {
|
|
|
self.state = CursorState::default();
|
|
|
- self.ancestors = SmallVec::new();
|
|
|
+ self.ancestors.clear();
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- /// Return the target index of the cursor.
|
|
|
- pub fn index(&self) -> u64 {
|
|
|
- self.index
|
|
|
+struct NodeMutRef<'a, I>
|
|
|
+where
|
|
|
+ I: ItemEntry,
|
|
|
+{
|
|
|
+ inner: DormantMutRef<'a, XNode<I>>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, I: ItemEntry> NodeMutRef<'a, I> {
|
|
|
+ fn new(node: &'a mut XNode<I>, operation_offset: u8) -> (&'a mut XEntry<I>, NodeMutRef<'a, I>) {
|
|
|
+ let (node, inner) = DormantMutRef::new(node);
|
|
|
+ (node.entry_mut(operation_offset), NodeMutRef { inner })
|
|
|
+ }
|
|
|
+
|
|
|
+ unsafe fn awaken(self) -> &'a mut XNode<I> {
|
|
|
+ self.inner.awaken()
|
|
|
}
|
|
|
|
|
|
- /// Determine whether the cursor arrive at the node that can operate target entry.
|
|
|
- /// It can only be used before or after traversing. Since the cursor will only either
|
|
|
- /// not yet started or has already reached the target node when not in a traversal,
|
|
|
- /// it is reasonable to determine whether the cursor has reached its destination node
|
|
|
- /// by checking if the cursor is positioned on a node.
|
|
|
- pub fn is_arrived(&self) -> bool {
|
|
|
- self.state.is_at_node()
|
|
|
+ unsafe fn awaken_modified(self, last_index: u64) -> (&'a mut XNode<I>, bool) {
|
|
|
+ let node = unsafe { self.inner.awaken() };
|
|
|
+ let changed = node.update_mark(node.height().height_offset(last_index));
|
|
|
+ (node, changed)
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -290,115 +390,50 @@ impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> Cursor<'a, I, L, M> {
|
|
|
///
|
|
|
/// When a CursorMut doing write operation on XArray, it should not be affected by other CursorMuts
|
|
|
/// or affect other Cursors.
|
|
|
-pub struct CursorMut<'a, I, L, M>
|
|
|
+pub struct CursorMut<'a, I, M>
|
|
|
where
|
|
|
I: ItemEntry,
|
|
|
- L: XLock,
|
|
|
M: Into<XMark>,
|
|
|
{
|
|
|
/// The `XArray` the cursor located in.
|
|
|
- xa: &'a mut XArray<I, L, M>,
|
|
|
+ xa: DormantMutRef<'a, XArray<I, M>>,
|
|
|
/// The target index of the cursor in the belonged `XArray`.
|
|
|
index: u64,
|
|
|
/// Represents the current state of the cursor.
|
|
|
- state: CursorState<'a, I, L, ReadWrite>,
|
|
|
+ state: CursorState<'a, I, ReadWrite>,
|
|
|
/// Record add nodes passed from the head node to the target position.
|
|
|
/// The index is the height of the recorded node.
|
|
|
- ancestors: SmallVec<[&'a XNode<I, L, ReadWrite>; MAX_HEIGHT]>,
|
|
|
-
|
|
|
- is_exclusive: bool,
|
|
|
-
|
|
|
- _marker: PhantomData<I>,
|
|
|
+ mut_ancestors: SmallVec<[NodeMutRef<'a, I>; MAX_HEIGHT]>,
|
|
|
+ ancestors: SmallVec<[&'a XNode<I>; MAX_HEIGHT]>,
|
|
|
}
|
|
|
|
|
|
-impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> CursorMut<'a, I, L, M> {
|
|
|
+impl<'a, I: ItemEntry, M: Into<XMark>> CursorMut<'a, I, M> {
|
|
|
/// Create an `CursorMut` to perform read and write operations on the `XArray`.
|
|
|
- pub(super) fn new(xa: &'a mut XArray<I, L, M>, index: u64) -> Self {
|
|
|
+ pub(super) fn new(xa: &'a mut XArray<I, M>, index: u64) -> Self {
|
|
|
let mut cursor = Self {
|
|
|
- xa,
|
|
|
+ xa: DormantMutRef::new(xa).1,
|
|
|
index,
|
|
|
state: CursorState::default(),
|
|
|
+ mut_ancestors: SmallVec::new(),
|
|
|
ancestors: SmallVec::new(),
|
|
|
- is_exclusive: false,
|
|
|
- _marker: PhantomData,
|
|
|
};
|
|
|
-
|
|
|
cursor.traverse_to_target();
|
|
|
- cursor
|
|
|
- }
|
|
|
-
|
|
|
- /// Get a reference to current operated entry of the CursorMut.
|
|
|
- fn ref_operated_entry(&self) -> Option<&'a XEntry<I, L>> {
|
|
|
- // SAFETY: The lifetime of the reference to the operated XEntry is equal to `&self`.
|
|
|
- // Hence when the reference existing there will not be other mutable operation in current `CursorMut`,
|
|
|
- // nor will there be any modification operations on the XNode where it resides.
|
|
|
- self.state
|
|
|
- .node_info()
|
|
|
- .map(|info| unsafe { info.0.ref_node_entry(self.is_exclusive, info.1) })
|
|
|
- }
|
|
|
-
|
|
|
- /// Reborrow the target node with a longer lifetime `'a`.
|
|
|
- ///
|
|
|
- /// # Safety
|
|
|
- ///
|
|
|
- /// Users must ensure when this reborrowed reference exists, its corresponding XNode will not be removed.
|
|
|
- pub(super) unsafe fn reborrow_node<Operation>(
|
|
|
- &self,
|
|
|
- node: &XNode<I, L, Operation>,
|
|
|
- ) -> &'a XNode<I, L, Operation> {
|
|
|
- &*(node as *const XNode<I, L, Operation>)
|
|
|
- }
|
|
|
|
|
|
- /// Move the `CursorMut` to the `XNode`, and update the cursor's state based on its target index.
|
|
|
- /// Return a reference to the `XEntry` within the slots of the current XNode that needs to be operated on next.
|
|
|
- fn move_to(&mut self, node: &'a XNode<I, L, ReadWrite>) -> &XEntry<I, L> {
|
|
|
- let offset = node.entry_offset(self.index);
|
|
|
- self.state.arrive_node(node, offset);
|
|
|
- self.ref_operated_entry().unwrap()
|
|
|
- }
|
|
|
-
|
|
|
- /// Initialize the Cursor to its initial state.
|
|
|
- pub fn init(&mut self) {
|
|
|
- self.state = CursorState::default();
|
|
|
- self.ancestors = SmallVec::new();
|
|
|
- self.is_exclusive = false;
|
|
|
+ cursor
|
|
|
}
|
|
|
|
|
|
/// Reset the target index of the Cursor. Once set, it will immediately attempt to move the
|
|
|
/// Cursor to touch the target XEntry.
|
|
|
pub fn reset_to(&mut self, index: u64) {
|
|
|
- self.init();
|
|
|
+ self.reset();
|
|
|
self.index = index;
|
|
|
- self.traverse_to_target();
|
|
|
- }
|
|
|
|
|
|
- /// Load the `XEntry` at the current cursor index within the `XArray`.
|
|
|
- ///
|
|
|
- /// Returns a reference to the `XEntry` at the target index if succeed.
|
|
|
- /// If the cursor cannot reach to the target index, the method will return `None`.
|
|
|
- pub fn load(&self) -> Option<ItemRef<'a, I>> {
|
|
|
- if let Some(entry) = self.ref_operated_entry() {
|
|
|
- entry.as_item_ref()
|
|
|
- } else {
|
|
|
- None
|
|
|
- }
|
|
|
+ self.traverse_to_target();
|
|
|
}
|
|
|
|
|
|
- /// Stores the provided `XEntry` in the `XArray` at the position indicated by the current cursor index.
|
|
|
- ///
|
|
|
- /// If the provided entry is the same as the current entry at the cursor position,
|
|
|
- /// the method returns the provided entry without making changes.
|
|
|
- /// Otherwise, it replaces the current entry with the provided one and returns the old entry.
|
|
|
- pub fn store(&mut self, item: I) -> Option<I> {
|
|
|
- self.ensure_exclusive_before_modify();
|
|
|
- let stored_entry = XEntry::from_item(item);
|
|
|
- let target_entry = self.expand_and_traverse_to_target();
|
|
|
- if stored_entry.raw() == target_entry.raw() {
|
|
|
- return XEntry::into_item(stored_entry);
|
|
|
- }
|
|
|
- let (current_node, operation_offset) = self.state.node_info().unwrap();
|
|
|
- let old_entry = current_node.set_entry(operation_offset, stored_entry);
|
|
|
- XEntry::into_item(old_entry)
|
|
|
+ /// Return the target index of the cursor.
|
|
|
+ pub fn index(&self) -> u64 {
|
|
|
+ self.index
|
|
|
}
|
|
|
|
|
|
/// Move the target index of the cursor to index + 1.
|
|
@@ -406,46 +441,80 @@ impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> CursorMut<'a, I, L, M> {
|
|
|
/// will move to touch the target XEntry. If the move fails, the cursor's state will be
|
|
|
/// set to `Inactive`.
|
|
|
pub fn next(&mut self) {
|
|
|
- // TODO: overflow;
|
|
|
- self.index += 1;
|
|
|
- self.is_exclusive = false;
|
|
|
- if !self.is_arrived() {
|
|
|
- return;
|
|
|
- }
|
|
|
+ self.index = self.index.checked_add(1).unwrap();
|
|
|
|
|
|
- if self.xa.max_index() < self.index {
|
|
|
- self.init();
|
|
|
+ if !self.state.is_at_node() {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
- let (mut current_node, mut operation_offset) = self.state.node_info().unwrap();
|
|
|
+ let (mut current_node, mut operation_offset) = core::mem::take(&mut self.state)
|
|
|
+ .into_node_maybe_mut()
|
|
|
+ .unwrap();
|
|
|
+
|
|
|
operation_offset += 1;
|
|
|
while operation_offset == SLOT_SIZE as u8 {
|
|
|
- operation_offset = current_node.offset_in_parent() + 1;
|
|
|
- if let Some(node) = self.ancestors.pop() {
|
|
|
- current_node = node;
|
|
|
- continue;
|
|
|
- }
|
|
|
+ let offset_in_parent = current_node.offset_in_parent();
|
|
|
+ drop(current_node);
|
|
|
+
|
|
|
+ let parent_node = if let Some(node) = self.ancestors.pop() {
|
|
|
+ NodeMaybeMut::Shared(node)
|
|
|
+ } else if let Some(node) = self.mut_ancestors.pop() {
|
|
|
+ NodeMaybeMut::Exclusive(unsafe { node.awaken_modified(self.index - 1).0 })
|
|
|
+ } else {
|
|
|
+ self.reset();
|
|
|
+ return;
|
|
|
+ };
|
|
|
|
|
|
- operation_offset = 0;
|
|
|
- break;
|
|
|
+ operation_offset = offset_in_parent + 1;
|
|
|
+ current_node = parent_node;
|
|
|
}
|
|
|
- self.state.arrive_node(current_node, operation_offset);
|
|
|
|
|
|
- while !current_node.is_leaf() {
|
|
|
- self.ancestors.push(current_node);
|
|
|
- let next_entry = self.ref_operated_entry().unwrap();
|
|
|
- if !next_entry.is_node() {
|
|
|
- self.init();
|
|
|
- return;
|
|
|
- }
|
|
|
+ self.state.move_to_maybe_mut(current_node, self.index);
|
|
|
+ self.continue_traverse_to_target();
|
|
|
+ }
|
|
|
|
|
|
- // SAFETY: Cursor will move to the `next_node` and the current XNode will not be
|
|
|
- // removed within this function.
|
|
|
- let next_node = unsafe { self.reborrow_node(next_entry.as_node_mut().unwrap()) };
|
|
|
- self.move_to(next_node);
|
|
|
- (current_node, _) = self.state.node_info().unwrap();
|
|
|
- }
|
|
|
+ /// Load the `XEntry` at the current cursor index within the `XArray`.
|
|
|
+ ///
|
|
|
+ /// Returns a reference to the `XEntry` at the target index if succeed.
|
|
|
+ /// If the cursor cannot reach to the target index, the method will return `None`.
|
|
|
+ pub fn load(&mut self) -> Option<ItemRef<'_, I>> {
|
|
|
+ self.traverse_to_target();
|
|
|
+ self.state
|
|
|
+ .as_node()
|
|
|
+ .and_then(|(node, off)| node.entry(off).as_item_ref())
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Judge if the target item is marked with the input `mark`.
|
|
|
+ /// If target item does not exist, the function will return `None`.
|
|
|
+ pub fn is_marked(&mut self, mark: M) -> bool {
|
|
|
+ self.traverse_to_target();
|
|
|
+ self.state
|
|
|
+ .as_node()
|
|
|
+ .map(|(node, off)| node.is_marked(off, mark.into().index()))
|
|
|
+ .unwrap_or(false)
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Stores the provided `XEntry` in the `XArray` at the position indicated by the current cursor index.
|
|
|
+ ///
|
|
|
+ /// If the provided entry is the same as the current entry at the cursor position,
|
|
|
+ /// the method returns the provided entry without making changes.
|
|
|
+ /// Otherwise, it replaces the current entry with the provided one and returns the old entry.
|
|
|
+ pub fn store(&mut self, item: I) -> Option<I> {
|
|
|
+ self.expand_and_traverse_to_target();
|
|
|
+ self.state
|
|
|
+ .as_node_mut()
|
|
|
+ .and_then(|(node, off)| node.set_entry(off, XEntry::from_item(item)).into_item())
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Removes the `XEntry` at the target index of the 'CursorMut' within the `XArray`.
|
|
|
+ ///
|
|
|
+ /// This is achieved by storing an empty `XEntry` at the target index using the `store` method.
|
|
|
+ /// The method returns the replaced `XEntry` that was previously stored at the target index.
|
|
|
+ pub fn remove(&mut self) -> Option<I> {
|
|
|
+ self.ensure_exclusive_before_modification();
|
|
|
+ self.state
|
|
|
+ .as_node_mut()
|
|
|
+ .and_then(|(node, off)| node.set_entry(off, XEntry::EMPTY).into_item())
|
|
|
}
|
|
|
|
|
|
/// Mark the item at the target index in the `XArray` with the input `mark`.
|
|
@@ -455,28 +524,12 @@ impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> CursorMut<'a, I, L, M> {
|
|
|
/// with the input `mark`, because a marked intermediate node should be equivalent to
|
|
|
/// having a child node that is marked.
|
|
|
pub fn set_mark(&mut self, mark: M) -> Result<(), ()> {
|
|
|
- self.ensure_exclusive_before_modify();
|
|
|
- if let Some((current_node, operation_offset)) = self.state.node_info() {
|
|
|
- let item_entry = self.ref_operated_entry().unwrap();
|
|
|
- if item_entry.is_null() {
|
|
|
- return Err(());
|
|
|
- }
|
|
|
-
|
|
|
- let mark_index = mark.into().index();
|
|
|
- current_node.set_mark(operation_offset, mark_index);
|
|
|
-
|
|
|
- let mut offset_in_parent = current_node.offset_in_parent();
|
|
|
- for ancestor in self.ancestors.iter().rev() {
|
|
|
- if ancestor.is_marked(offset_in_parent, mark_index) {
|
|
|
- break;
|
|
|
- }
|
|
|
- ancestor.set_mark(offset_in_parent, mark_index);
|
|
|
- offset_in_parent = ancestor.offset_in_parent();
|
|
|
- }
|
|
|
- Ok(())
|
|
|
- } else {
|
|
|
- Err(())
|
|
|
- }
|
|
|
+ self.ensure_exclusive_before_modification();
|
|
|
+ self.state
|
|
|
+ .as_node_mut()
|
|
|
+ .filter(|(node, off)| node.entry(*off).is_item())
|
|
|
+ .map(|(node, off)| node.set_mark(off, mark.into().index()))
|
|
|
+ .ok_or(())
|
|
|
}
|
|
|
|
|
|
/// Unset the input `mark` for the item at the target index in the `XArray`.
|
|
@@ -485,83 +538,33 @@ impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> CursorMut<'a, I, L, M> {
|
|
|
/// This operation will also unset the input `mark` for all nodes along the path from the head node
|
|
|
/// to the target node if the input `mark` have not marked any of their children.
|
|
|
pub fn unset_mark(&mut self, mark: M) -> Result<(), ()> {
|
|
|
- self.ensure_exclusive_before_modify();
|
|
|
- if let Some((current_node, operation_offset)) = self.state.node_info() {
|
|
|
- let item_entry = self.ref_operated_entry().unwrap();
|
|
|
- if item_entry.is_null() {
|
|
|
- return Err(());
|
|
|
- }
|
|
|
-
|
|
|
- let mark_index = mark.into().index();
|
|
|
- current_node.unset_mark(operation_offset, mark_index);
|
|
|
-
|
|
|
- if current_node.is_mark_clear(mark_index) {
|
|
|
- let mut offset_in_parent = current_node.offset_in_parent();
|
|
|
- for ancestor in self.ancestors.iter().rev() {
|
|
|
- ancestor.unset_mark(offset_in_parent, mark_index);
|
|
|
- if !ancestor.is_mark_clear(mark_index) {
|
|
|
- break;
|
|
|
- }
|
|
|
-
|
|
|
- offset_in_parent = ancestor.offset_in_parent();
|
|
|
- }
|
|
|
- }
|
|
|
- Ok(())
|
|
|
- } else {
|
|
|
- Err(())
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- /// Removes the `XEntry` at the target index of the 'CursorMut' within the `XArray`.
|
|
|
- ///
|
|
|
- /// This is achieved by storing an empty `XEntry` at the target index using the `store` method.
|
|
|
- /// The method returns the replaced `XEntry` that was previously stored at the target index.
|
|
|
- pub fn remove(&mut self) -> Option<I> {
|
|
|
- self.ensure_exclusive_before_modify();
|
|
|
- if let Some((current_node, operation_offset)) = self.state.node_info() {
|
|
|
- let old_entry = current_node.set_entry(operation_offset, XEntry::EMPTY);
|
|
|
- return XEntry::into_item(old_entry);
|
|
|
- }
|
|
|
- None
|
|
|
+ self.ensure_exclusive_before_modification();
|
|
|
+ self.state
|
|
|
+ .as_node_mut()
|
|
|
+ .filter(|(node, off)| node.entry(*off).is_item())
|
|
|
+ .map(|(node, off)| node.unset_mark(off, mark.into().index()))
|
|
|
+ .ok_or(())
|
|
|
}
|
|
|
|
|
|
/// Traverse the XArray and move to the node that can operate the target entry.
|
|
|
/// It then returns the reference to the `XEntry` stored in the slot corresponding to the target index.
|
|
|
/// A target operated XEntry must be an item entry.
|
|
|
/// If can not touch the target entry, the function will return `None`.
|
|
|
- fn traverse_to_target(&mut self) -> Option<&XEntry<I, L>> {
|
|
|
- if self.is_arrived() {
|
|
|
- return self.ref_operated_entry();
|
|
|
+ fn traverse_to_target(&mut self) {
|
|
|
+ if self.state.is_at_node() {
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- let max_index = self.xa.max_index();
|
|
|
- if max_index < self.index || max_index == 0 {
|
|
|
- return None;
|
|
|
- }
|
|
|
+ let xa = unsafe { self.xa.reborrow() };
|
|
|
|
|
|
- if self.is_exclusive {
|
|
|
- self.xa.ensure_head_exclusive();
|
|
|
+ let max_index = xa.max_index();
|
|
|
+ if max_index < self.index || max_index == 0 {
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- let head = self.xa.head().as_node_mut().unwrap();
|
|
|
- // SAFETY: Cursor will move to the `head` and the head will not be
|
|
|
- // removed within this function.
|
|
|
- self.move_to(unsafe { self.reborrow_node(head) });
|
|
|
-
|
|
|
- let (mut current_node, _) = self.state.node_info().unwrap();
|
|
|
- while !current_node.is_leaf() {
|
|
|
- self.ancestors.push(current_node);
|
|
|
- let operated_entry = self.ref_operated_entry().unwrap();
|
|
|
- if !operated_entry.is_node() {
|
|
|
- self.init();
|
|
|
- return None;
|
|
|
- }
|
|
|
- // SAFETY: Cursor will move to the `current_node` and it will not be
|
|
|
- // removed within this function.
|
|
|
- current_node = unsafe { self.reborrow_node(operated_entry.as_node_mut().unwrap()) };
|
|
|
- self.move_to(current_node);
|
|
|
- }
|
|
|
- self.ref_operated_entry()
|
|
|
+ let current_node = xa.head_mut().as_node_maybe_mut().unwrap();
|
|
|
+ self.state.move_to_maybe_mut(current_node, self.index);
|
|
|
+ self.continue_traverse_to_target();
|
|
|
}
|
|
|
|
|
|
/// Traverse the XArray and move to the node that can operate the target entry.
|
|
@@ -572,146 +575,119 @@ impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> CursorMut<'a, I, L, M> {
|
|
|
///
|
|
|
/// It then returns the reference to the `XEntry` stored in the slot corresponding to the target index.
|
|
|
/// A target operated XEntry must be an item entry.
|
|
|
- fn expand_and_traverse_to_target(&mut self) -> &XEntry<I, L> {
|
|
|
- if self.is_arrived() {
|
|
|
- return self.ref_operated_entry().unwrap();
|
|
|
+ fn expand_and_traverse_to_target(&mut self) {
|
|
|
+ if self.state.is_at_node() {
|
|
|
+ self.ensure_exclusive_before_modification();
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- self.expand_height();
|
|
|
+ let head = {
|
|
|
+ let xa = unsafe { self.xa.reborrow() };
|
|
|
+ xa.reserve(self.index);
|
|
|
+ xa.head_mut().as_node_mut_or_cow().unwrap()
|
|
|
+ };
|
|
|
+
|
|
|
+ self.state.move_to_mut(head, self.index);
|
|
|
+ self.exclusively_traverse_to_target();
|
|
|
+ }
|
|
|
|
|
|
- if self.is_exclusive {
|
|
|
- self.xa.ensure_head_exclusive();
|
|
|
+ fn ensure_exclusive_before_modification(&mut self) {
|
|
|
+ if self.state.is_at_node_mut() {
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- let head = self.xa.head().as_node_mut().unwrap();
|
|
|
- // SAFETY: Cursor will move to the `head` and the head will not be
|
|
|
- // removed within this function.
|
|
|
- self.move_to(unsafe { self.reborrow_node(head) });
|
|
|
+ self.state = CursorState::default();
|
|
|
+ self.ancestors.clear();
|
|
|
|
|
|
- let (mut current_node, _) = self.state.node_info().unwrap();
|
|
|
- while !current_node.is_leaf() {
|
|
|
- self.ancestors.push(current_node);
|
|
|
- let mut operated_entry = self.ref_operated_entry().unwrap();
|
|
|
- let current_height = current_node.height();
|
|
|
+ let node = match self.mut_ancestors.pop() {
|
|
|
+ Some(node) => unsafe { node.awaken() },
|
|
|
+ None => {
|
|
|
+ let xa = unsafe { self.xa.reborrow() };
|
|
|
|
|
|
- if !operated_entry.is_node() {
|
|
|
- let (current_node, operation_offset) = self.state.node_info().unwrap();
|
|
|
- let new_owned_entry =
|
|
|
- self.alloc_node(Height::new(*current_height - 1), operation_offset);
|
|
|
- let _ = current_node.set_entry(operation_offset, new_owned_entry);
|
|
|
- // SAFETY: The cursor will move towards the subtree of the current_node,
|
|
|
- // and no further modifications will be made to the current_node within the current function.
|
|
|
- operated_entry =
|
|
|
- unsafe { current_node.ref_node_entry(self.is_exclusive, operation_offset) };
|
|
|
+ let head = xa.head_mut();
|
|
|
+ if !head.is_node() {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ head.as_node_mut_or_cow().unwrap()
|
|
|
}
|
|
|
- // SAFETY: Cursor will move to the `current_node` and it will not be
|
|
|
- // removed within this function.
|
|
|
- current_node = unsafe { self.reborrow_node(operated_entry.as_node_mut().unwrap()) };
|
|
|
+ };
|
|
|
|
|
|
- self.move_to(current_node);
|
|
|
- }
|
|
|
- self.ref_operated_entry().unwrap()
|
|
|
+ self.state.move_to_mut(node, self.index);
|
|
|
+ self.exclusively_traverse_to_target();
|
|
|
}
|
|
|
|
|
|
- /// Increase the number of heights for XArray to expand its capacity, allowing it to accommodate
|
|
|
- /// the target index, and returns the height of the final head node.
|
|
|
- ///
|
|
|
- /// If the head node of the XArray does not exist, allocate a new head node of appropriate height
|
|
|
- /// directly. Otherwise, if needed, repeatedly insert new nodes on top of the current head node to
|
|
|
- /// serve as the new head.
|
|
|
- fn expand_height(&mut self) -> Height {
|
|
|
- if self.xa.head().is_null() {
|
|
|
- let mut head_height = Height::new(1);
|
|
|
- while self.index > head_height.max_index() {
|
|
|
- *head_height += 1;
|
|
|
- }
|
|
|
- let head = self.alloc_node(head_height, 0);
|
|
|
- self.xa.set_head(head);
|
|
|
- return head_height;
|
|
|
- } else {
|
|
|
- loop {
|
|
|
- let head_height = {
|
|
|
- if self.is_exclusive {
|
|
|
- self.xa.ensure_head_exclusive();
|
|
|
+ fn continue_traverse_to_target(&mut self) {
|
|
|
+ while !self.state.is_leaf() {
|
|
|
+ let (current_node, operation_offset) = core::mem::take(&mut self.state)
|
|
|
+ .into_node_maybe_mut()
|
|
|
+ .unwrap();
|
|
|
+
|
|
|
+ let next_node = match current_node {
|
|
|
+ NodeMaybeMut::Shared(node) => {
|
|
|
+ let operated_entry = node.entry(operation_offset);
|
|
|
+ if !operated_entry.is_node() {
|
|
|
+ self.reset();
|
|
|
+ return;
|
|
|
}
|
|
|
- let head = self.xa.head().as_node().unwrap();
|
|
|
- head.height()
|
|
|
- };
|
|
|
|
|
|
- if head_height.max_index() > self.index {
|
|
|
- return head_height;
|
|
|
- }
|
|
|
+ self.ancestors.push(node);
|
|
|
|
|
|
- let new_node_entry = self.alloc_node(Height::new(*head_height + 1), 0);
|
|
|
- let old_head_entry = self.xa.set_head(new_node_entry);
|
|
|
- let old_head = old_head_entry.as_node_mut().unwrap();
|
|
|
- let new_head = self.xa.head().as_node_mut().unwrap();
|
|
|
- for i in 0..3 {
|
|
|
- if !old_head.mark(i).is_clear() {
|
|
|
- new_head.set_mark(0, i);
|
|
|
+ NodeMaybeMut::Shared(operated_entry.as_node_ref().unwrap())
|
|
|
+ }
|
|
|
+ NodeMaybeMut::Exclusive(node) => {
|
|
|
+ let (operated_entry, dormant_node) = NodeMutRef::new(node, operation_offset);
|
|
|
+ if !operated_entry.is_node() {
|
|
|
+ self.reset();
|
|
|
+ return;
|
|
|
}
|
|
|
+
|
|
|
+ self.mut_ancestors.push(dormant_node);
|
|
|
+
|
|
|
+ operated_entry.as_node_maybe_mut().unwrap()
|
|
|
}
|
|
|
- let _empty = new_head.set_entry(0, old_head_entry);
|
|
|
- }
|
|
|
+ };
|
|
|
+
|
|
|
+ self.state.move_to_maybe_mut(next_node, self.index);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- /// Allocate a new XNode with the specified height and offset,
|
|
|
- /// then generate a node entry from it and return it to the caller.
|
|
|
- fn alloc_node(&mut self, height: Height, offset: u8) -> XEntry<I, L> {
|
|
|
- XEntry::from_node(XNode::<I, L, ReadWrite>::new(height, offset))
|
|
|
- }
|
|
|
+ fn exclusively_traverse_to_target(&mut self) {
|
|
|
+ while !self.state.is_leaf() {
|
|
|
+ let (current_node, operation_offset) =
|
|
|
+ core::mem::take(&mut self.state).into_node_mut().unwrap();
|
|
|
|
|
|
- fn ensure_exclusive_before_modify(&mut self) {
|
|
|
- if self.is_exclusive {
|
|
|
- return;
|
|
|
- }
|
|
|
+ if current_node.entry(operation_offset).is_null() {
|
|
|
+ let new_node = XNode::new(current_node.height().go_leaf(), operation_offset);
|
|
|
+ let new_entry = XEntry::from_node(new_node);
|
|
|
+ current_node.set_entry(operation_offset, new_entry);
|
|
|
+ }
|
|
|
|
|
|
- if !self.is_arrived() {
|
|
|
- self.is_exclusive = true;
|
|
|
- return;
|
|
|
- }
|
|
|
+ let (operated_entry, dormant_node) = NodeMutRef::new(current_node, operation_offset);
|
|
|
+ self.mut_ancestors.push(dormant_node);
|
|
|
|
|
|
- if self.xa.head().node_strong_count().unwrap() > 1 {
|
|
|
- self.init();
|
|
|
- self.is_exclusive = true;
|
|
|
- self.traverse_to_target();
|
|
|
- return;
|
|
|
+ let next_node = operated_entry.as_node_mut_or_cow().unwrap();
|
|
|
+ self.state.move_to_mut(next_node, self.index)
|
|
|
}
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Initialize the Cursor to its initial state.
|
|
|
+ fn reset(&mut self) {
|
|
|
+ self.state = CursorState::default();
|
|
|
+ self.ancestors.clear();
|
|
|
|
|
|
- let mut new_ancestors: SmallVec<[&'a XNode<I, L, ReadWrite>; MAX_HEIGHT]> = SmallVec::new();
|
|
|
- for ancestor in self.ancestors.iter() {
|
|
|
- let offset = ancestor.entry_offset(self.index);
|
|
|
- // SAFETY: The process involves descending from the top within the current ancestors to find the first shared node,
|
|
|
- // without modifying the contents of the XNode.
|
|
|
- let entry = unsafe { ancestor.ref_node_entry(false, offset) };
|
|
|
- if entry.node_strong_count().unwrap() > 1 {
|
|
|
- self.move_to(ancestor);
|
|
|
+ while let Some(node) = self.mut_ancestors.pop() {
|
|
|
+ let (_, changed) = unsafe { node.awaken_modified(self.index) };
|
|
|
+ if !changed {
|
|
|
+ self.mut_ancestors.clear();
|
|
|
break;
|
|
|
}
|
|
|
- new_ancestors.push(*ancestor);
|
|
|
- }
|
|
|
- self.ancestors = new_ancestors;
|
|
|
-
|
|
|
- let (mut current_node, _) = self.state.node_info().unwrap();
|
|
|
- self.is_exclusive = true;
|
|
|
- while !current_node.is_leaf() {
|
|
|
- // SAFETY: Cursor will move to the `next_node` and it will not be
|
|
|
- // removed within this function.
|
|
|
- let next_node = unsafe {
|
|
|
- self.reborrow_node(self.ref_operated_entry().unwrap().as_node_mut().unwrap())
|
|
|
- };
|
|
|
- self.ancestors.push(current_node);
|
|
|
- self.move_to(next_node);
|
|
|
- current_node = next_node;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl<'a, I: ItemEntry, L: XLock, M: Into<XMark>> Deref for CursorMut<'a, I, L, M> {
|
|
|
- type Target = Cursor<'a, I, L, M>;
|
|
|
-
|
|
|
- fn deref(&self) -> &Self::Target {
|
|
|
- unsafe { &*(self as *const CursorMut<'a, I, L, M> as *const Cursor<'a, I, L, M>) }
|
|
|
+impl<'a, I: ItemEntry, M: Into<XMark>> Drop for CursorMut<'a, I, M> {
|
|
|
+ fn drop(&mut self) {
|
|
|
+ self.reset();
|
|
|
}
|
|
|
}
|