|  | /* | 
|  | * 9p user. xattr callback | 
|  | * | 
|  | * Copyright IBM, Corp. 2010 | 
|  | * | 
|  | * Authors: | 
|  | * Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> | 
|  | * | 
|  | * This work is licensed under the terms of the GNU GPL, version 2.  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 "9p.h" | 
|  | #include "fsdev/file-op-9p.h" | 
|  | #include "9p-xattr.h" | 
|  |  | 
|  |  | 
|  | static ssize_t mp_user_getxattr(FsContext *ctx, const char *path, | 
|  | const char *name, void *value, size_t size) | 
|  | { | 
|  | if (strncmp(name, "user.virtfs.", 12) == 0) { | 
|  | /* | 
|  | * Don't allow fetch of user.virtfs namespace | 
|  | * in case of mapped security | 
|  | */ | 
|  | errno = ENOATTR; | 
|  | return -1; | 
|  | } | 
|  | return local_getxattr_nofollow(ctx, path, name, value, size); | 
|  | } | 
|  |  | 
|  | static ssize_t mp_user_listxattr(FsContext *ctx, const char *path, | 
|  | char *name, void *value, size_t size) | 
|  | { | 
|  | int name_size = strlen(name) + 1; | 
|  | if (strncmp(name, "user.virtfs.", 12) == 0) { | 
|  |  | 
|  | /*  check if it is a mapped posix acl */ | 
|  | if (strncmp(name, "user.virtfs.system.posix_acl_", 29) == 0) { | 
|  | /* adjust the name and size */ | 
|  | name += 12; | 
|  | name_size -= 12; | 
|  | } else { | 
|  | /* | 
|  | * Don't allow fetch of user.virtfs namespace | 
|  | * in case of mapped security | 
|  | */ | 
|  | return 0; | 
|  | } | 
|  | } | 
|  | if (!value) { | 
|  | return name_size; | 
|  | } | 
|  |  | 
|  | if (size < name_size) { | 
|  | errno = ERANGE; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | /* name_size includes the trailing NUL. */ | 
|  | memcpy(value, name, name_size); | 
|  | return name_size; | 
|  | } | 
|  |  | 
|  | static int mp_user_setxattr(FsContext *ctx, const char *path, const char *name, | 
|  | void *value, size_t size, int flags) | 
|  | { | 
|  | if (strncmp(name, "user.virtfs.", 12) == 0) { | 
|  | /* | 
|  | * Don't allow fetch of user.virtfs namespace | 
|  | * in case of mapped security | 
|  | */ | 
|  | errno = EACCES; | 
|  | return -1; | 
|  | } | 
|  | return local_setxattr_nofollow(ctx, path, name, value, size, flags); | 
|  | } | 
|  |  | 
|  | static int mp_user_removexattr(FsContext *ctx, | 
|  | const char *path, const char *name) | 
|  | { | 
|  | if (strncmp(name, "user.virtfs.", 12) == 0) { | 
|  | /* | 
|  | * Don't allow fetch of user.virtfs namespace | 
|  | * in case of mapped security | 
|  | */ | 
|  | errno = EACCES; | 
|  | return -1; | 
|  | } | 
|  | return local_removexattr_nofollow(ctx, path, name); | 
|  | } | 
|  |  | 
|  | XattrOperations mapped_user_xattr = { | 
|  | .name = "user.", | 
|  | .getxattr = mp_user_getxattr, | 
|  | .setxattr = mp_user_setxattr, | 
|  | .listxattr = mp_user_listxattr, | 
|  | .removexattr = mp_user_removexattr, | 
|  | }; | 
|  |  | 
|  | XattrOperations passthrough_user_xattr = { | 
|  | .name = "user.", | 
|  | .getxattr = pt_getxattr, | 
|  | .setxattr = pt_setxattr, | 
|  | .listxattr = pt_listxattr, | 
|  | .removexattr = pt_removexattr, | 
|  | }; |