2018-06-05 19:42:14 -07:00
// SPDX-License-Identifier: GPL-2.0
2008-06-23 13:34:09 +10:00
/*
* Copyright ( C ) 2008 Christoph Hellwig .
* Portions Copyright ( C ) 2000 - 2008 Silicon Graphics , Inc .
*/
# include "xfs.h"
2019-06-28 19:25:35 -07:00
# include "xfs_shared.h"
2013-10-23 10:51:50 +11:00
# include "xfs_format.h"
2013-08-12 20:49:23 +10:00
# include "xfs_log_format.h"
2013-10-15 09:17:51 +11:00
# include "xfs_da_format.h"
2008-06-23 13:34:09 +10:00
# include "xfs_inode.h"
# include "xfs_attr.h"
# include <linux/posix_acl_xattr.h>
# include <linux/xattr.h>
static int
2016-04-10 20:48:24 -04:00
xfs_xattr_get ( const struct xattr_handler * handler , struct dentry * unused ,
struct inode * inode , const char * name , void * value , size_t size )
2008-06-23 13:34:09 +10:00
{
2015-10-04 19:18:51 +02:00
int xflags = handler - > flags ;
2016-04-10 20:48:24 -04:00
struct xfs_inode * ip = XFS_I ( inode ) ;
2008-06-23 13:34:09 +10:00
int error , asize = size ;
/* Convert Linux syscall to XFS internal ATTR flags */
if ( ! size ) {
xflags | = ATTR_KERNOVAL ;
value = NULL ;
}
2014-06-25 14:58:08 +10:00
error = xfs_attr_get ( ip , ( unsigned char * ) name , value , & asize , xflags ) ;
2008-06-23 13:34:09 +10:00
if ( error )
return error ;
return asize ;
}
2015-11-03 12:56:17 +11:00
void
xfs_forget_acl (
struct inode * inode ,
const char * name ,
int xflags )
{
/*
* Invalidate any cached ACLs if the user has bypassed the ACL
* interface . We don ' t validate the content whatsoever so it is caller
* responsibility to provide data in valid format and ensure i_mode is
* consistent .
*/
if ( xflags & ATTR_ROOT ) {
# ifdef CONFIG_XFS_POSIX_ACL
if ( ! strcmp ( name , SGI_ACL_FILE ) )
forget_cached_acl ( inode , ACL_TYPE_ACCESS ) ;
else if ( ! strcmp ( name , SGI_ACL_DEFAULT ) )
forget_cached_acl ( inode , ACL_TYPE_DEFAULT ) ;
# endif
}
}
2008-06-23 13:34:09 +10:00
static int
2016-05-27 10:19:30 -04:00
xfs_xattr_set ( const struct xattr_handler * handler , struct dentry * unused ,
struct inode * inode , const char * name , const void * value ,
size_t size , int flags )
2008-06-23 13:34:09 +10:00
{
2015-11-13 18:02:30 -08:00
int xflags = handler - > flags ;
2016-05-27 10:19:30 -04:00
struct xfs_inode * ip = XFS_I ( inode ) ;
2015-11-03 12:40:59 +11:00
int error ;
2008-06-23 13:34:09 +10:00
/* Convert Linux syscall to XFS internal ATTR flags */
if ( flags & XATTR_CREATE )
xflags | = ATTR_CREATE ;
if ( flags & XATTR_REPLACE )
xflags | = ATTR_REPLACE ;
if ( ! value )
2014-06-25 14:58:08 +10:00
return xfs_attr_remove ( ip , ( unsigned char * ) name , xflags ) ;
2015-11-03 12:40:59 +11:00
error = xfs_attr_set ( ip , ( unsigned char * ) name ,
2010-01-20 10:47:48 +11:00
( void * ) value , size , xflags ) ;
2015-11-03 12:56:17 +11:00
if ( ! error )
2016-05-27 10:19:30 -04:00
xfs_forget_acl ( inode , name , xflags ) ;
2015-11-03 12:40:59 +11:00
return error ;
2008-06-23 13:34:09 +10:00
}
2010-05-13 17:53:20 -07:00
static const struct xattr_handler xfs_xattr_user_handler = {
2008-06-23 13:34:09 +10:00
. prefix = XATTR_USER_PREFIX ,
2009-11-13 09:52:56 +00:00
. flags = 0 , /* no flags implies user namespace */
. get = xfs_xattr_get ,
. set = xfs_xattr_set ,
2008-06-23 13:34:09 +10:00
} ;
2010-05-13 17:53:20 -07:00
static const struct xattr_handler xfs_xattr_trusted_handler = {
2008-06-23 13:34:09 +10:00
. prefix = XATTR_TRUSTED_PREFIX ,
2009-11-13 09:52:56 +00:00
. flags = ATTR_ROOT ,
. get = xfs_xattr_get ,
. set = xfs_xattr_set ,
2008-06-23 13:34:09 +10:00
} ;
2010-05-13 17:53:20 -07:00
static const struct xattr_handler xfs_xattr_security_handler = {
2008-06-23 13:34:09 +10:00
. prefix = XATTR_SECURITY_PREFIX ,
2009-11-13 09:52:56 +00:00
. flags = ATTR_SECURE ,
. get = xfs_xattr_get ,
. set = xfs_xattr_set ,
2008-06-23 13:34:09 +10:00
} ;
2010-05-13 17:53:20 -07:00
const struct xattr_handler * xfs_xattr_handlers [ ] = {
2008-06-23 13:34:09 +10:00
& xfs_xattr_user_handler ,
& xfs_xattr_trusted_handler ,
& xfs_xattr_security_handler ,
2009-06-10 17:07:47 +02:00
# ifdef CONFIG_XFS_POSIX_ACL
2013-12-20 05:16:50 -08:00
& posix_acl_access_xattr_handler ,
& posix_acl_default_xattr_handler ,
2009-06-10 17:07:47 +02:00
# endif
2008-06-23 13:34:09 +10:00
NULL
} ;
2016-12-05 12:32:14 +11:00
static void
2015-12-02 14:44:40 +01:00
__xfs_xattr_put_listent (
2010-01-20 10:47:48 +11:00
struct xfs_attr_list_context * context ,
2015-12-02 14:44:40 +01:00
char * prefix ,
int prefix_len ,
unsigned char * name ,
int namelen )
2008-06-23 13:34:09 +10:00
{
char * offset ;
int arraytop ;
2019-02-13 11:15:17 -08:00
if ( context - > count < 0 | | context - > seen_enough )
return ;
2015-12-02 14:44:40 +01:00
if ( ! context - > alist )
goto compute_size ;
2008-06-23 13:34:09 +10:00
arraytop = context - > count + prefix_len + namelen + 1 ;
if ( arraytop > context - > firstu ) {
context - > count = - 1 ; /* insufficient space */
2016-09-14 07:40:35 +10:00
context - > seen_enough = 1 ;
2016-12-05 12:32:14 +11:00
return ;
2008-06-23 13:34:09 +10:00
}
offset = ( char * ) context - > alist + context - > count ;
2015-12-02 14:44:40 +01:00
strncpy ( offset , prefix , prefix_len ) ;
2008-06-23 13:34:09 +10:00
offset + = prefix_len ;
2010-01-20 10:47:48 +11:00
strncpy ( offset , ( char * ) name , namelen ) ; /* real name */
2008-06-23 13:34:09 +10:00
offset + = namelen ;
* offset = ' \0 ' ;
2015-12-02 14:44:40 +01:00
compute_size :
2008-06-23 13:34:09 +10:00
context - > count + = prefix_len + namelen + 1 ;
2016-12-05 12:32:14 +11:00
return ;
2008-06-23 13:34:09 +10:00
}
2016-12-05 12:32:14 +11:00
static void
2015-12-02 14:44:40 +01:00
xfs_xattr_put_listent (
2010-01-20 10:47:48 +11:00
struct xfs_attr_list_context * context ,
int flags ,
unsigned char * name ,
int namelen ,
2016-04-06 07:57:32 +10:00
int valuelen )
2008-06-23 13:34:09 +10:00
{
2015-12-02 14:44:40 +01:00
char * prefix ;
int prefix_len ;
2008-06-23 13:34:09 +10:00
2015-12-02 14:44:40 +01:00
ASSERT ( context - > count > = 0 ) ;
2008-06-23 13:34:09 +10:00
2015-12-02 14:44:40 +01:00
if ( flags & XFS_ATTR_ROOT ) {
# ifdef CONFIG_XFS_POSIX_ACL
if ( namelen = = SGI_ACL_FILE_SIZE & &
strncmp ( name , SGI_ACL_FILE ,
SGI_ACL_FILE_SIZE ) = = 0 ) {
2016-12-05 12:32:14 +11:00
__xfs_xattr_put_listent (
2015-12-02 14:44:40 +01:00
context , XATTR_SYSTEM_PREFIX ,
XATTR_SYSTEM_PREFIX_LEN ,
XATTR_POSIX_ACL_ACCESS ,
strlen ( XATTR_POSIX_ACL_ACCESS ) ) ;
} else if ( namelen = = SGI_ACL_DEFAULT_SIZE & &
strncmp ( name , SGI_ACL_DEFAULT ,
SGI_ACL_DEFAULT_SIZE ) = = 0 ) {
2016-12-05 12:32:14 +11:00
__xfs_xattr_put_listent (
2015-12-02 14:44:40 +01:00
context , XATTR_SYSTEM_PREFIX ,
XATTR_SYSTEM_PREFIX_LEN ,
XATTR_POSIX_ACL_DEFAULT ,
strlen ( XATTR_POSIX_ACL_DEFAULT ) ) ;
}
# endif
2008-06-23 13:34:09 +10:00
2015-12-02 14:44:40 +01:00
/*
* Only show root namespace entries if we are actually allowed to
* see them .
*/
if ( ! capable ( CAP_SYS_ADMIN ) )
2016-12-05 12:32:14 +11:00
return ;
2015-12-02 14:44:40 +01:00
prefix = XATTR_TRUSTED_PREFIX ;
prefix_len = XATTR_TRUSTED_PREFIX_LEN ;
} else if ( flags & XFS_ATTR_SECURE ) {
prefix = XATTR_SECURITY_PREFIX ;
prefix_len = XATTR_SECURITY_PREFIX_LEN ;
} else {
prefix = XATTR_USER_PREFIX ;
prefix_len = XATTR_USER_PREFIX_LEN ;
}
2016-12-05 12:32:14 +11:00
__xfs_xattr_put_listent ( context , prefix , prefix_len , name ,
namelen ) ;
return ;
2008-06-23 13:34:09 +10:00
}
ssize_t
2016-04-06 07:57:18 +10:00
xfs_vn_listxattr (
struct dentry * dentry ,
char * data ,
size_t size )
2008-06-23 13:34:09 +10:00
{
struct xfs_attr_list_context context ;
struct attrlist_cursor_kern cursor = { 0 } ;
2016-04-06 07:57:18 +10:00
struct inode * inode = d_inode ( dentry ) ;
int error ;
2008-06-23 13:34:09 +10:00
/*
* First read the regular on - disk attributes .
*/
memset ( & context , 0 , sizeof ( context ) ) ;
context . dp = XFS_I ( inode ) ;
context . cursor = & cursor ;
context . resynch = 1 ;
2015-12-02 14:44:40 +01:00
context . alist = size ? data : NULL ;
2008-06-23 13:34:09 +10:00
context . bufsize = size ;
context . firstu = context . bufsize ;
2015-12-02 14:44:40 +01:00
context . put_listent = xfs_xattr_put_listent ;
2008-06-23 13:34:09 +10:00
2016-04-06 07:57:18 +10:00
error = xfs_attr_list_int ( & context ) ;
if ( error )
return error ;
2008-06-23 13:34:09 +10:00
if ( context . count < 0 )
return - ERANGE ;
return context . count ;
}