1
0
mirror of https://github.com/rclone/rclone.git synced 2025-01-13 20:38:12 +02:00

vfs: implement lock ordering between File and Dir to eliminate deadlocks

As part of this we take a copy of the directory path as calling
d.Path() violates the total locking order.

See the comment at the top of file.go for details
This commit is contained in:
Nick Craig-Wood 2020-04-14 17:55:18 +01:00
parent 1e4589db18
commit 268fcbb973
2 changed files with 36 additions and 15 deletions

View File

@ -288,7 +288,7 @@ func (d *Dir) _readDirFromEntries(entries fs.DirEntries, dirTree dirtree.DirTree
if file, ok := node.(*File); node != nil && ok { if file, ok := node.(*File); node != nil && ok {
file.setObjectNoUpdate(obj) file.setObjectNoUpdate(obj)
} else { } else {
node = newFile(d, obj, name) node = newFile(d, d.path, obj, name)
} }
case fs.Directory: case fs.Directory:
// Reuse old dir value if it exists // Reuse old dir value if it exists
@ -511,7 +511,7 @@ func (d *Dir) Create(name string, flags int) (*File, error) {
return nil, EROFS return nil, EROFS
} }
// This gets added to the directory when the file is opened for write // This gets added to the directory when the file is opened for write
return newFile(d, nil, name), nil return newFile(d, d.Path(), nil, name), nil
} }
// Mkdir creates a new directory // Mkdir creates a new directory

View File

@ -14,6 +14,18 @@ import (
"github.com/rclone/rclone/fs/operations" "github.com/rclone/rclone/fs/operations"
) )
// The File object is tightly coupled to the Dir object. Since they
// both have locks there is plenty of potential for deadlocks. In
// order to mitigate this, we use the following conventions
//
// File may read directly, without locking, from the read-only section
// of the Dir object, eg vfs, and f members. File may **not** read any
// other members directly.
//
// File may **not** call Dir methods with the File lock held. This
// preserves total lock ordering and makes File subordinate to Dir as
// far as locking is concerned, preventing deadlocks.
// File represents a file // File represents a file
type File struct { type File struct {
inode uint64 // inode number - read only inode uint64 // inode number - read only
@ -21,6 +33,7 @@ type File struct {
mu sync.RWMutex // protects the following mu sync.RWMutex // protects the following
d *Dir // parent directory d *Dir // parent directory
dPath string // path of parent directory. NB dir rename means all Files are flushed
o fs.Object // NB o may be nil if file is being written o fs.Object // NB o may be nil if file is being written
leaf string // leaf name of the object leaf string // leaf name of the object
rwOpenCount int // number of open files on this handle rwOpenCount int // number of open files on this handle
@ -39,9 +52,10 @@ type File struct {
// newFile creates a new File // newFile creates a new File
// //
// o may be nil // o may be nil
func newFile(d *Dir, o fs.Object, leaf string) *File { func newFile(d *Dir, dPath string, o fs.Object, leaf string) *File {
f := &File{ f := &File{
d: d, d: d,
dPath: dPath,
o: o, o: o,
leaf: leaf, leaf: leaf,
inode: newInode(), inode: newInode(),
@ -91,15 +105,15 @@ func (f *File) Name() (name string) {
// _path returns the full path of the file // _path returns the full path of the file
// use when lock is held // use when lock is held
func (f *File) _path() string { func (f *File) _path() string {
return path.Join(f.d.Path(), f.leaf) return path.Join(f.dPath, f.leaf)
} }
// Path returns the full path of the file // Path returns the full path of the file
func (f *File) Path() string { func (f *File) Path() string {
f.mu.RLock() f.mu.RLock()
d, leaf := f.d, f.leaf dPath, leaf := f.dPath, f.leaf
f.mu.RUnlock() f.mu.RUnlock()
return path.Join(d.Path(), leaf) return path.Join(dPath, leaf)
} }
// osPath returns the full path of the file in the cache in OS format // osPath returns the full path of the file in the cache in OS format
@ -154,7 +168,8 @@ func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
return err return err
} }
newPath := path.Join(destDir.path, newName) // File.mu is unlocked here to call Dir.Path()
newPath := path.Join(destDir.Path(), newName)
renameCall := func(ctx context.Context) error { renameCall := func(ctx context.Context) error {
// chain rename calls if any // chain rename calls if any
@ -207,8 +222,10 @@ func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
} }
// rename the file object // rename the file object
dPath := destDir.Path()
f.mu.Lock() f.mu.Lock()
f.d = destDir f.d = destDir
f.dPath = dPath
f.leaf = newName f.leaf = newName
writing := f._writingInProgress() writing := f._writingInProgress()
f.mu.Unlock() f.mu.Unlock()
@ -313,18 +330,19 @@ func (f *File) activeWriters() int {
// if NoModTime is set then it returns the mod time of the directory // if NoModTime is set then it returns the mod time of the directory
func (f *File) ModTime() (modTime time.Time) { func (f *File) ModTime() (modTime time.Time) {
f.mu.RLock() f.mu.RLock()
defer f.mu.RUnlock() d, o, pendingModTime := f.d, f.o, f.pendingModTime
f.mu.RUnlock()
if f.d.vfs.Opt.NoModTime { if d.vfs.Opt.NoModTime {
return f.d.modTime return d.ModTime()
} }
if !f.pendingModTime.IsZero() { if !pendingModTime.IsZero() {
return f.pendingModTime return pendingModTime
} }
if f.o == nil { if o == nil {
return f.d.modTime return d.ModTime()
} }
return f.o.ModTime(context.TODO()) return o.ModTime(context.TODO())
} }
// nonNegative returns 0 if i is -ve, i otherwise // nonNegative returns 0 if i is -ve, i otherwise
@ -417,6 +435,7 @@ func (f *File) setObject(o fs.Object) {
_ = f._applyPendingModTime() _ = f._applyPendingModTime()
f.mu.Unlock() f.mu.Unlock()
// Release File.mu before calling Dir method
f.d.addObject(f) f.d.addObject(f)
} }
@ -553,6 +572,7 @@ func (f *File) Remove() error {
f.muRW.Unlock() f.muRW.Unlock()
// Remove the item from the directory listing // Remove the item from the directory listing
// called with File.mu released
d.delObject(f.Name()) d.delObject(f.Name())
// Remove the object from the cache // Remove the object from the cache
if d.vfs.Opt.CacheMode >= CacheModeMinimal { if d.vfs.Opt.CacheMode >= CacheModeMinimal {
@ -676,6 +696,7 @@ func (f *File) Open(flags int) (fd Handle, err error) {
} }
// if creating a file, add the file to the directory // if creating a file, add the file to the directory
if err == nil && flags&os.O_CREATE != 0 { if err == nil && flags&os.O_CREATE != 0 {
// called without File.mu held
d.addObject(f) d.addObject(f)
} }
return fd, err return fd, err