package buf // Inspired by https://github.com/xtaci/smux/blob/master/alloc.go import ( "errors" "math/bits" "sync" ) var DefaultAllocator = newDefaultAllocator() type Allocator interface { Get(size int) []byte Put(buf []byte) error } // defaultAllocator for incoming frames, optimized to prevent overwriting after zeroing type defaultAllocator struct { buffers [11]sync.Pool } // NewAllocator initiates a []byte allocator for frames less than 65536 bytes, // the waste(memory fragmentation) of space allocation is guaranteed to be // no more than 50%. func newDefaultAllocator() Allocator { return &defaultAllocator{ buffers: [...]sync.Pool{ // 64B -> 64K {New: func() any { return new([1 << 6]byte) }}, {New: func() any { return new([1 << 7]byte) }}, {New: func() any { return new([1 << 8]byte) }}, {New: func() any { return new([1 << 9]byte) }}, {New: func() any { return new([1 << 10]byte) }}, {New: func() any { return new([1 << 11]byte) }}, {New: func() any { return new([1 << 12]byte) }}, {New: func() any { return new([1 << 13]byte) }}, {New: func() any { return new([1 << 14]byte) }}, {New: func() any { return new([1 << 15]byte) }}, {New: func() any { return new([1 << 16]byte) }}, }, } } // Get a []byte from pool with most appropriate cap func (alloc *defaultAllocator) Get(size int) []byte { if size <= 0 || size > 65536 { return nil } var index uint16 if size > 64 { index = msb(size) if size != 1< 65536 || cap(buf) != 1<