|
|
|
@ -1,5 +1,9 @@
|
|
|
|
|
package alloc
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"sync"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// Buffer is a recyclable allocation of a byte array. Buffer.Release() recycles
|
|
|
|
|
// the buffer into an internal buffer pool, in order to recreate a buffer more
|
|
|
|
|
// quickly.
|
|
|
|
@ -69,15 +73,16 @@ func (b *Buffer) Write(data []byte) (int, error) {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type bufferPool struct {
|
|
|
|
|
chain chan []byte
|
|
|
|
|
bufferSize int
|
|
|
|
|
buffers2Keep int
|
|
|
|
|
chain chan []byte
|
|
|
|
|
allocator *sync.Pool
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func newBufferPool(bufferSize, poolSize int) *bufferPool {
|
|
|
|
|
pool := &bufferPool{
|
|
|
|
|
chain: make(chan []byte, poolSize),
|
|
|
|
|
bufferSize: bufferSize,
|
|
|
|
|
chain: make(chan []byte, poolSize),
|
|
|
|
|
allocator: &sync.Pool{
|
|
|
|
|
New: func() interface{} { return make([]byte, bufferSize) },
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
for i := 0; i < poolSize; i++ {
|
|
|
|
|
pool.chain <- make([]byte, bufferSize)
|
|
|
|
@ -90,7 +95,7 @@ func (p *bufferPool) allocate() *Buffer {
|
|
|
|
|
select {
|
|
|
|
|
case b = <-p.chain:
|
|
|
|
|
default:
|
|
|
|
|
b = make([]byte, p.bufferSize)
|
|
|
|
|
b = p.allocator.Get().([]byte)
|
|
|
|
|
}
|
|
|
|
|
return &Buffer{
|
|
|
|
|
head: b,
|
|
|
|
@ -103,6 +108,7 @@ func (p *bufferPool) free(buffer *Buffer) {
|
|
|
|
|
select {
|
|
|
|
|
case p.chain <- buffer.head:
|
|
|
|
|
default:
|
|
|
|
|
p.allocator.Put(buffer.head)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|