node.rs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. use core::cmp::Ordering;
  2. use core::ops::{Deref, DerefMut};
  3. use crate::entry::{ItemEntry, XEntry};
  4. use crate::mark::{Mark, NUM_MARKS};
  5. use crate::xarray::{BITS_PER_LAYER, SLOT_MASK, SLOT_SIZE};
  6. /// The height of an `XNode` within an `XArray`.
  7. ///
  8. /// In an `XArray`, the head has the highest height, while the `XNode`s that directly store items
  9. /// are at the lowest height, with a height value of 1. Each level up from the bottom height
  10. /// increases the height number by 1. The height of an `XArray` is the height of its head.
  11. #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone)]
  12. pub(super) struct Height {
  13. height: u8,
  14. }
  15. impl Deref for Height {
  16. type Target = u8;
  17. fn deref(&self) -> &Self::Target {
  18. &self.height
  19. }
  20. }
  21. impl DerefMut for Height {
  22. fn deref_mut(&mut self) -> &mut Self::Target {
  23. &mut self.height
  24. }
  25. }
  26. impl PartialEq<u8> for Height {
  27. fn eq(&self, other: &u8) -> bool {
  28. self.height == *other
  29. }
  30. }
  31. impl PartialOrd<u8> for Height {
  32. fn partial_cmp(&self, other: &u8) -> Option<Ordering> {
  33. self.height.partial_cmp(other)
  34. }
  35. }
  36. impl Height {
  37. /// Creates a `Height` directly from a height value.
  38. pub fn new(height: u8) -> Self {
  39. Self { height }
  40. }
  41. /// Creates a `Height` which has the mininal height value but allows the `index`-th item to be
  42. /// stored.
  43. pub fn from_index(index: u64) -> Self {
  44. let mut height = Height::new(1);
  45. while index > height.max_index() {
  46. *height += 1;
  47. }
  48. height
  49. }
  50. /// Goes up, which increases the height velue by one.
  51. pub fn go_root(&self) -> Self {
  52. Self::new(self.height + 1)
  53. }
  54. /// Goes down, which decreases the height value by one.
  55. pub fn go_leaf(&self) -> Self {
  56. Self::new(self.height - 1)
  57. }
  58. fn height_shift(&self) -> u8 {
  59. (self.height - 1) * BITS_PER_LAYER as u8
  60. }
  61. /// Calculates the corresponding offset for the target index at the current height.
  62. pub fn height_offset(&self, index: u64) -> u8 {
  63. ((index >> self.height_shift()) & SLOT_MASK as u64) as u8
  64. }
  65. /// Calculates the maximum index that can be represented in an `XArray` with the current
  66. /// height.
  67. pub fn max_index(&self) -> u64 {
  68. ((SLOT_SIZE as u64) << self.height_shift()) - 1
  69. }
  70. }
  71. /// The `XNode` is the intermediate node in the tree-like structure of the `XArray`.
  72. ///
  73. /// It contains `SLOT_SIZE` number of `XEntry`s, meaning it can accommodate up to `SLOT_SIZE` child
  74. /// nodes. The `height` and `offset_in_parent` attributes of an `XNode` are determined at
  75. /// initialization and remain unchanged thereafter.
  76. #[derive(Clone, Debug)]
  77. pub(super) struct XNode<I>
  78. where
  79. I: ItemEntry,
  80. {
  81. /// The height of the subtree rooted at the current node. The height of a leaf node,
  82. /// which stores the user-given items, is 1.
  83. height: Height,
  84. /// This node is its parent's `offset_in_parent`-th child.
  85. ///
  86. /// This field will be zero if this node is the root, as the node will be the 0-th child of its
  87. /// parent once the height of `XArray` is increased.
  88. offset_in_parent: u8,
  89. /// The slots storing `XEntry`s, which point to user-given items for leaf nodes and other
  90. /// `XNode`s for interior nodes.
  91. slots: [XEntry<I>; SLOT_SIZE],
  92. /// The marks representing whether each slot is marked or not.
  93. ///
  94. /// Users can set mark or unset mark on user-given items, and a leaf node or an interior node
  95. /// is marked if and only if there is at least one marked item within the node.
  96. marks: [Mark; NUM_MARKS],
  97. }
  98. impl<I: ItemEntry> XNode<I> {
  99. pub fn new_root(height: Height) -> Self {
  100. Self::new(height, 0)
  101. }
  102. pub fn new(height: Height, offset: u8) -> Self {
  103. Self {
  104. height,
  105. offset_in_parent: offset,
  106. slots: [XEntry::EMPTY; SLOT_SIZE],
  107. marks: [Mark::EMPTY; NUM_MARKS],
  108. }
  109. }
  110. /// Get the slot offset at the current `XNode` for the target index `target_index`.
  111. pub fn entry_offset(&self, target_index: u64) -> u8 {
  112. self.height.height_offset(target_index)
  113. }
  114. pub fn height(&self) -> Height {
  115. self.height
  116. }
  117. pub fn offset_in_parent(&self) -> u8 {
  118. self.offset_in_parent
  119. }
  120. pub fn entry(&self, offset: u8) -> &XEntry<I> {
  121. &self.slots[offset as usize]
  122. }
  123. pub fn entry_mut(&mut self, offset: u8) -> &mut XEntry<I> {
  124. &mut self.slots[offset as usize]
  125. }
  126. pub fn entries_mut(&mut self) -> &mut [XEntry<I>] {
  127. &mut self.slots
  128. }
  129. pub fn is_marked(&self, offset: u8, mark: usize) -> bool {
  130. self.marks[mark].is_marked(offset)
  131. }
  132. pub fn is_mark_clear(&self, mark: usize) -> bool {
  133. self.marks[mark].is_clear()
  134. }
  135. pub fn mark(&self, mark: usize) -> Mark {
  136. self.marks[mark]
  137. }
  138. pub fn is_leaf(&self) -> bool {
  139. self.height == 1
  140. }
  141. /// Sets the slot at the given `offset` to the given `entry`.
  142. ///
  143. /// If `entry` represents an item, the old marks at the same offset will be cleared. Otherwise,
  144. /// if `entry` represents a node, the marks at the same offset will be updated according to
  145. /// whether the new node contains marked items.
  146. ///
  147. /// This method changes the mark _only_ on this `XNode'. It's the caller's responsibility to
  148. /// ensure that the marks on the ancestors of this `XNode' are up to date. See also
  149. /// [`XNode::update_mark`].
  150. pub fn set_entry(&mut self, offset: u8, entry: XEntry<I>) -> XEntry<I> {
  151. let is_new_node = entry.is_node();
  152. let old_entry = core::mem::replace(&mut self.slots[offset as usize], entry);
  153. if is_new_node {
  154. self.update_mark(offset);
  155. return old_entry;
  156. }
  157. for i in 0..NUM_MARKS {
  158. self.marks[i].unset(offset);
  159. }
  160. old_entry
  161. }
  162. /// Sets the input `mark` at the given `offset`.
  163. ///
  164. /// This method changes the mark _only_ on this `XNode'. It's the caller's responsibility to
  165. /// ensure that the marks on the ancestors of this `XNode' are up to date. See also
  166. /// [`XNode::update_mark`].
  167. pub fn set_mark(&mut self, offset: u8, mark: usize) {
  168. self.marks[mark].set(offset);
  169. }
  170. /// Unsets the input `mark` at the given `offset`.
  171. ///
  172. /// This method changes the mark _only_ on this `XNode'. It's the caller's responsibility to
  173. /// ensure that the marks on the ancestors of this `XNode' are up to date. See also
  174. /// [`XNode::update_mark`].
  175. pub fn unset_mark(&mut self, offset: u8, mark: usize) {
  176. self.marks[mark].unset(offset);
  177. }
  178. pub fn clear_mark(&mut self, mark: usize) {
  179. self.marks[mark].clear();
  180. }
  181. /// Updates the mark at the given `offset` and returns `true` if the mark is changed.
  182. ///
  183. /// This method does nothing if the slot at the given `offset` does not represent a node. It
  184. /// assumes the marks of the child node are up to date, and ensures the mark at the given
  185. /// `offset` is also up to date.
  186. ///
  187. /// Whenever a mark at the leaf node changes, this method should be invoked from the leaf node
  188. /// up to the root node, until the mark does not change on some node or the root node has been
  189. /// reached.
  190. pub fn update_mark(&mut self, offset: u8) -> bool {
  191. let Some(node) = self.slots[offset as usize].as_node_ref() else {
  192. return false;
  193. };
  194. let mut changed = false;
  195. for i in 0..NUM_MARKS {
  196. changed |= self.marks[i].update(offset, !node.is_mark_clear(i));
  197. }
  198. changed
  199. }
  200. }
  201. pub(super) trait TryClone
  202. where
  203. Self: Sized,
  204. {
  205. fn try_clone(&self) -> Option<Self>;
  206. }
  207. impl<I: ItemEntry> TryClone for XNode<I> {
  208. default fn try_clone(&self) -> Option<Self> {
  209. None
  210. }
  211. }
  212. impl<I: ItemEntry + Clone> TryClone for XNode<I> {
  213. fn try_clone(&self) -> Option<Self> {
  214. Some(self.clone())
  215. }
  216. }