Commit c27411d4 authored by Darrick J. Wong's avatar Darrick J. Wong

xfs: make attr removal an explicit operation

Parent pointers match attrs on name+value, unlike everything else which
matches on only the name.  Therefore, we cannot keep using the heuristic
that !value means remove.  Make this an explicit operation code.
Signed-off-by: default avatarDarrick J. Wong <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
parent 54275d84
...@@ -916,10 +916,6 @@ xfs_attr_defer_add( ...@@ -916,10 +916,6 @@ xfs_attr_defer_add(
trace_xfs_attr_defer_add(new->xattri_dela_state, args->dp); trace_xfs_attr_defer_add(new->xattri_dela_state, args->dp);
} }
/*
* Note: If args->value is NULL the attribute will be removed, just like the
* Linux ->setattr API.
*/
int int
xfs_attr_set( xfs_attr_set(
struct xfs_da_args *args, struct xfs_da_args *args,
...@@ -955,7 +951,10 @@ xfs_attr_set( ...@@ -955,7 +951,10 @@ xfs_attr_set(
args->op_flags = XFS_DA_OP_OKNOENT | args->op_flags = XFS_DA_OP_OKNOENT |
(args->op_flags & XFS_DA_OP_LOGGED); (args->op_flags & XFS_DA_OP_LOGGED);
if (args->value) { switch (op) {
case XFS_ATTRUPDATE_UPSERT:
case XFS_ATTRUPDATE_CREATE:
case XFS_ATTRUPDATE_REPLACE:
XFS_STATS_INC(mp, xs_attr_set); XFS_STATS_INC(mp, xs_attr_set);
args->total = xfs_attr_calc_size(args, &local); args->total = xfs_attr_calc_size(args, &local);
...@@ -975,9 +974,11 @@ xfs_attr_set( ...@@ -975,9 +974,11 @@ xfs_attr_set(
if (!local) if (!local)
rmt_blks = xfs_attr3_rmt_blocks(mp, args->valuelen); rmt_blks = xfs_attr3_rmt_blocks(mp, args->valuelen);
} else { break;
case XFS_ATTRUPDATE_REMOVE:
XFS_STATS_INC(mp, xs_attr_remove); XFS_STATS_INC(mp, xs_attr_remove);
rmt_blks = xfs_attr3_rmt_blocks(mp, XFS_XATTR_SIZE_MAX); rmt_blks = xfs_attr3_rmt_blocks(mp, XFS_XATTR_SIZE_MAX);
break;
} }
/* /*
...@@ -989,7 +990,7 @@ xfs_attr_set( ...@@ -989,7 +990,7 @@ xfs_attr_set(
if (error) if (error)
return error; return error;
if (args->value || xfs_inode_hasattr(dp)) { if (op != XFS_ATTRUPDATE_REMOVE || xfs_inode_hasattr(dp)) {
error = xfs_iext_count_may_overflow(dp, XFS_ATTR_FORK, error = xfs_iext_count_may_overflow(dp, XFS_ATTR_FORK,
XFS_IEXT_ATTR_MANIP_CNT(rmt_blks)); XFS_IEXT_ATTR_MANIP_CNT(rmt_blks));
if (error == -EFBIG) if (error == -EFBIG)
...@@ -1002,7 +1003,7 @@ xfs_attr_set( ...@@ -1002,7 +1003,7 @@ xfs_attr_set(
error = xfs_attr_lookup(args); error = xfs_attr_lookup(args);
switch (error) { switch (error) {
case -EEXIST: case -EEXIST:
if (!args->value) { if (op == XFS_ATTRUPDATE_REMOVE) {
/* if no value, we are performing a remove operation */ /* if no value, we are performing a remove operation */
xfs_attr_defer_add(args, XFS_ATTRI_OP_FLAGS_REMOVE); xfs_attr_defer_add(args, XFS_ATTRI_OP_FLAGS_REMOVE);
break; break;
...@@ -1015,7 +1016,7 @@ xfs_attr_set( ...@@ -1015,7 +1016,7 @@ xfs_attr_set(
break; break;
case -ENOATTR: case -ENOATTR:
/* Can't remove what isn't there. */ /* Can't remove what isn't there. */
if (!args->value) if (op == XFS_ATTRUPDATE_REMOVE)
goto out_trans_cancel; goto out_trans_cancel;
/* Pure replace fails if no existing attr to replace. */ /* Pure replace fails if no existing attr to replace. */
......
...@@ -546,7 +546,8 @@ int xfs_attr_get_ilocked(struct xfs_da_args *args); ...@@ -546,7 +546,8 @@ int xfs_attr_get_ilocked(struct xfs_da_args *args);
int xfs_attr_get(struct xfs_da_args *args); int xfs_attr_get(struct xfs_da_args *args);
enum xfs_attr_update { enum xfs_attr_update {
XFS_ATTRUPDATE_UPSERTR, /* set/remove value, replace any existing attr */ XFS_ATTRUPDATE_REMOVE, /* remove attr */
XFS_ATTRUPDATE_UPSERT, /* set value, replace any existing attr */
XFS_ATTRUPDATE_CREATE, /* set value, fail if attr already exists */ XFS_ATTRUPDATE_CREATE, /* set value, fail if attr already exists */
XFS_ATTRUPDATE_REPLACE, /* set value, fail if attr does not exist */ XFS_ATTRUPDATE_REPLACE, /* set value, fail if attr does not exist */
}; };
......
...@@ -201,16 +201,17 @@ __xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type) ...@@ -201,16 +201,17 @@ __xfs_set_acl(struct inode *inode, struct posix_acl *acl, int type)
if (!args.value) if (!args.value)
return -ENOMEM; return -ENOMEM;
xfs_acl_to_disk(args.value, acl); xfs_acl_to_disk(args.value, acl);
} error = xfs_attr_change(&args, XFS_ATTRUPDATE_UPSERT);
error = xfs_attr_change(&args, XFS_ATTRUPDATE_UPSERTR);
kvfree(args.value); kvfree(args.value);
} else {
error = xfs_attr_change(&args, XFS_ATTRUPDATE_REMOVE);
/* /*
* If the attribute didn't exist to start with that's fine. * If the attribute didn't exist to start with that's fine.
*/ */
if (!acl && error == -ENOATTR) if (error == -ENOATTR)
error = 0; error = 0;
}
if (!error) if (!error)
set_cached_acl(inode, type, acl); set_cached_acl(inode, type, acl);
return error; return error;
......
...@@ -363,13 +363,16 @@ xfs_attr_filter( ...@@ -363,13 +363,16 @@ xfs_attr_filter(
static inline enum xfs_attr_update static inline enum xfs_attr_update
xfs_xattr_flags( xfs_xattr_flags(
u32 ioc_flags) u32 ioc_flags,
void *value)
{ {
if (!value)
return XFS_ATTRUPDATE_REMOVE;
if (ioc_flags & XFS_IOC_ATTR_CREATE) if (ioc_flags & XFS_IOC_ATTR_CREATE)
return XFS_ATTRUPDATE_CREATE; return XFS_ATTRUPDATE_CREATE;
if (ioc_flags & XFS_IOC_ATTR_REPLACE) if (ioc_flags & XFS_IOC_ATTR_REPLACE)
return XFS_ATTRUPDATE_REPLACE; return XFS_ATTRUPDATE_REPLACE;
return XFS_ATTRUPDATE_UPSERTR; return XFS_ATTRUPDATE_UPSERT;
} }
int int
...@@ -526,7 +529,7 @@ xfs_attrmulti_attr_set( ...@@ -526,7 +529,7 @@ xfs_attrmulti_attr_set(
args.valuelen = len; args.valuelen = len;
} }
error = xfs_attr_change(&args, xfs_xattr_flags(flags)); error = xfs_attr_change(&args, xfs_xattr_flags(flags, args.value));
if (!error && (flags & XFS_IOC_ATTR_ROOT)) if (!error && (flags & XFS_IOC_ATTR_ROOT))
xfs_forget_acl(inode, name); xfs_forget_acl(inode, name);
kfree(args.value); kfree(args.value);
......
...@@ -63,7 +63,7 @@ xfs_initxattrs( ...@@ -63,7 +63,7 @@ xfs_initxattrs(
.value = xattr->value, .value = xattr->value,
.valuelen = xattr->value_len, .valuelen = xattr->value_len,
}; };
error = xfs_attr_change(&args, XFS_ATTRUPDATE_UPSERTR); error = xfs_attr_change(&args, XFS_ATTRUPDATE_UPSERT);
if (error < 0) if (error < 0)
break; break;
} }
......
...@@ -118,13 +118,16 @@ xfs_xattr_get(const struct xattr_handler *handler, struct dentry *unused, ...@@ -118,13 +118,16 @@ xfs_xattr_get(const struct xattr_handler *handler, struct dentry *unused,
static inline enum xfs_attr_update static inline enum xfs_attr_update
xfs_xattr_flags_to_op( xfs_xattr_flags_to_op(
int flags) int flags,
const void *value)
{ {
if (!value)
return XFS_ATTRUPDATE_REMOVE;
if (flags & XATTR_CREATE) if (flags & XATTR_CREATE)
return XFS_ATTRUPDATE_CREATE; return XFS_ATTRUPDATE_CREATE;
if (flags & XATTR_REPLACE) if (flags & XATTR_REPLACE)
return XFS_ATTRUPDATE_REPLACE; return XFS_ATTRUPDATE_REPLACE;
return XFS_ATTRUPDATE_UPSERTR; return XFS_ATTRUPDATE_UPSERT;
} }
static int static int
...@@ -143,7 +146,7 @@ xfs_xattr_set(const struct xattr_handler *handler, ...@@ -143,7 +146,7 @@ xfs_xattr_set(const struct xattr_handler *handler,
}; };
int error; int error;
error = xfs_attr_change(&args, xfs_xattr_flags_to_op(flags)); error = xfs_attr_change(&args, xfs_xattr_flags_to_op(flags, value));
if (!error && (handler->flags & XFS_ATTR_ROOT)) if (!error && (handler->flags & XFS_ATTR_ROOT))
xfs_forget_acl(inode, name); xfs_forget_acl(inode, name);
return error; return error;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment