2005-04-16 15:20:36 -07:00
/*
File : fs / xattr . c
Extended attribute handling .
Copyright ( C ) 2001 by Andreas Gruenbacher < a . gruenbacher @ computer . org >
Copyright ( C ) 2001 SGI - Silicon Graphics , Inc < linux - xfs @ oss . sgi . com >
Copyright ( c ) 2004 Red Hat , Inc . , James Morris < jmorris @ redhat . com >
*/
# include <linux/fs.h>
# include <linux/slab.h>
# include <linux/file.h>
# include <linux/xattr.h>
2008-02-15 14:37:38 -08:00
# include <linux/mount.h>
2005-04-16 15:20:36 -07:00
# include <linux/namei.h>
# include <linux/security.h>
2011-03-09 14:39:18 -05:00
# include <linux/evm.h>
2005-04-16 15:20:36 -07:00
# include <linux/syscalls.h>
2011-11-16 23:57:37 -05:00
# include <linux/export.h>
[PATCH] inotify
inotify is intended to correct the deficiencies of dnotify, particularly
its inability to scale and its terrible user interface:
* dnotify requires the opening of one fd per each directory
that you intend to watch. This quickly results in too many
open files and pins removable media, preventing unmount.
* dnotify is directory-based. You only learn about changes to
directories. Sure, a change to a file in a directory affects
the directory, but you are then forced to keep a cache of
stat structures.
* dnotify's interface to user-space is awful. Signals?
inotify provides a more usable, simple, powerful solution to file change
notification:
* inotify's interface is a system call that returns a fd, not SIGIO.
You get a single fd, which is select()-able.
* inotify has an event that says "the filesystem that the item
you were watching is on was unmounted."
* inotify can watch directories or files.
Inotify is currently used by Beagle (a desktop search infrastructure),
Gamin (a FAM replacement), and other projects.
See Documentation/filesystems/inotify.txt.
Signed-off-by: Robert Love <rml@novell.com>
Cc: John McCutchan <ttb@tentacle.dhs.org>
Cc: Christoph Hellwig <hch@lst.de>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-07-12 17:06:03 -04:00
# include <linux/fsnotify.h>
2005-11-03 16:00:25 +00:00
# include <linux/audit.h>
2012-04-05 14:25:07 -07:00
# include <linux/vmalloc.h>
2012-02-07 18:52:57 -08:00
# include <linux/posix_acl_xattr.h>
2005-04-16 15:20:36 -07:00
2012-04-05 14:25:07 -07:00
# include <asm/uaccess.h>
2006-01-09 20:51:55 -08:00
2006-01-09 20:51:56 -08:00
/*
* Check permissions for extended attribute access . This is a bit complicated
* because different namespaces have very different rules .
*/
static int
xattr_permission ( struct inode * inode , const char * name , int mask )
{
/*
* We can never set or remove an extended attribute on a read - only
* filesystem or on an immutable / append - only inode .
*/
if ( mask & MAY_WRITE ) {
if ( IS_IMMUTABLE ( inode ) | | IS_APPEND ( inode ) )
return - EPERM ;
}
/*
* No restriction for security . * and system . * from the VFS . Decision
* on these is left to the underlying filesystem / security module .
*/
if ( ! strncmp ( name , XATTR_SECURITY_PREFIX , XATTR_SECURITY_PREFIX_LEN ) | |
! strncmp ( name , XATTR_SYSTEM_PREFIX , XATTR_SYSTEM_PREFIX_LEN ) )
return 0 ;
/*
2011-05-27 14:50:36 +02:00
* The trusted . * namespace can only be accessed by privileged users .
2006-01-09 20:51:56 -08:00
*/
2011-05-27 14:50:36 +02:00
if ( ! strncmp ( name , XATTR_TRUSTED_PREFIX , XATTR_TRUSTED_PREFIX_LEN ) ) {
if ( ! capable ( CAP_SYS_ADMIN ) )
return ( mask & MAY_WRITE ) ? - EPERM : - ENODATA ;
return 0 ;
}
2006-01-09 20:51:56 -08:00
2011-05-27 14:50:36 +02:00
/*
* In the user . * namespace , only regular files and directories can have
2006-11-02 22:07:29 -08:00
* extended attributes . For sticky directories , only the owner and
2011-05-27 14:50:36 +02:00
* privileged users can write attributes .
2006-11-02 22:07:29 -08:00
*/
2006-01-09 20:51:56 -08:00
if ( ! strncmp ( name , XATTR_USER_PREFIX , XATTR_USER_PREFIX_LEN ) ) {
2006-11-02 22:07:29 -08:00
if ( ! S_ISREG ( inode - > i_mode ) & & ! S_ISDIR ( inode - > i_mode ) )
2011-05-27 14:50:36 +02:00
return ( mask & MAY_WRITE ) ? - EPERM : - ENODATA ;
2006-11-02 22:07:29 -08:00
if ( S_ISDIR ( inode - > i_mode ) & & ( inode - > i_mode & S_ISVTX ) & &
2011-03-23 16:43:26 -07:00
( mask & MAY_WRITE ) & & ! inode_owner_or_capable ( inode ) )
2006-01-09 20:51:56 -08:00
return - EPERM ;
}
2008-07-22 00:07:17 -04:00
return inode_permission ( inode , mask ) ;
2006-01-09 20:51:56 -08:00
}
2009-09-03 14:25:56 -04:00
/**
* __vfs_setxattr_noperm - perform setxattr operation without performing
* permission checks .
*
* @ dentry - object to perform setxattr on
* @ name - xattr name to set
* @ value - value to set @ name to
* @ size - size of @ value
* @ flags - flags to pass into filesystem operations
*
* returns the result of the internal setxattr or setsecurity operations .
*
* This function requires the caller to lock the inode ' s i_mutex before it
* is executed . It also assumes that the caller will make the appropriate
* permission checks .
*/
int __vfs_setxattr_noperm ( struct dentry * dentry , const char * name ,
const void * value , size_t size , int flags )
2006-01-09 20:51:55 -08:00
{
struct inode * inode = dentry - > d_inode ;
2009-09-03 14:25:56 -04:00
int error = - EOPNOTSUPP ;
2011-05-28 08:25:51 -07:00
int issec = ! strncmp ( name , XATTR_SECURITY_PREFIX ,
XATTR_SECURITY_PREFIX_LEN ) ;
2006-01-09 20:51:56 -08:00
2011-05-28 08:25:51 -07:00
if ( issec )
inode - > i_flags & = ~ S_NOSEC ;
2006-01-09 20:51:55 -08:00
if ( inode - > i_op - > setxattr ) {
error = inode - > i_op - > setxattr ( dentry , name , value , size , flags ) ;
if ( ! error ) {
fsnotify_xattr ( dentry ) ;
security_inode_post_setxattr ( dentry , name , value ,
size , flags ) ;
}
2011-05-28 08:25:51 -07:00
} else if ( issec ) {
2006-01-09 20:51:56 -08:00
const char * suffix = name + XATTR_SECURITY_PREFIX_LEN ;
2006-01-09 20:51:55 -08:00
error = security_inode_setsecurity ( inode , suffix , value ,
size , flags ) ;
if ( ! error )
fsnotify_xattr ( dentry ) ;
}
2009-09-03 14:25:56 -04:00
return error ;
}
int
vfs_setxattr ( struct dentry * dentry , const char * name , const void * value ,
size_t size , int flags )
{
struct inode * inode = dentry - > d_inode ;
int error ;
error = xattr_permission ( inode , name , MAY_WRITE ) ;
if ( error )
return error ;
2016-01-22 15:40:57 -05:00
inode_lock ( inode ) ;
2009-09-03 14:25:56 -04:00
error = security_inode_setxattr ( dentry , name , value , size , flags ) ;
if ( error )
goto out ;
error = __vfs_setxattr_noperm ( dentry , name , value , size , flags ) ;
2006-01-09 20:51:55 -08:00
out :
2016-01-22 15:40:57 -05:00
inode_unlock ( inode ) ;
2006-01-09 20:51:55 -08:00
return error ;
}
EXPORT_SYMBOL_GPL ( vfs_setxattr ) ;
2008-02-04 22:29:39 -08:00
ssize_t
xattr_getsecurity ( struct inode * inode , const char * name , void * value ,
size_t size )
{
void * buffer = NULL ;
ssize_t len ;
if ( ! value | | ! size ) {
len = security_inode_getsecurity ( inode , name , & buffer , false ) ;
goto out_noalloc ;
}
len = security_inode_getsecurity ( inode , name , & buffer , true ) ;
if ( len < 0 )
return len ;
if ( size < len ) {
len = - ERANGE ;
goto out ;
}
memcpy ( value , buffer , len ) ;
out :
security_release_secctx ( buffer , len ) ;
out_noalloc :
return len ;
}
EXPORT_SYMBOL_GPL ( xattr_getsecurity ) ;
2011-03-09 14:23:34 -05:00
/*
* 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 ;
2016-04-11 00:48:00 -04:00
error = inode - > i_op - > getxattr ( dentry , inode , name , NULL , 0 ) ;
2011-03-09 14:23:34 -05:00
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 ) ;
}
2016-04-11 00:48:00 -04:00
error = inode - > i_op - > getxattr ( dentry , inode , name , value , error ) ;
2011-03-09 14:23:34 -05:00
* xattr_value = value ;
return error ;
}
2006-01-09 20:51:55 -08:00
ssize_t
2008-04-29 00:59:41 -07:00
vfs_getxattr ( struct dentry * dentry , const char * name , void * value , size_t size )
2006-01-09 20:51:55 -08:00
{
struct inode * inode = dentry - > d_inode ;
int error ;
2006-01-09 20:51:56 -08:00
error = xattr_permission ( inode , name , MAY_READ ) ;
if ( error )
return error ;
2006-01-09 20:51:55 -08:00
error = security_inode_getxattr ( dentry , name ) ;
if ( error )
return error ;
if ( ! strncmp ( name , XATTR_SECURITY_PREFIX ,
2006-01-09 20:51:56 -08:00
XATTR_SECURITY_PREFIX_LEN ) ) {
const char * suffix = name + XATTR_SECURITY_PREFIX_LEN ;
2008-02-04 22:29:39 -08:00
int ret = xattr_getsecurity ( inode , suffix , value , size ) ;
2006-01-09 20:51:55 -08:00
/*
* Only overwrite the return value if a security module
* is actually active .
*/
2008-02-04 22:29:40 -08:00
if ( ret = = - EOPNOTSUPP )
goto nolsm ;
return ret ;
2006-01-09 20:51:55 -08:00
}
2008-02-04 22:29:40 -08:00
nolsm :
if ( inode - > i_op - > getxattr )
2016-04-11 00:48:00 -04:00
error = inode - > i_op - > getxattr ( dentry , inode , name , value , size ) ;
2008-02-04 22:29:40 -08:00
else
error = - EOPNOTSUPP ;
2006-01-09 20:51:55 -08:00
return error ;
}
EXPORT_SYMBOL_GPL ( vfs_getxattr ) ;
2006-10-09 16:10:48 -04:00
ssize_t
vfs_listxattr ( struct dentry * d , char * list , size_t size )
{
ssize_t error ;
error = security_inode_listxattr ( d ) ;
if ( error )
return error ;
error = - EOPNOTSUPP ;
2008-12-04 10:06:33 -05:00
if ( d - > d_inode - > i_op - > listxattr ) {
2006-10-09 16:10:48 -04:00
error = d - > d_inode - > i_op - > listxattr ( d , list , size ) ;
} else {
error = security_inode_listsecurity ( d - > d_inode , list , size ) ;
if ( size & & error > size )
error = - ERANGE ;
}
return error ;
}
EXPORT_SYMBOL_GPL ( vfs_listxattr ) ;
2006-01-09 20:51:55 -08:00
int
2008-04-29 00:59:41 -07:00
vfs_removexattr ( struct dentry * dentry , const char * name )
2006-01-09 20:51:55 -08:00
{
struct inode * inode = dentry - > d_inode ;
int error ;
if ( ! inode - > i_op - > removexattr )
return - EOPNOTSUPP ;
2006-01-09 20:51:56 -08:00
error = xattr_permission ( inode , name , MAY_WRITE ) ;
if ( error )
return error ;
2016-01-22 15:40:57 -05:00
inode_lock ( inode ) ;
2006-01-09 20:51:55 -08:00
error = security_inode_removexattr ( dentry , name ) ;
2014-11-20 16:31:01 +02:00
if ( error )
goto out ;
2006-01-09 20:51:55 -08:00
error = inode - > i_op - > removexattr ( dentry , name ) ;
2011-03-09 14:39:18 -05:00
if ( ! error ) {
2006-01-09 20:51:55 -08:00
fsnotify_xattr ( dentry ) ;
2011-03-09 14:39:18 -05:00
evm_inode_post_removexattr ( dentry , name ) ;
}
2014-11-20 16:31:01 +02:00
out :
2016-01-22 15:40:57 -05:00
inode_unlock ( inode ) ;
2006-01-09 20:51:55 -08:00
return error ;
}
EXPORT_SYMBOL_GPL ( vfs_removexattr ) ;
2005-04-16 15:20:36 -07:00
/*
* Extended attribute SET operations
*/
static long
2008-04-29 00:59:41 -07:00
setxattr ( struct dentry * d , const char __user * name , const void __user * value ,
2005-04-16 15:20:36 -07:00
size_t size , int flags )
{
int error ;
void * kvalue = NULL ;
char kname [ XATTR_NAME_MAX + 1 ] ;
if ( flags & ~ ( XATTR_CREATE | XATTR_REPLACE ) )
return - EINVAL ;
error = strncpy_from_user ( kname , name , sizeof ( kname ) ) ;
if ( error = = 0 | | error = = sizeof ( kname ) )
error = - ERANGE ;
if ( error < 0 )
return error ;
if ( size ) {
if ( size > XATTR_SIZE_MAX )
return - E2BIG ;
2012-04-05 14:25:07 -07:00
kvalue = kmalloc ( size , GFP_KERNEL | __GFP_NOWARN ) ;
if ( ! kvalue ) {
2016-01-02 23:09:47 +01:00
kvalue = vmalloc ( size ) ;
if ( ! kvalue )
2012-04-05 14:25:07 -07:00
return - ENOMEM ;
}
if ( copy_from_user ( kvalue , value , size ) ) {
error = - EFAULT ;
goto out ;
}
2012-02-07 18:52:57 -08:00
if ( ( strcmp ( kname , XATTR_NAME_POSIX_ACL_ACCESS ) = = 0 ) | |
( strcmp ( kname , XATTR_NAME_POSIX_ACL_DEFAULT ) = = 0 ) )
posix_acl_fix_xattr_from_user ( kvalue , size ) ;
2005-04-16 15:20:36 -07:00
}
2006-01-09 20:51:55 -08:00
error = vfs_setxattr ( d , kname , kvalue , size , flags ) ;
2012-04-05 14:25:07 -07:00
out :
2016-01-02 23:09:47 +01:00
kvfree ( kvalue ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
static int path_setxattr ( const char __user * pathname ,
const char __user * name , const void __user * value ,
size_t size , int flags , unsigned int lookup_flags )
2005-04-16 15:20:36 -07:00
{
2008-07-22 09:59:21 -04:00
struct path path ;
2005-04-16 15:20:36 -07:00
int error ;
2012-12-11 12:10:15 -05:00
retry :
error = user_path_at ( AT_FDCWD , pathname , lookup_flags , & path ) ;
2005-04-16 15:20:36 -07:00
if ( error )
return error ;
2008-07-22 09:59:21 -04:00
error = mnt_want_write ( path . mnt ) ;
2008-02-15 14:37:38 -08:00
if ( ! error ) {
2008-07-22 09:59:21 -04:00
error = setxattr ( path . dentry , name , value , size , flags ) ;
mnt_drop_write ( path . mnt ) ;
2008-02-15 14:37:38 -08:00
}
2008-07-22 09:59:21 -04:00
path_put ( & path ) ;
2012-12-11 12:10:15 -05:00
if ( retry_estale ( error , lookup_flags ) ) {
lookup_flags | = LOOKUP_REVAL ;
goto retry ;
}
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
SYSCALL_DEFINE5 ( setxattr , const char __user * , pathname ,
const char __user * , name , const void __user * , value ,
size_t , size , int , flags )
{
return path_setxattr ( pathname , name , value , size , flags , LOOKUP_FOLLOW ) ;
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE5 ( lsetxattr , const char __user * , pathname ,
const char __user * , name , const void __user * , value ,
size_t , size , int , flags )
2005-04-16 15:20:36 -07:00
{
2014-10-12 11:59:58 -05:00
return path_setxattr ( pathname , name , value , size , flags , 0 ) ;
2005-04-16 15:20:36 -07:00
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE5 ( fsetxattr , int , fd , const char __user * , name ,
const void __user * , value , size_t , size , int , flags )
2005-04-16 15:20:36 -07:00
{
2012-08-28 12:52:22 -04:00
struct fd f = fdget ( fd ) ;
2005-04-16 15:20:36 -07:00
int error = - EBADF ;
2012-08-28 12:52:22 -04:00
if ( ! f . file )
2005-04-16 15:20:36 -07:00
return error ;
2014-10-31 17:44:57 -04:00
audit_file ( f . file ) ;
2012-08-28 12:52:22 -04:00
error = mnt_want_write_file ( f . file ) ;
2008-02-15 14:37:38 -08:00
if ( ! error ) {
2014-10-31 17:44:57 -04:00
error = setxattr ( f . file - > f_path . dentry , name , value , size , flags ) ;
2012-08-28 12:52:22 -04:00
mnt_drop_write_file ( f . file ) ;
2008-02-15 14:37:38 -08:00
}
2012-08-28 12:52:22 -04:00
fdput ( f ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
/*
* Extended attribute GET operations
*/
static ssize_t
2008-04-29 00:59:41 -07:00
getxattr ( struct dentry * d , const char __user * name , void __user * value ,
size_t size )
2005-04-16 15:20:36 -07:00
{
ssize_t error ;
void * kvalue = NULL ;
char kname [ XATTR_NAME_MAX + 1 ] ;
error = strncpy_from_user ( kname , name , sizeof ( kname ) ) ;
if ( error = = 0 | | error = = sizeof ( kname ) )
error = - ERANGE ;
if ( error < 0 )
return error ;
if ( size ) {
if ( size > XATTR_SIZE_MAX )
size = XATTR_SIZE_MAX ;
2012-07-30 14:39:13 -07:00
kvalue = kzalloc ( size , GFP_KERNEL | __GFP_NOWARN ) ;
if ( ! kvalue ) {
2016-01-02 23:09:47 +01:00
kvalue = vmalloc ( size ) ;
if ( ! kvalue )
2012-07-30 14:39:13 -07:00
return - ENOMEM ;
}
2005-04-16 15:20:36 -07:00
}
2006-01-09 20:51:55 -08:00
error = vfs_getxattr ( d , kname , kvalue , size ) ;
2005-09-03 15:55:18 -07:00
if ( error > 0 ) {
2012-02-07 18:52:57 -08:00
if ( ( strcmp ( kname , XATTR_NAME_POSIX_ACL_ACCESS ) = = 0 ) | |
( strcmp ( kname , XATTR_NAME_POSIX_ACL_DEFAULT ) = = 0 ) )
posix_acl_fix_xattr_to_user ( kvalue , size ) ;
2005-09-03 15:55:18 -07:00
if ( size & & copy_to_user ( value , kvalue , error ) )
error = - EFAULT ;
} else if ( error = = - ERANGE & & size > = XATTR_SIZE_MAX ) {
/* The file system tried to returned a value bigger
than XATTR_SIZE_MAX bytes . Not possible . */
error = - E2BIG ;
2005-04-16 15:20:36 -07:00
}
2016-01-02 23:09:47 +01:00
kvfree ( kvalue ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
static ssize_t path_getxattr ( const char __user * pathname ,
const char __user * name , void __user * value ,
size_t size , unsigned int lookup_flags )
2005-04-16 15:20:36 -07:00
{
2008-07-22 09:59:21 -04:00
struct path path ;
2005-04-16 15:20:36 -07:00
ssize_t error ;
2012-12-11 12:10:16 -05:00
retry :
error = user_path_at ( AT_FDCWD , pathname , lookup_flags , & path ) ;
2005-04-16 15:20:36 -07:00
if ( error )
return error ;
2008-07-22 09:59:21 -04:00
error = getxattr ( path . dentry , name , value , size ) ;
path_put ( & path ) ;
2012-12-11 12:10:16 -05:00
if ( retry_estale ( error , lookup_flags ) ) {
lookup_flags | = LOOKUP_REVAL ;
goto retry ;
}
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
SYSCALL_DEFINE4 ( getxattr , const char __user * , pathname ,
const char __user * , name , void __user * , value , size_t , size )
{
return path_getxattr ( pathname , name , value , size , LOOKUP_FOLLOW ) ;
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE4 ( lgetxattr , const char __user * , pathname ,
const char __user * , name , void __user * , value , size_t , size )
2005-04-16 15:20:36 -07:00
{
2014-10-12 11:59:58 -05:00
return path_getxattr ( pathname , name , value , size , 0 ) ;
2005-04-16 15:20:36 -07:00
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE4 ( fgetxattr , int , fd , const char __user * , name ,
void __user * , value , size_t , size )
2005-04-16 15:20:36 -07:00
{
2012-08-28 12:52:22 -04:00
struct fd f = fdget ( fd ) ;
2005-04-16 15:20:36 -07:00
ssize_t error = - EBADF ;
2012-08-28 12:52:22 -04:00
if ( ! f . file )
2005-04-16 15:20:36 -07:00
return error ;
2014-10-31 17:44:57 -04:00
audit_file ( f . file ) ;
2012-08-28 12:52:22 -04:00
error = getxattr ( f . file - > f_path . dentry , name , value , size ) ;
fdput ( f ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
/*
* Extended attribute LIST operations
*/
static ssize_t
listxattr ( struct dentry * d , char __user * list , size_t size )
{
ssize_t error ;
char * klist = NULL ;
if ( size ) {
if ( size > XATTR_LIST_MAX )
size = XATTR_LIST_MAX ;
2012-04-05 14:25:06 -07:00
klist = kmalloc ( size , __GFP_NOWARN | GFP_KERNEL ) ;
2012-04-05 14:25:07 -07:00
if ( ! klist ) {
2016-01-02 23:09:47 +01:00
klist = vmalloc ( size ) ;
if ( ! klist )
2012-04-05 14:25:07 -07:00
return - ENOMEM ;
}
2005-04-16 15:20:36 -07:00
}
2006-10-09 16:10:48 -04:00
error = vfs_listxattr ( d , klist , size ) ;
2005-09-03 15:55:18 -07:00
if ( error > 0 ) {
if ( size & & copy_to_user ( list , klist , error ) )
error = - EFAULT ;
} else if ( error = = - ERANGE & & size > = XATTR_LIST_MAX ) {
/* The file system tried to returned a list bigger
than XATTR_LIST_MAX bytes . Not possible . */
error = - E2BIG ;
2005-04-16 15:20:36 -07:00
}
2016-01-02 23:09:47 +01:00
kvfree ( klist ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
static ssize_t path_listxattr ( const char __user * pathname , char __user * list ,
size_t size , unsigned int lookup_flags )
2005-04-16 15:20:36 -07:00
{
2008-07-22 09:59:21 -04:00
struct path path ;
2005-04-16 15:20:36 -07:00
ssize_t error ;
2012-12-11 12:10:16 -05:00
retry :
error = user_path_at ( AT_FDCWD , pathname , lookup_flags , & path ) ;
2005-04-16 15:20:36 -07:00
if ( error )
return error ;
2008-07-22 09:59:21 -04:00
error = listxattr ( path . dentry , list , size ) ;
path_put ( & path ) ;
2012-12-11 12:10:16 -05:00
if ( retry_estale ( error , lookup_flags ) ) {
lookup_flags | = LOOKUP_REVAL ;
goto retry ;
}
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
SYSCALL_DEFINE3 ( listxattr , const char __user * , pathname , char __user * , list ,
size_t , size )
{
return path_listxattr ( pathname , list , size , LOOKUP_FOLLOW ) ;
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE3 ( llistxattr , const char __user * , pathname , char __user * , list ,
size_t , size )
2005-04-16 15:20:36 -07:00
{
2014-10-12 11:59:58 -05:00
return path_listxattr ( pathname , list , size , 0 ) ;
2005-04-16 15:20:36 -07:00
}
2009-01-14 14:14:14 +01:00
SYSCALL_DEFINE3 ( flistxattr , int , fd , char __user * , list , size_t , size )
2005-04-16 15:20:36 -07:00
{
2012-08-28 12:52:22 -04:00
struct fd f = fdget ( fd ) ;
2005-04-16 15:20:36 -07:00
ssize_t error = - EBADF ;
2012-08-28 12:52:22 -04:00
if ( ! f . file )
2005-04-16 15:20:36 -07:00
return error ;
2014-10-31 17:44:57 -04:00
audit_file ( f . file ) ;
2012-08-28 12:52:22 -04:00
error = listxattr ( f . file - > f_path . dentry , list , size ) ;
fdput ( f ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
/*
* Extended attribute REMOVE operations
*/
static long
2008-04-29 00:59:41 -07:00
removexattr ( struct dentry * d , const char __user * name )
2005-04-16 15:20:36 -07:00
{
int error ;
char kname [ XATTR_NAME_MAX + 1 ] ;
error = strncpy_from_user ( kname , name , sizeof ( kname ) ) ;
if ( error = = 0 | | error = = sizeof ( kname ) )
error = - ERANGE ;
if ( error < 0 )
return error ;
2006-01-09 20:51:55 -08:00
return vfs_removexattr ( d , kname ) ;
2005-04-16 15:20:36 -07:00
}
2014-10-12 11:59:58 -05:00
static int path_removexattr ( const char __user * pathname ,
const char __user * name , unsigned int lookup_flags )
2005-04-16 15:20:36 -07:00
{
2008-07-22 09:59:21 -04:00
struct path path ;
2005-04-16 15:20:36 -07:00
int error ;
2012-12-11 12:10:17 -05:00
retry :
error = user_path_at ( AT_FDCWD , pathname , lookup_flags , & path ) ;
2005-04-16 15:20:36 -07:00
if ( error )
return error ;
2008-07-22 09:59:21 -04:00
error = mnt_want_write ( path . mnt ) ;
2008-02-15 14:37:38 -08:00
if ( ! error ) {
2008-07-22 09:59:21 -04:00
error = removexattr ( path . dentry , name ) ;
mnt_drop_write ( path . mnt ) ;
2008-02-15 14:37:38 -08:00
}
2008-07-22 09:59:21 -04:00
path_put ( & path ) ;
2012-12-11 12:10:17 -05:00
if ( retry_estale ( error , lookup_flags ) ) {
lookup_flags | = LOOKUP_REVAL ;
goto retry ;
}
2005-04-16 15:20:36 -07:00
return error ;
}
2014-10-12 11:59:58 -05:00
SYSCALL_DEFINE2 ( removexattr , const char __user * , pathname ,
const char __user * , name )
{
return path_removexattr ( pathname , name , LOOKUP_FOLLOW ) ;
}
2009-01-14 14:14:15 +01:00
SYSCALL_DEFINE2 ( lremovexattr , const char __user * , pathname ,
const char __user * , name )
2005-04-16 15:20:36 -07:00
{
2014-10-12 11:59:58 -05:00
return path_removexattr ( pathname , name , 0 ) ;
2005-04-16 15:20:36 -07:00
}
2009-01-14 14:14:15 +01:00
SYSCALL_DEFINE2 ( fremovexattr , int , fd , const char __user * , name )
2005-04-16 15:20:36 -07:00
{
2012-08-28 12:52:22 -04:00
struct fd f = fdget ( fd ) ;
2005-04-16 15:20:36 -07:00
int error = - EBADF ;
2012-08-28 12:52:22 -04:00
if ( ! f . file )
2005-04-16 15:20:36 -07:00
return error ;
2014-10-31 17:44:57 -04:00
audit_file ( f . file ) ;
2012-08-28 12:52:22 -04:00
error = mnt_want_write_file ( f . file ) ;
2008-02-15 14:37:38 -08:00
if ( ! error ) {
2014-10-31 17:44:57 -04:00
error = removexattr ( f . file - > f_path . dentry , name ) ;
2012-08-28 12:52:22 -04:00
mnt_drop_write_file ( f . file ) ;
2008-02-15 14:37:38 -08:00
}
2012-08-28 12:52:22 -04:00
fdput ( f ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
static const char *
strcmp_prefix ( const char * a , const char * a_prefix )
{
while ( * a_prefix & & * a = = * a_prefix ) {
a + + ;
a_prefix + + ;
}
return * a_prefix ? NULL : a ;
}
/*
* In order to implement different sets of xattr operations for each xattr
* prefix with the generic xattr API , a filesystem should create a
* null - terminated array of struct xattr_handler ( one for each prefix ) and
* hang a pointer to it off of the s_xattr field of the superblock .
*
* The generic_fooxattr ( ) functions will use this list to dispatch xattr
* operations to the correct xattr_handler .
*/
# define for_each_xattr_handler(handlers, handler) \
2016-05-25 17:34:41 -04:00
if ( handlers ) \
2005-04-16 15:20:36 -07:00
for ( ( handler ) = * ( handlers ) + + ; \
( handler ) ! = NULL ; \
( handler ) = * ( handlers ) + + )
/*
* Find the xattr_handler with the matching prefix .
*/
2010-05-13 17:53:14 -07:00
static const struct xattr_handler *
xattr_resolve_name ( const struct xattr_handler * * handlers , const char * * name )
2005-04-16 15:20:36 -07:00
{
2010-05-13 17:53:14 -07:00
const struct xattr_handler * handler ;
2005-04-16 15:20:36 -07:00
if ( ! * name )
2016-05-09 13:28:49 +02:00
return ERR_PTR ( - EINVAL ) ;
2005-04-16 15:20:36 -07:00
for_each_xattr_handler ( handlers , handler ) {
2015-12-02 14:44:36 +01:00
const char * n ;
n = strcmp_prefix ( * name , xattr_prefix ( handler ) ) ;
2005-04-16 15:20:36 -07:00
if ( n ) {
2015-12-02 14:44:36 +01:00
if ( ! handler - > prefix ^ ! * n ) {
if ( * n )
continue ;
return ERR_PTR ( - EINVAL ) ;
}
2005-04-16 15:20:36 -07:00
* name = n ;
2015-12-02 14:44:36 +01:00
return handler ;
2005-04-16 15:20:36 -07:00
}
}
2015-12-02 14:44:36 +01:00
return ERR_PTR ( - EOPNOTSUPP ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Find the handler for the prefix and dispatch its get ( ) operation .
*/
ssize_t
2016-04-11 00:48:00 -04:00
generic_getxattr ( struct dentry * dentry , struct inode * inode ,
const char * name , void * buffer , size_t size )
2005-04-16 15:20:36 -07:00
{
2010-05-13 17:53:14 -07:00
const struct xattr_handler * handler ;
2005-04-16 15:20:36 -07:00
2009-11-13 09:52:56 +00:00
handler = xattr_resolve_name ( dentry - > d_sb - > s_xattr , & name ) ;
2015-12-02 14:44:36 +01:00
if ( IS_ERR ( handler ) )
return PTR_ERR ( handler ) ;
2016-04-11 00:48:00 -04:00
return handler - > get ( handler , dentry , inode ,
2016-04-10 20:48:24 -04:00
name , buffer , size ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Combine the results of the list ( ) operation from every xattr_handler in the
* list .
*/
ssize_t
generic_listxattr ( struct dentry * dentry , char * buffer , size_t buffer_size )
{
2010-05-13 17:53:14 -07:00
const struct xattr_handler * handler , * * handlers = dentry - > d_sb - > s_xattr ;
2005-04-16 15:20:36 -07:00
unsigned int size = 0 ;
if ( ! buffer ) {
2009-11-13 09:52:56 +00:00
for_each_xattr_handler ( handlers , handler ) {
2015-12-02 14:44:43 +01:00
if ( ! handler - > name | |
( handler - > list & & ! handler - > list ( dentry ) ) )
2015-12-02 14:44:41 +01:00
continue ;
2015-12-02 14:44:43 +01:00
size + = strlen ( handler - > name ) + 1 ;
2009-11-13 09:52:56 +00:00
}
2005-04-16 15:20:36 -07:00
} else {
char * buf = buffer ;
2015-12-02 14:44:43 +01:00
size_t len ;
2005-04-16 15:20:36 -07:00
for_each_xattr_handler ( handlers , handler ) {
2015-12-02 14:44:43 +01:00
if ( ! handler - > name | |
( handler - > list & & ! handler - > list ( dentry ) ) )
2015-12-02 14:44:41 +01:00
continue ;
2015-12-02 14:44:43 +01:00
len = strlen ( handler - > name ) ;
if ( len + 1 > buffer_size )
2005-04-16 15:20:36 -07:00
return - ERANGE ;
2015-12-02 14:44:43 +01:00
memcpy ( buf , handler - > name , len + 1 ) ;
buf + = len + 1 ;
buffer_size - = len + 1 ;
2005-04-16 15:20:36 -07:00
}
size = buf - buffer ;
}
return size ;
}
/*
* Find the handler for the prefix and dispatch its set ( ) operation .
*/
int
generic_setxattr ( struct dentry * dentry , const char * name , const void * value , size_t size , int flags )
{
2010-05-13 17:53:14 -07:00
const struct xattr_handler * handler ;
2005-04-16 15:20:36 -07:00
if ( size = = 0 )
value = " " ; /* empty EA, do not remove */
2009-11-13 09:52:56 +00:00
handler = xattr_resolve_name ( dentry - > d_sb - > s_xattr , & name ) ;
2015-12-02 14:44:36 +01:00
if ( IS_ERR ( handler ) )
return PTR_ERR ( handler ) ;
2016-05-27 10:19:30 -04:00
return handler - > set ( handler , dentry , d_inode ( dentry ) , name , value ,
size , flags ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Find the handler for the prefix and dispatch its set ( ) operation to remove
* any associated extended attribute .
*/
int
generic_removexattr ( struct dentry * dentry , const char * name )
{
2010-05-13 17:53:14 -07:00
const struct xattr_handler * handler ;
2005-04-16 15:20:36 -07:00
2009-11-13 09:52:56 +00:00
handler = xattr_resolve_name ( dentry - > d_sb - > s_xattr , & name ) ;
2015-12-02 14:44:36 +01:00
if ( IS_ERR ( handler ) )
return PTR_ERR ( handler ) ;
2016-05-27 10:19:30 -04:00
return handler - > set ( handler , dentry , d_inode ( dentry ) , name , NULL ,
0 , XATTR_REPLACE ) ;
2005-04-16 15:20:36 -07:00
}
EXPORT_SYMBOL ( generic_getxattr ) ;
EXPORT_SYMBOL ( generic_listxattr ) ;
EXPORT_SYMBOL ( generic_setxattr ) ;
EXPORT_SYMBOL ( generic_removexattr ) ;
2012-08-23 16:53:28 -04:00
2015-10-04 19:18:52 +02:00
/**
* xattr_full_name - Compute full attribute name from suffix
*
* @ handler : handler of the xattr_handler operation
* @ name : name passed to the xattr_handler operation
*
* The get and set xattr handler operations are called with the remainder of
* the attribute name after skipping the handler ' s prefix : for example , " foo "
* is passed to the get operation of a handler with prefix " user. " to get
* attribute " user.foo " . The full name is still " there " in the name though .
*
* Note : the list xattr handler operation when called from the vfs is passed a
* NULL name ; some file systems use this operation internally , with varying
* semantics .
*/
const char * xattr_full_name ( const struct xattr_handler * handler ,
const char * name )
{
2015-12-02 14:44:36 +01:00
size_t prefix_len = strlen ( xattr_prefix ( handler ) ) ;
2015-10-04 19:18:52 +02:00
return name - prefix_len ;
}
EXPORT_SYMBOL ( xattr_full_name ) ;
2012-08-23 16:53:28 -04:00
/*
* Allocate new xattr and copy in the value ; but leave the name to callers .
*/
struct simple_xattr * simple_xattr_alloc ( const void * value , size_t size )
{
struct simple_xattr * new_xattr ;
size_t len ;
/* wrap around? */
len = sizeof ( * new_xattr ) + size ;
2014-07-23 14:00:17 -07:00
if ( len < sizeof ( * new_xattr ) )
2012-08-23 16:53:28 -04:00
return NULL ;
new_xattr = kmalloc ( len , GFP_KERNEL ) ;
if ( ! new_xattr )
return NULL ;
new_xattr - > size = size ;
memcpy ( new_xattr - > value , value , size ) ;
return new_xattr ;
}
/*
* xattr GET operation for in - memory / pseudo filesystems
*/
int simple_xattr_get ( struct simple_xattrs * xattrs , const char * name ,
void * buffer , size_t size )
{
struct simple_xattr * xattr ;
int ret = - ENODATA ;
spin_lock ( & xattrs - > lock ) ;
list_for_each_entry ( xattr , & xattrs - > head , list ) {
if ( strcmp ( name , xattr - > name ) )
continue ;
ret = xattr - > size ;
if ( buffer ) {
if ( size < xattr - > size )
ret = - ERANGE ;
else
memcpy ( buffer , xattr - > value , xattr - > size ) ;
}
break ;
}
spin_unlock ( & xattrs - > lock ) ;
return ret ;
}
2015-12-02 14:44:38 +01:00
/**
* simple_xattr_set - xattr SET operation for in - memory / pseudo filesystems
* @ xattrs : target simple_xattr list
* @ name : name of the extended attribute
* @ value : value of the xattr . If % NULL , will remove the attribute .
* @ size : size of the new xattr
* @ flags : % XATTR_ { CREATE | REPLACE }
*
* % XATTR_CREATE is set , the xattr shouldn ' t exist already ; otherwise fails
* with - EEXIST . If % XATTR_REPLACE is set , the xattr should exist ;
* otherwise , fails with - ENODATA .
*
* Returns 0 on success , - errno on failure .
*/
int simple_xattr_set ( struct simple_xattrs * xattrs , const char * name ,
const void * value , size_t size , int flags )
2012-08-23 16:53:28 -04:00
{
struct simple_xattr * xattr ;
2012-10-17 20:41:15 -07:00
struct simple_xattr * new_xattr = NULL ;
2012-08-23 16:53:28 -04:00
int err = 0 ;
/* value == NULL means remove */
if ( value ) {
new_xattr = simple_xattr_alloc ( value , size ) ;
if ( ! new_xattr )
return - ENOMEM ;
new_xattr - > name = kstrdup ( name , GFP_KERNEL ) ;
if ( ! new_xattr - > name ) {
kfree ( new_xattr ) ;
return - ENOMEM ;
}
}
spin_lock ( & xattrs - > lock ) ;
list_for_each_entry ( xattr , & xattrs - > head , list ) {
if ( ! strcmp ( name , xattr - > name ) ) {
if ( flags & XATTR_CREATE ) {
xattr = new_xattr ;
err = - EEXIST ;
} else if ( new_xattr ) {
list_replace ( & xattr - > list , & new_xattr - > list ) ;
} else {
list_del ( & xattr - > list ) ;
}
goto out ;
}
}
if ( flags & XATTR_REPLACE ) {
xattr = new_xattr ;
err = - ENODATA ;
} else {
list_add ( & new_xattr - > list , & xattrs - > head ) ;
xattr = NULL ;
}
out :
spin_unlock ( & xattrs - > lock ) ;
if ( xattr ) {
kfree ( xattr - > name ) ;
kfree ( xattr ) ;
}
return err ;
}
static bool xattr_is_trusted ( const char * name )
{
return ! strncmp ( name , XATTR_TRUSTED_PREFIX , XATTR_TRUSTED_PREFIX_LEN ) ;
}
2015-12-02 14:44:39 +01:00
static int xattr_list_one ( char * * buffer , ssize_t * remaining_size ,
const char * name )
{
size_t len = strlen ( name ) + 1 ;
if ( * buffer ) {
if ( * remaining_size < len )
return - ERANGE ;
memcpy ( * buffer , name , len ) ;
* buffer + = len ;
}
* remaining_size - = len ;
return 0 ;
}
2012-08-23 16:53:28 -04:00
/*
* xattr LIST operation for in - memory / pseudo filesystems
*/
2015-12-02 14:44:39 +01:00
ssize_t simple_xattr_list ( struct inode * inode , struct simple_xattrs * xattrs ,
char * buffer , size_t size )
2012-08-23 16:53:28 -04:00
{
bool trusted = capable ( CAP_SYS_ADMIN ) ;
struct simple_xattr * xattr ;
2015-12-02 14:44:39 +01:00
ssize_t remaining_size = size ;
2016-02-04 02:56:30 +01:00
int err = 0 ;
2015-12-02 14:44:39 +01:00
# ifdef CONFIG_FS_POSIX_ACL
if ( inode - > i_acl ) {
err = xattr_list_one ( & buffer , & remaining_size ,
XATTR_NAME_POSIX_ACL_ACCESS ) ;
if ( err )
return err ;
}
if ( inode - > i_default_acl ) {
err = xattr_list_one ( & buffer , & remaining_size ,
XATTR_NAME_POSIX_ACL_DEFAULT ) ;
if ( err )
return err ;
}
# endif
2012-08-23 16:53:28 -04:00
spin_lock ( & xattrs - > lock ) ;
list_for_each_entry ( xattr , & xattrs - > head , list ) {
/* skip "trusted." attributes for unprivileged callers */
if ( ! trusted & & xattr_is_trusted ( xattr - > name ) )
continue ;
2015-12-02 14:44:39 +01:00
err = xattr_list_one ( & buffer , & remaining_size , xattr - > name ) ;
if ( err )
2016-02-04 02:56:30 +01:00
break ;
2012-08-23 16:53:28 -04:00
}
spin_unlock ( & xattrs - > lock ) ;
2016-02-04 02:56:30 +01:00
return err ? err : size - remaining_size ;
2012-08-23 16:53:28 -04:00
}
2012-09-11 16:28:11 -04:00
/*
* Adds an extended attribute to the list
*/
2012-08-23 16:53:28 -04:00
void simple_xattr_list_add ( struct simple_xattrs * xattrs ,
struct simple_xattr * new_xattr )
{
spin_lock ( & xattrs - > lock ) ;
list_add ( & new_xattr - > list , & xattrs - > head ) ;
spin_unlock ( & xattrs - > lock ) ;
}