freelru: Fix GetAndRefreshOrAdd

This commit is contained in:
世界 2025-01-09 15:57:55 +08:00
parent be9840c70f
commit 4dabb9be97
No known key found for this signature in database
GPG key ID: CD109927C34A63C4
5 changed files with 23 additions and 17 deletions

View file

@ -51,7 +51,7 @@ func New(handler N.UDPConnectionHandlerEx, prepare PrepareFunc, timeout time.Dur
} }
func (s *Service) NewPacket(bufferSlices [][]byte, source M.Socksaddr, destination M.Socksaddr, userData any) { func (s *Service) NewPacket(bufferSlices [][]byte, source M.Socksaddr, destination M.Socksaddr, userData any) {
conn, loaded := s.cache.GetAndRefreshOrAdd(source.AddrPort(), func() (*natConn, bool) { conn, _, ok := s.cache.GetAndRefreshOrAdd(source.AddrPort(), func() (*natConn, bool) {
ok, ctx, writer, onClose := s.prepare(source, destination, userData) ok, ctx, writer, onClose := s.prepare(source, destination, userData)
if !ok { if !ok {
return nil, false return nil, false
@ -67,10 +67,8 @@ func (s *Service) NewPacket(bufferSlices [][]byte, source M.Socksaddr, destinati
go s.handler.NewPacketConnectionEx(ctx, newConn, source, destination, onClose) go s.handler.NewPacketConnectionEx(ctx, newConn, source, destination, onClose)
return newConn, true return newConn, true
}) })
if !loaded { if !ok {
if conn == nil { return
return
}
} }
buffer := conn.readWaitOptions.NewPacketBuffer() buffer := conn.readWaitOptions.NewPacketBuffer()
for _, bufferSlice := range bufferSlices { for _, bufferSlice := range bufferSlices {

View file

@ -54,7 +54,7 @@ type Cache[K comparable, V comparable] interface {
// The lifetime of the found cache item is refreshed, even if it was already expired. // The lifetime of the found cache item is refreshed, even if it was already expired.
GetAndRefresh(key K) (V, bool) GetAndRefresh(key K) (V, bool)
GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (V, bool) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (V, bool, bool)
// Peek looks up a key's value from the cache, without changing its recent-ness. // Peek looks up a key's value from the cache, without changing its recent-ness.
// If the found entry is already expired, the evict function is called. // If the found entry is already expired, the evict function is called.

View file

@ -522,11 +522,15 @@ func (lru *LRU[K, V]) getAndRefresh(hash uint32, key K) (value V, ok bool) {
return return
} }
func (lru *LRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (V, bool) { func (lru *LRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (V, bool, bool) {
return lru.getAndRefreshOrAdd(lru.hash(key), key, constructor) value, updated, ok := lru.getAndRefreshOrAdd(lru.hash(key), key, constructor)
if !updated && ok {
lru.PurgeExpired()
}
return value, updated, ok
} }
func (lru *LRU[K, V]) getAndRefreshOrAdd(hash uint32, key K, constructor func() (V, bool)) (value V, ok bool) { func (lru *LRU[K, V]) getAndRefreshOrAdd(hash uint32, key K, constructor func() (V, bool)) (value V, updated bool, ok bool) {
if pos, ok := lru.findKeyNoExpire(hash, key); ok { if pos, ok := lru.findKeyNoExpire(hash, key); ok {
if pos != lru.head { if pos != lru.head {
lru.unlinkElement(pos) lru.unlinkElement(pos)
@ -534,17 +538,15 @@ func (lru *LRU[K, V]) getAndRefreshOrAdd(hash uint32, key K, constructor func()
} }
lru.metrics.Hits++ lru.metrics.Hits++
lru.elements[pos].expire = expire(lru.lifetime) lru.elements[pos].expire = expire(lru.lifetime)
return lru.elements[pos].value, ok return lru.elements[pos].value, true, true
} }
lru.metrics.Misses++ lru.metrics.Misses++
value, ok = constructor() value, ok = constructor()
if !ok { if !ok {
return return
} }
lru.addWithLifetime(hash, key, value, lru.lifetime) lru.addWithLifetime(hash, key, value, lru.lifetime)
lru.PurgeExpired() return value, false, true
return value, false
} }
// Peek looks up a key's value from the cache, without changing its recent-ness. // Peek looks up a key's value from the cache, without changing its recent-ness.

View file

@ -201,14 +201,17 @@ func (lru *ShardedLRU[K, V]) GetAndRefresh(key K) (value V, ok bool) {
return return
} }
func (lru *ShardedLRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (value V, updated bool) { func (lru *ShardedLRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (value V, updated bool, ok bool) {
hash := lru.hash(key) hash := lru.hash(key)
shard := (hash >> 16) & lru.mask shard := (hash >> 16) & lru.mask
lru.mus[shard].Lock() lru.mus[shard].Lock()
value, updated = lru.lrus[shard].getAndRefreshOrAdd(hash, key, constructor) value, updated, ok = lru.lrus[shard].getAndRefreshOrAdd(hash, key, constructor)
lru.mus[shard].Unlock() lru.mus[shard].Unlock()
if !updated && ok {
lru.PurgeExpired()
}
return return
} }

View file

@ -121,11 +121,14 @@ func (lru *SyncedLRU[K, V]) GetAndRefresh(key K) (value V, ok bool) {
return return
} }
func (lru *SyncedLRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (value V, updated bool) { func (lru *SyncedLRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (value V, updated bool, ok bool) {
hash := lru.lru.hash(key) hash := lru.lru.hash(key)
lru.mu.Lock() lru.mu.Lock()
value, updated = lru.lru.getAndRefreshOrAdd(hash, key, constructor) value, updated, ok = lru.lru.getAndRefreshOrAdd(hash, key, constructor)
if !updated && ok {
lru.lru.PurgeExpired()
}
lru.mu.Unlock() lru.mu.Unlock()
return return