forked from TrueCloudLab/rclone
139 lines
3.2 KiB
Go
139 lines
3.2 KiB
Go
// Copyright 2016 the Go-FUSE Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package fuse
|
|
|
|
import (
|
|
"bytes"
|
|
"os"
|
|
"syscall"
|
|
"unsafe"
|
|
)
|
|
|
|
// TODO - move these into Go's syscall package.
|
|
|
|
func sys_writev(fd int, iovecs *syscall.Iovec, cnt int) (n int, err error) {
|
|
n1, _, e1 := syscall.Syscall(
|
|
syscall.SYS_WRITEV,
|
|
uintptr(fd), uintptr(unsafe.Pointer(iovecs)), uintptr(cnt))
|
|
n = int(n1)
|
|
if e1 != 0 {
|
|
err = syscall.Errno(e1)
|
|
}
|
|
return
|
|
}
|
|
|
|
func writev(fd int, packet [][]byte) (n int, err error) {
|
|
iovecs := make([]syscall.Iovec, 0, len(packet))
|
|
|
|
for _, v := range packet {
|
|
if len(v) == 0 {
|
|
continue
|
|
}
|
|
vec := syscall.Iovec{
|
|
Base: &v[0],
|
|
}
|
|
vec.SetLen(len(v))
|
|
iovecs = append(iovecs, vec)
|
|
}
|
|
|
|
sysErr := handleEINTR(func() error {
|
|
var err error
|
|
n, err = sys_writev(fd, &iovecs[0], len(iovecs))
|
|
return err
|
|
})
|
|
if sysErr != nil {
|
|
err = os.NewSyscallError("writev", sysErr)
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
func getxattr(path string, attr string, dest []byte) (sz int, errno int) {
|
|
pathBs := syscall.StringBytePtr(path)
|
|
attrBs := syscall.StringBytePtr(attr)
|
|
size, _, errNo := syscall.Syscall6(
|
|
syscall.SYS_GETXATTR,
|
|
uintptr(unsafe.Pointer(pathBs)),
|
|
uintptr(unsafe.Pointer(attrBs)),
|
|
uintptr(unsafe.Pointer(&dest[0])),
|
|
uintptr(len(dest)),
|
|
0, 0)
|
|
return int(size), int(errNo)
|
|
}
|
|
|
|
func GetXAttr(path string, attr string, dest []byte) (value []byte, errno int) {
|
|
sz, errno := getxattr(path, attr, dest)
|
|
|
|
for sz > cap(dest) && errno == 0 {
|
|
dest = make([]byte, sz)
|
|
sz, errno = getxattr(path, attr, dest)
|
|
}
|
|
|
|
if errno != 0 {
|
|
return nil, errno
|
|
}
|
|
|
|
return dest[:sz], errno
|
|
}
|
|
|
|
func listxattr(path string, dest []byte) (sz int, errno int) {
|
|
pathbs := syscall.StringBytePtr(path)
|
|
var destPointer unsafe.Pointer
|
|
if len(dest) > 0 {
|
|
destPointer = unsafe.Pointer(&dest[0])
|
|
}
|
|
size, _, errNo := syscall.Syscall(
|
|
syscall.SYS_LISTXATTR,
|
|
uintptr(unsafe.Pointer(pathbs)),
|
|
uintptr(destPointer),
|
|
uintptr(len(dest)))
|
|
|
|
return int(size), int(errNo)
|
|
}
|
|
|
|
func ListXAttr(path string) (attributes []string, errno int) {
|
|
dest := make([]byte, 0)
|
|
sz, errno := listxattr(path, dest)
|
|
if errno != 0 {
|
|
return nil, errno
|
|
}
|
|
|
|
for sz > cap(dest) && errno == 0 {
|
|
dest = make([]byte, sz)
|
|
sz, errno = listxattr(path, dest)
|
|
}
|
|
|
|
// -1 to drop the final empty slice.
|
|
dest = dest[:sz-1]
|
|
attributesBytes := bytes.Split(dest, []byte{0})
|
|
attributes = make([]string, len(attributesBytes))
|
|
for i, v := range attributesBytes {
|
|
attributes[i] = string(v)
|
|
}
|
|
return attributes, errno
|
|
}
|
|
|
|
func Setxattr(path string, attr string, data []byte, flags int) (errno int) {
|
|
pathbs := syscall.StringBytePtr(path)
|
|
attrbs := syscall.StringBytePtr(attr)
|
|
_, _, errNo := syscall.Syscall6(
|
|
syscall.SYS_SETXATTR,
|
|
uintptr(unsafe.Pointer(pathbs)),
|
|
uintptr(unsafe.Pointer(attrbs)),
|
|
uintptr(unsafe.Pointer(&data[0])),
|
|
uintptr(len(data)),
|
|
uintptr(flags), 0)
|
|
|
|
return int(errNo)
|
|
}
|
|
|
|
func Removexattr(path string, attr string) (errno int) {
|
|
pathbs := syscall.StringBytePtr(path)
|
|
attrbs := syscall.StringBytePtr(attr)
|
|
_, _, errNo := syscall.Syscall(
|
|
syscall.SYS_REMOVEXATTR,
|
|
uintptr(unsafe.Pointer(pathbs)),
|
|
uintptr(unsafe.Pointer(attrbs)), 0)
|
|
return int(errNo)
|
|
}
|