mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2024-01-19 02:48:24 +00:00
add back writes to swap file when too many in memory chunks are used.
This commit is contained in:
parent
fc0a583a4d
commit
011a41b561
|
@ -30,7 +30,10 @@ func newMemoryChunkPages(fh *FileHandle, chunkSize int64) *ChunkedDirtyPages {
|
||||||
fh: fh,
|
fh: fh,
|
||||||
}
|
}
|
||||||
|
|
||||||
dirtyPages.uploadPipeline = page_writer.NewUploadPipeline(fh.wfs.concurrentWriters, chunkSize, dirtyPages.saveChunkedFileIntevalToStorage, fh.wfs.option.ConcurrentWriters)
|
swapFileDir := fh.wfs.option.getTempFilePageDir()
|
||||||
|
|
||||||
|
dirtyPages.uploadPipeline = page_writer.NewUploadPipeline(fh.wfs.concurrentWriters, chunkSize,
|
||||||
|
dirtyPages.saveChunkedFileIntevalToStorage, fh.wfs.option.ConcurrentWriters, swapFileDir)
|
||||||
|
|
||||||
return dirtyPages
|
return dirtyPages
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,6 @@ func newPageWriter(fh *FileHandle, chunkSize int64) *PageWriter {
|
||||||
fh: fh,
|
fh: fh,
|
||||||
chunkSize: chunkSize,
|
chunkSize: chunkSize,
|
||||||
randomWriter: newMemoryChunkPages(fh, chunkSize),
|
randomWriter: newMemoryChunkPages(fh, chunkSize),
|
||||||
// randomWriter: newTempFileDirtyPages(fh.f, chunkSize),
|
|
||||||
}
|
}
|
||||||
return pw
|
return pw
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,10 +3,13 @@ package page_writer
|
||||||
import (
|
import (
|
||||||
"github.com/chrislusf/seaweedfs/weed/util"
|
"github.com/chrislusf/seaweedfs/weed/util"
|
||||||
"github.com/chrislusf/seaweedfs/weed/util/mem"
|
"github.com/chrislusf/seaweedfs/weed/util/mem"
|
||||||
|
"sync/atomic"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
_ = PageChunk(&MemChunk{})
|
_ = PageChunk(&MemChunk{})
|
||||||
|
|
||||||
|
memChunkCounter int64
|
||||||
)
|
)
|
||||||
|
|
||||||
type MemChunk struct {
|
type MemChunk struct {
|
||||||
|
@ -17,6 +20,7 @@ type MemChunk struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewMemChunk(logicChunkIndex LogicChunkIndex, chunkSize int64) *MemChunk {
|
func NewMemChunk(logicChunkIndex LogicChunkIndex, chunkSize int64) *MemChunk {
|
||||||
|
atomic.AddInt64(&memChunkCounter, 1)
|
||||||
return &MemChunk{
|
return &MemChunk{
|
||||||
logicChunkIndex: logicChunkIndex,
|
logicChunkIndex: logicChunkIndex,
|
||||||
chunkSize: chunkSize,
|
chunkSize: chunkSize,
|
||||||
|
@ -26,6 +30,7 @@ func NewMemChunk(logicChunkIndex LogicChunkIndex, chunkSize int64) *MemChunk {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MemChunk) FreeResource() {
|
func (mc *MemChunk) FreeResource() {
|
||||||
|
atomic.AddInt64(&memChunkCounter, -1)
|
||||||
mem.Free(mc.buf)
|
mem.Free(mc.buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,7 @@ type UploadPipeline struct {
|
||||||
activeReadChunks map[LogicChunkIndex]int
|
activeReadChunks map[LogicChunkIndex]int
|
||||||
activeReadChunksLock sync.Mutex
|
activeReadChunksLock sync.Mutex
|
||||||
bufferChunkLimit int
|
bufferChunkLimit int
|
||||||
|
swapFile *SwapFile
|
||||||
}
|
}
|
||||||
|
|
||||||
type SealedChunk struct {
|
type SealedChunk struct {
|
||||||
|
@ -40,7 +41,7 @@ func (sc *SealedChunk) FreeReference(messageOnFree string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewUploadPipeline(writers *util.LimitedConcurrentExecutor, chunkSize int64, saveToStorageFn SaveToStorageFunc, bufferChunkLimit int) *UploadPipeline {
|
func NewUploadPipeline(writers *util.LimitedConcurrentExecutor, chunkSize int64, saveToStorageFn SaveToStorageFunc, bufferChunkLimit int, swapFileDir string) *UploadPipeline {
|
||||||
return &UploadPipeline{
|
return &UploadPipeline{
|
||||||
ChunkSize: chunkSize,
|
ChunkSize: chunkSize,
|
||||||
writableChunks: make(map[LogicChunkIndex]PageChunk),
|
writableChunks: make(map[LogicChunkIndex]PageChunk),
|
||||||
|
@ -50,6 +51,7 @@ func NewUploadPipeline(writers *util.LimitedConcurrentExecutor, chunkSize int64,
|
||||||
saveToStorageFn: saveToStorageFn,
|
saveToStorageFn: saveToStorageFn,
|
||||||
activeReadChunks: make(map[LogicChunkIndex]int),
|
activeReadChunks: make(map[LogicChunkIndex]int),
|
||||||
bufferChunkLimit: bufferChunkLimit,
|
bufferChunkLimit: bufferChunkLimit,
|
||||||
|
swapFile: NewSwapFile(swapFileDir, chunkSize),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,10 +61,14 @@ func (up *UploadPipeline) SaveDataAt(p []byte, off int64) (n int) {
|
||||||
|
|
||||||
logicChunkIndex := LogicChunkIndex(off / up.ChunkSize)
|
logicChunkIndex := LogicChunkIndex(off / up.ChunkSize)
|
||||||
|
|
||||||
memChunk, found := up.writableChunks[logicChunkIndex]
|
pageChunk, found := up.writableChunks[logicChunkIndex]
|
||||||
if !found {
|
if !found {
|
||||||
if len(up.writableChunks) < up.bufferChunkLimit {
|
if atomic.LoadInt64(&memChunkCounter) > 4*int64(up.bufferChunkLimit) {
|
||||||
memChunk = NewMemChunk(logicChunkIndex, up.ChunkSize)
|
// if total number of chunks is over 4 times of per file buffer count limit
|
||||||
|
pageChunk = up.swapFile.NewTempFileChunk(logicChunkIndex)
|
||||||
|
} else if len(up.writableChunks) < up.bufferChunkLimit {
|
||||||
|
// if current file chunks is still under the per file buffer count limit
|
||||||
|
pageChunk = NewMemChunk(logicChunkIndex, up.ChunkSize)
|
||||||
} else {
|
} else {
|
||||||
fullestChunkIndex, fullness := LogicChunkIndex(-1), int64(0)
|
fullestChunkIndex, fullness := LogicChunkIndex(-1), int64(0)
|
||||||
for lci, mc := range up.writableChunks {
|
for lci, mc := range up.writableChunks {
|
||||||
|
@ -75,12 +81,12 @@ func (up *UploadPipeline) SaveDataAt(p []byte, off int64) (n int) {
|
||||||
up.moveToSealed(up.writableChunks[fullestChunkIndex], fullestChunkIndex)
|
up.moveToSealed(up.writableChunks[fullestChunkIndex], fullestChunkIndex)
|
||||||
delete(up.writableChunks, fullestChunkIndex)
|
delete(up.writableChunks, fullestChunkIndex)
|
||||||
// fmt.Printf("flush chunk %d with %d bytes written\n", logicChunkIndex, fullness)
|
// fmt.Printf("flush chunk %d with %d bytes written\n", logicChunkIndex, fullness)
|
||||||
memChunk = NewMemChunk(logicChunkIndex, up.ChunkSize)
|
pageChunk = NewMemChunk(logicChunkIndex, up.ChunkSize)
|
||||||
}
|
}
|
||||||
up.writableChunks[logicChunkIndex] = memChunk
|
up.writableChunks[logicChunkIndex] = pageChunk
|
||||||
}
|
}
|
||||||
n = memChunk.WriteDataAt(p, off)
|
n = pageChunk.WriteDataAt(p, off)
|
||||||
up.maybeMoveToSealed(memChunk, logicChunkIndex)
|
up.maybeMoveToSealed(pageChunk, logicChunkIndex)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -179,6 +185,7 @@ func (up *UploadPipeline) moveToSealed(memChunk PageChunk, logicChunkIndex Logic
|
||||||
}
|
}
|
||||||
|
|
||||||
func (up *UploadPipeline) Shutdown() {
|
func (up *UploadPipeline) Shutdown() {
|
||||||
|
up.swapFile.FreeResource()
|
||||||
for logicChunkIndex, sealedChunk := range up.sealedChunks {
|
for logicChunkIndex, sealedChunk := range up.sealedChunks {
|
||||||
sealedChunk.FreeReference(fmt.Sprintf("%s uploadpipeline shutdown chunk %d", up.filepath, logicChunkIndex))
|
sealedChunk.FreeReference(fmt.Sprintf("%s uploadpipeline shutdown chunk %d", up.filepath, logicChunkIndex))
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@ import (
|
||||||
|
|
||||||
func TestUploadPipeline(t *testing.T) {
|
func TestUploadPipeline(t *testing.T) {
|
||||||
|
|
||||||
uploadPipeline := NewUploadPipeline(nil, 2*1024*1024, nil, 16)
|
uploadPipeline := NewUploadPipeline(nil, 2*1024*1024, nil, 16, "")
|
||||||
|
|
||||||
writeRange(uploadPipeline, 0, 131072)
|
writeRange(uploadPipeline, 0, 131072)
|
||||||
writeRange(uploadPipeline, 131072, 262144)
|
writeRange(uploadPipeline, 131072, 262144)
|
||||||
|
|
|
@ -16,6 +16,7 @@ import (
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"path/filepath"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/hanwen/go-fuse/v2/fs"
|
"github.com/hanwen/go-fuse/v2/fs"
|
||||||
|
@ -51,6 +52,7 @@ type Option struct {
|
||||||
UidGidMapper *meta_cache.UidGidMapper
|
UidGidMapper *meta_cache.UidGidMapper
|
||||||
|
|
||||||
uniqueCacheDir string
|
uniqueCacheDir string
|
||||||
|
uniqueCacheTempPageDir string
|
||||||
}
|
}
|
||||||
|
|
||||||
type WFS struct {
|
type WFS struct {
|
||||||
|
@ -177,7 +179,12 @@ func (wfs *WFS) getCurrentFiler() pb.ServerAddress {
|
||||||
func (option *Option) setupUniqueCacheDirectory() {
|
func (option *Option) setupUniqueCacheDirectory() {
|
||||||
cacheUniqueId := util.Md5String([]byte(option.MountDirectory + string(option.FilerAddresses[0]) + option.FilerMountRootPath + util.Version()))[0:8]
|
cacheUniqueId := util.Md5String([]byte(option.MountDirectory + string(option.FilerAddresses[0]) + option.FilerMountRootPath + util.Version()))[0:8]
|
||||||
option.uniqueCacheDir = path.Join(option.CacheDir, cacheUniqueId)
|
option.uniqueCacheDir = path.Join(option.CacheDir, cacheUniqueId)
|
||||||
os.MkdirAll(option.uniqueCacheDir, os.FileMode(0777)&^option.Umask)
|
option.uniqueCacheTempPageDir = filepath.Join(option.uniqueCacheDir, "swap")
|
||||||
|
os.MkdirAll(option.uniqueCacheTempPageDir, os.FileMode(0777)&^option.Umask)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (option *Option) getTempFilePageDir() string {
|
||||||
|
return option.uniqueCacheTempPageDir
|
||||||
}
|
}
|
||||||
|
|
||||||
func (option *Option) getUniqueCacheDir() string {
|
func (option *Option) getUniqueCacheDir() string {
|
||||||
|
|
Loading…
Reference in a new issue