136 lines
2.6 KiB
Go
Raw Normal View History

// Copyright (c) 2014, Suryandaru Triandana <syndtr@gmail.com>
// All rights reserved.
//
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package util
import (
"fmt"
"sync"
"sync/atomic"
)
// BufferPool is a 'buffer pool'.
type BufferPool struct {
2022-01-18 17:19:43 -04:00
pool [6]sync.Pool
baseline [5]int
get uint32
put uint32
less uint32
equal uint32
greater uint32
miss uint32
}
func (p *BufferPool) poolNum(n int) int {
for i, x := range p.baseline {
if n <= x {
2022-01-18 17:19:43 -04:00
return i
}
}
2022-01-18 17:19:43 -04:00
return len(p.baseline)
}
// Get returns buffer with length of n.
func (p *BufferPool) Get(n int) []byte {
if p == nil {
return make([]byte, n)
}
2022-01-18 17:19:43 -04:00
atomic.AddUint32(&p.get, 1)
2022-01-18 17:19:43 -04:00
poolNum := p.poolNum(n)
2022-01-18 17:19:43 -04:00
b := p.pool[poolNum].Get().(*[]byte)
2022-01-18 17:19:43 -04:00
if cap(*b) == 0 {
// If we grabbed nothing, increment the miss stats.
atomic.AddUint32(&p.miss, 1)
2022-01-18 17:19:43 -04:00
if poolNum == len(p.baseline) {
*b = make([]byte, n)
return *b
}
2022-01-18 17:19:43 -04:00
*b = make([]byte, p.baseline[poolNum])
*b = (*b)[:n]
return *b
} else {
2022-01-18 17:19:43 -04:00
// If there is enough capacity in the bytes grabbed, resize the length
// to n and return.
if n < cap(*b) {
atomic.AddUint32(&p.less, 1)
*b = (*b)[:n]
return *b
} else if n == cap(*b) {
atomic.AddUint32(&p.equal, 1)
*b = (*b)[:n]
return *b
} else if n > cap(*b) {
atomic.AddUint32(&p.greater, 1)
}
2022-01-18 17:19:43 -04:00
}
2022-01-18 17:19:43 -04:00
if poolNum == len(p.baseline) {
*b = make([]byte, n)
return *b
}
2022-01-18 17:19:43 -04:00
*b = make([]byte, p.baseline[poolNum])
*b = (*b)[:n]
return *b
}
// Put adds given buffer to the pool.
func (p *BufferPool) Put(b []byte) {
if p == nil {
return
}
2022-01-18 17:19:43 -04:00
poolNum := p.poolNum(cap(b))
atomic.AddUint32(&p.put, 1)
2022-01-18 17:19:43 -04:00
p.pool[poolNum].Put(&b)
}
func (p *BufferPool) String() string {
if p == nil {
return "<nil>"
}
2022-01-18 17:19:43 -04:00
return fmt.Sprintf("BufferPool{B·%d G·%d P·%d <·%d =·%d >·%d M·%d}",
p.baseline, p.get, p.put, p.less, p.equal, p.greater, p.miss)
}
// NewBufferPool creates a new initialized 'buffer pool'.
func NewBufferPool(baseline int) *BufferPool {
if baseline <= 0 {
panic("baseline can't be <= 0")
}
2022-01-18 17:19:43 -04:00
bufPool := &BufferPool{
baseline: [...]int{baseline / 4, baseline / 2, baseline, baseline * 2, baseline * 4},
pool: [6]sync.Pool{
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
{
2022-01-18 17:19:43 -04:00
New: func() interface{} { return new([]byte) },
},
},
}
2022-01-18 17:19:43 -04:00
return bufPool
}