2016-04-12 10:52:57 -04:00
|
|
|
package alloc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
2016-07-28 10:24:15 -04:00
|
|
|
type Pool interface {
|
|
|
|
Allocate() *Buffer
|
|
|
|
Free(*Buffer)
|
|
|
|
}
|
|
|
|
|
2016-04-12 10:52:57 -04:00
|
|
|
type BufferPool struct {
|
|
|
|
chain chan []byte
|
|
|
|
allocator *sync.Pool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBufferPool(bufferSize, poolSize int) *BufferPool {
|
|
|
|
pool := &BufferPool{
|
|
|
|
chain: make(chan []byte, poolSize),
|
|
|
|
allocator: &sync.Pool{
|
|
|
|
New: func() interface{} { return make([]byte, bufferSize) },
|
|
|
|
},
|
|
|
|
}
|
2016-07-15 09:17:06 -04:00
|
|
|
for i := 0; i < poolSize; i++ {
|
2016-04-12 10:52:57 -04:00
|
|
|
pool.chain <- make([]byte, bufferSize)
|
|
|
|
}
|
|
|
|
return pool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *BufferPool) Allocate() *Buffer {
|
|
|
|
var b []byte
|
|
|
|
select {
|
|
|
|
case b = <-p.chain:
|
|
|
|
default:
|
|
|
|
b = p.allocator.Get().([]byte)
|
|
|
|
}
|
2016-07-15 08:24:20 -04:00
|
|
|
return CreateBuffer(b, p)
|
2016-04-12 10:52:57 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *BufferPool) Free(buffer *Buffer) {
|
|
|
|
rawBuffer := buffer.head
|
|
|
|
if rawBuffer == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case p.chain <- rawBuffer:
|
|
|
|
default:
|
|
|
|
p.allocator.Put(rawBuffer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-11 13:54:20 -04:00
|
|
|
const (
|
2016-06-18 11:14:30 -04:00
|
|
|
SmallBufferSize = 1600 - defaultOffset
|
2016-05-11 13:54:20 -04:00
|
|
|
BufferSize = 8*1024 - defaultOffset
|
|
|
|
LargeBufferSize = 64*1024 - defaultOffset
|
|
|
|
)
|
|
|
|
|
2016-08-01 11:47:31 -04:00
|
|
|
var smallPool = NewBufferPool(1600, 256)
|
|
|
|
var mediumPool = NewBufferPool(8*1024, 1024)
|
2016-07-15 09:16:35 -04:00
|
|
|
var largePool = NewBufferPool(64*1024, 32)
|