2013-09-11 14:24:29 -07:00
/*
* linux / fs / hfsplus / posix_acl . c
*
* Vyacheslav Dubeyko < slava @ dubeyko . com >
*
* Handler for Posix Access Control Lists ( ACLs ) support .
*/
# include "hfsplus_fs.h"
# include "xattr.h"
# include "acl.h"
struct posix_acl * hfsplus_get_posix_acl ( struct inode * inode , int type )
{
struct posix_acl * acl ;
char * xattr_name ;
char * value = NULL ;
ssize_t size ;
2013-12-20 05:16:46 -08:00
hfs_dbg ( ACL_MOD , " [%s]: ino %lu \n " , __func__ , inode - > i_ino ) ;
2013-09-11 14:24:29 -07:00
switch ( type ) {
case ACL_TYPE_ACCESS :
xattr_name = POSIX_ACL_XATTR_ACCESS ;
break ;
case ACL_TYPE_DEFAULT :
xattr_name = POSIX_ACL_XATTR_DEFAULT ;
break ;
default :
return ERR_PTR ( - EINVAL ) ;
}
size = __hfsplus_getxattr ( inode , xattr_name , NULL , 0 ) ;
if ( size > 0 ) {
value = ( char * ) hfsplus_alloc_attr_entry ( ) ;
if ( unlikely ( ! value ) )
return ERR_PTR ( - ENOMEM ) ;
size = __hfsplus_getxattr ( inode , xattr_name , value , size ) ;
}
if ( size > 0 )
acl = posix_acl_from_xattr ( & init_user_ns , value , size ) ;
else if ( size = = - ENODATA )
acl = NULL ;
else
acl = ERR_PTR ( size ) ;
hfsplus_destroy_attr_entry ( ( hfsplus_attr_entry * ) value ) ;
if ( ! IS_ERR ( acl ) )
set_cached_acl ( inode , type , acl ) ;
return acl ;
}
2013-12-20 05:16:46 -08:00
int hfsplus_set_posix_acl ( struct inode * inode , struct posix_acl * acl ,
int type )
2013-09-11 14:24:29 -07:00
{
int err ;
char * xattr_name ;
size_t size = 0 ;
char * value = NULL ;
2013-12-20 05:16:46 -08:00
hfs_dbg ( ACL_MOD , " [%s]: ino %lu \n " , __func__ , inode - > i_ino ) ;
2013-09-11 14:24:29 -07:00
switch ( type ) {
case ACL_TYPE_ACCESS :
xattr_name = POSIX_ACL_XATTR_ACCESS ;
if ( acl ) {
err = posix_acl_equiv_mode ( acl , & inode - > i_mode ) ;
if ( err < 0 )
return err ;
}
err = 0 ;
break ;
case ACL_TYPE_DEFAULT :
xattr_name = POSIX_ACL_XATTR_DEFAULT ;
if ( ! S_ISDIR ( inode - > i_mode ) )
return acl ? - EACCES : 0 ;
break ;
default :
return - EINVAL ;
}
if ( acl ) {
size = posix_acl_xattr_size ( acl - > a_count ) ;
if ( unlikely ( size > HFSPLUS_MAX_INLINE_DATA_SIZE ) )
return - ENOMEM ;
value = ( char * ) hfsplus_alloc_attr_entry ( ) ;
if ( unlikely ( ! value ) )
return - ENOMEM ;
err = posix_acl_to_xattr ( & init_user_ns , acl , value , size ) ;
if ( unlikely ( err < 0 ) )
goto end_set_acl ;
}
err = __hfsplus_setxattr ( inode , xattr_name , value , size , 0 ) ;
end_set_acl :
hfsplus_destroy_attr_entry ( ( hfsplus_attr_entry * ) value ) ;
if ( ! err )
set_cached_acl ( inode , type , acl ) ;
return err ;
}
int hfsplus_init_posix_acl ( struct inode * inode , struct inode * dir )
{
int err = 0 ;
2013-12-20 05:16:46 -08:00
struct posix_acl * default_acl , * acl ;
2013-09-11 14:24:29 -07:00
hfs_dbg ( ACL_MOD ,
" [%s]: ino %lu, dir->ino %lu \n " ,
__func__ , inode - > i_ino , dir - > i_ino ) ;
if ( S_ISLNK ( inode - > i_mode ) )
return 0 ;
2013-12-20 05:16:46 -08:00
err = posix_acl_create ( dir , & inode - > i_mode , & default_acl , & acl ) ;
if ( err )
2013-09-11 14:24:29 -07:00
return err ;
2013-12-20 05:16:46 -08:00
if ( default_acl ) {
err = hfsplus_set_posix_acl ( inode , default_acl ,
ACL_TYPE_DEFAULT ) ;
posix_acl_release ( default_acl ) ;
2013-09-11 14:24:29 -07:00
}
2013-12-20 05:16:46 -08:00
if ( acl ) {
if ( ! err )
err = hfsplus_set_posix_acl ( inode , acl ,
ACL_TYPE_ACCESS ) ;
posix_acl_release ( acl ) ;
}
2013-09-11 14:24:29 -07:00
return err ;
}