Mostly updated flushfs.

geesefs-0-30-9
Aaron Jacobs 2015-03-24 16:17:10 +11:00
parent 9190eea002
commit 9a704e1cd1
1 changed files with 80 additions and 63 deletions

View File

@ -16,12 +16,12 @@ package flushfs
import ( import (
"fmt" "fmt"
"io"
"os" "os"
"sync" "sync"
"github.com/jacobsa/fuse" "github.com/jacobsa/fuse"
"github.com/jacobsa/fuse/fuseops" "github.com/jacobsa/fuse/fuseops"
"golang.org/x/net/context"
) )
// Create a file system whose sole contents are a file named "foo" and a // Create a file system whose sole contents are a file named "foo" and a
@ -86,45 +86,76 @@ func (fs *flushFS) barAttributes() fuseops.InodeAttributes {
} }
// LOCKS_REQUIRED(fs.mu) // LOCKS_REQUIRED(fs.mu)
func (fs *flushFS) ServeOps(c *fuse.Connection) func (fs *flushFS) ServeOps(c *fuse.Connection) {
for {
op, err := c.ReadOp()
if err == io.EOF {
break
}
if err != nil {
panic(err)
}
switch typed := op.(type) {
case *fuseops.InitOp:
fs.init(typed)
case *fuseops.LookUpInodeOp:
fs.lookUpInode(typed)
case *fuseops.GetInodeAttributesOp:
fs.getInodeAttributes(typed)
case *fuseops.OpenDirOp:
fs.openDir(typed)
case *fuseops.OpenFileOp:
fs.openFile(typed)
case *fuseops.ReadFileOp:
fs.readFile(typed)
default:
typed.Respond(fuse.ENOSYS)
}
}
}
//////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////
// Op methods // Op methods
//////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////
func (fs *flushFS) Init( func (fs *flushFS) init(op *fuseops.InitOp) {
ctx context.Context, var err error
req *fuse.InitRequest) ( defer func() { op.Respond(err) }()
resp *fuse.InitResponse, err error) {
resp = &fuse.InitResponse{}
return return
} }
func (fs *flushFS) LookUpInode( func (fs *flushFS) lookUpInode(op *fuseops.LookUpInodeOp) {
ctx context.Context, var err error
req *fuse.LookUpInodeRequest) ( defer func() { op.Respond(err) }()
resp *fuse.LookUpInodeResponse, err error) {
resp = &fuse.LookUpInodeResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
// Sanity check. // Sanity check.
if req.Parent != fuseops.RootInodeID { if op.Parent != fuseops.RootInodeID {
err = fuse.ENOENT err = fuse.ENOENT
return return
} }
// Set up the entry. // Set up the entry.
switch req.Name { switch op.Name {
case "foo": case "foo":
resp.Entry = fuse.ChildInodeEntry{ op.Entry = fuseops.ChildInodeEntry{
Child: fooID, Child: fooID,
Attributes: fs.fooAttributes(), Attributes: fs.fooAttributes(),
} }
case "bar": case "bar":
resp.Entry = fuse.ChildInodeEntry{ op.Entry = fuseops.ChildInodeEntry{
Child: barID, Child: barID,
Attributes: fs.barAttributes(), Attributes: fs.barAttributes(),
} }
@ -137,26 +168,24 @@ func (fs *flushFS) LookUpInode(
return return
} }
func (fs *flushFS) GetInodeAttributes( func (fs *flushFS) getInodeAttributes(op *fuseops.GetInodeAttributesOp) {
ctx context.Context, var err error
req *fuse.GetInodeAttributesRequest) ( defer func() { op.Respond(err) }()
resp *fuse.GetInodeAttributesResponse, err error) {
resp = &fuse.GetInodeAttributesResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
switch req.Inode { switch op.Inode {
case fuseops.RootInodeID: case fuseops.RootInodeID:
resp.Attributes = fs.rootAttributes() op.Attributes = fs.rootAttributes()
return return
case fooID: case fooID:
resp.Attributes = fs.fooAttributes() op.Attributes = fs.fooAttributes()
return return
case barID: case barID:
resp.Attributes = fs.barAttributes() op.Attributes = fs.barAttributes()
return return
default: default:
@ -165,17 +194,15 @@ func (fs *flushFS) GetInodeAttributes(
} }
} }
func (fs *flushFS) OpenFile( func (fs *flushFS) openFile(op *fuseops.OpenFileOp) {
ctx context.Context, var err error
req *fuse.OpenFileRequest) ( defer func() { op.Respond(err) }()
resp *fuse.OpenFileResponse, err error) {
resp = &fuse.OpenFileResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
// Sanity check. // Sanity check.
if req.Inode != fooID { if op.Inode != fooID {
err = fuse.ENOSYS err = fuse.ENOSYS
return return
} }
@ -183,59 +210,53 @@ func (fs *flushFS) OpenFile(
return return
} }
func (fs *flushFS) ReadFile( func (fs *flushFS) readFile(op *fuseops.ReadFileOp) {
ctx context.Context, var err error
req *fuse.ReadFileRequest) ( defer func() { op.Respond(err) }()
resp *fuse.ReadFileResponse, err error) {
resp = &fuse.ReadFileResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
// Ensure the offset is in range. // Ensure the offset is in range.
if req.Offset > int64(len(fs.fooContents)) { if op.Offset > int64(len(fs.fooContents)) {
return return
} }
// Read what we can. // Read what we can.
resp.Data = make([]byte, req.Size) op.Data = make([]byte, op.Size)
copy(resp.Data, fs.fooContents[req.Offset:]) copy(op.Data, fs.fooContents[op.Offset:])
return return
} }
func (fs *flushFS) WriteFile( func (fs *flushFS) writeFile(op *fuseops.WriteFileOp) {
ctx context.Context, var err error
req *fuse.WriteFileRequest) ( defer func() { op.Respond(err) }()
resp *fuse.WriteFileResponse, err error) {
resp = &fuse.WriteFileResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
// Ensure that the contents slice is long enough. // Ensure that the contents slice is long enough.
newLen := int(req.Offset) + len(req.Data) newLen := int(op.Offset) + len(op.Data)
if len(fs.fooContents) < newLen { if len(fs.fooContents) < newLen {
padding := make([]byte, newLen-len(fs.fooContents)) padding := make([]byte, newLen-len(fs.fooContents))
fs.fooContents = append(fs.fooContents, padding...) fs.fooContents = append(fs.fooContents, padding...)
} }
// Copy in the data. // Copy in the data.
n := copy(fs.fooContents[req.Offset:], req.Data) n := copy(fs.fooContents[op.Offset:], op.Data)
// Sanity check. // Sanity check.
if n != len(req.Data) { if n != len(op.Data) {
panic(fmt.Sprintf("Unexpected short copy: %v", n)) panic(fmt.Sprintf("Unexpected short copy: %v", n))
} }
return return
} }
func (fs *flushFS) SyncFile( func (fs *flushFS) syncFile(op *fuseops.SyncFileOp) {
ctx context.Context, var err error
req *fuse.SyncFileRequest) ( defer func() { op.Respond(err) }()
resp *fuse.SyncFileResponse, err error) {
resp = &fuse.SyncFileResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
@ -244,11 +265,9 @@ func (fs *flushFS) SyncFile(
return return
} }
func (fs *flushFS) FlushFile( func (fs *flushFS) flushFile(op *fuseops.FlushFileOp) {
ctx context.Context, var err error
req *fuse.FlushFileRequest) ( defer func() { op.Respond(err) }()
resp *fuse.FlushFileResponse, err error) {
resp = &fuse.FlushFileResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
@ -257,17 +276,15 @@ func (fs *flushFS) FlushFile(
return return
} }
func (fs *flushFS) OpenDir( func (fs *flushFS) openDir(op *fuseops.OpenDirOp) {
ctx context.Context, var err error
req *fuse.OpenDirRequest) ( defer func() { op.Respond(err) }()
resp *fuse.OpenDirResponse, err error) {
resp = &fuse.OpenDirResponse{}
fs.mu.Lock() fs.mu.Lock()
defer fs.mu.Unlock() defer fs.mu.Unlock()
// Sanity check. // Sanity check.
if req.Inode != barID { if op.Inode != barID {
err = fuse.ENOSYS err = fuse.ENOSYS
return return
} }