fusefrontend: relay Utimens to go-fuse
Commit af5441dcd9033e81da43ab77887a7b5aac693ab6 has caused a regression ( https://github.com/rfjakob/gocryptfs/issues/35 ) that is fixed by this commit. The go-fuse library by now has all the syscall wrappers in place to correctly handle Utimens, also for symlinks. Instead of duplicating the effort here just call into go-fuse. Closes #35
This commit is contained in:
parent
8f1e51789d
commit
fca1b82417
@ -33,18 +33,16 @@ type file struct {
|
||||
// Every FUSE entrypoint should RLock(). The only user of Lock() is
|
||||
// Release(), which closes the fd and sets "released" to true.
|
||||
fdLock sync.RWMutex
|
||||
|
||||
// Was the file opened O_WRONLY?
|
||||
writeOnly bool
|
||||
|
||||
// Content encryption helper
|
||||
contentEnc *contentenc.ContentEnc
|
||||
|
||||
// Inode number
|
||||
ino uint64
|
||||
|
||||
// File header
|
||||
header *contentenc.FileHeader
|
||||
// go-fuse nodefs.loopbackFile
|
||||
loopbackFile nodefs.File
|
||||
}
|
||||
|
||||
func NewFile(fd *os.File, writeOnly bool, contentEnc *contentenc.ContentEnc) (nodefs.File, fuse.Status) {
|
||||
@ -61,6 +59,7 @@ func NewFile(fd *os.File, writeOnly bool, contentEnc *contentenc.ContentEnc) (no
|
||||
writeOnly: writeOnly,
|
||||
contentEnc: contentEnc,
|
||||
ino: st.Ino,
|
||||
loopbackFile: nodefs.NewLoopbackFile(fd),
|
||||
}, fuse.OK
|
||||
}
|
||||
|
||||
@ -386,41 +385,8 @@ func (f *file) GetAttr(a *fuse.Attr) fuse.Status {
|
||||
return fuse.OK
|
||||
}
|
||||
|
||||
const _UTIME_OMIT = ((1 << 30) - 2)
|
||||
|
||||
// utimeToTimespec converts a "time.Time" pointer as passed to Utimens to a
|
||||
// Timespec that can be passed to the utimensat syscall.
|
||||
func utimeToTimespec(t *time.Time) (ts syscall.Timespec) {
|
||||
if t == nil {
|
||||
ts.Nsec = _UTIME_OMIT
|
||||
} else {
|
||||
ts = syscall.NsecToTimespec(t.UnixNano())
|
||||
// For dates before 1970, NsecToTimespec incorrectly returns negative
|
||||
// nanoseconds. Ticket: https://github.com/golang/go/issues/12777
|
||||
if ts.Nsec < 0 {
|
||||
ts.Nsec = 0
|
||||
}
|
||||
}
|
||||
return ts
|
||||
}
|
||||
|
||||
func (f *file) Utimens(a *time.Time, m *time.Time) fuse.Status {
|
||||
f.fdLock.RLock()
|
||||
defer f.fdLock.RUnlock()
|
||||
|
||||
ts := make([]syscall.Timespec, 2)
|
||||
ts[0] = utimeToTimespec(a)
|
||||
ts[1] = utimeToTimespec(m)
|
||||
|
||||
fn := fmt.Sprintf("/proc/self/fd/%d", f.fd.Fd())
|
||||
err := syscall.UtimesNano(fn, ts)
|
||||
if err != nil {
|
||||
tlog.Debug.Printf("UtimesNano on %q failed: %v", fn, err)
|
||||
}
|
||||
if err == syscall.ENOENT {
|
||||
// If /proc/self/fd/X did not exist, the actual error is that the file
|
||||
// descriptor was invalid.
|
||||
return fuse.EBADF
|
||||
}
|
||||
return fuse.ToStatus(err)
|
||||
return f.loopbackFile.Utimens(a, m)
|
||||
}
|
||||
|
@ -234,14 +234,11 @@ func (fs *FS) Utimens(path string, a *time.Time, m *time.Time, context *fuse.Con
|
||||
if fs.isFiltered(path) {
|
||||
return fuse.EPERM
|
||||
}
|
||||
cPath, err := fs.getBackingPath(path)
|
||||
cPath, err := fs.encryptPath(path)
|
||||
if err != nil {
|
||||
return fuse.ToStatus(err)
|
||||
}
|
||||
ts := make([]syscall.Timespec, 2)
|
||||
ts[0] = utimeToTimespec(a)
|
||||
ts[1] = utimeToTimespec(m)
|
||||
return fuse.ToStatus(syscall.UtimesNano(cPath, ts))
|
||||
return fs.FileSystem.Utimens(cPath, a, m, context)
|
||||
}
|
||||
|
||||
func (fs *FS) StatFs(path string) *fuse.StatfsOut {
|
||||
|
Loading…
x
Reference in New Issue
Block a user