mirror of
https://github.com/rclone/rclone
synced 2024-12-25 17:03:45 +01:00
628ff8e524
Co-authored-by: Volodymyr Kit <v.kit@maytech.net>
109 lines
2.4 KiB
Go
109 lines
2.4 KiB
Go
package quatrix
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/rclone/rclone/fs"
|
|
)
|
|
|
|
// UploadMemoryManager dynamically calculates every chunk size for the transfer and increases or decreases it
|
|
// depending on the upload speed. This makes general upload time smaller, because transfers that are faster
|
|
// does not have to wait for the slower ones until they finish upload.
|
|
type UploadMemoryManager struct {
|
|
m sync.Mutex
|
|
useDynamicSize bool
|
|
shared int64
|
|
reserved int64
|
|
effectiveTime time.Duration
|
|
fileUsage map[string]int64
|
|
}
|
|
|
|
// NewUploadMemoryManager is a constructor for UploadMemoryManager
|
|
func NewUploadMemoryManager(ci *fs.ConfigInfo, opt *Options) *UploadMemoryManager {
|
|
useDynamicSize := true
|
|
|
|
sharedMemory := int64(opt.MaximalSummaryChunkSize) - int64(opt.MinimalChunkSize)*int64(ci.Transfers)
|
|
if sharedMemory <= 0 {
|
|
sharedMemory = 0
|
|
useDynamicSize = false
|
|
}
|
|
|
|
return &UploadMemoryManager{
|
|
useDynamicSize: useDynamicSize,
|
|
shared: sharedMemory,
|
|
reserved: int64(opt.MinimalChunkSize),
|
|
effectiveTime: time.Duration(opt.EffectiveUploadTime),
|
|
fileUsage: map[string]int64{},
|
|
}
|
|
}
|
|
|
|
// Consume -- decide amount of memory to consume
|
|
func (u *UploadMemoryManager) Consume(fileID string, neededMemory int64, speed float64) int64 {
|
|
if !u.useDynamicSize {
|
|
if neededMemory < u.reserved {
|
|
return neededMemory
|
|
}
|
|
|
|
return u.reserved
|
|
}
|
|
|
|
u.m.Lock()
|
|
defer u.m.Unlock()
|
|
|
|
borrowed, found := u.fileUsage[fileID]
|
|
if found {
|
|
u.shared += borrowed
|
|
borrowed = 0
|
|
}
|
|
|
|
defer func() { u.fileUsage[fileID] = borrowed }()
|
|
|
|
effectiveChunkSize := int64(speed * u.effectiveTime.Seconds())
|
|
|
|
if effectiveChunkSize < u.reserved {
|
|
effectiveChunkSize = u.reserved
|
|
}
|
|
|
|
if neededMemory < effectiveChunkSize {
|
|
effectiveChunkSize = neededMemory
|
|
}
|
|
|
|
if effectiveChunkSize <= u.reserved {
|
|
return effectiveChunkSize
|
|
}
|
|
|
|
toBorrow := effectiveChunkSize - u.reserved
|
|
|
|
if toBorrow <= u.shared {
|
|
u.shared -= toBorrow
|
|
borrowed = toBorrow
|
|
|
|
return effectiveChunkSize
|
|
}
|
|
|
|
borrowed = u.shared
|
|
u.shared = 0
|
|
|
|
return borrowed + u.reserved
|
|
}
|
|
|
|
// Return returns consumed memory for the previous chunk upload to the memory pool
|
|
func (u *UploadMemoryManager) Return(fileID string) {
|
|
if !u.useDynamicSize {
|
|
return
|
|
}
|
|
|
|
u.m.Lock()
|
|
defer u.m.Unlock()
|
|
|
|
borrowed, found := u.fileUsage[fileID]
|
|
if !found {
|
|
return
|
|
}
|
|
|
|
u.shared += borrowed
|
|
|
|
delete(u.fileUsage, fileID)
|
|
}
|