From a6a0a8cc9f6def84161bafef0580e1eb88d31d92 Mon Sep 17 00:00:00 2001 From: Changgyoo Park Date: Fri, 12 Apr 2024 21:21:22 +0200 Subject: [PATCH] feat(tree_index): relax trait bounds --- .gitignore | 3 + CHANGELOG.md | 4 + Cargo.toml | 2 +- src/hash_table/bucket_array.rs | 34 +-- src/tree_index.rs | 157 +++++------- src/tree_index/internal_node.rs | 132 ++++------ src/tree_index/leaf.rs | 439 +++++++++++++++----------------- src/tree_index/leaf_node.rs | 133 ++++------ src/tree_index/node.rs | 24 +- 9 files changed, 411 insertions(+), 517 deletions(-) diff --git a/.gitignore b/.gitignore index 81c6bb08..fc675721 100644 --- a/.gitignore +++ b/.gitignore @@ -9,6 +9,9 @@ Cargo.lock # These are backup files generated by rustfmt **/*.rs.bk +# IntelliJ +.idea + # macOS **/.DS_Store diff --git a/CHANGELOG.md b/CHANGELOG.md index 12fc1f40..f50092ac 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,10 @@ ## Version 2 +2.0.20 + +* Relax trait bounds of `TreeIndex`. + 2.0.19 * Remove unnecessary trait bounds from type definitions of `HashCache`, `HashIndex`, `HashMap`, and `HashSet`. diff --git a/Cargo.toml b/Cargo.toml index d5e35f4f..c865d8ca 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -2,7 +2,7 @@ name = "scc" description = "High performance containers and utilities for concurrent and asynchronous programming" documentation = "https://docs.rs/scc" -version = "2.0.19" +version = "2.0.20" authors = ["wvwwvwwv "] edition = "2021" rust-version = "1.65.0" diff --git a/src/hash_table/bucket_array.rs b/src/hash_table/bucket_array.rs index 986ab65f..64efd716 100644 --- a/src/hash_table/bucket_array.rs +++ b/src/hash_table/bucket_array.rs @@ -24,6 +24,13 @@ impl BucketArray { self.array_len } + /// Returns a reference to a [`Bucket`] at the given position. + #[inline] + pub(crate) fn bucket(&self, index: usize) -> &Bucket { + debug_assert!(index < self.num_buckets()); + unsafe { &(*(self.bucket_ptr.add(index))) } + } + /// Returns a mutable reference to a [`Bucket`] at the given position. #[allow(clippy::mut_from_ref)] #[inline] @@ -53,19 +60,6 @@ impl BucketArray { } impl BucketArray { - /// Returns the minimum capacity. - #[inline] - pub const fn minimum_capacity() -> usize { - BUCKET_LEN << 1 - } - - /// Returns the partial hash value of the given hash. - #[allow(clippy::cast_possible_truncation)] - #[inline] - pub(crate) const fn partial_hash(hash: u64) -> u8 { - (hash % (1 << 8)) as u8 - } - /// Creates a new [`BucketArray`] of the given capacity. /// /// `capacity` is the desired number entries, not the number of [`Bucket`] instances. @@ -132,11 +126,17 @@ impl BucketArray { } } - /// Returns a reference to a [`Bucket`] at the given position. + /// Returns the minimum capacity. #[inline] - pub(crate) fn bucket(&self, index: usize) -> &Bucket { - debug_assert!(index < self.num_buckets()); - unsafe { &(*(self.bucket_ptr.add(index))) } + pub const fn minimum_capacity() -> usize { + BUCKET_LEN << 1 + } + + /// Returns the partial hash value of the given hash. + #[allow(clippy::cast_possible_truncation)] + #[inline] + pub(crate) const fn partial_hash(hash: u64) -> u8 { + (hash % (1 << 8)) as u8 } /// Returns a reference to its rehashing metadata. diff --git a/src/tree_index.rs b/src/tree_index.rs index d590588c..6deb118e 100644 --- a/src/tree_index.rs +++ b/src/tree_index.rs @@ -55,11 +55,7 @@ use std::sync::atomic::Ordering::{AcqRel, Acquire, Relaxed}; /// /// [`TreeIndex`] is impervious to out-of-memory errors and panics in user specified code on one /// condition; `K::drop` and `V::drop` must not panic. -pub struct TreeIndex -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct TreeIndex { root: AtomicShared>, } @@ -67,23 +63,14 @@ where /// /// An [`Iter`] iterates over all the entries that survive the [`Iter`] in monotonically increasing /// order. -pub struct Iter<'t, 'g, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct Iter<'t, 'g, K, V> { root: &'t AtomicShared>, leaf_scanner: Option>, guard: &'g Guard, } /// An iterator over a sub-range of entries in a [`TreeIndex`]. -pub struct Range<'t, 'g, K, V, R> -where - K: 'static + Clone + Ord, - V: 'static + Clone, - R: RangeBounds, -{ +pub struct Range<'t, 'g, K, V, R: RangeBounds> { root: &'t AtomicShared>, leaf_scanner: Option>, range: R, @@ -92,11 +79,7 @@ where guard: &'g Guard, } -impl TreeIndex -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl TreeIndex { /// Creates an empty [`TreeIndex`]. /// /// # Examples @@ -114,6 +97,48 @@ where } } + /// Clears the [`TreeIndex`]. + /// + /// # Examples + /// + /// ``` + /// use scc::TreeIndex; + /// + /// let treeindex: TreeIndex = TreeIndex::new(); + /// + /// treeindex.clear(); + /// assert_eq!(treeindex.len(), 0); + /// ``` + #[inline] + pub fn clear(&self) { + self.root.swap((None, Tag::None), Relaxed); + } + + /// Returns the depth of the [`TreeIndex`]. + /// + /// # Examples + /// + /// ``` + /// use scc::TreeIndex; + /// + /// let treeindex: TreeIndex = TreeIndex::new(); + /// assert_eq!(treeindex.depth(), 0); + /// ``` + #[inline] + pub fn depth(&self) -> usize { + let guard = Guard::new(); + self.root + .load(Acquire, &guard) + .as_ref() + .map_or(0, |root_ref| root_ref.depth(1, &guard)) + } +} + +impl TreeIndex +where + K: 'static + Clone + Ord, + V: 'static + Clone, +{ /// Inserts a key-value pair. /// /// # Errors @@ -572,23 +597,6 @@ where self.peek(key, &Guard::new()).is_some() } - /// Clears the [`TreeIndex`]. - /// - /// # Examples - /// - /// ``` - /// use scc::TreeIndex; - /// - /// let treeindex: TreeIndex = TreeIndex::new(); - /// - /// treeindex.clear(); - /// assert_eq!(treeindex.len(), 0); - /// ``` - #[inline] - pub fn clear(&self) { - self.root.swap((None, Tag::None), Relaxed); - } - /// Returns the size of the [`TreeIndex`]. /// /// It internally scans all the leaf nodes, and therefore the time complexity is O(N). @@ -624,25 +632,6 @@ where !self.iter(&guard).any(|_| true) } - /// Returns the depth of the [`TreeIndex`]. - /// - /// # Examples - /// - /// ``` - /// use scc::TreeIndex; - /// - /// let treeindex: TreeIndex = TreeIndex::new(); - /// assert_eq!(treeindex.depth(), 0); - /// ``` - #[inline] - pub fn depth(&self) -> usize { - let guard = Guard::new(); - self.root - .load(Acquire, &guard) - .as_ref() - .map_or(0, |root_ref| root_ref.depth(1, &guard)) - } - /// Returns an [`Iter`]. /// /// The returned [`Iter`] starts scanning from the minimum key-value pair. Key-value pairs @@ -721,11 +710,7 @@ where } } -impl Default for TreeIndex -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Default for TreeIndex { /// Creates a [`TreeIndex`] with the default parameters. /// /// # Examples @@ -754,11 +739,7 @@ where } } -impl<'t, 'g, K, V> Iter<'t, 'g, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl<'t, 'g, K, V> Iter<'t, 'g, K, V> { #[inline] fn new(root: &'t AtomicShared>, guard: &'g Guard) -> Iter<'t, 'g, K, V> { Iter::<'t, 'g, K, V> { @@ -769,11 +750,7 @@ where } } -impl<'t, 'g, K, V> Debug for Iter<'t, 'g, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl<'t, 'g, K, V> Debug for Iter<'t, 'g, K, V> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("Iter") .field("root", &self.root) @@ -829,19 +806,9 @@ where { } -impl<'t, 'g, K, V> UnwindSafe for Iter<'t, 'g, K, V> -where - K: 'static + Clone + Ord + UnwindSafe, - V: 'static + Clone + UnwindSafe, -{ -} +impl<'t, 'g, K, V> UnwindSafe for Iter<'t, 'g, K, V> {} -impl<'t, 'g, K, V, R> Range<'t, 'g, K, V, R> -where - K: 'static + Clone + Ord, - V: 'static + Clone, - R: RangeBounds, -{ +impl<'t, 'g, K, V, R: RangeBounds> Range<'t, 'g, K, V, R> { #[inline] fn new( root: &'t AtomicShared>, @@ -857,7 +824,14 @@ where guard, } } +} +impl<'t, 'g, K, V, R> Range<'t, 'g, K, V, R> +where + K: 'static + Clone + Ord, + V: 'static + Clone, + R: RangeBounds, +{ #[inline] fn next_unbounded(&mut self) -> Option<(&'g K, &'g V)> { if self.leaf_scanner.is_none() { @@ -929,12 +903,7 @@ where } } -impl<'t, 'g, K, V, R> Debug for Range<'t, 'g, K, V, R> -where - K: 'static + Clone + Ord, - V: 'static + Clone, - R: RangeBounds, -{ +impl<'t, 'g, K, V, R: RangeBounds> Debug for Range<'t, 'g, K, V, R> { #[inline] fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("Range") @@ -1005,10 +974,4 @@ where { } -impl<'t, 'g, K, V, R> UnwindSafe for Range<'t, 'g, K, V, R> -where - K: 'static + Clone + Ord + UnwindSafe, - V: 'static + Clone + UnwindSafe, - R: RangeBounds + UnwindSafe, -{ -} +impl<'t, 'g, K, V, R> UnwindSafe for Range<'t, 'g, K, V, R> where R: RangeBounds + UnwindSafe {} diff --git a/src/tree_index/internal_node.rs b/src/tree_index/internal_node.rs index 7ed2ef28..c2301e01 100644 --- a/src/tree_index/internal_node.rs +++ b/src/tree_index/internal_node.rs @@ -16,11 +16,7 @@ use std::sync::atomic::{AtomicPtr, AtomicU8}; /// Internal node. /// /// The layout of an internal node: `|ptr(children)/max(child keys)|...|ptr(children)|`. -pub struct InternalNode -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct InternalNode { /// Children of the [`InternalNode`]. pub(super) children: Leaf>>, @@ -41,11 +37,7 @@ where } /// [`Locker`] holds exclusive ownership of a [`InternalNode`]. -pub(super) struct Locker<'n, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub(super) struct Locker<'n, K, V> { internal_node: &'n InternalNode, } @@ -53,11 +45,7 @@ where /// /// `AtomicPtr` members may point to values under the protection of the [`Guard`] used for the /// split operation. -struct StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +struct StructuralChange { origin_node_key: AtomicPtr, origin_node: AtomicShared>, low_key_node: AtomicShared>, @@ -65,11 +53,7 @@ where high_key_node: AtomicShared>, } -impl InternalNode -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl InternalNode { /// Creates a new empty internal node. #[inline] pub(super) fn new() -> InternalNode { @@ -98,6 +82,51 @@ where self.unbounded_child.tag(mo) == RETIRED } + /// Waits for the lock on the [`InternalNode`] to be released. + #[inline] + pub(super) fn wait(&self, async_wait: &mut D) { + let waiter = || { + if self.latch.load(Relaxed) == LOCKED.into() { + // The `InternalNode` is being split or locked. + return Err(()); + } + Ok(()) + }; + + if let Some(async_wait) = async_wait.derive() { + let _result = self.wait_queue.push_async_entry(async_wait, waiter); + } else { + let _result = self.wait_queue.wait_sync(waiter); + } + } + + /// Tries to lock the [`InternalNode`]. + fn try_lock(&self) -> bool { + self.latch + .compare_exchange(Tag::None.into(), LOCKED.into(), Acquire, Relaxed) + .is_ok() + } + + /// Unlocks the [`InternalNode`]. + fn unlock(&self) { + debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); + self.latch.store(Tag::None.into(), Release); + self.wait_queue.signal(); + } + + /// Retires itself. + fn retire(&self) { + debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); + self.latch.store(RETIRED.into(), Release); + self.wait_queue.signal(); + } +} + +impl InternalNode +where + K: 'static + Clone + Ord, + V: 'static + Clone, +{ /// Searches for an entry associated with the given key. #[inline] pub(super) fn search<'g, Q>(&self, key: &Q, guard: &'g Guard) -> Option<&'g V> @@ -878,24 +907,6 @@ where false } - /// Waits for the lock on the [`InternalNode`] to be released. - #[inline] - pub(super) fn wait(&self, async_wait: &mut D) { - let waiter = || { - if self.latch.load(Relaxed) == LOCKED.into() { - // The `InternalNode` is being split or locked. - return Err(()); - } - Ok(()) - }; - - if let Some(async_wait) = async_wait.derive() { - let _result = self.wait_queue.push_async_entry(async_wait, waiter); - } else { - let _result = self.wait_queue.wait_sync(waiter); - } - } - /// Tries to coalesce nodes. fn coalesce(&self, guard: &Guard) -> RemoveResult where @@ -1004,34 +1015,9 @@ where } false } - - /// Tries to lock the [`InternalNode`]. - fn try_lock(&self) -> bool { - self.latch - .compare_exchange(Tag::None.into(), LOCKED.into(), Acquire, Relaxed) - .is_ok() - } - - /// Unlocks the [`InternalNode`]. - fn unlock(&self) { - debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); - self.latch.store(Tag::None.into(), Release); - self.wait_queue.signal(); - } - - /// Retires itself. - fn retire(&self) { - debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); - self.latch.store(RETIRED.into(), Release); - self.wait_queue.signal(); - } } -impl<'n, K, V> Locker<'n, K, V> -where - K: Clone + Ord, - V: Clone, -{ +impl<'n, K, V> Locker<'n, K, V> { /// Acquires exclusive lock on the [`InternalNode`]. #[inline] pub(super) fn try_lock(internal_node: &'n InternalNode) -> Option> { @@ -1049,22 +1035,14 @@ where } } -impl<'n, K, V> Drop for Locker<'n, K, V> -where - K: Clone + Ord, - V: Clone, -{ +impl<'n, K, V> Drop for Locker<'n, K, V> { #[inline] fn drop(&mut self) { self.internal_node.unlock(); } } -impl StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl StructuralChange { fn reset(&self) -> Option>> { self.origin_node_key.store(ptr::null_mut(), Relaxed); self.low_key_node.swap((None, Tag::None), Relaxed); @@ -1074,11 +1052,7 @@ where } } -impl Default for StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Default for StructuralChange { #[inline] fn default() -> Self { Self { diff --git a/src/tree_index/leaf.rs b/src/tree_index/leaf.rs index 8197658c..631e7382 100644 --- a/src/tree_index/leaf.rs +++ b/src/tree_index/leaf.rs @@ -11,11 +11,7 @@ use std::sync::atomic::Ordering::{AcqRel, Acquire, Relaxed, Release}; /// /// A constructed key-value pair entry is never dropped until the entire [`Leaf`] instance is /// dropped. -pub struct Leaf -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct Leaf { /// The metadata containing information about the [`Leaf`] and individual entries. /// /// The state of each entry is as follows. @@ -85,11 +81,7 @@ pub enum RemoveResult { Frozen, } -impl Leaf -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Leaf { /// Creates a new [`Leaf`]. #[inline] pub(super) const fn new() -> Leaf { @@ -167,6 +159,154 @@ where None } + /// Inserts a key value pair at the specified position without checking the metadata. + /// + /// `rank` is calculated as `index + 1`. + #[inline] + pub(super) fn insert_unchecked(&self, key: K, val: V, index: usize) { + debug_assert!(index < DIMENSION.num_entries); + let metadata = self.metadata.load(Relaxed); + let new_metadata = DIMENSION.augment(metadata, index, index + 1); + self.write(index, key, val); + self.metadata.store(new_metadata, Release); + } + + /// Compares the given metadata value with the current one. + #[inline] + pub(super) fn validate(&self, metadata: usize) -> bool { + // `Relaxed` is sufficient as long as the caller has read-acquired its contents. + self.metadata.load(Relaxed) == metadata + } + + /// Freezes the [`Leaf`] temporarily. + /// + /// A frozen [`Leaf`] cannot store more entries, and on-going insertion is canceled. + #[inline] + pub(super) fn freeze(&self) -> bool { + self.metadata + .fetch_update(AcqRel, Acquire, |p| { + if Dimension::frozen(p) { + None + } else { + Some(Dimension::freeze(p)) + } + }) + .is_ok() + } + + /// Returns the recommended number of entries that the left-side node shall store when a + /// [`Leaf`] is split. + /// + /// Returns a number in `[1, len(leaf))` that represents the recommended number of entries in + /// the left-side node. The number is calculated as, for each adjacent slots, + /// - Initial `score = len(leaf)`. + /// - Rank increased: `score -= 1`. + /// - Rank decreased: `score += 1`. + /// - Clamp `score` in `[len(leaf) / 2 + 1, len(leaf) / 2 + len(leaf) - 1)`. + /// - Take `score - len(leaf) / 2`. + /// + /// For instance, when the length of a [`Leaf`] is 7, + /// - Returns 6 for `rank = [1, 2, 3, 4, 5, 6, 7]`. + /// - Returns 1 for `rank = [7, 6, 5, 4, 3, 2, 1]`. + #[inline] + pub(super) fn optimal_boundary(mut mutable_metadata: usize) -> usize { + let mut boundary: usize = DIMENSION.num_entries; + let mut prev_rank = 1; + for _ in 0..DIMENSION.num_entries { + let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); + if rank != 0 && rank != DIMENSION.removed_rank() { + if prev_rank < rank { + boundary += 1; + } else { + boundary -= 1; + } + prev_rank = rank; + } + mutable_metadata >>= DIMENSION.num_bits_per_entry; + } + boundary.clamp( + DIMENSION.num_entries / 2 + 1, + DIMENSION.num_entries + DIMENSION.num_entries / 2 - 1, + ) - DIMENSION.num_entries / 2 + } + + const fn key_at(&self, index: usize) -> &K { + unsafe { &*self.entry_array.0[index].as_ptr() } + } + + const fn value_at(&self, index: usize) -> &V { + unsafe { &*self.entry_array.1[index].as_ptr() } + } + + fn rollback(&self, index: usize) -> InsertResult { + let (key, val) = self.take(index); + let result = self + .metadata + .fetch_and(!DIMENSION.rank_mask(index), Relaxed) + & (!DIMENSION.rank_mask(index)); + if Dimension::retired(result) { + InsertResult::Retired(key, val) + } else if Dimension::frozen(result) { + InsertResult::Frozen(key, val) + } else { + InsertResult::Duplicate(key, val) + } + } + + fn take(&self, index: usize) -> (K, V) { + unsafe { + ( + self.entry_array.0[index].as_ptr().read(), + self.entry_array.1[index].as_ptr().read(), + ) + } + } + + fn write(&self, index: usize, key: K, val: V) { + unsafe { + (self.entry_array.0[index].as_ptr().cast_mut()).write(key); + (self.entry_array.1[index].as_ptr().cast_mut()).write(val); + } + } + + /// Returns the index of the corresponding entry of the next higher ranked entry. + fn next(index: usize, mut mutable_metadata: usize) -> usize { + debug_assert_ne!(index, usize::MAX); + let current_entry_rank = if index == DIMENSION.num_entries { + 0 + } else { + DIMENSION.rank(mutable_metadata, index) + }; + let mut next_index = DIMENSION.num_entries; + if current_entry_rank < DIMENSION.num_entries { + let mut next_rank = DIMENSION.removed_rank(); + for i in 0..DIMENSION.num_entries { + if mutable_metadata == 0 { + break; + } + if i != index { + let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); + if rank != Dimension::uninit_rank() && rank < next_rank { + if rank == current_entry_rank + 1 { + return i; + } else if rank > current_entry_rank { + next_rank = rank; + next_index = i; + } + } + } + mutable_metadata >>= DIMENSION.num_bits_per_entry; + } + } + next_index + } +} + +impl Leaf +where + K: 'static + Clone + Ord, + V: 'static + Clone, +{ /// Inserts a key value pair. #[inline] pub(super) fn insert(&self, key: K, val: V) -> InsertResult { @@ -208,18 +348,6 @@ where } } - /// Inserts a key value pair at the specified position without checking the metadata. - /// - /// `rank` is calculated as `index + 1`. - #[inline] - pub(super) fn insert_unchecked(&self, key: K, val: V, index: usize) { - debug_assert!(index < DIMENSION.num_entries); - let metadata = self.metadata.load(Relaxed); - let new_metadata = DIMENSION.augment(metadata, index, index + 1); - self.write(index, key, val); - self.metadata.store(new_metadata, Release); - } - /// Removes the key if the condition is met. #[inline] pub(super) fn remove_if bool>( @@ -416,29 +544,6 @@ where (None, metadata) } - /// Compares the given metadata value with the current one. - #[inline] - pub(super) fn validate(&self, metadata: usize) -> bool { - // `Relaxed` is sufficient as long as the caller has read-acquired its contents. - self.metadata.load(Relaxed) == metadata - } - - /// Freezes the [`Leaf`] temporarily. - /// - /// A frozen [`Leaf`] cannot store more entries, and on-going insertion is canceled. - #[inline] - pub(super) fn freeze(&self) -> bool { - self.metadata - .fetch_update(AcqRel, Acquire, |p| { - if Dimension::frozen(p) { - None - } else { - Some(Dimension::freeze(p)) - } - }) - .is_ok() - } - /// Freezes the [`Leaf`] and distribute entries to two new leaves. #[inline] pub(super) fn freeze_and_distribute( @@ -477,77 +582,6 @@ where } } - /// Returns the recommended number of entries that the left-side node shall store when a - /// [`Leaf`] is split. - /// - /// Returns a number in `[1, len(leaf))` that represents the recommended number of entries in - /// the left-side node. The number is calculated as, for each adjacent slots, - /// - Initial `score = len(leaf)`. - /// - Rank increased: `score -= 1`. - /// - Rank decreased: `score += 1`. - /// - Clamp `score` in `[len(leaf) / 2 + 1, len(leaf) / 2 + len(leaf) - 1)`. - /// - Take `score - len(leaf) / 2`. - /// - /// For instance, when the length of a [`Leaf`] is 7, - /// - Returns 6 for `rank = [1, 2, 3, 4, 5, 6, 7]`. - /// - Returns 1 for `rank = [7, 6, 5, 4, 3, 2, 1]`. - #[inline] - pub(super) fn optimal_boundary(mut mutable_metadata: usize) -> usize { - let mut boundary: usize = DIMENSION.num_entries; - let mut prev_rank = 1; - for _ in 0..DIMENSION.num_entries { - let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); - if rank != 0 && rank != DIMENSION.removed_rank() { - if prev_rank < rank { - boundary += 1; - } else { - boundary -= 1; - } - prev_rank = rank; - } - mutable_metadata >>= DIMENSION.num_bits_per_entry; - } - boundary.clamp( - DIMENSION.num_entries / 2 + 1, - DIMENSION.num_entries + DIMENSION.num_entries / 2 - 1, - ) - DIMENSION.num_entries / 2 - } - - /// Searches for a slot in which the key is stored. - fn search_slot(&self, key: &Q, mut mutable_metadata: usize) -> Option - where - K: Borrow, - Q: Ord + ?Sized, - { - let mut min_max_rank = DIMENSION.removed_rank(); - let mut max_min_rank = 0; - for i in 0..DIMENSION.num_entries { - if mutable_metadata == 0 { - break; - } - let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); - if rank < min_max_rank && rank > max_min_rank { - match self.compare(i, key) { - Ordering::Less => { - if max_min_rank < rank { - max_min_rank = rank; - } - } - Ordering::Greater => { - if min_max_rank > rank { - min_max_rank = rank; - } - } - Ordering::Equal => { - return Some(i); - } - } - } - mutable_metadata >>= DIMENSION.num_bits_per_entry; - } - None - } - /// Post-processing after reserving a free slot. fn post_insert(&self, free_slot_index: usize, mut prev_metadata: usize) -> InsertResult { let key = self.key_at(free_slot_index); @@ -602,91 +636,51 @@ where } } - fn rollback(&self, index: usize) -> InsertResult { - let (key, val) = self.take(index); - let result = self - .metadata - .fetch_and(!DIMENSION.rank_mask(index), Relaxed) - & (!DIMENSION.rank_mask(index)); - if Dimension::retired(result) { - InsertResult::Retired(key, val) - } else if Dimension::frozen(result) { - InsertResult::Frozen(key, val) - } else { - InsertResult::Duplicate(key, val) - } - } - - fn compare(&self, index: usize, key: &Q) -> Ordering + /// Searches for a slot in which the key is stored. + fn search_slot(&self, key: &Q, mut mutable_metadata: usize) -> Option where K: Borrow, Q: Ord + ?Sized, { - self.key_at(index).borrow().cmp(key) - } - - fn take(&self, index: usize) -> (K, V) { - unsafe { - ( - self.entry_array.0[index].as_ptr().read(), - self.entry_array.1[index].as_ptr().read(), - ) - } - } - - fn write(&self, index: usize, key: K, val: V) { - unsafe { - (self.entry_array.0[index].as_ptr().cast_mut()).write(key); - (self.entry_array.1[index].as_ptr().cast_mut()).write(val); - } - } - - const fn key_at(&self, index: usize) -> &K { - unsafe { &*self.entry_array.0[index].as_ptr() } - } - - const fn value_at(&self, index: usize) -> &V { - unsafe { &*self.entry_array.1[index].as_ptr() } - } - - /// Returns the index of the corresponding entry of the next higher ranked entry. - fn next(index: usize, mut mutable_metadata: usize) -> usize { - debug_assert_ne!(index, usize::MAX); - let current_entry_rank = if index == DIMENSION.num_entries { - 0 - } else { - DIMENSION.rank(mutable_metadata, index) - }; - let mut next_index = DIMENSION.num_entries; - if current_entry_rank < DIMENSION.num_entries { - let mut next_rank = DIMENSION.removed_rank(); - for i in 0..DIMENSION.num_entries { - if mutable_metadata == 0 { - break; - } - if i != index { - let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); - if rank != Dimension::uninit_rank() && rank < next_rank { - if rank == current_entry_rank + 1 { - return i; - } else if rank > current_entry_rank { - next_rank = rank; - next_index = i; + let mut min_max_rank = DIMENSION.removed_rank(); + let mut max_min_rank = 0; + for i in 0..DIMENSION.num_entries { + if mutable_metadata == 0 { + break; + } + let rank = mutable_metadata % (1_usize << DIMENSION.num_bits_per_entry); + if rank < min_max_rank && rank > max_min_rank { + match self.compare(i, key) { + Ordering::Less => { + if max_min_rank < rank { + max_min_rank = rank; } } + Ordering::Greater => { + if min_max_rank > rank { + min_max_rank = rank; + } + } + Ordering::Equal => { + return Some(i); + } } - mutable_metadata >>= DIMENSION.num_bits_per_entry; } + mutable_metadata >>= DIMENSION.num_bits_per_entry; } - next_index + None + } + + fn compare(&self, index: usize, key: &Q) -> Ordering + where + K: Borrow, + Q: Ord + ?Sized, + { + self.key_at(index).borrow().cmp(key) } } -impl Drop for Leaf -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Drop for Leaf { #[inline] fn drop(&mut self) { if needs_drop::<(K, V)>() { @@ -707,11 +701,7 @@ where } /// [`LinkedList`] implementation for [`Leaf`]. -impl LinkedList for Leaf -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl LinkedList for Leaf { #[inline] fn link_ref(&self) -> &AtomicShared> { &self.link @@ -812,21 +802,13 @@ pub type EntryArray = ( ); /// Leaf scanner. -pub struct Scanner<'l, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct Scanner<'l, K, V> { leaf: &'l Leaf, metadata: usize, entry_index: usize, } -impl<'l, K, V> Scanner<'l, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl<'l, K, V> Scanner<'l, K, V> { /// Creates a new [`Scanner`]. #[inline] pub(super) fn new(leaf: &'l Leaf) -> Scanner<'l, K, V> { @@ -836,25 +818,6 @@ where entry_index: DIMENSION.num_entries, } } - /// Returns a [`Scanner`] pointing to the max-less entry if there is one. - #[inline] - pub(super) fn max_less(leaf: &'l Leaf, key: &Q) -> Option> - where - K: Borrow, - Q: Ord + ?Sized, - { - let metadata = leaf.metadata.load(Acquire); - let index = leaf.max_less(metadata, key); - if index == DIMENSION.num_entries { - None - } else { - Some(Scanner { - leaf, - metadata, - entry_index: index, - }) - } - } /// Returns the metadata that the [`Scanner`] is currently using. #[inline] @@ -931,11 +894,33 @@ where } } -impl<'l, K, V> Debug for Scanner<'l, K, V> +impl<'l, K, V> Scanner<'l, K, V> where K: 'static + Clone + Ord, V: 'static + Clone, { + /// Returns a [`Scanner`] pointing to the max-less entry if there is one. + #[inline] + pub(super) fn max_less(leaf: &'l Leaf, key: &Q) -> Option> + where + K: Borrow, + Q: Ord + ?Sized, + { + let metadata = leaf.metadata.load(Acquire); + let index = leaf.max_less(metadata, key); + if index == DIMENSION.num_entries { + None + } else { + Some(Scanner { + leaf, + metadata, + entry_index: index, + }) + } + } +} + +impl<'l, K, V> Debug for Scanner<'l, K, V> { #[inline] fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("Scanner") @@ -945,11 +930,7 @@ where } } -impl<'l, K, V> Iterator for Scanner<'l, K, V> -where - K: Clone + Ord, - V: Clone, -{ +impl<'l, K, V> Iterator for Scanner<'l, K, V> { type Item = (&'l K, &'l V); #[inline] diff --git a/src/tree_index/leaf_node.rs b/src/tree_index/leaf_node.rs index ca2ab8c1..7dd8fca8 100644 --- a/src/tree_index/leaf_node.rs +++ b/src/tree_index/leaf_node.rs @@ -21,11 +21,7 @@ pub const LOCKED: Tag = Tag::Second; /// [`LeafNode`] contains a list of instances of `K, V` [`Leaf`]. /// /// The layout of a leaf node: `|ptr(entry array)/max(child keys)|...|ptr(entry array)|` -pub struct LeafNode -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub struct LeafNode { /// Children of the [`LeafNode`]. pub(super) children: Leaf>>, @@ -46,11 +42,7 @@ where } /// [`Locker`] holds exclusive ownership of a [`Leaf`]. -pub(super) struct Locker<'n, K, V> -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub(super) struct Locker<'n, K, V> { leaf_node: &'n LeafNode, } @@ -76,11 +68,7 @@ pub(super) enum RemoveRangeState { /// /// `AtomicPtr` members may point to values under the protection of the [`Guard`] used for the /// split operation. -pub(super) struct StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub(super) struct StructuralChange { origin_leaf_key: AtomicPtr, origin_leaf: AtomicShared>, low_key_leaf: AtomicShared>, @@ -89,11 +77,7 @@ where high_key_leaf_node: AtomicPtr>, } -impl LeafNode -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl LeafNode { /// Creates a new empty [`LeafNode`]. #[inline] pub(super) fn new() -> LeafNode { @@ -112,6 +96,51 @@ where self.unbounded_child.tag(mo) == RETIRED } + /// Waits for the lock on the [`LeafNode`] to be released. + #[inline] + pub(super) fn wait(&self, async_wait: &mut D) { + let waiter = || { + if self.latch.load(Relaxed) == LOCKED.into() { + // The `LeafNode` is being split or locked. + return Err(()); + } + Ok(()) + }; + + if let Some(async_wait) = async_wait.derive() { + let _result = self.wait_queue.push_async_entry(async_wait, waiter); + } else { + let _result = self.wait_queue.wait_sync(waiter); + } + } + + /// Tries to lock the [`LeafNode`]. + fn try_lock(&self) -> bool { + self.latch + .compare_exchange(Tag::None.into(), LOCKED.into(), Acquire, Relaxed) + .is_ok() + } + + /// Unlocks the [`LeafNode`]. + fn unlock(&self) { + debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); + self.latch.store(Tag::None.into(), Release); + self.wait_queue.signal(); + } + + /// Retires itself. + fn retire(&self) { + debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); + self.latch.store(RETIRED.into(), Release); + self.wait_queue.signal(); + } +} + +impl LeafNode +where + K: 'static + Clone + Ord, + V: 'static + Clone, +{ /// Searches for an entry associated with the given key. #[inline] pub(super) fn search<'g, Q>(&self, key: &Q, guard: &'g Guard) -> Option<&'g V> @@ -724,24 +753,6 @@ where true } - /// Waits for the lock on the [`LeafNode`] to be released. - #[inline] - pub(super) fn wait(&self, async_wait: &mut D) { - let waiter = || { - if self.latch.load(Relaxed) == LOCKED.into() { - // The `LeafNode` is being split or locked. - return Err(()); - } - Ok(()) - }; - - if let Some(async_wait) = async_wait.derive() { - let _result = self.wait_queue.push_async_entry(async_wait, waiter); - } else { - let _result = self.wait_queue.wait_sync(waiter); - } - } - /// Splits a full leaf. /// /// # Errors @@ -1009,34 +1020,9 @@ where } false } - - /// Tries to lock the [`LeafNode`]. - fn try_lock(&self) -> bool { - self.latch - .compare_exchange(Tag::None.into(), LOCKED.into(), Acquire, Relaxed) - .is_ok() - } - - /// Unlocks the [`LeafNode`]. - fn unlock(&self) { - debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); - self.latch.store(Tag::None.into(), Release); - self.wait_queue.signal(); - } - - /// Retires itself. - fn retire(&self) { - debug_assert_eq!(self.latch.load(Relaxed), LOCKED.into()); - self.latch.store(RETIRED.into(), Release); - self.wait_queue.signal(); - } } -impl<'n, K, V> Locker<'n, K, V> -where - K: Clone + Ord, - V: Clone, -{ +impl<'n, K, V> Locker<'n, K, V> { /// Acquires exclusive lock on the [`LeafNode`]. #[inline] pub(super) fn try_lock(leaf_node: &'n LeafNode) -> Option> { @@ -1048,11 +1034,7 @@ where } } -impl<'n, K, V> Drop for Locker<'n, K, V> -where - K: Clone + Ord, - V: Clone, -{ +impl<'n, K, V> Drop for Locker<'n, K, V> { #[inline] fn drop(&mut self) { self.leaf_node.unlock(); @@ -1103,11 +1085,8 @@ impl RemoveRangeState { } } } -impl StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ + +impl StructuralChange { fn reset(&self) -> Option>> { self.origin_leaf_key.store(ptr::null_mut(), Relaxed); self.low_key_leaf.swap((None, Tag::None), Relaxed); @@ -1118,11 +1097,7 @@ where } } -impl Default for StructuralChange -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Default for StructuralChange { #[inline] fn default() -> Self { Self { diff --git a/src/tree_index/node.rs b/src/tree_index/node.rs index 6d9ad07b..c1436727 100644 --- a/src/tree_index/node.rs +++ b/src/tree_index/node.rs @@ -9,11 +9,7 @@ use std::ops::RangeBounds; use std::sync::atomic::Ordering::{self, Acquire, Relaxed, Release}; /// [`Node`] is either [`Self::Internal`] or [`Self::Leaf`]. -pub enum Node -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +pub enum Node { /// Internal node. Internal(InternalNode), @@ -21,11 +17,7 @@ where Leaf(LeafNode), } -impl Node -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Node { /// Creates a new [`InternalNode`]. #[inline] pub(super) fn new_internal_node() -> Self { @@ -55,7 +47,13 @@ where Self::Leaf(leaf_node) => leaf_node.retired(mo), } } +} +impl Node +where + K: 'static + Clone + Ord, + V: 'static + Clone, +{ /// Searches for an entry associated with the given key. #[inline] pub(super) fn search<'g, Q>(&self, key: &Q, guard: &'g Guard) -> Option<&'g V> @@ -339,11 +337,7 @@ where } } -impl Debug for Node -where - K: 'static + Clone + Ord, - V: 'static + Clone, -{ +impl Debug for Node { #[inline] fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self {