mirror of
https://github.com/qemu/qemu.git
synced 2025-08-09 19:15:32 +00:00
tools/virtiofsd: xattr name mappings: Map client xattr names
Map xattr names originating at the client; from get/set/remove xattr. Signed-off-by: Dr. David Alan Gilbert <dgilbert@redhat.com> Message-Id: <20201023165812.36028-3-dgilbert@redhat.com> Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com> Signed-off-by: Dr. David Alan Gilbert <dgilbert@redhat.com>
This commit is contained in:
parent
6084633dff
commit
4f088dbf98
@ -2188,20 +2188,80 @@ static void parse_xattrmap(struct lo_data *lo)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void lo_getxattr(fuse_req_t req, fuse_ino_t ino, const char *name,
|
/*
|
||||||
|
* For use with getxattr/setxattr/removexattr, where the client
|
||||||
|
* gives us a name and we may need to choose a different one.
|
||||||
|
* Allocates a buffer for the result placing it in *out_name.
|
||||||
|
* If there's no change then *out_name is not set.
|
||||||
|
* Returns 0 on success
|
||||||
|
* Can return -EPERM to indicate we block a given attribute
|
||||||
|
* (in which case out_name is not allocated)
|
||||||
|
* Can return -ENOMEM to indicate out_name couldn't be allocated.
|
||||||
|
*/
|
||||||
|
static int xattr_map_client(const struct lo_data *lo, const char *client_name,
|
||||||
|
char **out_name)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
for (i = 0; i < lo->xattr_map_nentries; i++) {
|
||||||
|
const XattrMapEntry *cur_entry = lo->xattr_map_list + i;
|
||||||
|
|
||||||
|
if ((cur_entry->flags & XATTR_MAP_FLAG_CLIENT) &&
|
||||||
|
(strstart(client_name, cur_entry->key, NULL))) {
|
||||||
|
if (cur_entry->flags & XATTR_MAP_FLAG_BAD) {
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
if (cur_entry->flags & XATTR_MAP_FLAG_OK) {
|
||||||
|
/* Unmodified name */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (cur_entry->flags & XATTR_MAP_FLAG_PREFIX) {
|
||||||
|
*out_name = g_try_malloc(strlen(client_name) +
|
||||||
|
strlen(cur_entry->prepend) + 1);
|
||||||
|
if (!*out_name) {
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
sprintf(*out_name, "%s%s", cur_entry->prepend, client_name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void lo_getxattr(fuse_req_t req, fuse_ino_t ino, const char *in_name,
|
||||||
size_t size)
|
size_t size)
|
||||||
{
|
{
|
||||||
struct lo_data *lo = lo_data(req);
|
struct lo_data *lo = lo_data(req);
|
||||||
char *value = NULL;
|
char *value = NULL;
|
||||||
char procname[64];
|
char procname[64];
|
||||||
|
const char *name;
|
||||||
|
char *mapped_name;
|
||||||
struct lo_inode *inode;
|
struct lo_inode *inode;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
int saverr;
|
int saverr;
|
||||||
int fd = -1;
|
int fd = -1;
|
||||||
|
|
||||||
|
mapped_name = NULL;
|
||||||
|
name = in_name;
|
||||||
|
if (lo->xattrmap) {
|
||||||
|
ret = xattr_map_client(lo, in_name, &mapped_name);
|
||||||
|
if (ret < 0) {
|
||||||
|
if (ret == -EPERM) {
|
||||||
|
ret = -ENODATA;
|
||||||
|
}
|
||||||
|
fuse_reply_err(req, -ret);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mapped_name) {
|
||||||
|
name = mapped_name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inode = lo_inode(req, ino);
|
inode = lo_inode(req, ino);
|
||||||
if (!inode) {
|
if (!inode) {
|
||||||
fuse_reply_err(req, EBADF);
|
fuse_reply_err(req, EBADF);
|
||||||
|
g_free(mapped_name);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2266,6 +2326,7 @@ out_err:
|
|||||||
saverr = errno;
|
saverr = errno;
|
||||||
out:
|
out:
|
||||||
fuse_reply_err(req, saverr);
|
fuse_reply_err(req, saverr);
|
||||||
|
g_free(mapped_name);
|
||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2343,19 +2404,35 @@ out:
|
|||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void lo_setxattr(fuse_req_t req, fuse_ino_t ino, const char *name,
|
static void lo_setxattr(fuse_req_t req, fuse_ino_t ino, const char *in_name,
|
||||||
const char *value, size_t size, int flags)
|
const char *value, size_t size, int flags)
|
||||||
{
|
{
|
||||||
char procname[64];
|
char procname[64];
|
||||||
|
const char *name;
|
||||||
|
char *mapped_name;
|
||||||
struct lo_data *lo = lo_data(req);
|
struct lo_data *lo = lo_data(req);
|
||||||
struct lo_inode *inode;
|
struct lo_inode *inode;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
int saverr;
|
int saverr;
|
||||||
int fd = -1;
|
int fd = -1;
|
||||||
|
|
||||||
|
mapped_name = NULL;
|
||||||
|
name = in_name;
|
||||||
|
if (lo->xattrmap) {
|
||||||
|
ret = xattr_map_client(lo, in_name, &mapped_name);
|
||||||
|
if (ret < 0) {
|
||||||
|
fuse_reply_err(req, -ret);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mapped_name) {
|
||||||
|
name = mapped_name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inode = lo_inode(req, ino);
|
inode = lo_inode(req, ino);
|
||||||
if (!inode) {
|
if (!inode) {
|
||||||
fuse_reply_err(req, EBADF);
|
fuse_reply_err(req, EBADF);
|
||||||
|
g_free(mapped_name);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2390,21 +2467,38 @@ out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
lo_inode_put(lo, &inode);
|
lo_inode_put(lo, &inode);
|
||||||
|
g_free(mapped_name);
|
||||||
fuse_reply_err(req, saverr);
|
fuse_reply_err(req, saverr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void lo_removexattr(fuse_req_t req, fuse_ino_t ino, const char *name)
|
static void lo_removexattr(fuse_req_t req, fuse_ino_t ino, const char *in_name)
|
||||||
{
|
{
|
||||||
char procname[64];
|
char procname[64];
|
||||||
|
const char *name;
|
||||||
|
char *mapped_name;
|
||||||
struct lo_data *lo = lo_data(req);
|
struct lo_data *lo = lo_data(req);
|
||||||
struct lo_inode *inode;
|
struct lo_inode *inode;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
int saverr;
|
int saverr;
|
||||||
int fd = -1;
|
int fd = -1;
|
||||||
|
|
||||||
|
mapped_name = NULL;
|
||||||
|
name = in_name;
|
||||||
|
if (lo->xattrmap) {
|
||||||
|
ret = xattr_map_client(lo, in_name, &mapped_name);
|
||||||
|
if (ret < 0) {
|
||||||
|
fuse_reply_err(req, -ret);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mapped_name) {
|
||||||
|
name = mapped_name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inode = lo_inode(req, ino);
|
inode = lo_inode(req, ino);
|
||||||
if (!inode) {
|
if (!inode) {
|
||||||
fuse_reply_err(req, EBADF);
|
fuse_reply_err(req, EBADF);
|
||||||
|
g_free(mapped_name);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2439,6 +2533,7 @@ out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
lo_inode_put(lo, &inode);
|
lo_inode_put(lo, &inode);
|
||||||
|
g_free(mapped_name);
|
||||||
fuse_reply_err(req, saverr);
|
fuse_reply_err(req, saverr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user