Skip to content

Commit

Permalink
Revert "Only hash keys once, taking advantage of hashbrown raw entry …
Browse files Browse the repository at this point in the history
…api (#259)"

This reverts commit 3448b6f.
  • Loading branch information
xacrimon committed Aug 29, 2023
1 parent dfb7b9a commit febc45d
Show file tree
Hide file tree
Showing 2 changed files with 67 additions and 62 deletions.
104 changes: 50 additions & 54 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,6 @@ use core::fmt;
use core::hash::{BuildHasher, Hash, Hasher};
use core::iter::FromIterator;
use core::ops::{BitAnd, BitOr, Shl, Shr, Sub};
use hashbrown::hash_map::RawEntryMut;
use iter::{Iter, IterMut, OwningIter};
use mapref::entry::{Entry, OccupiedEntry, VacantEntry};
use mapref::multiple::RefMulti;
Expand Down Expand Up @@ -296,15 +295,11 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: BuildHasher + Clone> DashMap<K, V, S> {
/// Hash a given item to produce a usize.
/// Uses the provided or default HashBuilder.
pub fn hash_usize<T: Hash>(&self, item: &T) -> usize {
self.hash_u64(item) as usize
}

fn hash_u64<T: Hash>(&self, item: &T) -> u64 {
let mut hasher = self.hasher.build_hasher();

item.hash(&mut hasher);

hasher.finish()
hasher.finish() as usize
}

cfg_if! {
Expand Down Expand Up @@ -921,9 +916,9 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
}

fn _insert(&self, key: K, value: V) -> Option<V> {
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let mut shard = unsafe { self._yield_write_shard(idx) };

Expand All @@ -937,9 +932,9 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let mut shard = unsafe { self._yield_write_shard(idx) };

Expand All @@ -951,20 +946,22 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let mut shard = unsafe { self._yield_write_shard(idx) };

if let RawEntryMut::Occupied(entry) =
shard.raw_entry_mut().from_key_hashed_nocheck(hash, key)
{
if f(entry.key(), entry.get().get()) {
let (k, v) = entry.remove_entry();
Some((k, v.into_inner()))
} else {
None
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();

if f(&*kptr, &mut *vptr) {
shard.remove_entry(key).map(|(k, v)| (k, v.into_inner()))
} else {
None
}
}
} else {
None
Expand All @@ -976,21 +973,22 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let mut shard = unsafe { self._yield_write_shard(idx) };

if let RawEntryMut::Occupied(mut entry) =
shard.raw_entry_mut().from_key_hashed_nocheck(hash, key)
{
let (k, v) = entry.get_key_value_mut();
if f(k, v.get_mut()) {
let (k, v) = entry.remove_entry();
Some((k, v.into_inner()))
} else {
None
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();

if f(&*kptr, &mut *vptr) {
shard.remove_entry(key).map(|(k, v)| (k, v.into_inner()))
} else {
None
}
}
} else {
None
Expand All @@ -1010,13 +1008,13 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let shard = unsafe { self._yield_read_shard(idx) };

if let Some((kptr, vptr)) = shard.raw_entry().from_key_hashed_nocheck(hash, key) {
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *const V = vptr.get();
Expand All @@ -1032,18 +1030,16 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let mut shard = unsafe { self._yield_write_shard(idx) };
let shard = unsafe { self._yield_write_shard(idx) };

if let RawEntryMut::Occupied(mut entry) =
shard.raw_entry_mut().from_key_hashed_nocheck(hash, key)
{
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = entry.key();
let vptr: *mut V = entry.get_mut().as_ptr();
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();
Some(RefMut::new(shard, kptr, vptr))
}
} else {
Expand All @@ -1056,16 +1052,16 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let shard = match unsafe { self._try_yield_read_shard(idx) } {
Some(shard) => shard,
None => return TryResult::Locked,
};

if let Some((kptr, vptr)) = shard.raw_entry().from_key_hashed_nocheck(hash, key) {
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *const V = vptr.get();
Expand All @@ -1081,16 +1077,16 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let shard = match unsafe { self._try_yield_write_shard(idx) } {
Some(shard) => shard,
None => return TryResult::Locked,
};

if let Some((kptr, vptr)) = shard.raw_entry().from_key_hashed_nocheck(hash, key) {
if let Some((kptr, vptr)) = shard.get_key_value(key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();
Expand Down Expand Up @@ -1149,13 +1145,13 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
}

fn _entry(&'a self, key: K) -> Entry<'a, K, V, S> {
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let shard = unsafe { self._yield_write_shard(idx) };

if let Some((kptr, vptr)) = shard.raw_entry().from_key_hashed_nocheck(hash, &key) {
if let Some((kptr, vptr)) = shard.get_key_value(&key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();
Expand All @@ -1167,16 +1163,16 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: 'a + BuildHasher + Clone> Map<'a, K, V, S>
}

fn _try_entry(&'a self, key: K) -> Option<Entry<'a, K, V, S>> {
let hash = self.hash_u64(&key);
let hash = self.hash_usize(&key);

let idx = self.determine_shard(hash as usize);
let idx = self.determine_shard(hash);

let shard = match unsafe { self._try_yield_write_shard(idx) } {
Some(shard) => shard,
None => return None,
};

if let Some((kptr, vptr)) = shard.raw_entry().from_key_hashed_nocheck(hash, &key) {
if let Some((kptr, vptr)) = shard.get_key_value(&key) {
unsafe {
let kptr: *const K = kptr;
let vptr: *mut V = vptr.as_ptr();
Expand Down
25 changes: 17 additions & 8 deletions src/read_only.rs
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,13 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: BuildHasher + Clone> ReadOnlyView<K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.get(key).is_some()
let hash = self.map.hash_usize(&key);

let idx = self.map.determine_shard(hash);

let shard = unsafe { self.map._get_read_shard(idx) };

shard.contains_key(key)
}

/// Returns a reference to the value corresponding to the key.
Expand All @@ -70,7 +76,13 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: BuildHasher + Clone> ReadOnlyView<K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.get_key_value(key).map(|(_k, v)| v)
let hash = self.map.hash_usize(&key);

let idx = self.map.determine_shard(hash);

let shard = unsafe { self.map._get_read_shard(idx) };

shard.get(key).map(|v| v.get())
}

/// Returns the key-value pair corresponding to the supplied key.
Expand All @@ -79,16 +91,13 @@ impl<'a, K: 'a + Eq + Hash, V: 'a, S: BuildHasher + Clone> ReadOnlyView<K, V, S>
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let hash = self.map.hash_u64(&key);
let hash = self.map.hash_usize(&key);

let idx = self.map.determine_shard(hash as usize);
let idx = self.map.determine_shard(hash);

let shard = unsafe { self.map._get_read_shard(idx) };

shard
.raw_entry()
.from_key_hashed_nocheck(hash, key)
.map(|(k, v)| (k, v.get()))
shard.get_key_value(key).map(|(k, v)| (k, v.get()))
}

fn shard_read_iter(&'a self) -> impl Iterator<Item = &'a HashMap<K, V, S>> + 'a {
Expand Down

0 comments on commit febc45d

Please sign in to comment.