From 636b7af969231d17f6eb614c2c87697f50ef49fd Mon Sep 17 00:00:00 2001 From: Changgyoo Park Date: Fri, 3 May 2024 20:51:29 +0200 Subject: [PATCH] chore(*): finalize SCC 2.1.1 --- .config/glossary.txt | 3 +++ CHANGELOG.md | 4 ++++ Cargo.toml | 2 +- src/hash_cache.rs | 29 ++++++++++++++++++++++++++++- src/hash_index.rs | 15 +++++++++++++++ src/hash_map.rs | 33 ++++++++++++++++++++++++--------- src/lib.rs | 2 +- 7 files changed, 76 insertions(+), 12 deletions(-) diff --git a/.config/glossary.txt b/.config/glossary.txt index 226ccf5..ea29dfc 100644 --- a/.config/glossary.txt +++ b/.config/glossary.txt @@ -21,6 +21,9 @@ Linearizability LinkedList LRU M1 +M2 +M3 +M4 metadata overcount Resize diff --git a/CHANGELOG.md b/CHANGELOG.md index 86c1f09..65b5ff9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,10 @@ ## Version 2 +2.1.1 + +* Implement `Deref` and `DerefMut` for `OccupiedEntry`: [#140](https://github.com/wvwwvwwv/scalable-concurrent-containers/issues/140) by [gituser-rs](https://github.com/gituser-rs) + 2.1.0 * Use [`sdd`](https://crates.io/crates/sdd) as the memory reclaimer. diff --git a/Cargo.toml b/Cargo.toml index b4f50da..01f2976 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.1.0" +version = "2.1.1" authors = ["wvwwvwwv "] edition = "2021" rust-version = "1.65.0" diff --git a/src/hash_cache.rs b/src/hash_cache.rs index 9180202..4e99c82 100644 --- a/src/hash_cache.rs +++ b/src/hash_cache.rs @@ -11,7 +11,7 @@ use std::collections::hash_map::RandomState; use std::fmt::{self, Debug}; use std::hash::{BuildHasher, Hash}; use std::mem::replace; -use std::ops::RangeInclusive; +use std::ops::{Deref, DerefMut, RangeInclusive}; use std::pin::Pin; use std::sync::atomic::AtomicUsize; use std::sync::atomic::Ordering::{Acquire, Relaxed}; @@ -363,6 +363,9 @@ where /// assert!(hashcache.get(&1).is_none()); /// assert!(hashcache.put(1, 10).is_ok()); /// assert_eq!(*hashcache.get(&1).unwrap().get(), 10); + /// + /// *hashcache.get(&1).unwrap() = 11; + /// assert_eq!(*hashcache.get(&1).unwrap(), 11); /// ``` #[inline] pub fn get(&self, key: &Q) -> Option> @@ -1602,6 +1605,30 @@ where } } +impl<'h, K, V, H> Deref for OccupiedEntry<'h, K, V, H> +where + K: Eq + Hash, + H: BuildHasher, +{ + type Target = V; + + #[inline] + fn deref(&self) -> &Self::Target { + self.get() + } +} + +impl<'h, K, V, H> DerefMut for OccupiedEntry<'h, K, V, H> +where + K: Eq + Hash, + H: BuildHasher, +{ + #[inline] + fn deref_mut(&mut self) -> &mut Self::Target { + self.get_mut() + } +} + impl<'h, K, V, H> VacantEntry<'h, K, V, H> where K: Eq + Hash, diff --git a/src/hash_index.rs b/src/hash_index.rs index 056013d..461c690 100644 --- a/src/hash_index.rs +++ b/src/hash_index.rs @@ -588,6 +588,7 @@ where /// assert!(hashindex.get(&1).is_none()); /// assert!(hashindex.insert(1, 10).is_ok()); /// assert_eq!(*hashindex.get(&1).unwrap().get(), 10); + /// assert_eq!(*hashindex.get(&1).unwrap(), 10); /// ``` #[inline] pub fn get(&self, key: &Q) -> Option> @@ -1609,6 +1610,20 @@ where } } +impl<'h, K, V, H> Deref for OccupiedEntry<'h, K, V, H> +where + K: 'static + Clone + Debug + Eq + Hash, + V: 'static + Clone + Debug, + H: BuildHasher, +{ + type Target = V; + + #[inline] + fn deref(&self) -> &Self::Target { + self.get() + } +} + impl<'h, K, V, H> VacantEntry<'h, K, V, H> where K: 'static + Clone + Eq + Hash, diff --git a/src/hash_map.rs b/src/hash_map.rs index fde81ea..2ab98c0 100644 --- a/src/hash_map.rs +++ b/src/hash_map.rs @@ -10,7 +10,7 @@ use std::collections::hash_map::RandomState; use std::fmt::{self, Debug}; use std::hash::{BuildHasher, Hash}; use std::mem::replace; -use std::ops::{Deref, RangeInclusive}; +use std::ops::{Deref, DerefMut, RangeInclusive}; use std::pin::Pin; use std::sync::atomic::AtomicUsize; use std::sync::atomic::Ordering::{Acquire, Relaxed}; @@ -659,6 +659,9 @@ where /// assert!(hashmap.get(&1).is_none()); /// assert!(hashmap.insert(1, 10).is_ok()); /// assert_eq!(*hashmap.get(&1).unwrap().get(), 10); + /// + /// *hashmap.get(&1).unwrap() = 11; + /// assert_eq!(*hashmap.get(&1).unwrap(), 11); /// ``` #[inline] pub fn get(&self, key: &Q) -> Option> @@ -1921,6 +1924,21 @@ where } } +impl<'h, K, V, H> Debug for OccupiedEntry<'h, K, V, H> +where + K: Debug + Eq + Hash, + V: Debug, + H: BuildHasher, +{ + #[inline] + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + f.debug_struct("OccupiedEntry") + .field("key", self.key()) + .field("value", self.get()) + .finish_non_exhaustive() + } +} + impl<'h, K, V, H> Deref for OccupiedEntry<'h, K, V, H> where K: Eq + Hash, @@ -1928,23 +1946,20 @@ where { type Target = V; + #[inline] fn deref(&self) -> &Self::Target { self.get() } } -impl<'h, K, V, H> Debug for OccupiedEntry<'h, K, V, H> +impl<'h, K, V, H> DerefMut for OccupiedEntry<'h, K, V, H> where - K: Debug + Eq + Hash, - V: Debug, + K: Eq + Hash, H: BuildHasher, { #[inline] - fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - f.debug_struct("OccupiedEntry") - .field("key", self.key()) - .field("value", self.get()) - .finish_non_exhaustive() + fn deref_mut(&mut self) -> &mut Self::Target { + self.get_mut() } } diff --git a/src/lib.rs b/src/lib.rs index 1ebd43d..2499ef5 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -29,7 +29,7 @@ pub use stack::Stack; pub mod tree_index; pub use tree_index::TreeIndex; -/// Re-export the [`sdd`](https://crates.io/crates/sdd) crate for backward compatibility. +/// Re-exports the [`sdd`](https://crates.io/crates/sdd) crate for backward compatibility. pub use sdd as ebr; mod exit_guard;