1
mirror of https://github.com/rclone/rclone synced 2025-01-04 05:06:24 +01:00

vfs: consistently use f.Path() or f._path() in File logs to avoid deadlocks

Previously we were using f which calls f.String() which calls f.Path()
which can cause a deadlock if uses carelessly.

This patch explicitly calls f.Path() or f._path() to bring attention
to the fact that there is a call to a method.
This commit is contained in:
Nick Craig-Wood 2020-04-15 12:17:28 +01:00
parent 19db0df639
commit 1f50b70919

View File

@ -152,7 +152,7 @@ func (f *File) applyPendingRename() {
if fun == nil || writing { if fun == nil || writing {
return return
} }
fs.Debugf(f.o, "Running delayed rename now") fs.Debugf(f.Path(), "Running delayed rename now")
if err := fun(context.TODO()); err != nil { if err := fun(context.TODO()); err != nil {
fs.Errorf(f.Path(), "delayed File.Rename error: %v", err) fs.Errorf(f.Path(), "delayed File.Rename error: %v", err)
} }
@ -164,7 +164,6 @@ func (f *File) applyPendingRename() {
func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error { func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
f.mu.RLock() f.mu.RLock()
d := f.d d := f.d
o := f.o
oldPendingRenameFun := f.pendingRenameFun oldPendingRenameFun := f.pendingRenameFun
f.mu.RUnlock() f.mu.RUnlock()
@ -211,7 +210,7 @@ func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
return err return err
} }
// Update the node with the new details // Update the node with the new details
fs.Debugf(o, "Updating file with %v %p", newObject, f) fs.Debugf(f.Path(), "Updating file with %v %p", newObject, f)
// f.rename(destDir, newObject) // f.rename(destDir, newObject)
f.mu.Lock() f.mu.Lock()
f.o = newObject f.o = newObject
@ -237,7 +236,7 @@ func (f *File) rename(ctx context.Context, destDir *Dir, newName string) error {
f.mu.Unlock() f.mu.Unlock()
if writing { if writing {
fs.Debugf(o, "File is currently open, delaying rename %p", f) fs.Debugf(f.Path(), "File is currently open, delaying rename %p", f)
f.mu.Lock() f.mu.Lock()
f.pendingRenameFun = renameCall f.pendingRenameFun = renameCall
f.mu.Unlock() f.mu.Unlock()
@ -274,7 +273,7 @@ func (f *File) delWriter(h Handle, modifiedCacheFile bool) (lastWriterAndModifie
f.writers = append(f.writers[:found], f.writers[found+1:]...) f.writers = append(f.writers[:found], f.writers[found+1:]...)
atomic.AddInt32(&f.nwriters, -1) atomic.AddInt32(&f.nwriters, -1)
} else { } else {
fs.Debugf(f.o, "File.delWriter couldn't find handle") fs.Debugf(f._path(), "File.delWriter couldn't find handle")
} }
if _, ok := h.(*RWFileHandle); ok { if _, ok := h.(*RWFileHandle); ok {
f.readWriters-- f.readWriters--
@ -391,7 +390,7 @@ func (f *File) SetModTime(modTime time.Time) error {
} }
// Apply a pending mod time // Apply a pending mod time
// Call with the write mutex held // Call with the mutex held
func (f *File) _applyPendingModTime() error { func (f *File) _applyPendingModTime() error {
if f.pendingModTime.IsZero() { if f.pendingModTime.IsZero() {
return nil return nil
@ -412,11 +411,11 @@ func (f *File) _applyPendingModTime() error {
err := f.o.SetModTime(context.TODO(), f.pendingModTime) err := f.o.SetModTime(context.TODO(), f.pendingModTime)
switch err { switch err {
case nil: case nil:
fs.Debugf(f.o, "File._applyPendingModTime OK") fs.Debugf(f._path(), "File._applyPendingModTime OK")
case fs.ErrorCantSetModTime, fs.ErrorCantSetModTimeWithoutDelete: case fs.ErrorCantSetModTime, fs.ErrorCantSetModTimeWithoutDelete:
// do nothing, in order to not break "touch somefile" if it exists already // do nothing, in order to not break "touch somefile" if it exists already
default: default:
fs.Errorf(f, "File._applyPendingModTime error: %v", err) fs.Debugf(f._path(), "File._applyPendingModTime error: %v", err)
return err return err
} }
@ -496,11 +495,11 @@ func (f *File) openRead() (fh *ReadFileHandle, err error) {
if err != nil { if err != nil {
return nil, err return nil, err
} }
// fs.Debugf(o, "File.openRead") // fs.Debugf(f.Path(), "File.openRead")
fh, err = newReadFileHandle(f) fh, err = newReadFileHandle(f)
if err != nil { if err != nil {
fs.Errorf(f, "File.openRead failed: %v", err) fs.Debugf(f.Path(), "File.openRead failed: %v", err)
return nil, err return nil, err
} }
return fh, nil return fh, nil
@ -515,11 +514,11 @@ func (f *File) openWrite(flags int) (fh *WriteFileHandle, err error) {
if d.vfs.Opt.ReadOnly { if d.vfs.Opt.ReadOnly {
return nil, EROFS return nil, EROFS
} }
// fs.Debugf(o, "File.openWrite") // fs.Debugf(f.Path(), "File.openWrite")
fh, err = newWriteFileHandle(d, f, f.Path(), flags) fh, err = newWriteFileHandle(d, f, f.Path(), flags)
if err != nil { if err != nil {
fs.Errorf(f, "File.openWrite failed: %v", err) fs.Debugf(f.Path(), "File.openWrite failed: %v", err)
return nil, err return nil, err
} }
return fh, nil return fh, nil
@ -537,11 +536,11 @@ func (f *File) openRW(flags int) (fh *RWFileHandle, err error) {
if flags&accessModeMask != os.O_RDONLY && d.vfs.Opt.ReadOnly { if flags&accessModeMask != os.O_RDONLY && d.vfs.Opt.ReadOnly {
return nil, EROFS return nil, EROFS
} }
// fs.Debugf(o, "File.openRW") // fs.Debugf(f.Path(), "File.openRW")
fh, err = newRWFileHandle(d, f, flags) fh, err = newRWFileHandle(d, f, flags)
if err != nil { if err != nil {
fs.Errorf(f, "File.openRW failed: %v", err) fs.Debugf(f.Path(), "File.openRW failed: %v", err)
return nil, err return nil, err
} }
return fh, nil return fh, nil
@ -568,7 +567,7 @@ func (f *File) Remove() error {
if f.o != nil { if f.o != nil {
err := f.o.Remove(context.TODO()) err := f.o.Remove(context.TODO())
if err != nil { if err != nil {
fs.Errorf(f, "File.Remove file error: %v", err) fs.Debugf(f._path(), "File.Remove file error: %v", err)
f.mu.Unlock() f.mu.Unlock()
f.muRW.Unlock() f.muRW.Unlock()
return err return err
@ -634,7 +633,7 @@ func (f *File) Fs() fs.Fs {
// //
// We ignore O_SYNC and O_EXCL // We ignore O_SYNC and O_EXCL
func (f *File) Open(flags int) (fd Handle, err error) { func (f *File) Open(flags int) (fd Handle, err error) {
defer log.Trace(f, "flags=%s", decodeOpenFlags(flags))("fd=%v, err=%v", &fd, &err) defer log.Trace(f.Path(), "flags=%s", decodeOpenFlags(flags))("fd=%v, err=%v", &fd, &err)
var ( var (
write bool // if set need write support write bool // if set need write support
read bool // if set need read support read bool // if set need read support
@ -658,7 +657,7 @@ func (f *File) Open(flags int) (fd Handle, err error) {
read = true read = true
write = true write = true
default: default:
fs.Errorf(f, "Can't figure out how to open with flags: 0x%X", flags) fs.Debugf(f.Path(), "Can't figure out how to open with flags: 0x%X", flags)
return nil, EPERM return nil, EPERM
} }
@ -704,7 +703,7 @@ func (f *File) Open(flags int) (fd Handle, err error) {
fd, err = f.openRead() fd, err = f.openRead()
} }
} else { } else {
fs.Errorf(f, "Can't figure out how to open with flags: 0x%X", flags) fs.Debugf(f.Path(), "Can't figure out how to open with flags: 0x%X", flags)
return nil, EPERM return nil, EPERM
} }
// if creating a file, add the file to the directory // if creating a file, add the file to the directory
@ -729,7 +728,7 @@ func (f *File) Truncate(size int64) (err error) {
// If have writers then call truncate for each writer // If have writers then call truncate for each writer
if len(writers) != 0 { if len(writers) != 0 {
fs.Debugf(f, "Truncating %d file handles", len(writers)) fs.Debugf(f.Path(), "Truncating %d file handles", len(writers))
for _, h := range writers { for _, h := range writers {
truncateErr := h.Truncate(size) truncateErr := h.Truncate(size)
if truncateErr != nil { if truncateErr != nil {
@ -744,7 +743,7 @@ func (f *File) Truncate(size int64) (err error) {
return nil return nil
} }
fs.Debugf(f, "Truncating file") fs.Debugf(f.Path(), "Truncating file")
// Otherwise if no writers then truncate the file by opening // Otherwise if no writers then truncate the file by opening
// the file and truncating it. // the file and truncating it.