From 5fc37f1fd402ca08e63c42610877fc4e442cd1d1 Mon Sep 17 00:00:00 2001 From: Yiling-J Date: Mon, 28 Oct 2024 21:27:47 +0800 Subject: [PATCH] fix race when using entry pool --- internal/store.go | 32 +++++++++++++++++++------------- 1 file changed, 19 insertions(+), 13 deletions(-) diff --git a/internal/store.go b/internal/store.go index 139eda4..662aecc 100644 --- a/internal/store.go +++ b/internal/store.go @@ -380,7 +380,6 @@ func (s *Store[K, V]) setShard(shard *Shard[K, V], hash uint64, key K, value V, entry.expire.Store(expire) entry.weight.Store(cost) entry.policyWeight = 0 - entry.flag = Flag{} s.setEntry(hash, shard, cost, entry, nvmClean) return shard, entry, true @@ -474,9 +473,10 @@ func (s *Store[K, V]) index(key K) (uint64, int) { } func (s *Store[K, V]) postDelete(entry *Entry[K, V]) { - var zero V - entry.value = zero if s.entryPool != nil { + var zero V + entry.value = zero + entry.flag = Flag{} s.entryPool.Put(entry) } } @@ -512,12 +512,15 @@ func (s *Store[K, V]) removeEntry(entry *Entry[K, V], reason RemoveReason) { } if rn <= s.probability { - s.secondaryCacheBuf <- SecondaryCacheItem[K, V]{ + select { + case s.secondaryCacheBuf <- SecondaryCacheItem[K, V]{ entry: entry, reason: reason, shard: shard, + }: + return + default: } - return } } shard.mu.Lock() @@ -606,6 +609,14 @@ func (s *Store[K, V]) sinkWrite(item WriteBufItem[K, V]) { case EVICTE: s.removeEntry(entry, EVICTED) case UPDATE: + // recheck hash if entry pool enabled to avoid race + if s.entryPool != nil { + hh := s.hasher.hash(entry.key) + if hh != item.hash { + return + } + } + // update entry policy weight entry.policyWeight += item.costChange @@ -619,13 +630,6 @@ func (s *Store[K, V]) sinkWrite(item WriteBufItem[K, V]) { } if item.costChange != 0 { - // recheck hash if entry pool enabled to avoid race - if s.entryPool != nil { - hh := s.hasher.hash(entry.key) - if hh != item.hash { - return - } - } // update policy weight s.policy.UpdateCost(entry, item.costChange) } @@ -836,10 +840,12 @@ func (s *Store[K, V]) processSecondary() { if item.reason == EVICTED { item.shard.mu.Lock() deleted := item.shard.delete(item.entry) + item.shard.mu.Unlock() if deleted { + s.policyMu.Lock() s.postDelete(item.entry) + s.policyMu.Unlock() } - item.shard.mu.Unlock() } } else { item.shard.mu.RUnlock(tk)