197 lines
5.3 KiB
Go
197 lines
5.3 KiB
Go
// +build freebsd netbsd
|
|
|
|
package xattr
|
|
|
|
import (
|
|
"os"
|
|
"syscall"
|
|
"unsafe"
|
|
)
|
|
|
|
const (
|
|
EXTATTR_NAMESPACE_USER = 1
|
|
|
|
// ENOATTR is not exported by the syscall package on Linux, because it is
|
|
// an alias for ENODATA. We export it here so it is available on all
|
|
// our supported platforms.
|
|
ENOATTR = syscall.ENOATTR
|
|
)
|
|
|
|
func getxattr(path string, name string, data []byte) (int, error) {
|
|
return sysGet(syscall.SYS_EXTATTR_GET_FILE, path, name, data)
|
|
}
|
|
|
|
func lgetxattr(path string, name string, data []byte) (int, error) {
|
|
return sysGet(syscall.SYS_EXTATTR_GET_LINK, path, name, data)
|
|
}
|
|
|
|
func fgetxattr(f *os.File, name string, data []byte) (int, error) {
|
|
return getxattr(f.Name(), name, data)
|
|
}
|
|
|
|
// sysGet is called by getxattr and lgetxattr with the appropriate syscall
|
|
// number. This works because syscalls have the same signature and return
|
|
// values.
|
|
func sysGet(syscallNum uintptr, path string, name string, data []byte) (int, error) {
|
|
ptr, nbytes := bytePtrFromSlice(data)
|
|
/*
|
|
ssize_t extattr_get_file(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname,
|
|
void *data,
|
|
size_t nbytes);
|
|
|
|
ssize_t extattr_get_link(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname,
|
|
void *data,
|
|
size_t nbytes);
|
|
*/
|
|
r0, _, err := syscall.Syscall6(syscallNum, uintptr(unsafe.Pointer(syscall.StringBytePtr(path))),
|
|
EXTATTR_NAMESPACE_USER, uintptr(unsafe.Pointer(syscall.StringBytePtr(name))),
|
|
uintptr(unsafe.Pointer(ptr)), uintptr(nbytes), 0)
|
|
if err != syscall.Errno(0) {
|
|
return int(r0), err
|
|
}
|
|
return int(r0), nil
|
|
}
|
|
|
|
func setxattr(path string, name string, data []byte, flags int) error {
|
|
return sysSet(syscall.SYS_EXTATTR_SET_FILE, path, name, data)
|
|
}
|
|
|
|
func lsetxattr(path string, name string, data []byte, flags int) error {
|
|
return sysSet(syscall.SYS_EXTATTR_SET_LINK, path, name, data)
|
|
}
|
|
|
|
func fsetxattr(f *os.File, name string, data []byte, flags int) error {
|
|
return setxattr(f.Name(), name, data, flags)
|
|
}
|
|
|
|
// sysSet is called by setxattr and lsetxattr with the appropriate syscall
|
|
// number. This works because syscalls have the same signature and return
|
|
// values.
|
|
func sysSet(syscallNum uintptr, path string, name string, data []byte) error {
|
|
ptr, nbytes := bytePtrFromSlice(data)
|
|
/*
|
|
ssize_t extattr_set_file(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname,
|
|
const void *data,
|
|
size_t nbytes
|
|
);
|
|
|
|
ssize_t extattr_set_link(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname,
|
|
const void *data,
|
|
size_t nbytes
|
|
);
|
|
*/
|
|
r0, _, err := syscall.Syscall6(syscallNum, uintptr(unsafe.Pointer(syscall.StringBytePtr(path))),
|
|
EXTATTR_NAMESPACE_USER, uintptr(unsafe.Pointer(syscall.StringBytePtr(name))),
|
|
uintptr(unsafe.Pointer(ptr)), uintptr(nbytes), 0)
|
|
if err != syscall.Errno(0) {
|
|
return err
|
|
}
|
|
if int(r0) != nbytes {
|
|
return syscall.E2BIG
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func removexattr(path string, name string) error {
|
|
return sysRemove(syscall.SYS_EXTATTR_DELETE_FILE, path, name)
|
|
}
|
|
|
|
func lremovexattr(path string, name string) error {
|
|
return sysRemove(syscall.SYS_EXTATTR_DELETE_LINK, path, name)
|
|
}
|
|
|
|
func fremovexattr(f *os.File, name string) error {
|
|
return removexattr(f.Name(), name)
|
|
}
|
|
|
|
// sysSet is called by removexattr and lremovexattr with the appropriate syscall
|
|
// number. This works because syscalls have the same signature and return
|
|
// values.
|
|
func sysRemove(syscallNum uintptr, path string, name string) error {
|
|
/*
|
|
int extattr_delete_file(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname
|
|
);
|
|
|
|
int extattr_delete_link(
|
|
const char *path,
|
|
int attrnamespace,
|
|
const char *attrname
|
|
);
|
|
*/
|
|
_, _, err := syscall.Syscall(syscallNum, uintptr(unsafe.Pointer(syscall.StringBytePtr(path))),
|
|
EXTATTR_NAMESPACE_USER, uintptr(unsafe.Pointer(syscall.StringBytePtr(name))),
|
|
)
|
|
if err != syscall.Errno(0) {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func listxattr(path string, data []byte) (int, error) {
|
|
return sysList(syscall.SYS_EXTATTR_LIST_FILE, path, data)
|
|
}
|
|
|
|
func llistxattr(path string, data []byte) (int, error) {
|
|
return sysList(syscall.SYS_EXTATTR_LIST_LINK, path, data)
|
|
}
|
|
|
|
func flistxattr(f *os.File, data []byte) (int, error) {
|
|
return listxattr(f.Name(), data)
|
|
}
|
|
|
|
// sysSet is called by listxattr and llistxattr with the appropriate syscall
|
|
// number. This works because syscalls have the same signature and return
|
|
// values.
|
|
func sysList(syscallNum uintptr, path string, data []byte) (int, error) {
|
|
ptr, nbytes := bytePtrFromSlice(data)
|
|
/*
|
|
ssize_t extattr_list_file(
|
|
const char *path,
|
|
int attrnamespace,
|
|
void *data,
|
|
size_t nbytes
|
|
);
|
|
|
|
ssize_t extattr_list_link(
|
|
const char *path,
|
|
int attrnamespace,
|
|
void *data,
|
|
size_t nbytes
|
|
);
|
|
*/
|
|
r0, _, err := syscall.Syscall6(syscallNum, uintptr(unsafe.Pointer(syscall.StringBytePtr(path))),
|
|
EXTATTR_NAMESPACE_USER, uintptr(unsafe.Pointer(ptr)), uintptr(nbytes), 0, 0)
|
|
if err != syscall.Errno(0) {
|
|
return int(r0), err
|
|
}
|
|
return int(r0), nil
|
|
}
|
|
|
|
// stringsFromByteSlice converts a sequence of attributes to a []string.
|
|
// On FreeBSD, each entry consists of a single byte containing the length
|
|
// of the attribute name, followed by the attribute name.
|
|
// The name is _not_ terminated by NULL.
|
|
func stringsFromByteSlice(buf []byte) (result []string) {
|
|
index := 0
|
|
for index < len(buf) {
|
|
next := index + 1 + int(buf[index])
|
|
result = append(result, string(buf[index+1:next]))
|
|
index = next
|
|
}
|
|
return
|
|
}
|