Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 264479
b: refs/heads/master
c: 1601fba
h: refs/heads/master
i:
  264477: e573483
  264475: 00dd51e
  264471: 140cdca
  264463: c83c7ea
  264447: fdfc63b
v: v3
  • Loading branch information
Mimi Zohar committed Jul 18, 2011
1 parent 4a2ac8e commit db9f3de
Show file tree
Hide file tree
Showing 3 changed files with 63 additions and 2 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: f381c272224f5f158f5cff64f8f3481fa0eee8b3
refs/heads/master: 1601fbad2b14e0b8d4dbb55e749bfe31e972818a
58 changes: 58 additions & 0 deletions trunk/fs/xattr.c
Original file line number Diff line number Diff line change
Expand Up @@ -166,6 +166,64 @@ xattr_getsecurity(struct inode *inode, const char *name, void *value,
}
EXPORT_SYMBOL_GPL(xattr_getsecurity);

/*
* vfs_getxattr_alloc - allocate memory, if necessary, before calling getxattr
*
* Allocate memory, if not already allocated, or re-allocate correct size,
* before retrieving the extended attribute.
*
* Returns the result of alloc, if failed, or the getxattr operation.
*/
ssize_t
vfs_getxattr_alloc(struct dentry *dentry, const char *name, char **xattr_value,
size_t xattr_size, gfp_t flags)
{
struct inode *inode = dentry->d_inode;
char *value = *xattr_value;
int error;

error = xattr_permission(inode, name, MAY_READ);
if (error)
return error;

if (!inode->i_op->getxattr)
return -EOPNOTSUPP;

error = inode->i_op->getxattr(dentry, name, NULL, 0);
if (error < 0)
return error;

if (!value || (error > xattr_size)) {
value = krealloc(*xattr_value, error + 1, flags);
if (!value)
return -ENOMEM;
memset(value, 0, error + 1);
}

error = inode->i_op->getxattr(dentry, name, value, error);
*xattr_value = value;
return error;
}

/* Compare an extended attribute value with the given value */
int vfs_xattr_cmp(struct dentry *dentry, const char *xattr_name,
const char *value, size_t size, gfp_t flags)
{
char *xattr_value = NULL;
int rc;

rc = vfs_getxattr_alloc(dentry, xattr_name, &xattr_value, 0, flags);
if (rc < 0)
return rc;

if ((rc != size) || (memcmp(xattr_value, value, rc) != 0))
rc = -EINVAL;
else
rc = 0;
kfree(xattr_value);
return rc;
}

ssize_t
vfs_getxattr(struct dentry *dentry, const char *name, void *value, size_t size)
{
Expand Down
5 changes: 4 additions & 1 deletion trunk/include/linux/xattr.h
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,10 @@ ssize_t generic_getxattr(struct dentry *dentry, const char *name, void *buffer,
ssize_t generic_listxattr(struct dentry *dentry, char *buffer, size_t buffer_size);
int generic_setxattr(struct dentry *dentry, const char *name, const void *value, size_t size, int flags);
int generic_removexattr(struct dentry *dentry, const char *name);

ssize_t vfs_getxattr_alloc(struct dentry *dentry, const char *name,
char **xattr_value, size_t size, gfp_t flags);
int vfs_xattr_cmp(struct dentry *dentry, const char *xattr_name,
const char *value, size_t size, gfp_t flags);
#endif /* __KERNEL__ */

#endif /* _LINUX_XATTR_H */

0 comments on commit db9f3de

Please sign in to comment.