Commit e90f5d22 authored by Han-Wen Nienhuys's avatar Han-Wen Nienhuys

nodefs: issue GetAttr on file if we have open files

Fix and test Nlink==0 test (issue #137)

Do not overwrite Attr.Ino, otherwise recycled inodes risk giving tools
like RSync the impression there are more hardlinks than there are.
parent 6950891f
...@@ -44,6 +44,7 @@ type rawBridge struct { ...@@ -44,6 +44,7 @@ type rawBridge struct {
} }
// newInode creates creates new inode pointing to node. // newInode creates creates new inode pointing to node.
// XXX - should store the Ino number we expose in GetAttr too ?
func (b *rawBridge) newInode(node Node, mode uint32, id FileID, persistent bool) *Inode { func (b *rawBridge) newInode(node Node, mode uint32, id FileID, persistent bool) *Inode {
b.mu.Lock() b.mu.Lock()
defer b.mu.Unlock() defer b.mu.Unlock()
...@@ -213,10 +214,6 @@ func (b *rawBridge) addNewChild(parent *Inode, name string, child *Inode, file F ...@@ -213,10 +214,6 @@ func (b *rawBridge) addNewChild(parent *Inode, name string, child *Inode, file F
} }
out.NodeId = child.nodeID out.NodeId = child.nodeID
// NOSUBMIT - or should let FS expose Attr.Ino? This makes
// testing semantics hard though, because os.Lstat doesn't
// reflect the FUSE FS
out.Attr.Ino = child.nodeID
out.Generation = b.nodes[out.NodeId].generation out.Generation = b.nodes[out.NodeId].generation
b.mu.Unlock() b.mu.Unlock()
unlockNodes(parent, child) unlockNodes(parent, child)
...@@ -300,16 +297,8 @@ func (b *rawBridge) GetAttr(input *fuse.GetAttrIn, out *fuse.AttrOut) fuse.Statu ...@@ -300,16 +297,8 @@ func (b *rawBridge) GetAttr(input *fuse.GetAttrIn, out *fuse.AttrOut) fuse.Statu
f = nil f = nil
} }
code = n.node.GetAttr(context.TODO(), f, out) code := n.node.GetAttr(context.TODO(), f, out)
if out.Nlink == 0 {
// With Nlink == 0, newer kernels will refuse link
// operations.
out.Nlink = 1
}
b.setAttrTimeout(out) b.setAttrTimeout(out)
out.Ino = input.NodeId
return code return code
} }
......
...@@ -14,11 +14,11 @@ import ( ...@@ -14,11 +14,11 @@ import (
"github.com/hanwen/go-fuse/fuse" "github.com/hanwen/go-fuse/fuse"
) )
// LoopbackFile delegates all operations back to an underlying os.File. func newLoopbackFile(f *os.File) *loopbackFile {
func NewLoopbackFile(f *os.File) File {
return &loopbackFile{File: f} return &loopbackFile{File: f}
} }
// loopbackFile delegates all operations back to an underlying os.File.
type loopbackFile struct { type loopbackFile struct {
File *os.File File *os.File
......
...@@ -21,14 +21,18 @@ type parentData struct { ...@@ -21,14 +21,18 @@ type parentData struct {
} }
// FileID provides a identifier for file objects defined by FUSE // FileID provides a identifier for file objects defined by FUSE
// filesystems. // filesystems. The identifier is divided into a (Device, Inode) pair,
// so files in underlying file systems can easily be represented, but
// FUSE filesystems are free to use any other information as 128-bit
// key.
// //
// XXX name: PersistentID ? NodeID ? // XXX name: PersistentID ? NodeID ?
type FileID struct { type FileID struct {
Dev uint64 // XXX Rdev? Dev uint64
Ino uint64 Ino uint64
} }
// Zero returns if the FileID is zeroed out
func (i *FileID) Zero() bool { func (i *FileID) Zero() bool {
return i.Dev == 0 && i.Ino == 0 return i.Dev == 0 && i.Ino == 0
} }
...@@ -39,10 +43,12 @@ func (i *FileID) Zero() bool { ...@@ -39,10 +43,12 @@ func (i *FileID) Zero() bool {
// "persistent" Inodes. // "persistent" Inodes.
type Inode struct { type Inode struct {
// The filetype bits from the mode. // The filetype bits from the mode.
mode uint32 mode uint32
opaqueID FileID opaqueID FileID
node Node
bridge *rawBridge node Node
bridge *rawBridge
// Following data is mutable. // Following data is mutable.
...@@ -51,10 +57,8 @@ type Inode struct { ...@@ -51,10 +57,8 @@ type Inode struct {
// lockNodes/unlockNodes // lockNodes/unlockNodes
mu sync.Mutex mu sync.Mutex
// ID of the inode; 0 if inode was forgotten. Forgotten // ID of the inode for talking to the kernel, 0 if the kernel
// inodes could be persistent, not yet are unlinked from // does not know this inode.
// parent and children, but could be still not yet removed
// from bridge.nodes .
nodeID uint64 nodeID uint64
// persistent indicates that this node should not be removed // persistent indicates that this node should not be removed
...@@ -127,7 +131,9 @@ func lockNodes(ns ...*Inode) { ...@@ -127,7 +131,9 @@ func lockNodes(ns ...*Inode) {
// lockNode2 locks a and b in order consistent with lockNodes. // lockNode2 locks a and b in order consistent with lockNodes.
func lockNode2(a, b *Inode) { func lockNode2(a, b *Inode) {
if nodeLess(a, b) { if a == b {
a.mu.Lock()
} else if nodeLess(a, b) {
a.mu.Lock() a.mu.Lock()
b.mu.Lock() b.mu.Lock()
} else { } else {
...@@ -138,8 +144,12 @@ func lockNode2(a, b *Inode) { ...@@ -138,8 +144,12 @@ func lockNode2(a, b *Inode) {
// unlockNode2 unlocks a and b // unlockNode2 unlocks a and b
func unlockNode2(a, b *Inode) { func unlockNode2(a, b *Inode) {
a.mu.Unlock() if a == b {
b.mu.Unlock() a.mu.Unlock()
} else {
a.mu.Unlock()
b.mu.Unlock()
}
} }
// unlockNodes releases locks taken by lockNodes. // unlockNodes releases locks taken by lockNodes.
......
...@@ -8,6 +8,7 @@ import ( ...@@ -8,6 +8,7 @@ import (
"context" "context"
"os" "os"
"path/filepath" "path/filepath"
"sync"
"syscall" "syscall"
"github.com/hanwen/go-fuse/fuse" "github.com/hanwen/go-fuse/fuse"
...@@ -19,6 +20,13 @@ type loopbackRoot struct { ...@@ -19,6 +20,13 @@ type loopbackRoot struct {
root string root string
} }
func (n *loopbackRoot) newLoopbackNode() *loopbackNode {
return &loopbackNode{
rootNode: n,
openFiles: map[*loopbackFile]struct{}{},
}
}
func (n *loopbackRoot) GetAttr(ctx context.Context, f File, out *fuse.AttrOut) fuse.Status { func (n *loopbackRoot) GetAttr(ctx context.Context, f File, out *fuse.AttrOut) fuse.Status {
var err error = nil var err error = nil
st := syscall.Stat_t{} st := syscall.Stat_t{}
...@@ -34,6 +42,19 @@ type loopbackNode struct { ...@@ -34,6 +42,19 @@ type loopbackNode struct {
DefaultNode DefaultNode
rootNode *loopbackRoot rootNode *loopbackRoot
mu sync.Mutex
openFiles map[*loopbackFile]struct{}
}
func (n *loopbackNode) Release(ctx context.Context, f File) {
if f != nil {
n.mu.Lock()
defer n.mu.Unlock()
lf := f.(*loopbackFile)
delete(n.openFiles, lf)
f.Release(ctx)
}
} }
func (n *loopbackNode) path() string { func (n *loopbackNode) path() string {
...@@ -57,7 +78,7 @@ func (n *loopbackNode) Lookup(ctx context.Context, name string, out *fuse.EntryO ...@@ -57,7 +78,7 @@ func (n *loopbackNode) Lookup(ctx context.Context, name string, out *fuse.EntryO
Ino: out.Attr.Ino, Ino: out.Attr.Ino,
} }
node := &loopbackNode{rootNode: n.rootNode} node := n.rootNode.newLoopbackNode()
ch := n.inode().NewInode(node, out.Attr.Mode, opaque) ch := n.inode().NewInode(node, out.Attr.Mode, opaque)
return ch, fuse.OK return ch, fuse.OK
} }
...@@ -76,7 +97,7 @@ func (n *loopbackNode) Mknod(ctx context.Context, name string, mode, rdev uint32 ...@@ -76,7 +97,7 @@ func (n *loopbackNode) Mknod(ctx context.Context, name string, mode, rdev uint32
out.Attr.FromStat(&st) out.Attr.FromStat(&st)
node := &loopbackNode{rootNode: n.rootNode} node := n.rootNode.newLoopbackNode()
opaque := FileID{ opaque := FileID{
Dev: uint64(out.Attr.Rdev), Dev: uint64(out.Attr.Rdev),
Ino: out.Attr.Ino, Ino: out.Attr.Ino,
...@@ -101,7 +122,7 @@ func (n *loopbackNode) Mkdir(ctx context.Context, name string, mode uint32, out ...@@ -101,7 +122,7 @@ func (n *loopbackNode) Mkdir(ctx context.Context, name string, mode uint32, out
out.Attr.FromStat(&st) out.Attr.FromStat(&st)
node := &loopbackNode{rootNode: n.rootNode} node := n.rootNode.newLoopbackNode()
opaque := FileID{ opaque := FileID{
Dev: uint64(out.Attr.Rdev), Dev: uint64(out.Attr.Rdev),
Ino: out.Attr.Ino, Ino: out.Attr.Ino,
...@@ -156,14 +177,18 @@ func (n *loopbackNode) Create(ctx context.Context, name string, flags uint32, mo ...@@ -156,14 +177,18 @@ func (n *loopbackNode) Create(ctx context.Context, name string, flags uint32, mo
return nil, nil, 0, fuse.ToStatus(err) return nil, nil, 0, fuse.ToStatus(err)
} }
node := &loopbackNode{rootNode: n.rootNode} node := n.rootNode.newLoopbackNode()
opaque := FileID{ opaque := FileID{
Dev: st.Rdev, Dev: st.Rdev,
Ino: st.Ino, Ino: st.Ino,
} }
ch := n.inode().NewInode(node, st.Mode, opaque) ch := n.inode().NewInode(node, st.Mode, opaque)
return ch, NewLoopbackFile(f), 0, fuse.OK lf := newLoopbackFile(f)
n.mu.Lock()
defer n.mu.Unlock()
n.openFiles[lf] = struct{}{}
return ch, lf, 0, fuse.OK
} }
func (n *loopbackNode) Open(ctx context.Context, flags uint32) (fh File, fuseFlags uint32, code fuse.Status) { func (n *loopbackNode) Open(ctx context.Context, flags uint32) (fh File, fuseFlags uint32, code fuse.Status) {
...@@ -172,12 +197,32 @@ func (n *loopbackNode) Open(ctx context.Context, flags uint32) (fh File, fuseFla ...@@ -172,12 +197,32 @@ func (n *loopbackNode) Open(ctx context.Context, flags uint32) (fh File, fuseFla
if err != nil { if err != nil {
return nil, 0, fuse.ToStatus(err) return nil, 0, fuse.ToStatus(err)
} }
return NewLoopbackFile(f), 0, fuse.OK n.mu.Lock()
defer n.mu.Unlock()
lf := newLoopbackFile(f)
n.openFiles[lf] = struct{}{}
return lf, 0, fuse.OK
}
func (n *loopbackNode) fGetAttr(ctx context.Context, out *fuse.AttrOut) (fuse.Status, bool) {
n.mu.Lock()
defer n.mu.Unlock()
for f := range n.openFiles {
if f != nil {
return f.GetAttr(ctx, out), true
}
}
return fuse.EBADF, false
} }
func (n *loopbackNode) GetAttr(ctx context.Context, f File, out *fuse.AttrOut) fuse.Status { func (n *loopbackNode) GetAttr(ctx context.Context, f File, out *fuse.AttrOut) fuse.Status {
if f != nil { if f != nil {
// this never happens because the kernel never sends FH on getattr.
return f.GetAttr(ctx, out) return f.GetAttr(ctx, out)
}
if code, ok := n.fGetAttr(ctx, out); ok {
return code
} }
p := n.path() p := n.path()
...@@ -197,6 +242,6 @@ func NewLoopback(root string) Node { ...@@ -197,6 +242,6 @@ func NewLoopback(root string) Node {
root: root, root: root,
} }
n.rootNode = n n.rootNode = n
n.openFiles = map[*loopbackFile]struct{}{}
return n return n
} }
...@@ -7,9 +7,11 @@ package nodefs ...@@ -7,9 +7,11 @@ package nodefs
import ( import (
"bytes" "bytes"
"io/ioutil" "io/ioutil"
"log"
"os" "os"
"path/filepath" "path/filepath"
"runtime" "runtime"
"sync"
"syscall" "syscall"
"testing" "testing"
"time" "time"
...@@ -20,6 +22,8 @@ import ( ...@@ -20,6 +22,8 @@ import (
"github.com/hanwen/go-fuse/internal/testutil" "github.com/hanwen/go-fuse/internal/testutil"
) )
var _ = log.Println
type testCase struct { type testCase struct {
*testing.T *testing.T
...@@ -335,3 +339,71 @@ func TestRenameNoOverwrite(t *testing.T) { ...@@ -335,3 +339,71 @@ func TestRenameNoOverwrite(t *testing.T) {
t.Errorf("got %v (%T) want %v (%T)", err, err, syscall.EINVAL, syscall.EINVAL) t.Errorf("got %v (%T) want %v (%T)", err, err, syscall.EINVAL, syscall.EINVAL)
} }
} }
func TestNlinkZero(t *testing.T) {
// xfstest generic/035.
tc := newTestCase(t)
defer tc.Clean()
src := tc.mntDir + "/src"
dst := tc.mntDir + "/dst"
if err := ioutil.WriteFile(src, []byte("source"), 0644); err != nil {
t.Fatalf("WriteFile: %v", err)
}
if err := ioutil.WriteFile(dst, []byte("dst"), 0644); err != nil {
t.Fatalf("WriteFile: %v", err)
}
f, err := os.Open(dst)
if err != nil {
t.Fatalf("Open: %v", err)
}
defer f.Close()
var st syscall.Stat_t
if err := syscall.Fstat(int(f.Fd()), &st); err != nil {
t.Errorf("Fstat before: %v", err)
} else if st.Nlink != 1 {
t.Errorf("Nlink of file: got %d, want 1", st.Nlink)
}
if err := os.Rename(src, dst); err != nil {
t.Fatalf("Rename: %v", err)
}
if err := syscall.Fstat(int(f.Fd()), &st); err != nil {
t.Errorf("Fstat after: %v", err)
} else if st.Nlink != 0 {
t.Errorf("Nlink of overwritten file: got %d, want 0", st.Nlink)
}
}
func TestParallelFileOpen(t *testing.T) {
tc := newTestCase(t)
defer tc.Clean()
fn := tc.mntDir + "/file"
if err := ioutil.WriteFile(fn, []byte("content"), 0644); err != nil {
t.Fatalf("WriteFile: %v", err)
}
var wg sync.WaitGroup
one := func(b byte) {
f, err := os.OpenFile(fn, os.O_RDWR, 0644)
if err != nil {
t.Fatalf("OpenFile: %v", err)
}
var buf [10]byte
f.Read(buf[:])
buf[0] = b
f.WriteAt(buf[0:1], 2)
f.Close()
wg.Done()
}
for i := 0; i < 10; i++ {
wg.Add(1)
go one(byte(i))
}
wg.Wait()
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment