123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- use core::cmp::Ordering;
- use core::ops::{Deref, DerefMut};
- use crate::entry::{ItemEntry, XEntry};
- use crate::mark::{Mark, NUM_MARKS};
- use crate::xarray::{BITS_PER_LAYER, SLOT_MASK, SLOT_SIZE};
- #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Copy, Clone)]
- pub(super) struct Height {
- height: u8,
- }
- impl Deref for Height {
- type Target = u8;
- fn deref(&self) -> &Self::Target {
- &self.height
- }
- }
- impl DerefMut for Height {
- fn deref_mut(&mut self) -> &mut Self::Target {
- &mut self.height
- }
- }
- impl PartialEq<u8> for Height {
- fn eq(&self, other: &u8) -> bool {
- self.height == *other
- }
- }
- impl PartialOrd<u8> for Height {
- fn partial_cmp(&self, other: &u8) -> Option<Ordering> {
- self.height.partial_cmp(other)
- }
- }
- impl Height {
-
- pub fn new(height: u8) -> Self {
- Self { height }
- }
-
-
- pub fn from_index(index: u64) -> Self {
- let mut height = Height::new(1);
- while index > height.max_index() {
- *height += 1;
- }
- height
- }
-
- pub fn go_root(&self) -> Self {
- Self::new(self.height + 1)
- }
-
- pub fn go_leaf(&self) -> Self {
- Self::new(self.height - 1)
- }
- fn height_shift(&self) -> u8 {
- (self.height - 1) * BITS_PER_LAYER as u8
- }
-
- pub fn height_offset(&self, index: u64) -> u8 {
- ((index >> self.height_shift()) & SLOT_MASK as u64) as u8
- }
-
-
- pub fn max_index(&self) -> u64 {
- ((SLOT_SIZE as u64) << self.height_shift()) - 1
- }
- }
- #[derive(Clone, Debug)]
- pub(super) struct XNode<I>
- where
- I: ItemEntry,
- {
-
-
- height: Height,
-
-
-
-
- offset_in_parent: u8,
-
-
- slots: [XEntry<I>; SLOT_SIZE],
-
-
-
-
- marks: [Mark; NUM_MARKS],
- }
- impl<I: ItemEntry> XNode<I> {
- pub fn new_root(height: Height) -> Self {
- Self::new(height, 0)
- }
- pub fn new(height: Height, offset: u8) -> Self {
- Self {
- height,
- offset_in_parent: offset,
- slots: [XEntry::EMPTY; SLOT_SIZE],
- marks: [Mark::EMPTY; NUM_MARKS],
- }
- }
-
- pub fn entry_offset(&self, target_index: u64) -> u8 {
- self.height.height_offset(target_index)
- }
- pub fn height(&self) -> Height {
- self.height
- }
- pub fn offset_in_parent(&self) -> u8 {
- self.offset_in_parent
- }
- pub fn entry(&self, offset: u8) -> &XEntry<I> {
- &self.slots[offset as usize]
- }
- pub fn entry_mut(&mut self, offset: u8) -> &mut XEntry<I> {
- &mut self.slots[offset as usize]
- }
- pub fn entries_mut(&mut self) -> &mut [XEntry<I>] {
- &mut self.slots
- }
- pub fn is_marked(&self, offset: u8, mark: usize) -> bool {
- self.marks[mark].is_marked(offset)
- }
- pub fn is_mark_clear(&self, mark: usize) -> bool {
- self.marks[mark].is_clear()
- }
- pub fn mark(&self, mark: usize) -> Mark {
- self.marks[mark]
- }
- pub fn is_leaf(&self) -> bool {
- self.height == 1
- }
-
-
-
-
-
-
-
-
-
- pub fn set_entry(&mut self, offset: u8, entry: XEntry<I>) -> XEntry<I> {
- let is_new_node = entry.is_node();
- let old_entry = core::mem::replace(&mut self.slots[offset as usize], entry);
- if is_new_node {
- self.update_mark(offset);
- return old_entry;
- }
- for i in 0..NUM_MARKS {
- self.marks[i].unset(offset);
- }
- old_entry
- }
-
-
-
-
-
- pub fn set_mark(&mut self, offset: u8, mark: usize) {
- self.marks[mark].set(offset);
- }
-
-
-
-
-
- pub fn unset_mark(&mut self, offset: u8, mark: usize) {
- self.marks[mark].unset(offset);
- }
- pub fn clear_mark(&mut self, mark: usize) {
- self.marks[mark].clear();
- }
-
-
-
-
-
-
-
-
-
- pub fn update_mark(&mut self, offset: u8) -> bool {
- let Some(node) = self.slots[offset as usize].as_node_ref() else {
- return false;
- };
- let mut changed = false;
- for i in 0..NUM_MARKS {
- changed |= self.marks[i].update(offset, !node.is_mark_clear(i));
- }
- changed
- }
- }
- pub(super) trait TryClone
- where
- Self: Sized,
- {
- fn try_clone(&self) -> Option<Self>;
- }
- impl<I: ItemEntry> TryClone for XNode<I> {
- default fn try_clone(&self) -> Option<Self> {
- None
- }
- }
- impl<I: ItemEntry + Clone> TryClone for XNode<I> {
- fn try_clone(&self) -> Option<Self> {
- Some(self.clone())
- }
- }
|