xref: /openbmc/linux/fs/ubifs/xattr.c (revision 422edacec0927b091ea5c32384f9d5b89efec1c1)
11e51764aSArtem Bityutskiy /*
21e51764aSArtem Bityutskiy  * This file is part of UBIFS.
31e51764aSArtem Bityutskiy  *
41e51764aSArtem Bityutskiy  * Copyright (C) 2006-2008 Nokia Corporation.
51e51764aSArtem Bityutskiy  *
61e51764aSArtem Bityutskiy  * This program is free software; you can redistribute it and/or modify it
71e51764aSArtem Bityutskiy  * under the terms of the GNU General Public License version 2 as published by
81e51764aSArtem Bityutskiy  * the Free Software Foundation.
91e51764aSArtem Bityutskiy  *
101e51764aSArtem Bityutskiy  * This program is distributed in the hope that it will be useful, but WITHOUT
111e51764aSArtem Bityutskiy  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
121e51764aSArtem Bityutskiy  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
131e51764aSArtem Bityutskiy  * more details.
141e51764aSArtem Bityutskiy  *
151e51764aSArtem Bityutskiy  * You should have received a copy of the GNU General Public License along with
161e51764aSArtem Bityutskiy  * this program; if not, write to the Free Software Foundation, Inc., 51
171e51764aSArtem Bityutskiy  * Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
181e51764aSArtem Bityutskiy  *
191e51764aSArtem Bityutskiy  * Authors: Artem Bityutskiy (Битюцкий Артём)
201e51764aSArtem Bityutskiy  *          Adrian Hunter
211e51764aSArtem Bityutskiy  */
221e51764aSArtem Bityutskiy 
231e51764aSArtem Bityutskiy /*
241e51764aSArtem Bityutskiy  * This file implements UBIFS extended attributes support.
251e51764aSArtem Bityutskiy  *
261e51764aSArtem Bityutskiy  * Extended attributes are implemented as regular inodes with attached data,
271e51764aSArtem Bityutskiy  * which limits extended attribute size to UBIFS block size (4KiB). Names of
281e51764aSArtem Bityutskiy  * extended attributes are described by extended attribute entries (xentries),
291e51764aSArtem Bityutskiy  * which are almost identical to directory entries, but have different key type.
301e51764aSArtem Bityutskiy  *
311e51764aSArtem Bityutskiy  * In other words, the situation with extended attributes is very similar to
321e51764aSArtem Bityutskiy  * directories. Indeed, any inode (but of course not xattr inodes) may have a
331e51764aSArtem Bityutskiy  * number of associated xentries, just like directory inodes have associated
341e51764aSArtem Bityutskiy  * directory entries. Extended attribute entries store the name of the extended
351e51764aSArtem Bityutskiy  * attribute, the host inode number, and the extended attribute inode number.
361e51764aSArtem Bityutskiy  * Similarly, direntries store the name, the parent and the target inode
371e51764aSArtem Bityutskiy  * numbers. Thus, most of the common UBIFS mechanisms may be re-used for
381e51764aSArtem Bityutskiy  * extended attributes.
391e51764aSArtem Bityutskiy  *
401e51764aSArtem Bityutskiy  * The number of extended attributes is not limited, but there is Linux
411e51764aSArtem Bityutskiy  * limitation on the maximum possible size of the list of all extended
421e51764aSArtem Bityutskiy  * attributes associated with an inode (%XATTR_LIST_MAX), so UBIFS makes sure
431e51764aSArtem Bityutskiy  * the sum of all extended attribute names of the inode does not exceed that
441e51764aSArtem Bityutskiy  * limit.
451e51764aSArtem Bityutskiy  *
461e51764aSArtem Bityutskiy  * Extended attributes are synchronous, which means they are written to the
471e51764aSArtem Bityutskiy  * flash media synchronously and there is no write-back for extended attribute
481e51764aSArtem Bityutskiy  * inodes. The extended attribute values are not stored in compressed form on
491e51764aSArtem Bityutskiy  * the media.
501e51764aSArtem Bityutskiy  *
511e51764aSArtem Bityutskiy  * Since extended attributes are represented by regular inodes, they are cached
521e51764aSArtem Bityutskiy  * in the VFS inode cache. The xentries are cached in the LNC cache (see
531e51764aSArtem Bityutskiy  * tnc.c).
541e51764aSArtem Bityutskiy  *
551e51764aSArtem Bityutskiy  * ACL support is not implemented.
561e51764aSArtem Bityutskiy  */
571e51764aSArtem Bityutskiy 
58073aaa1bSAl Viro #include "ubifs.h"
597dcda1c9SJens Axboe #include <linux/fs.h>
605a0e3ad6STejun Heo #include <linux/slab.h>
611e51764aSArtem Bityutskiy #include <linux/xattr.h>
621e51764aSArtem Bityutskiy 
631e51764aSArtem Bityutskiy /*
641e51764aSArtem Bityutskiy  * Limit the number of extended attributes per inode so that the total size
65c78c7e35SArtem Bityutskiy  * (@xattr_size) is guaranteeded to fit in an 'unsigned int'.
661e51764aSArtem Bityutskiy  */
671e51764aSArtem Bityutskiy #define MAX_XATTRS_PER_INODE 65535
681e51764aSArtem Bityutskiy 
691e51764aSArtem Bityutskiy /*
701e51764aSArtem Bityutskiy  * Extended attribute type constants.
711e51764aSArtem Bityutskiy  *
721e51764aSArtem Bityutskiy  * USER_XATTR: user extended attribute ("user.*")
731e51764aSArtem Bityutskiy  * TRUSTED_XATTR: trusted extended attribute ("trusted.*)
741e51764aSArtem Bityutskiy  * SECURITY_XATTR: security extended attribute ("security.*")
751e51764aSArtem Bityutskiy  */
761e51764aSArtem Bityutskiy enum {
771e51764aSArtem Bityutskiy 	USER_XATTR,
781e51764aSArtem Bityutskiy 	TRUSTED_XATTR,
791e51764aSArtem Bityutskiy 	SECURITY_XATTR,
801e51764aSArtem Bityutskiy };
811e51764aSArtem Bityutskiy 
8214ffd5d0SSedat Dilek static const struct inode_operations empty_iops;
8314ffd5d0SSedat Dilek static const struct file_operations empty_fops;
841e51764aSArtem Bityutskiy 
851e51764aSArtem Bityutskiy /**
861e51764aSArtem Bityutskiy  * create_xattr - create an extended attribute.
871e51764aSArtem Bityutskiy  * @c: UBIFS file-system description object
881e51764aSArtem Bityutskiy  * @host: host inode
891e51764aSArtem Bityutskiy  * @nm: extended attribute name
901e51764aSArtem Bityutskiy  * @value: extended attribute value
911e51764aSArtem Bityutskiy  * @size: size of extended attribute value
921e51764aSArtem Bityutskiy  *
931e51764aSArtem Bityutskiy  * This is a helper function which creates an extended attribute of name @nm
941e51764aSArtem Bityutskiy  * and value @value for inode @host. The host inode is also updated on flash
951e51764aSArtem Bityutskiy  * because the ctime and extended attribute accounting data changes. This
961e51764aSArtem Bityutskiy  * function returns zero in case of success and a negative error code in case
971e51764aSArtem Bityutskiy  * of failure.
981e51764aSArtem Bityutskiy  */
991e51764aSArtem Bityutskiy static int create_xattr(struct ubifs_info *c, struct inode *host,
100f4f61d2cSRichard Weinberger 			const struct fscrypt_name *nm, const void *value, int size)
1011e51764aSArtem Bityutskiy {
102fee1756dSSubodh Nijsure 	int err, names_len;
1031e51764aSArtem Bityutskiy 	struct inode *inode;
1041e51764aSArtem Bityutskiy 	struct ubifs_inode *ui, *host_ui = ubifs_inode(host);
1051e51764aSArtem Bityutskiy 	struct ubifs_budget_req req = { .new_ino = 1, .new_dent = 1,
1065acd6ff8SZoltan Sogor 				.new_ino_d = ALIGN(size, 8), .dirtied_ino = 1,
107dab4b4d2SArtem Bityutskiy 				.dirtied_ino_d = ALIGN(host_ui->data_len, 8) };
1081e51764aSArtem Bityutskiy 
109fee1756dSSubodh Nijsure 	if (host_ui->xattr_cnt >= MAX_XATTRS_PER_INODE) {
110235c362bSSheng Yong 		ubifs_err(c, "inode %lu already has too many xattrs (%d), cannot create more",
111fee1756dSSubodh Nijsure 			  host->i_ino, host_ui->xattr_cnt);
1121e51764aSArtem Bityutskiy 		return -ENOSPC;
113fee1756dSSubodh Nijsure 	}
1141e51764aSArtem Bityutskiy 	/*
1151e51764aSArtem Bityutskiy 	 * Linux limits the maximum size of the extended attribute names list
116c78c7e35SArtem Bityutskiy 	 * to %XATTR_LIST_MAX. This means we should not allow creating more
1171e51764aSArtem Bityutskiy 	 * extended attributes if the name list becomes larger. This limitation
1181e51764aSArtem Bityutskiy 	 * is artificial for UBIFS, though.
1191e51764aSArtem Bityutskiy 	 */
120f4f61d2cSRichard Weinberger 	names_len = host_ui->xattr_names + host_ui->xattr_cnt + fname_len(nm) + 1;
121fee1756dSSubodh Nijsure 	if (names_len > XATTR_LIST_MAX) {
122235c362bSSheng Yong 		ubifs_err(c, "cannot add one more xattr name to inode %lu, total names length would become %d, max. is %d",
123fee1756dSSubodh Nijsure 			  host->i_ino, names_len, XATTR_LIST_MAX);
1241e51764aSArtem Bityutskiy 		return -ENOSPC;
125fee1756dSSubodh Nijsure 	}
1261e51764aSArtem Bityutskiy 
1271e51764aSArtem Bityutskiy 	err = ubifs_budget_space(c, &req);
1281e51764aSArtem Bityutskiy 	if (err)
1291e51764aSArtem Bityutskiy 		return err;
1301e51764aSArtem Bityutskiy 
1311e51764aSArtem Bityutskiy 	inode = ubifs_new_inode(c, host, S_IFREG | S_IRWXUGO);
1321e51764aSArtem Bityutskiy 	if (IS_ERR(inode)) {
1331e51764aSArtem Bityutskiy 		err = PTR_ERR(inode);
1341e51764aSArtem Bityutskiy 		goto out_budg;
1351e51764aSArtem Bityutskiy 	}
1361e51764aSArtem Bityutskiy 
1371e51764aSArtem Bityutskiy 	/* Re-define all operations to be "nothing" */
1387dcda1c9SJens Axboe 	inode->i_mapping->a_ops = &empty_aops;
13914ffd5d0SSedat Dilek 	inode->i_op = &empty_iops;
14014ffd5d0SSedat Dilek 	inode->i_fop = &empty_fops;
1411e51764aSArtem Bityutskiy 
142*422edaceSWang Shilong 	inode->i_flags |= S_SYNC | S_NOATIME | S_NOCMTIME;
1431e51764aSArtem Bityutskiy 	ui = ubifs_inode(inode);
1441e51764aSArtem Bityutskiy 	ui->xattr = 1;
1451e51764aSArtem Bityutskiy 	ui->flags |= UBIFS_XATTR_FL;
146eaecf43aSThomas Meyer 	ui->data = kmemdup(value, size, GFP_NOFS);
1471e51764aSArtem Bityutskiy 	if (!ui->data) {
1481e51764aSArtem Bityutskiy 		err = -ENOMEM;
149c78c7e35SArtem Bityutskiy 		goto out_free;
1501e51764aSArtem Bityutskiy 	}
151c78c7e35SArtem Bityutskiy 	inode->i_size = ui->ui_size = size;
152c78c7e35SArtem Bityutskiy 	ui->data_len = size;
153c78c7e35SArtem Bityutskiy 
154c78c7e35SArtem Bityutskiy 	mutex_lock(&host_ui->ui_mutex);
155607a11adSDeepa Dinamani 	host->i_ctime = current_time(host);
1561e51764aSArtem Bityutskiy 	host_ui->xattr_cnt += 1;
157f4f61d2cSRichard Weinberger 	host_ui->xattr_size += CALC_DENT_SIZE(fname_len(nm));
1581e51764aSArtem Bityutskiy 	host_ui->xattr_size += CALC_XATTR_BYTES(size);
159f4f61d2cSRichard Weinberger 	host_ui->xattr_names += fname_len(nm);
1601e51764aSArtem Bityutskiy 
161d475a507SRichard Weinberger 	/*
162d475a507SRichard Weinberger 	 * We handle UBIFS_XATTR_NAME_ENCRYPTION_CONTEXT here because we
163d475a507SRichard Weinberger 	 * have to set the UBIFS_CRYPT_FL flag on the host inode.
164d475a507SRichard Weinberger 	 * To avoid multiple updates of the same inode in the same operation,
165d475a507SRichard Weinberger 	 * let's do it here.
166d475a507SRichard Weinberger 	 */
167f4f61d2cSRichard Weinberger 	if (strcmp(fname_name(nm), UBIFS_XATTR_NAME_ENCRYPTION_CONTEXT) == 0)
168d475a507SRichard Weinberger 		host_ui->flags |= UBIFS_CRYPT_FL;
169d475a507SRichard Weinberger 
1701e51764aSArtem Bityutskiy 	err = ubifs_jnl_update(c, host, nm, inode, 0, 1);
1711e51764aSArtem Bityutskiy 	if (err)
1721e51764aSArtem Bityutskiy 		goto out_cancel;
1732ee6a576SEric Biggers 	ubifs_set_inode_flags(host);
1741e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
1751e51764aSArtem Bityutskiy 
1761e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
1771e51764aSArtem Bityutskiy 	insert_inode_hash(inode);
1781e51764aSArtem Bityutskiy 	iput(inode);
1791e51764aSArtem Bityutskiy 	return 0;
1801e51764aSArtem Bityutskiy 
1811e51764aSArtem Bityutskiy out_cancel:
1821e51764aSArtem Bityutskiy 	host_ui->xattr_cnt -= 1;
183f4f61d2cSRichard Weinberger 	host_ui->xattr_size -= CALC_DENT_SIZE(fname_len(nm));
1841e51764aSArtem Bityutskiy 	host_ui->xattr_size -= CALC_XATTR_BYTES(size);
185f4f61d2cSRichard Weinberger 	host_ui->xattr_names -= fname_len(nm);
186d475a507SRichard Weinberger 	host_ui->flags &= ~UBIFS_CRYPT_FL;
1871e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
188c78c7e35SArtem Bityutskiy out_free:
1891e51764aSArtem Bityutskiy 	make_bad_inode(inode);
1901e51764aSArtem Bityutskiy 	iput(inode);
1911e51764aSArtem Bityutskiy out_budg:
1921e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
1931e51764aSArtem Bityutskiy 	return err;
1941e51764aSArtem Bityutskiy }
1951e51764aSArtem Bityutskiy 
1961e51764aSArtem Bityutskiy /**
1971e51764aSArtem Bityutskiy  * change_xattr - change an extended attribute.
1981e51764aSArtem Bityutskiy  * @c: UBIFS file-system description object
1991e51764aSArtem Bityutskiy  * @host: host inode
2001e51764aSArtem Bityutskiy  * @inode: extended attribute inode
2011e51764aSArtem Bityutskiy  * @value: extended attribute value
2021e51764aSArtem Bityutskiy  * @size: size of extended attribute value
2031e51764aSArtem Bityutskiy  *
2041e51764aSArtem Bityutskiy  * This helper function changes the value of extended attribute @inode with new
2051e51764aSArtem Bityutskiy  * data from @value. Returns zero in case of success and a negative error code
2061e51764aSArtem Bityutskiy  * in case of failure.
2071e51764aSArtem Bityutskiy  */
2081e51764aSArtem Bityutskiy static int change_xattr(struct ubifs_info *c, struct inode *host,
2091e51764aSArtem Bityutskiy 			struct inode *inode, const void *value, int size)
2101e51764aSArtem Bityutskiy {
2111e51764aSArtem Bityutskiy 	int err;
2121e51764aSArtem Bityutskiy 	struct ubifs_inode *host_ui = ubifs_inode(host);
2131e51764aSArtem Bityutskiy 	struct ubifs_inode *ui = ubifs_inode(inode);
214ab92a20bSDongsheng Yang 	void *buf = NULL;
21574e9c700SPascal Eberhard 	int old_size;
2161e51764aSArtem Bityutskiy 	struct ubifs_budget_req req = { .dirtied_ino = 2,
2175acd6ff8SZoltan Sogor 		.dirtied_ino_d = ALIGN(size, 8) + ALIGN(host_ui->data_len, 8) };
2181e51764aSArtem Bityutskiy 
2191e51764aSArtem Bityutskiy 	ubifs_assert(ui->data_len == inode->i_size);
2201e51764aSArtem Bityutskiy 	err = ubifs_budget_space(c, &req);
2211e51764aSArtem Bityutskiy 	if (err)
2221e51764aSArtem Bityutskiy 		return err;
2231e51764aSArtem Bityutskiy 
224ab92a20bSDongsheng Yang 	buf = kmemdup(value, size, GFP_NOFS);
225ab92a20bSDongsheng Yang 	if (!buf) {
2261e51764aSArtem Bityutskiy 		err = -ENOMEM;
227c78c7e35SArtem Bityutskiy 		goto out_free;
2281e51764aSArtem Bityutskiy 	}
229ab92a20bSDongsheng Yang 	mutex_lock(&ui->ui_mutex);
230ab92a20bSDongsheng Yang 	kfree(ui->data);
231ab92a20bSDongsheng Yang 	ui->data = buf;
2321e51764aSArtem Bityutskiy 	inode->i_size = ui->ui_size = size;
23374e9c700SPascal Eberhard 	old_size = ui->data_len;
2341e51764aSArtem Bityutskiy 	ui->data_len = size;
235ab92a20bSDongsheng Yang 	mutex_unlock(&ui->ui_mutex);
2361e51764aSArtem Bityutskiy 
237c78c7e35SArtem Bityutskiy 	mutex_lock(&host_ui->ui_mutex);
238607a11adSDeepa Dinamani 	host->i_ctime = current_time(host);
23974e9c700SPascal Eberhard 	host_ui->xattr_size -= CALC_XATTR_BYTES(old_size);
240c78c7e35SArtem Bityutskiy 	host_ui->xattr_size += CALC_XATTR_BYTES(size);
241c78c7e35SArtem Bityutskiy 
2421e51764aSArtem Bityutskiy 	/*
2431e51764aSArtem Bityutskiy 	 * It is important to write the host inode after the xattr inode
2441e51764aSArtem Bityutskiy 	 * because if the host inode gets synchronized (via 'fsync()'), then
2451e51764aSArtem Bityutskiy 	 * the extended attribute inode gets synchronized, because it goes
2461e51764aSArtem Bityutskiy 	 * before the host inode in the write-buffer.
2471e51764aSArtem Bityutskiy 	 */
2481e51764aSArtem Bityutskiy 	err = ubifs_jnl_change_xattr(c, inode, host);
2491e51764aSArtem Bityutskiy 	if (err)
2501e51764aSArtem Bityutskiy 		goto out_cancel;
2511e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
2521e51764aSArtem Bityutskiy 
2531e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
2541e51764aSArtem Bityutskiy 	return 0;
2551e51764aSArtem Bityutskiy 
2561e51764aSArtem Bityutskiy out_cancel:
2571e51764aSArtem Bityutskiy 	host_ui->xattr_size -= CALC_XATTR_BYTES(size);
25874e9c700SPascal Eberhard 	host_ui->xattr_size += CALC_XATTR_BYTES(old_size);
2591e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
260c78c7e35SArtem Bityutskiy 	make_bad_inode(inode);
261c78c7e35SArtem Bityutskiy out_free:
2621e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
2631e51764aSArtem Bityutskiy 	return err;
2641e51764aSArtem Bityutskiy }
2651e51764aSArtem Bityutskiy 
2661e51764aSArtem Bityutskiy static struct inode *iget_xattr(struct ubifs_info *c, ino_t inum)
2671e51764aSArtem Bityutskiy {
2681e51764aSArtem Bityutskiy 	struct inode *inode;
2691e51764aSArtem Bityutskiy 
2701e51764aSArtem Bityutskiy 	inode = ubifs_iget(c->vfs_sb, inum);
2711e51764aSArtem Bityutskiy 	if (IS_ERR(inode)) {
272235c362bSSheng Yong 		ubifs_err(c, "dead extended attribute entry, error %d",
2731e51764aSArtem Bityutskiy 			  (int)PTR_ERR(inode));
2741e51764aSArtem Bityutskiy 		return inode;
2751e51764aSArtem Bityutskiy 	}
2761e51764aSArtem Bityutskiy 	if (ubifs_inode(inode)->xattr)
2771e51764aSArtem Bityutskiy 		return inode;
278235c362bSSheng Yong 	ubifs_err(c, "corrupt extended attribute entry");
2791e51764aSArtem Bityutskiy 	iput(inode);
2801e51764aSArtem Bityutskiy 	return ERR_PTR(-EINVAL);
2811e51764aSArtem Bityutskiy }
2821e51764aSArtem Bityutskiy 
283ade46c3aSRichard Weinberger int ubifs_xattr_set(struct inode *host, const char *name, const void *value,
284d8db5b1cSXiaolei Li 		    size_t size, int flags, bool check_lock)
2851e51764aSArtem Bityutskiy {
286895d9db2SSubodh Nijsure 	struct inode *inode;
2871e51764aSArtem Bityutskiy 	struct ubifs_info *c = host->i_sb->s_fs_info;
288f4f61d2cSRichard Weinberger 	struct fscrypt_name nm = { .disk_name = FSTR_INIT((char *)name, strlen(name))};
2891e51764aSArtem Bityutskiy 	struct ubifs_dent_node *xent;
2901e51764aSArtem Bityutskiy 	union ubifs_key key;
2912b88fc21SAndreas Gruenbacher 	int err;
2921e51764aSArtem Bityutskiy 
293d8db5b1cSXiaolei Li 	if (check_lock)
2945955102cSAl Viro 		ubifs_assert(inode_is_locked(host));
2951e51764aSArtem Bityutskiy 
2961e51764aSArtem Bityutskiy 	if (size > UBIFS_MAX_INO_DATA)
2971e51764aSArtem Bityutskiy 		return -ERANGE;
2981e51764aSArtem Bityutskiy 
299f4f61d2cSRichard Weinberger 	if (fname_len(&nm) > UBIFS_MAX_NLEN)
3002b88fc21SAndreas Gruenbacher 		return -ENAMETOOLONG;
3011e51764aSArtem Bityutskiy 
3021e51764aSArtem Bityutskiy 	xent = kmalloc(UBIFS_MAX_XENT_NODE_SZ, GFP_NOFS);
3031e51764aSArtem Bityutskiy 	if (!xent)
3041e51764aSArtem Bityutskiy 		return -ENOMEM;
3051e51764aSArtem Bityutskiy 
3061e51764aSArtem Bityutskiy 	/*
3071e51764aSArtem Bityutskiy 	 * The extended attribute entries are stored in LNC, so multiple
3081e51764aSArtem Bityutskiy 	 * look-ups do not involve reading the flash.
3091e51764aSArtem Bityutskiy 	 */
3101e51764aSArtem Bityutskiy 	xent_key_init(c, &key, host->i_ino, &nm);
3111e51764aSArtem Bityutskiy 	err = ubifs_tnc_lookup_nm(c, &key, xent, &nm);
3121e51764aSArtem Bityutskiy 	if (err) {
3131e51764aSArtem Bityutskiy 		if (err != -ENOENT)
3141e51764aSArtem Bityutskiy 			goto out_free;
3151e51764aSArtem Bityutskiy 
3161e51764aSArtem Bityutskiy 		if (flags & XATTR_REPLACE)
3171e51764aSArtem Bityutskiy 			/* We are asked not to create the xattr */
3181e51764aSArtem Bityutskiy 			err = -ENODATA;
3191e51764aSArtem Bityutskiy 		else
3201e51764aSArtem Bityutskiy 			err = create_xattr(c, host, &nm, value, size);
3211e51764aSArtem Bityutskiy 		goto out_free;
3221e51764aSArtem Bityutskiy 	}
3231e51764aSArtem Bityutskiy 
3241e51764aSArtem Bityutskiy 	if (flags & XATTR_CREATE) {
3251e51764aSArtem Bityutskiy 		/* We are asked not to replace the xattr */
3261e51764aSArtem Bityutskiy 		err = -EEXIST;
3271e51764aSArtem Bityutskiy 		goto out_free;
3281e51764aSArtem Bityutskiy 	}
3291e51764aSArtem Bityutskiy 
3301e51764aSArtem Bityutskiy 	inode = iget_xattr(c, le64_to_cpu(xent->inum));
3311e51764aSArtem Bityutskiy 	if (IS_ERR(inode)) {
3321e51764aSArtem Bityutskiy 		err = PTR_ERR(inode);
3331e51764aSArtem Bityutskiy 		goto out_free;
3341e51764aSArtem Bityutskiy 	}
3351e51764aSArtem Bityutskiy 
3361e51764aSArtem Bityutskiy 	err = change_xattr(c, host, inode, value, size);
3371e51764aSArtem Bityutskiy 	iput(inode);
3381e51764aSArtem Bityutskiy 
3391e51764aSArtem Bityutskiy out_free:
3401e51764aSArtem Bityutskiy 	kfree(xent);
3411e51764aSArtem Bityutskiy 	return err;
3421e51764aSArtem Bityutskiy }
3431e51764aSArtem Bityutskiy 
344ade46c3aSRichard Weinberger ssize_t ubifs_xattr_get(struct inode *host, const char *name, void *buf,
345ade46c3aSRichard Weinberger 			size_t size)
3461e51764aSArtem Bityutskiy {
347ce23e640SAl Viro 	struct inode *inode;
3481e51764aSArtem Bityutskiy 	struct ubifs_info *c = host->i_sb->s_fs_info;
349f4f61d2cSRichard Weinberger 	struct fscrypt_name nm = { .disk_name = FSTR_INIT((char *)name, strlen(name))};
3501e51764aSArtem Bityutskiy 	struct ubifs_inode *ui;
3511e51764aSArtem Bityutskiy 	struct ubifs_dent_node *xent;
3521e51764aSArtem Bityutskiy 	union ubifs_key key;
3531e51764aSArtem Bityutskiy 	int err;
3541e51764aSArtem Bityutskiy 
355f4f61d2cSRichard Weinberger 	if (fname_len(&nm) > UBIFS_MAX_NLEN)
3562b88fc21SAndreas Gruenbacher 		return -ENAMETOOLONG;
3571e51764aSArtem Bityutskiy 
3581e51764aSArtem Bityutskiy 	xent = kmalloc(UBIFS_MAX_XENT_NODE_SZ, GFP_NOFS);
3591e51764aSArtem Bityutskiy 	if (!xent)
3601e51764aSArtem Bityutskiy 		return -ENOMEM;
3611e51764aSArtem Bityutskiy 
3621e51764aSArtem Bityutskiy 	xent_key_init(c, &key, host->i_ino, &nm);
3631e51764aSArtem Bityutskiy 	err = ubifs_tnc_lookup_nm(c, &key, xent, &nm);
3641e51764aSArtem Bityutskiy 	if (err) {
3651e51764aSArtem Bityutskiy 		if (err == -ENOENT)
3661e51764aSArtem Bityutskiy 			err = -ENODATA;
3671e51764aSArtem Bityutskiy 		goto out_unlock;
3681e51764aSArtem Bityutskiy 	}
3691e51764aSArtem Bityutskiy 
3701e51764aSArtem Bityutskiy 	inode = iget_xattr(c, le64_to_cpu(xent->inum));
3711e51764aSArtem Bityutskiy 	if (IS_ERR(inode)) {
3721e51764aSArtem Bityutskiy 		err = PTR_ERR(inode);
3731e51764aSArtem Bityutskiy 		goto out_unlock;
3741e51764aSArtem Bityutskiy 	}
3751e51764aSArtem Bityutskiy 
3761e51764aSArtem Bityutskiy 	ui = ubifs_inode(inode);
3771e51764aSArtem Bityutskiy 	ubifs_assert(inode->i_size == ui->data_len);
3781e51764aSArtem Bityutskiy 	ubifs_assert(ubifs_inode(host)->xattr_size > ui->data_len);
3791e51764aSArtem Bityutskiy 
380ab92a20bSDongsheng Yang 	mutex_lock(&ui->ui_mutex);
3811e51764aSArtem Bityutskiy 	if (buf) {
3821e51764aSArtem Bityutskiy 		/* If @buf is %NULL we are supposed to return the length */
3831e51764aSArtem Bityutskiy 		if (ui->data_len > size) {
3841e51764aSArtem Bityutskiy 			err = -ERANGE;
3851e51764aSArtem Bityutskiy 			goto out_iput;
3861e51764aSArtem Bityutskiy 		}
3871e51764aSArtem Bityutskiy 
3881e51764aSArtem Bityutskiy 		memcpy(buf, ui->data, ui->data_len);
3891e51764aSArtem Bityutskiy 	}
3901e51764aSArtem Bityutskiy 	err = ui->data_len;
3911e51764aSArtem Bityutskiy 
3921e51764aSArtem Bityutskiy out_iput:
393ab92a20bSDongsheng Yang 	mutex_unlock(&ui->ui_mutex);
3941e51764aSArtem Bityutskiy 	iput(inode);
3951e51764aSArtem Bityutskiy out_unlock:
3961e51764aSArtem Bityutskiy 	kfree(xent);
3971e51764aSArtem Bityutskiy 	return err;
3981e51764aSArtem Bityutskiy }
3991e51764aSArtem Bityutskiy 
40043b113feSRichard Weinberger static bool xattr_visible(const char *name)
40143b113feSRichard Weinberger {
40243b113feSRichard Weinberger 	/* File encryption related xattrs are for internal use only */
40343b113feSRichard Weinberger 	if (strcmp(name, UBIFS_XATTR_NAME_ENCRYPTION_CONTEXT) == 0)
40443b113feSRichard Weinberger 		return false;
40543b113feSRichard Weinberger 
40643b113feSRichard Weinberger 	/* Show trusted namespace only for "power" users */
40743b113feSRichard Weinberger 	if (strncmp(name, XATTR_TRUSTED_PREFIX,
40843b113feSRichard Weinberger 		    XATTR_TRUSTED_PREFIX_LEN) == 0 && !capable(CAP_SYS_ADMIN))
40943b113feSRichard Weinberger 		return false;
41043b113feSRichard Weinberger 
41143b113feSRichard Weinberger 	return true;
41243b113feSRichard Weinberger }
41343b113feSRichard Weinberger 
4141e51764aSArtem Bityutskiy ssize_t ubifs_listxattr(struct dentry *dentry, char *buffer, size_t size)
4151e51764aSArtem Bityutskiy {
4161e51764aSArtem Bityutskiy 	union ubifs_key key;
4172b0143b5SDavid Howells 	struct inode *host = d_inode(dentry);
4181e51764aSArtem Bityutskiy 	struct ubifs_info *c = host->i_sb->s_fs_info;
4191e51764aSArtem Bityutskiy 	struct ubifs_inode *host_ui = ubifs_inode(host);
4201e51764aSArtem Bityutskiy 	struct ubifs_dent_node *xent, *pxent = NULL;
4211e51764aSArtem Bityutskiy 	int err, len, written = 0;
422f4f61d2cSRichard Weinberger 	struct fscrypt_name nm = {0};
4231e51764aSArtem Bityutskiy 
4244cb2a01dSAl Viro 	dbg_gen("ino %lu ('%pd'), buffer size %zd", host->i_ino,
4254cb2a01dSAl Viro 		dentry, size);
4261e51764aSArtem Bityutskiy 
4271e51764aSArtem Bityutskiy 	len = host_ui->xattr_names + host_ui->xattr_cnt;
4281e51764aSArtem Bityutskiy 	if (!buffer)
4291e51764aSArtem Bityutskiy 		/*
4301e51764aSArtem Bityutskiy 		 * We should return the minimum buffer size which will fit a
4311e51764aSArtem Bityutskiy 		 * null-terminated list of all the extended attribute names.
4321e51764aSArtem Bityutskiy 		 */
4331e51764aSArtem Bityutskiy 		return len;
4341e51764aSArtem Bityutskiy 
4351e51764aSArtem Bityutskiy 	if (len > size)
4361e51764aSArtem Bityutskiy 		return -ERANGE;
4371e51764aSArtem Bityutskiy 
4381e51764aSArtem Bityutskiy 	lowest_xent_key(c, &key, host->i_ino);
4391e51764aSArtem Bityutskiy 	while (1) {
4401e51764aSArtem Bityutskiy 		xent = ubifs_tnc_next_ent(c, &key, &nm);
4418d47aef4SHirofumi Nakagawa 		if (IS_ERR(xent)) {
4421e51764aSArtem Bityutskiy 			err = PTR_ERR(xent);
4431e51764aSArtem Bityutskiy 			break;
4441e51764aSArtem Bityutskiy 		}
4451e51764aSArtem Bityutskiy 
446f4f61d2cSRichard Weinberger 		fname_name(&nm) = xent->name;
447f4f61d2cSRichard Weinberger 		fname_len(&nm) = le16_to_cpu(xent->nlen);
4481e51764aSArtem Bityutskiy 
44943b113feSRichard Weinberger 		if (xattr_visible(xent->name)) {
450f4f61d2cSRichard Weinberger 			memcpy(buffer + written, fname_name(&nm), fname_len(&nm) + 1);
451f4f61d2cSRichard Weinberger 			written += fname_len(&nm) + 1;
4521e51764aSArtem Bityutskiy 		}
4531e51764aSArtem Bityutskiy 
4541e51764aSArtem Bityutskiy 		kfree(pxent);
4551e51764aSArtem Bityutskiy 		pxent = xent;
4561e51764aSArtem Bityutskiy 		key_read(c, &xent->key, &key);
4571e51764aSArtem Bityutskiy 	}
4581e51764aSArtem Bityutskiy 
4591e51764aSArtem Bityutskiy 	kfree(pxent);
4601e51764aSArtem Bityutskiy 	if (err != -ENOENT) {
461235c362bSSheng Yong 		ubifs_err(c, "cannot find next direntry, error %d", err);
4621e51764aSArtem Bityutskiy 		return err;
4631e51764aSArtem Bityutskiy 	}
4641e51764aSArtem Bityutskiy 
4651e51764aSArtem Bityutskiy 	ubifs_assert(written <= size);
4661e51764aSArtem Bityutskiy 	return written;
4671e51764aSArtem Bityutskiy }
4681e51764aSArtem Bityutskiy 
4691e51764aSArtem Bityutskiy static int remove_xattr(struct ubifs_info *c, struct inode *host,
470f4f61d2cSRichard Weinberger 			struct inode *inode, const struct fscrypt_name *nm)
4711e51764aSArtem Bityutskiy {
4721e51764aSArtem Bityutskiy 	int err;
4731e51764aSArtem Bityutskiy 	struct ubifs_inode *host_ui = ubifs_inode(host);
4741e51764aSArtem Bityutskiy 	struct ubifs_inode *ui = ubifs_inode(inode);
4755acd6ff8SZoltan Sogor 	struct ubifs_budget_req req = { .dirtied_ino = 2, .mod_dent = 1,
4765acd6ff8SZoltan Sogor 				.dirtied_ino_d = ALIGN(host_ui->data_len, 8) };
4771e51764aSArtem Bityutskiy 
4781e51764aSArtem Bityutskiy 	ubifs_assert(ui->data_len == inode->i_size);
4791e51764aSArtem Bityutskiy 
4801e51764aSArtem Bityutskiy 	err = ubifs_budget_space(c, &req);
4811e51764aSArtem Bityutskiy 	if (err)
4821e51764aSArtem Bityutskiy 		return err;
4831e51764aSArtem Bityutskiy 
4841e51764aSArtem Bityutskiy 	mutex_lock(&host_ui->ui_mutex);
485607a11adSDeepa Dinamani 	host->i_ctime = current_time(host);
4861e51764aSArtem Bityutskiy 	host_ui->xattr_cnt -= 1;
487f4f61d2cSRichard Weinberger 	host_ui->xattr_size -= CALC_DENT_SIZE(fname_len(nm));
4881e51764aSArtem Bityutskiy 	host_ui->xattr_size -= CALC_XATTR_BYTES(ui->data_len);
489f4f61d2cSRichard Weinberger 	host_ui->xattr_names -= fname_len(nm);
4901e51764aSArtem Bityutskiy 
4911e51764aSArtem Bityutskiy 	err = ubifs_jnl_delete_xattr(c, host, inode, nm);
4921e51764aSArtem Bityutskiy 	if (err)
4931e51764aSArtem Bityutskiy 		goto out_cancel;
4941e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
4951e51764aSArtem Bityutskiy 
4961e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
4971e51764aSArtem Bityutskiy 	return 0;
4981e51764aSArtem Bityutskiy 
4991e51764aSArtem Bityutskiy out_cancel:
5001e51764aSArtem Bityutskiy 	host_ui->xattr_cnt += 1;
501f4f61d2cSRichard Weinberger 	host_ui->xattr_size += CALC_DENT_SIZE(fname_len(nm));
5021e51764aSArtem Bityutskiy 	host_ui->xattr_size += CALC_XATTR_BYTES(ui->data_len);
503f4f61d2cSRichard Weinberger 	host_ui->xattr_names += fname_len(nm);
5041e51764aSArtem Bityutskiy 	mutex_unlock(&host_ui->ui_mutex);
5051e51764aSArtem Bityutskiy 	ubifs_release_budget(c, &req);
5061e51764aSArtem Bityutskiy 	make_bad_inode(inode);
5071e51764aSArtem Bityutskiy 	return err;
5081e51764aSArtem Bityutskiy }
5091e51764aSArtem Bityutskiy 
510272eda82SRichard Weinberger /**
511272eda82SRichard Weinberger  * ubifs_evict_xattr_inode - Evict an xattr inode.
512272eda82SRichard Weinberger  * @c: UBIFS file-system description object
513272eda82SRichard Weinberger  * @xattr_inum: xattr inode number
514272eda82SRichard Weinberger  *
515272eda82SRichard Weinberger  * When an inode that hosts xattrs is being removed we have to make sure
516272eda82SRichard Weinberger  * that cached inodes of the xattrs also get removed from the inode cache
517272eda82SRichard Weinberger  * otherwise we'd waste memory. This function looks up an inode from the
518272eda82SRichard Weinberger  * inode cache and clears the link counter such that iput() will evict
519272eda82SRichard Weinberger  * the inode.
520272eda82SRichard Weinberger  */
521272eda82SRichard Weinberger void ubifs_evict_xattr_inode(struct ubifs_info *c, ino_t xattr_inum)
522272eda82SRichard Weinberger {
523272eda82SRichard Weinberger 	struct inode *inode;
524272eda82SRichard Weinberger 
525272eda82SRichard Weinberger 	inode = ilookup(c->vfs_sb, xattr_inum);
526272eda82SRichard Weinberger 	if (inode) {
527272eda82SRichard Weinberger 		clear_nlink(inode);
528272eda82SRichard Weinberger 		iput(inode);
529272eda82SRichard Weinberger 	}
530272eda82SRichard Weinberger }
531272eda82SRichard Weinberger 
532ade46c3aSRichard Weinberger static int ubifs_xattr_remove(struct inode *host, const char *name)
5331e51764aSArtem Bityutskiy {
5342b88fc21SAndreas Gruenbacher 	struct inode *inode;
5351e51764aSArtem Bityutskiy 	struct ubifs_info *c = host->i_sb->s_fs_info;
536f4f61d2cSRichard Weinberger 	struct fscrypt_name nm = { .disk_name = FSTR_INIT((char *)name, strlen(name))};
5371e51764aSArtem Bityutskiy 	struct ubifs_dent_node *xent;
5381e51764aSArtem Bityutskiy 	union ubifs_key key;
5391e51764aSArtem Bityutskiy 	int err;
5401e51764aSArtem Bityutskiy 
5415955102cSAl Viro 	ubifs_assert(inode_is_locked(host));
5421e51764aSArtem Bityutskiy 
543f4f61d2cSRichard Weinberger 	if (fname_len(&nm) > UBIFS_MAX_NLEN)
5442b88fc21SAndreas Gruenbacher 		return -ENAMETOOLONG;
5451e51764aSArtem Bityutskiy 
5461e51764aSArtem Bityutskiy 	xent = kmalloc(UBIFS_MAX_XENT_NODE_SZ, GFP_NOFS);
5471e51764aSArtem Bityutskiy 	if (!xent)
5481e51764aSArtem Bityutskiy 		return -ENOMEM;
5491e51764aSArtem Bityutskiy 
5501e51764aSArtem Bityutskiy 	xent_key_init(c, &key, host->i_ino, &nm);
5511e51764aSArtem Bityutskiy 	err = ubifs_tnc_lookup_nm(c, &key, xent, &nm);
5521e51764aSArtem Bityutskiy 	if (err) {
5531e51764aSArtem Bityutskiy 		if (err == -ENOENT)
5541e51764aSArtem Bityutskiy 			err = -ENODATA;
5551e51764aSArtem Bityutskiy 		goto out_free;
5561e51764aSArtem Bityutskiy 	}
5571e51764aSArtem Bityutskiy 
5581e51764aSArtem Bityutskiy 	inode = iget_xattr(c, le64_to_cpu(xent->inum));
5591e51764aSArtem Bityutskiy 	if (IS_ERR(inode)) {
5601e51764aSArtem Bityutskiy 		err = PTR_ERR(inode);
5611e51764aSArtem Bityutskiy 		goto out_free;
5621e51764aSArtem Bityutskiy 	}
5631e51764aSArtem Bityutskiy 
5641e51764aSArtem Bityutskiy 	ubifs_assert(inode->i_nlink == 1);
5656d6b77f1SMiklos Szeredi 	clear_nlink(inode);
5661e51764aSArtem Bityutskiy 	err = remove_xattr(c, host, inode, &nm);
5671e51764aSArtem Bityutskiy 	if (err)
568bfe86848SMiklos Szeredi 		set_nlink(inode, 1);
5691e51764aSArtem Bityutskiy 
5701e51764aSArtem Bityutskiy 	/* If @i_nlink is 0, 'iput()' will delete the inode */
5711e51764aSArtem Bityutskiy 	iput(inode);
5721e51764aSArtem Bityutskiy 
5731e51764aSArtem Bityutskiy out_free:
5741e51764aSArtem Bityutskiy 	kfree(xent);
5751e51764aSArtem Bityutskiy 	return err;
5761e51764aSArtem Bityutskiy }
577d7f0b70dSSubodh Nijsure 
5788326c1eeSHyunchul Lee #ifdef CONFIG_UBIFS_FS_SECURITY
579d7f0b70dSSubodh Nijsure static int init_xattrs(struct inode *inode, const struct xattr *xattr_array,
580d7f0b70dSSubodh Nijsure 		      void *fs_info)
581d7f0b70dSSubodh Nijsure {
582d7f0b70dSSubodh Nijsure 	const struct xattr *xattr;
583d7f0b70dSSubodh Nijsure 	char *name;
584d7f0b70dSSubodh Nijsure 	int err = 0;
585d7f0b70dSSubodh Nijsure 
586d7f0b70dSSubodh Nijsure 	for (xattr = xattr_array; xattr->name != NULL; xattr++) {
587d7f0b70dSSubodh Nijsure 		name = kmalloc(XATTR_SECURITY_PREFIX_LEN +
588d7f0b70dSSubodh Nijsure 			       strlen(xattr->name) + 1, GFP_NOFS);
589d7f0b70dSSubodh Nijsure 		if (!name) {
590d7f0b70dSSubodh Nijsure 			err = -ENOMEM;
591d7f0b70dSSubodh Nijsure 			break;
592d7f0b70dSSubodh Nijsure 		}
593d7f0b70dSSubodh Nijsure 		strcpy(name, XATTR_SECURITY_PREFIX);
594d7f0b70dSSubodh Nijsure 		strcpy(name + XATTR_SECURITY_PREFIX_LEN, xattr->name);
595d8db5b1cSXiaolei Li 		/*
596d8db5b1cSXiaolei Li 		 * creating a new inode without holding the inode rwsem,
597d8db5b1cSXiaolei Li 		 * no need to check whether inode is locked.
598d8db5b1cSXiaolei Li 		 */
599ade46c3aSRichard Weinberger 		err = ubifs_xattr_set(inode, name, xattr->value,
600d8db5b1cSXiaolei Li 				      xattr->value_len, 0, false);
601d7f0b70dSSubodh Nijsure 		kfree(name);
602d7f0b70dSSubodh Nijsure 		if (err < 0)
603d7f0b70dSSubodh Nijsure 			break;
604d7f0b70dSSubodh Nijsure 	}
605d7f0b70dSSubodh Nijsure 
606d7f0b70dSSubodh Nijsure 	return err;
607d7f0b70dSSubodh Nijsure }
608d7f0b70dSSubodh Nijsure 
609d7f0b70dSSubodh Nijsure int ubifs_init_security(struct inode *dentry, struct inode *inode,
610d7f0b70dSSubodh Nijsure 			const struct qstr *qstr)
611d7f0b70dSSubodh Nijsure {
612d7f0b70dSSubodh Nijsure 	int err;
613d7f0b70dSSubodh Nijsure 
614d7f0b70dSSubodh Nijsure 	err = security_inode_init_security(inode, dentry, qstr,
615d7f0b70dSSubodh Nijsure 					   &init_xattrs, 0);
616235c362bSSheng Yong 	if (err) {
617235c362bSSheng Yong 		struct ubifs_info *c = dentry->i_sb->s_fs_info;
618235c362bSSheng Yong 		ubifs_err(c, "cannot initialize security for inode %lu, error %d",
619fee1756dSSubodh Nijsure 			  inode->i_ino, err);
620235c362bSSheng Yong 	}
621d7f0b70dSSubodh Nijsure 	return err;
622d7f0b70dSSubodh Nijsure }
6238326c1eeSHyunchul Lee #endif
6242b88fc21SAndreas Gruenbacher 
625ade46c3aSRichard Weinberger static int xattr_get(const struct xattr_handler *handler,
6262b88fc21SAndreas Gruenbacher 			   struct dentry *dentry, struct inode *inode,
6272b88fc21SAndreas Gruenbacher 			   const char *name, void *buffer, size_t size)
6282b88fc21SAndreas Gruenbacher {
6292b88fc21SAndreas Gruenbacher 	dbg_gen("xattr '%s', ino %lu ('%pd'), buf size %zd", name,
6302b88fc21SAndreas Gruenbacher 		inode->i_ino, dentry, size);
6312b88fc21SAndreas Gruenbacher 
63217ce1eb0SRichard Weinberger 	name = xattr_full_name(handler, name);
633ade46c3aSRichard Weinberger 	return ubifs_xattr_get(inode, name, buffer, size);
6342b88fc21SAndreas Gruenbacher }
6352b88fc21SAndreas Gruenbacher 
636ade46c3aSRichard Weinberger static int xattr_set(const struct xattr_handler *handler,
63759301226SAl Viro 			   struct dentry *dentry, struct inode *inode,
63859301226SAl Viro 			   const char *name, const void *value,
63959301226SAl Viro 			   size_t size, int flags)
6402b88fc21SAndreas Gruenbacher {
6412b88fc21SAndreas Gruenbacher 	dbg_gen("xattr '%s', host ino %lu ('%pd'), size %zd",
6422b88fc21SAndreas Gruenbacher 		name, inode->i_ino, dentry, size);
6432b88fc21SAndreas Gruenbacher 
64417ce1eb0SRichard Weinberger 	name = xattr_full_name(handler, name);
64517ce1eb0SRichard Weinberger 
6462b88fc21SAndreas Gruenbacher 	if (value)
647d8db5b1cSXiaolei Li 		return ubifs_xattr_set(inode, name, value, size, flags, true);
6482b88fc21SAndreas Gruenbacher 	else
649ade46c3aSRichard Weinberger 		return ubifs_xattr_remove(inode, name);
6502b88fc21SAndreas Gruenbacher }
6512b88fc21SAndreas Gruenbacher 
652dfaf8d2aSBen Dooks static const struct xattr_handler ubifs_user_xattr_handler = {
6532b88fc21SAndreas Gruenbacher 	.prefix = XATTR_USER_PREFIX,
654ade46c3aSRichard Weinberger 	.get = xattr_get,
655ade46c3aSRichard Weinberger 	.set = xattr_set,
6562b88fc21SAndreas Gruenbacher };
6572b88fc21SAndreas Gruenbacher 
658dfaf8d2aSBen Dooks static const struct xattr_handler ubifs_trusted_xattr_handler = {
6592b88fc21SAndreas Gruenbacher 	.prefix = XATTR_TRUSTED_PREFIX,
660ade46c3aSRichard Weinberger 	.get = xattr_get,
661ade46c3aSRichard Weinberger 	.set = xattr_set,
6622b88fc21SAndreas Gruenbacher };
6632b88fc21SAndreas Gruenbacher 
6648326c1eeSHyunchul Lee #ifdef CONFIG_UBIFS_FS_SECURITY
665dfaf8d2aSBen Dooks static const struct xattr_handler ubifs_security_xattr_handler = {
6662b88fc21SAndreas Gruenbacher 	.prefix = XATTR_SECURITY_PREFIX,
667ade46c3aSRichard Weinberger 	.get = xattr_get,
668ade46c3aSRichard Weinberger 	.set = xattr_set,
6692b88fc21SAndreas Gruenbacher };
6708326c1eeSHyunchul Lee #endif
6712b88fc21SAndreas Gruenbacher 
6722b88fc21SAndreas Gruenbacher const struct xattr_handler *ubifs_xattr_handlers[] = {
6732b88fc21SAndreas Gruenbacher 	&ubifs_user_xattr_handler,
6742b88fc21SAndreas Gruenbacher 	&ubifs_trusted_xattr_handler,
6758326c1eeSHyunchul Lee #ifdef CONFIG_UBIFS_FS_SECURITY
6762b88fc21SAndreas Gruenbacher 	&ubifs_security_xattr_handler,
6778326c1eeSHyunchul Lee #endif
6782b88fc21SAndreas Gruenbacher 	NULL
6792b88fc21SAndreas Gruenbacher };
680