Commit d20c4665 authored by Ka-Hing Cheung's avatar Ka-Hing Cheung

GetXattr/ListXattr implementation

parent 8aade5c7
...@@ -434,6 +434,10 @@ func (c *Connection) shouldLogError( ...@@ -434,6 +434,10 @@ func (c *Connection) shouldLogError(
return false return false
} }
case *fuseops.GetXattrOp:
if err == syscall.ENODATA || err == syscall.ERANGE {
return false
}
case *unknownOp: case *unknownOp:
// Don't bother the user with methods we intentionally don't support. // Don't bother the user with methods we intentionally don't support.
if err == syscall.ENOSYS { if err == syscall.ENOSYS {
...@@ -489,7 +493,7 @@ func (c *Connection) Reply(ctx context.Context, opErr error) { ...@@ -489,7 +493,7 @@ func (c *Connection) Reply(ctx context.Context, opErr error) {
if !noResponse { if !noResponse {
err := c.writeMessage(outMsg.Bytes()) err := c.writeMessage(outMsg.Bytes())
if err != nil && c.errorLogger != nil { if err != nil && c.errorLogger != nil {
c.errorLogger.Printf("writeMessage: %v", err) c.errorLogger.Printf("writeMessage: %v %v", err, outMsg.Bytes())
} }
} }
} }
......
...@@ -433,6 +433,61 @@ func convertInMessage( ...@@ -433,6 +433,61 @@ func convertInMessage(
Name: string(buf[:n-1]), Name: string(buf[:n-1]),
} }
case fusekernel.OpGetxattr:
type input fusekernel.GetxattrIn
in := (*input)(inMsg.Consume(unsafe.Sizeof(input{})))
if in == nil {
err = errors.New("Corrupt OpGetxattr")
return
}
name := inMsg.ConsumeBytes(inMsg.Len())
i := bytes.IndexByte(name, '\x00')
if i < 0 {
err = errors.New("Corrupt OpGetxattr")
return
}
name = name[:i]
to := &fuseops.GetXattrOp{
Inode: fuseops.InodeID(inMsg.Header().Nodeid),
Name: string(name),
}
o = to
readSize := int(in.Size)
p := outMsg.GrowNoZero(readSize)
if p == nil {
err = fmt.Errorf("Can't grow for %d-byte read", readSize)
return
}
sh := (*reflect.SliceHeader)(unsafe.Pointer(&to.Dst))
sh.Data = uintptr(p)
sh.Len = readSize
sh.Cap = readSize
case fusekernel.OpListxattr:
type input fusekernel.ListxattrIn
in := (*input)(inMsg.Consume(unsafe.Sizeof(input{})))
if in == nil {
err = errors.New("Corrupt OpListxattr")
return
}
o = &fuseops.ListXattrOp{
Inode: fuseops.InodeID(inMsg.Header().Nodeid),
}
readSize := int(in.Size)
if readSize != 0 {
p := outMsg.GrowNoZero(readSize)
if p == nil {
err = fmt.Errorf("Can't grow for %d-byte read", readSize)
return
}
}
default: default:
o = &unknownOp{ o = &unknownOp{
OpCode: inMsg.Header().Opcode, OpCode: inMsg.Header().Opcode,
...@@ -472,17 +527,32 @@ func (c *Connection) kernelResponse( ...@@ -472,17 +527,32 @@ func (c *Connection) kernelResponse(
// If the user returned the error, fill in the error field of the outgoing // If the user returned the error, fill in the error field of the outgoing
// message header. // message header.
if opErr != nil { if opErr != nil {
m.OutHeader().Error = -int32(syscall.EIO) handled := false
if errno, ok := opErr.(syscall.Errno); ok {
m.OutHeader().Error = -int32(errno) if opErr == syscall.ERANGE {
switch o := op.(type) {
case *fuseops.GetXattrOp:
writeXattrSize(m, uint32(o.BytesRead))
handled = true
case *fuseops.ListXattrOp:
writeXattrSize(m, uint32(o.BytesRead))
handled = true
}
}
if !handled {
m.OutHeader().Error = -int32(syscall.EIO)
if errno, ok := opErr.(syscall.Errno); ok {
m.OutHeader().Error = -int32(errno)
}
// Special case: for some types, convertInMessage grew the message in order
// to obtain a destination buffer. Make sure that we shrink back to just
// the header, because on OS X the kernel otherwise returns EINVAL when we
// attempt to write an error response with a length that extends beyond the
// header.
m.ShrinkTo(buffer.OutMessageHeaderSize)
} }
// Special case: for some types, convertInMessage grew the message in order
// to obtain a destination buffer. Make sure that we shrink back to just
// the header, because on OS X the kernel otherwise returns EINVAL when we
// attempt to write an error response with a length that extends beyond the
// header.
m.ShrinkTo(buffer.OutMessageHeaderSize)
} }
// Otherwise, fill in the rest of the response. // Otherwise, fill in the rest of the response.
...@@ -639,6 +709,23 @@ func (c *Connection) kernelResponseForOp( ...@@ -639,6 +709,23 @@ func (c *Connection) kernelResponseForOp(
case *fuseops.RemoveXattrOp: case *fuseops.RemoveXattrOp:
// Empty response // Empty response
case *fuseops.GetXattrOp:
// convertInMessage already set up the destination buffer to be at the end
// of the out message. We need only shrink to the right size based on how
// much the user read.
if o.BytesRead == 0 {
writeXattrSize(m, uint32(o.BytesRead))
} else {
m.ShrinkTo(buffer.OutMessageHeaderSize + o.BytesRead)
}
case *fuseops.ListXattrOp:
if o.BytesRead == 0 {
writeXattrSize(m, uint32(o.BytesRead))
} else {
m.ShrinkTo(buffer.OutMessageHeaderSize + o.BytesRead)
}
case *initOp: case *initOp:
out := (*fusekernel.InitOut)(m.Grow(int(unsafe.Sizeof(fusekernel.InitOut{})))) out := (*fusekernel.InitOut)(m.Grow(int(unsafe.Sizeof(fusekernel.InitOut{}))))
...@@ -760,3 +847,8 @@ func convertFileMode(unixMode uint32) os.FileMode { ...@@ -760,3 +847,8 @@ func convertFileMode(unixMode uint32) os.FileMode {
} }
return mode return mode
} }
func writeXattrSize(m *buffer.OutMessage, size uint32) {
out := (*fusekernel.GetxattrOut)(m.Grow(int(unsafe.Sizeof(fusekernel.GetxattrOut{}))))
out.Size = size
}
...@@ -89,6 +89,12 @@ func describeRequest(op interface{}) (s string) { ...@@ -89,6 +89,12 @@ func describeRequest(op interface{}) (s string) {
addComponent("handle %d", typed.Handle) addComponent("handle %d", typed.Handle)
addComponent("offset %d", typed.Offset) addComponent("offset %d", typed.Offset)
addComponent("%d bytes", len(typed.Data)) addComponent("%d bytes", len(typed.Data))
case *fuseops.RemoveXattrOp:
addComponent("name %s", typed.Name)
case *fuseops.GetXattrOp:
addComponent("name %s", typed.Name)
} }
// Use just the name if there is no extra info. // Use just the name if there is no extra info.
......
...@@ -780,3 +780,38 @@ type RemoveXattrOp struct { ...@@ -780,3 +780,38 @@ type RemoveXattrOp struct {
// The name of the extended attribute // The name of the extended attribute
Name string Name string
} }
// Get an extended attribute
type GetXattrOp struct {
// The inode that we are reading
Inode InodeID
// The name of the extended attribute
Name string
// The destination buffer. If the size is too small for the
// value, the ERANGE error should be sent.
Dst []byte
// Set by the file system: the number of bytes read into Dst, or
// the number of bytes that would have been read into Dst if Dst was
// big enough
BytesRead int
}
type ListXattrOp struct {
// The inode that we are reading
Inode InodeID
// The destination buffer. If the size is too small for the
// value, the ERANGE error should be sent.
//
// The output data should consist of a sequence of NUL-terminated strings,
// one for each xattr
Dst []byte
// Set by the file system: the number of bytes read into Dst, or
// the number of bytes that would have been read into Dst if Dst was
// big enough
BytesRead int
}
...@@ -58,6 +58,8 @@ type FileSystem interface { ...@@ -58,6 +58,8 @@ type FileSystem interface {
ReleaseFileHandle(context.Context, *fuseops.ReleaseFileHandleOp) error ReleaseFileHandle(context.Context, *fuseops.ReleaseFileHandleOp) error
ReadSymlink(context.Context, *fuseops.ReadSymlinkOp) error ReadSymlink(context.Context, *fuseops.ReadSymlinkOp) error
RemoveXattr(context.Context, *fuseops.RemoveXattrOp) error RemoveXattr(context.Context, *fuseops.RemoveXattrOp) error
GetXattr(context.Context, *fuseops.GetXattrOp) error
ListXattr(context.Context, *fuseops.ListXattrOp) error
// Regard all inodes (including the root inode) as having their lookup counts // Regard all inodes (including the root inode) as having their lookup counts
// decremented to zero, and clean up any resources associated with the file // decremented to zero, and clean up any resources associated with the file
...@@ -190,6 +192,12 @@ func (s *fileSystemServer) handleOp( ...@@ -190,6 +192,12 @@ func (s *fileSystemServer) handleOp(
case *fuseops.RemoveXattrOp: case *fuseops.RemoveXattrOp:
err = s.fs.RemoveXattr(ctx, typed) err = s.fs.RemoveXattr(ctx, typed)
case *fuseops.GetXattrOp:
err = s.fs.GetXattr(ctx, typed)
case *fuseops.ListXattrOp:
err = s.fs.ListXattr(ctx, typed)
} }
c.Reply(ctx, err) c.Reply(ctx, err)
......
...@@ -190,5 +190,19 @@ func (fs *NotImplementedFileSystem) RemoveXattr( ...@@ -190,5 +190,19 @@ func (fs *NotImplementedFileSystem) RemoveXattr(
return return
} }
func (fs *NotImplementedFileSystem) GetXattr(
ctx context.Context,
op *fuseops.GetXattrOp) (err error) {
err = fuse.ENOSYS
return
}
func (fs *NotImplementedFileSystem) ListXattr(
ctx context.Context,
op *fuseops.ListXattrOp) (err error) {
err = fuse.ENOSYS
return
}
func (fs *NotImplementedFileSystem) Destroy() { func (fs *NotImplementedFileSystem) Destroy() {
} }
...@@ -660,6 +660,11 @@ type GetxattrOut struct { ...@@ -660,6 +660,11 @@ type GetxattrOut struct {
Padding uint32 Padding uint32
} }
type ListxattrIn struct {
Size uint32
Padding uint32
}
type LkIn struct { type LkIn struct {
Fh uint64 Fh uint64
Owner uint64 Owner uint64
......
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