GoDoxy/internal/utils/synk/pool.go
yusing 8469b6406c
Some checks failed
Docker Image CI (socket-proxy) / build (push) Has been cancelled
tweak: consolidate bytes pool management and enhance CopyClose functionality for improved performance
2025-05-25 16:20:12 +08:00

133 lines
2.3 KiB
Go

package synk
import (
"runtime"
"unsafe"
)
type weakBuf = unsafe.Pointer
func makeWeak(b *[]byte) weakBuf {
ptr := runtime_registerWeakPointer(unsafe.Pointer(b))
addCleanup(b, addGCed, cap(*b))
runtime.KeepAlive(ptr)
return weakBuf(ptr)
}
func getBufFromWeak(w weakBuf) []byte {
ptr := (*[]byte)(runtime_makeStrongFromWeak(w))
if ptr == nil {
return nil
}
return *ptr
}
//go:linkname runtime_registerWeakPointer weak.runtime_registerWeakPointer
func runtime_registerWeakPointer(unsafe.Pointer) unsafe.Pointer
//go:linkname runtime_makeStrongFromWeak weak.runtime_makeStrongFromWeak
func runtime_makeStrongFromWeak(unsafe.Pointer) unsafe.Pointer
type BytesPool struct {
sizedPool chan weakBuf
unsizedPool chan weakBuf
initSize int
}
const (
kb = 1024
mb = 1024 * kb
)
const (
InPoolLimit = 32 * mb
UnsizedAvg = 4 * kb
SizedPoolThreshold = 256 * kb
DropThreshold = 4 * mb
SizedPoolSize = InPoolLimit * 8 / 10 / SizedPoolThreshold
UnsizedPoolSize = InPoolLimit * 2 / 10 / UnsizedAvg
)
var bytesPool = &BytesPool{
sizedPool: make(chan weakBuf, SizedPoolSize),
unsizedPool: make(chan weakBuf, UnsizedPoolSize),
initSize: UnsizedAvg,
}
func NewBytesPool() *BytesPool {
return bytesPool
}
func (p *BytesPool) Get() []byte {
for {
select {
case bWeak := <-p.unsizedPool:
bPtr := getBufFromWeak(bWeak)
if bPtr == nil {
continue
}
addReused(cap(bPtr))
return bPtr
default:
return make([]byte, 0)
}
}
}
func (p *BytesPool) GetSized(size int) []byte {
if size <= SizedPoolThreshold {
addNonPooled(size)
return make([]byte, size)
}
for {
select {
case bWeak := <-p.sizedPool:
bPtr := getBufFromWeak(bWeak)
if bPtr == nil {
continue
}
capB := cap(bPtr)
if capB >= size {
addReused(capB)
return (bPtr)[:size]
}
select {
case p.sizedPool <- bWeak:
default:
// just drop it
}
default:
}
addNonPooled(size)
return make([]byte, size)
}
}
func (p *BytesPool) Put(b []byte) {
size := cap(b)
if size > DropThreshold {
return
}
b = b[:0]
w := makeWeak(&b)
if size <= SizedPoolThreshold {
p.put(w, p.unsizedPool)
} else {
p.put(w, p.sizedPool)
}
}
//go:inline
func (p *BytesPool) put(w weakBuf, pool chan weakBuf) {
select {
case pool <- w:
default:
// just drop it
}
}
func init() {
initPoolStats()
}