162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * linux/fs/hfsplus/xattr_trusted.c 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Vyacheslav Dubeyko <slava@dubeyko.com> 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Handler for storing security labels as extended attributes. 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <linux/security.h> 1162306a36Sopenharmony_ci#include <linux/nls.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include "hfsplus_fs.h" 1462306a36Sopenharmony_ci#include "xattr.h" 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_cistatic int hfsplus_security_getxattr(const struct xattr_handler *handler, 1762306a36Sopenharmony_ci struct dentry *unused, struct inode *inode, 1862306a36Sopenharmony_ci const char *name, void *buffer, size_t size) 1962306a36Sopenharmony_ci{ 2062306a36Sopenharmony_ci return hfsplus_getxattr(inode, name, buffer, size, 2162306a36Sopenharmony_ci XATTR_SECURITY_PREFIX, 2262306a36Sopenharmony_ci XATTR_SECURITY_PREFIX_LEN); 2362306a36Sopenharmony_ci} 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_cistatic int hfsplus_security_setxattr(const struct xattr_handler *handler, 2662306a36Sopenharmony_ci struct mnt_idmap *idmap, 2762306a36Sopenharmony_ci struct dentry *unused, struct inode *inode, 2862306a36Sopenharmony_ci const char *name, const void *buffer, 2962306a36Sopenharmony_ci size_t size, int flags) 3062306a36Sopenharmony_ci{ 3162306a36Sopenharmony_ci return hfsplus_setxattr(inode, name, buffer, size, flags, 3262306a36Sopenharmony_ci XATTR_SECURITY_PREFIX, 3362306a36Sopenharmony_ci XATTR_SECURITY_PREFIX_LEN); 3462306a36Sopenharmony_ci} 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_cistatic int hfsplus_initxattrs(struct inode *inode, 3762306a36Sopenharmony_ci const struct xattr *xattr_array, 3862306a36Sopenharmony_ci void *fs_info) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci const struct xattr *xattr; 4162306a36Sopenharmony_ci char *xattr_name; 4262306a36Sopenharmony_ci int err = 0; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci xattr_name = kmalloc(NLS_MAX_CHARSET_SIZE * HFSPLUS_ATTR_MAX_STRLEN + 1, 4562306a36Sopenharmony_ci GFP_KERNEL); 4662306a36Sopenharmony_ci if (!xattr_name) 4762306a36Sopenharmony_ci return -ENOMEM; 4862306a36Sopenharmony_ci for (xattr = xattr_array; xattr->name != NULL; xattr++) { 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci if (!strcmp(xattr->name, "")) 5162306a36Sopenharmony_ci continue; 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci strcpy(xattr_name, XATTR_SECURITY_PREFIX); 5462306a36Sopenharmony_ci strcpy(xattr_name + 5562306a36Sopenharmony_ci XATTR_SECURITY_PREFIX_LEN, xattr->name); 5662306a36Sopenharmony_ci memset(xattr_name + 5762306a36Sopenharmony_ci XATTR_SECURITY_PREFIX_LEN + strlen(xattr->name), 0, 1); 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci err = __hfsplus_setxattr(inode, xattr_name, 6062306a36Sopenharmony_ci xattr->value, xattr->value_len, 0); 6162306a36Sopenharmony_ci if (err) 6262306a36Sopenharmony_ci break; 6362306a36Sopenharmony_ci } 6462306a36Sopenharmony_ci kfree(xattr_name); 6562306a36Sopenharmony_ci return err; 6662306a36Sopenharmony_ci} 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ciint hfsplus_init_security(struct inode *inode, struct inode *dir, 6962306a36Sopenharmony_ci const struct qstr *qstr) 7062306a36Sopenharmony_ci{ 7162306a36Sopenharmony_ci return security_inode_init_security(inode, dir, qstr, 7262306a36Sopenharmony_ci &hfsplus_initxattrs, NULL); 7362306a36Sopenharmony_ci} 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ciconst struct xattr_handler hfsplus_xattr_security_handler = { 7662306a36Sopenharmony_ci .prefix = XATTR_SECURITY_PREFIX, 7762306a36Sopenharmony_ci .get = hfsplus_security_getxattr, 7862306a36Sopenharmony_ci .set = hfsplus_security_setxattr, 7962306a36Sopenharmony_ci}; 80