2016-04-16 02:56:53 +08:00
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"sync/atomic"
|
|
|
|
|
|
|
|
"github.com/hashicorp/golang-lru"
|
|
|
|
|
|
|
|
"github.com/chrislusf/seaweedfs/go/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
bytesCache *lru.Cache
|
|
|
|
bytesPool *util.BytesPool
|
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
There are one level of caching, and one level of pooling.
|
|
|
|
|
|
|
|
In pooling, all []byte are fetched and returned to the pool bytesPool.
|
|
|
|
|
2016-04-16 03:06:43 +08:00
|
|
|
In caching, the string~[]byte mapping is cached
|
2016-04-16 02:56:53 +08:00
|
|
|
*/
|
|
|
|
func init() {
|
|
|
|
bytesPool = util.NewBytesPool()
|
2016-04-16 03:06:43 +08:00
|
|
|
bytesCache, _ = lru.NewWithEvict(512, func(key interface{}, value interface{}) {
|
2016-04-16 02:56:53 +08:00
|
|
|
value.(*Block).decreaseReference()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
type Block struct {
|
|
|
|
Bytes []byte
|
|
|
|
refCount int32
|
|
|
|
}
|
|
|
|
|
|
|
|
func (block *Block) decreaseReference() {
|
|
|
|
if atomic.AddInt32(&block.refCount, -1) == 0 {
|
|
|
|
bytesPool.Put(block.Bytes)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
func (block *Block) increaseReference() {
|
|
|
|
atomic.AddInt32(&block.refCount, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
// get bytes from the LRU cache of []byte first, then from the bytes pool
|
|
|
|
// when []byte in LRU cache is evicted, it will be put back to the bytes pool
|
2016-04-18 03:03:45 +08:00
|
|
|
func getBytesForFileBlock(r *os.File, offset int64, readSize int) (dataSlice []byte, block *Block, err error) {
|
2016-04-16 02:56:53 +08:00
|
|
|
// check cache, return if found
|
|
|
|
cacheKey := fmt.Sprintf("%d:%d:%d", r.Fd(), offset>>3, readSize)
|
|
|
|
if obj, found := bytesCache.Get(cacheKey); found {
|
|
|
|
block = obj.(*Block)
|
|
|
|
block.increaseReference()
|
2016-04-18 03:03:45 +08:00
|
|
|
dataSlice = block.Bytes[0:readSize]
|
|
|
|
return dataSlice, block, nil
|
2016-04-16 02:56:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// get the []byte from pool
|
|
|
|
b := bytesPool.Get(readSize)
|
|
|
|
// refCount = 2, one by the bytesCache, one by the actual needle object
|
|
|
|
block = &Block{Bytes: b, refCount: 2}
|
2016-04-18 03:03:45 +08:00
|
|
|
dataSlice = block.Bytes[0:readSize]
|
|
|
|
_, err = r.ReadAt(dataSlice, offset)
|
2016-04-16 02:56:53 +08:00
|
|
|
bytesCache.Add(cacheKey, block)
|
2016-04-18 03:03:45 +08:00
|
|
|
return dataSlice, block, err
|
2016-04-16 02:56:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (n *Needle) ReleaseMemory() {
|
|
|
|
n.rawBlock.decreaseReference()
|
|
|
|
}
|
|
|
|
func ReleaseBytes(b []byte) {
|
|
|
|
bytesPool.Put(b)
|
|
|
|
}
|