mirror of
https://github.com/rclone/rclone.git
synced 2025-01-09 15:58:28 +01:00
vfs: fix directory locking caused by slow directory listings
Before this fix we took the directory lock to read the ModTime of the directory. This was causing locking on directories which were being re-read from the backend. This commit gives the modtime its own lock so it can be read even when the directory is being updated. See: https://forum.rclone.org/t/high-cpu-load-with-rclone-mount/17604
This commit is contained in:
parent
811b30d116
commit
ddfde68140
16
vfs/dir.go
16
vfs/dir.go
@ -29,12 +29,14 @@ type Dir struct {
|
|||||||
mu sync.RWMutex // protects the following
|
mu sync.RWMutex // protects the following
|
||||||
parent *Dir // parent, nil for root
|
parent *Dir // parent, nil for root
|
||||||
path string
|
path string
|
||||||
modTime time.Time
|
|
||||||
entry fs.Directory
|
entry fs.Directory
|
||||||
read time.Time // time directory entry last read
|
read time.Time // time directory entry last read
|
||||||
items map[string]Node // directory entries - can be empty but not nil
|
items map[string]Node // directory entries - can be empty but not nil
|
||||||
virtual map[string]vState // virtual directory entries - may be nil
|
virtual map[string]vState // virtual directory entries - may be nil
|
||||||
sys atomic.Value // user defined info to be attached here
|
sys atomic.Value // user defined info to be attached here
|
||||||
|
|
||||||
|
modTimeMu sync.Mutex // protects the following
|
||||||
|
modTime time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
//go:generate stringer -type=vState
|
//go:generate stringer -type=vState
|
||||||
@ -270,11 +272,13 @@ func (d *Dir) _age(when time.Time) (age time.Duration, stale bool) {
|
|||||||
// reading everything again
|
// reading everything again
|
||||||
func (d *Dir) rename(newParent *Dir, fsDir fs.Directory) {
|
func (d *Dir) rename(newParent *Dir, fsDir fs.Directory) {
|
||||||
d.ForgetAll()
|
d.ForgetAll()
|
||||||
|
d.modTimeMu.Lock()
|
||||||
|
d.modTime = fsDir.ModTime(context.TODO())
|
||||||
|
d.modTimeMu.Unlock()
|
||||||
d.mu.Lock()
|
d.mu.Lock()
|
||||||
d.parent = newParent
|
d.parent = newParent
|
||||||
d.entry = fsDir
|
d.entry = fsDir
|
||||||
d.path = fsDir.Remote()
|
d.path = fsDir.Remote()
|
||||||
d.modTime = fsDir.ModTime(context.TODO())
|
|
||||||
d.read = time.Time{}
|
d.read = time.Time{}
|
||||||
d.mu.Unlock()
|
d.mu.Unlock()
|
||||||
}
|
}
|
||||||
@ -550,8 +554,8 @@ func (d *Dir) isEmpty() (bool, error) {
|
|||||||
|
|
||||||
// ModTime returns the modification time of the directory
|
// ModTime returns the modification time of the directory
|
||||||
func (d *Dir) ModTime() time.Time {
|
func (d *Dir) ModTime() time.Time {
|
||||||
d.mu.RLock()
|
d.modTimeMu.Lock()
|
||||||
defer d.mu.RUnlock()
|
defer d.modTimeMu.Unlock()
|
||||||
// fs.Debugf(d.path, "Dir.ModTime %v", d.modTime)
|
// fs.Debugf(d.path, "Dir.ModTime %v", d.modTime)
|
||||||
return d.modTime
|
return d.modTime
|
||||||
}
|
}
|
||||||
@ -566,9 +570,9 @@ func (d *Dir) SetModTime(modTime time.Time) error {
|
|||||||
if d.vfs.Opt.ReadOnly {
|
if d.vfs.Opt.ReadOnly {
|
||||||
return EROFS
|
return EROFS
|
||||||
}
|
}
|
||||||
d.mu.Lock()
|
d.modTimeMu.Lock()
|
||||||
d.modTime = modTime
|
d.modTime = modTime
|
||||||
d.mu.Unlock()
|
d.modTimeMu.Unlock()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user