mirror of
https://github.com/XTLS/Xray-core.git
synced 2026-06-27 19:52:43 +00:00
Some checks failed
Build and Release for Windows 7 / check-assets (push) Has been cancelled
Build and Release / check-assets (push) Has been cancelled
Tests and Checkings / check-assets (push) Has been cancelled
Tests and Checkings / check-proto (push) Has been cancelled
Tests and Checkings / check-format (push) Has been cancelled
Build and Release for Windows 7 / build (win7-32, 386, windows) (push) Has been cancelled
Build and Release for Windows 7 / build (win7-64, amd64, windows) (push) Has been cancelled
Build and Release / build (386, freebsd, ) (push) Has been cancelled
Build and Release / build (386, linux, ) (push) Has been cancelled
Build and Release / build (386, openbsd, ) (push) Has been cancelled
Build and Release / build (386, windows, ) (push) Has been cancelled
Build and Release / build (amd64, android, android-amd64) (push) Has been cancelled
Build and Release / build (amd64, darwin, ) (push) Has been cancelled
Build and Release / build (amd64, freebsd, ) (push) Has been cancelled
Build and Release / build (amd64, linux, ) (push) Has been cancelled
Build and Release / build (amd64, openbsd, ) (push) Has been cancelled
Build and Release / build (amd64, windows, ) (push) Has been cancelled
Build and Release / build (arm, 5, linux) (push) Has been cancelled
Build and Release / build (arm, 6, linux) (push) Has been cancelled
Build and Release / build (arm, 7, freebsd) (push) Has been cancelled
Build and Release / build (arm, 7, linux) (push) Has been cancelled
Build and Release / build (arm, 7, openbsd) (push) Has been cancelled
Build and Release / build (arm64, android) (push) Has been cancelled
Build and Release / build (arm64, darwin) (push) Has been cancelled
Build and Release / build (arm64, freebsd) (push) Has been cancelled
Build and Release / build (arm64, linux) (push) Has been cancelled
Build and Release / build (arm64, openbsd) (push) Has been cancelled
Build and Release / build (arm64, windows) (push) Has been cancelled
Build and Release / build (loong64, linux) (push) Has been cancelled
Build and Release / build (mips, linux) (push) Has been cancelled
Build and Release / build (mips64, linux) (push) Has been cancelled
Build and Release / build (mips64le, linux) (push) Has been cancelled
Build and Release / build (mipsle, linux) (push) Has been cancelled
Build and Release / build (ppc64, linux) (push) Has been cancelled
Build and Release / build (ppc64le, linux) (push) Has been cancelled
Build and Release / build (riscv64, linux) (push) Has been cancelled
Build and Release / build (s390x, linux) (push) Has been cancelled
Tests and Checkings / test (macos-latest) (push) Has been cancelled
Tests and Checkings / test (ubuntu-latest) (push) Has been cancelled
Tests and Checkings / test (windows-latest) (push) Has been cancelled
59 lines
1.2 KiB
Go
59 lines
1.2 KiB
Go
package utils
|
|
|
|
import (
|
|
"maps"
|
|
"runtime"
|
|
"sync"
|
|
"weak"
|
|
)
|
|
|
|
// WeakCacheMap is a map that holds weak references to values.
|
|
// Use for shared expensive objects and automatic cleanup when no longer used.
|
|
// This object can be GC and no goroutine is used for cleanup.
|
|
type WeakCacheMap[K comparable, V any] struct {
|
|
mu sync.Mutex
|
|
m map[K]weak.Pointer[V]
|
|
}
|
|
|
|
func NewWeakCacheMap[K comparable, V any]() *WeakCacheMap[K, V] {
|
|
return &WeakCacheMap[K, V]{
|
|
m: make(map[K]weak.Pointer[V]),
|
|
}
|
|
}
|
|
|
|
func (c *WeakCacheMap[K, V]) Load(key K) (value *V, ok bool) {
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
weakPtr := c.m[key].Value()
|
|
if weakPtr != nil {
|
|
return weakPtr, true
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
func (c *WeakCacheMap[K, V]) Store(key K, value *V) {
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
weakPtr := weak.Make(value)
|
|
c.m[key] = weakPtr
|
|
runtime.AddCleanup(value, func(struct{}) {
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
if c.m[key] == weakPtr {
|
|
delete(c.m, key)
|
|
}
|
|
}, struct{}{})
|
|
}
|
|
|
|
func (c *WeakCacheMap[K, V]) Range(f func(K, *V) bool) {
|
|
c.mu.Lock()
|
|
snapshot := maps.Clone(c.m)
|
|
c.mu.Unlock()
|
|
for k, v := range snapshot {
|
|
if value := v.Value(); value != nil {
|
|
if !f(k, value) {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|