| /* |
| * 9p utilities (FreeBSD Implementation) |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| * See the COPYING file in the top-level directory. |
| */ |
| |
| /* |
| * Not so fast! You might want to read the 9p developer docs first: |
| * https://wiki.qemu.org/Documentation/9p |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "qemu/xattr.h" |
| #include "9p-util.h" |
| |
| static int mangle_xattr_name(const char **namep) |
| { |
| const char *name = *namep; |
| |
| /* |
| * ZFS forbids attributes in starting with "user." or "system.". |
| */ |
| if (strncmp(name, "system.", 7) == 0) { |
| *namep = name + 7; |
| return EXTATTR_NAMESPACE_SYSTEM; |
| } |
| if (strncmp(name, "user.", 5) == 0) { |
| *namep = name + 5; |
| } |
| return EXTATTR_NAMESPACE_USER; |
| } |
| |
| ssize_t fgetxattr(int fd, const char *name, void *value, size_t size) |
| { |
| int namespace; |
| |
| namespace = mangle_xattr_name(&name); |
| return extattr_get_fd(fd, namespace, name, value, size); |
| } |
| |
| ssize_t fgetxattrat_nofollow(int dirfd, const char *filename, const char *name, |
| void *value, size_t size) |
| { |
| ssize_t ret; |
| int fd, namespace; |
| |
| fd = openat_file(dirfd, filename, |
| O_RDONLY | O_PATH_9P_UTIL | O_NOFOLLOW, 0); |
| if (fd == -1) { |
| return -1; |
| } |
| namespace = mangle_xattr_name(&name); |
| ret = extattr_get_fd(fd, namespace, name, value, size); |
| close_preserve_errno(fd); |
| return ret; |
| } |
| |
| ssize_t flistxattrat_nofollow(int dirfd, const char *filename, |
| char *list, size_t size) |
| { |
| ssize_t ret; |
| int fd; |
| |
| fd = openat_file(dirfd, filename, |
| O_RDONLY | O_PATH_9P_UTIL | O_NOFOLLOW, 0); |
| if (fd == -1) { |
| return -1; |
| } |
| ret = extattr_list_fd(fd, EXTATTR_NAMESPACE_USER, list, size); |
| close_preserve_errno(fd); |
| return ret; |
| } |
| |
| ssize_t fremovexattrat_nofollow(int dirfd, const char *filename, |
| const char *name) |
| { |
| int fd, namespace, ret; |
| |
| fd = openat_file(dirfd, filename, |
| O_RDONLY | O_PATH_9P_UTIL | O_NOFOLLOW, 0); |
| if (fd == -1) { |
| return -1; |
| } |
| namespace = mangle_xattr_name(&name); |
| ret = extattr_delete_fd(fd, namespace, name); |
| close_preserve_errno(fd); |
| return ret; |
| } |
| |
| int fsetxattrat_nofollow(int dirfd, const char *filename, const char *name, |
| void *value, size_t size, int flags) |
| { |
| ssize_t ret; |
| int fd, namespace; |
| |
| namespace = mangle_xattr_name(&name); |
| if (flags == (XATTR_CREATE | XATTR_REPLACE)) { |
| errno = EINVAL; |
| return -1; |
| } |
| fd = openat_file(dirfd, filename, |
| O_RDONLY | O_PATH_9P_UTIL | O_NOFOLLOW, 0); |
| if (fd == -1) { |
| return -1; |
| } |
| if (flags & (XATTR_CREATE | XATTR_REPLACE)) { |
| ret = extattr_get_fd(fd, namespace, name, NULL, 0); |
| if (ret == -1 && errno != ENOATTR) { |
| close_preserve_errno(fd); |
| return -1; |
| } |
| if (ret >= 0 && (flags & XATTR_CREATE)) { |
| errno = EEXIST; |
| close_preserve_errno(fd); |
| return -1; |
| } |
| if (ret == -1 && (flags & XATTR_REPLACE)) { |
| errno = ENOATTR; |
| close_preserve_errno(fd); |
| return -1; |
| } |
| } |
| ret = extattr_set_fd(fd, namespace, name, value, size); |
| close_preserve_errno(fd); |
| return ret; |
| } |
| |
| int qemu_mknodat(int dirfd, const char *filename, mode_t mode, dev_t dev) |
| { |
| return mknodat(dirfd, filename, mode, dev); |
| } |