2013-10-09 21:52:15 +04:00
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd .
Copyright 2013 Intel Corporation
Author : Auke Kok < auke - jan . h . kok @ intel . com >
systemd is free software ; you can redistribute it and / or modify it
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation ; either version 2.1 of the License , or
( at your option ) any later version .
systemd is distributed in the hope that it will be useful , but
WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
Lesser General Public License for more details .
You should have received a copy of the GNU Lesser General Public License
along with systemd ; If not , see < http : //www.gnu.org/licenses/>.
* * */
2014-05-28 13:36:40 +04:00
# include <sys/xattr.h>
2013-10-09 21:52:15 +04:00
2014-10-23 12:23:45 +04:00
# include "util.h"
# include "path-util.h"
2014-11-24 14:46:20 +03:00
# include "fileio.h"
2013-10-10 18:35:44 +04:00
# include "smack-util.h"
2013-10-09 21:52:15 +04:00
2014-10-23 20:40:03 +04:00
# define SMACK_FLOOR_LABEL "_"
# define SMACK_STAR_LABEL "*"
2014-10-23 19:34:30 +04:00
bool mac_smack_use ( void ) {
2013-10-10 18:35:44 +04:00
# ifdef HAVE_SMACK
2014-10-23 19:34:30 +04:00
static int cached_use = - 1 ;
2013-10-10 18:35:44 +04:00
2014-10-23 19:34:30 +04:00
if ( cached_use < 0 )
cached_use = access ( " /sys/fs/smackfs/ " , F_OK ) > = 0 ;
2013-10-09 21:52:15 +04:00
2014-10-23 19:34:30 +04:00
return cached_use ;
2013-10-10 18:35:44 +04:00
# else
return false ;
# endif
2013-10-09 21:52:15 +04:00
}
2013-10-11 11:47:31 +04:00
2014-10-23 19:49:29 +04:00
int mac_smack_apply ( const char * path , const char * label ) {
2014-10-23 20:06:51 +04:00
int r = 0 ;
assert ( path ) ;
2013-10-11 11:47:31 +04:00
# ifdef HAVE_SMACK
2014-10-23 19:34:30 +04:00
if ( ! mac_smack_use ( ) )
2013-10-11 11:47:31 +04:00
return 0 ;
if ( label )
2014-10-23 20:32:22 +04:00
r = lsetxattr ( path , " security.SMACK64 " , label , strlen ( label ) , 0 ) ;
2013-10-11 11:47:31 +04:00
else
2014-10-23 20:06:51 +04:00
r = lremovexattr ( path , " security.SMACK64 " ) ;
if ( r < 0 )
return - errno ;
2013-10-11 11:47:31 +04:00
# endif
2014-10-23 20:06:51 +04:00
return r ;
2013-10-11 11:47:31 +04:00
}
2014-10-23 19:49:29 +04:00
int mac_smack_apply_fd ( int fd , const char * label ) {
2014-10-23 20:06:51 +04:00
int r = 0 ;
assert ( fd > = 0 ) ;
2013-10-11 11:47:31 +04:00
# ifdef HAVE_SMACK
2014-10-23 19:34:30 +04:00
if ( ! mac_smack_use ( ) )
2013-10-11 11:47:31 +04:00
return 0 ;
2014-10-23 20:06:51 +04:00
if ( label )
r = fsetxattr ( fd , " security.SMACK64 " , label , strlen ( label ) , 0 ) ;
else
r = fremovexattr ( fd , " security.SMACK64 " ) ;
if ( r < 0 )
return - errno ;
2013-10-11 11:47:31 +04:00
# endif
2014-10-23 20:06:51 +04:00
return r ;
2013-10-11 11:47:31 +04:00
}
2014-10-23 19:49:29 +04:00
int mac_smack_apply_ip_out_fd ( int fd , const char * label ) {
2014-10-23 20:06:51 +04:00
int r = 0 ;
assert ( fd > = 0 ) ;
2013-10-11 11:47:31 +04:00
# ifdef HAVE_SMACK
2014-10-23 19:34:30 +04:00
if ( ! mac_smack_use ( ) )
2013-10-11 11:47:31 +04:00
return 0 ;
2014-10-23 20:06:51 +04:00
if ( label )
r = fsetxattr ( fd , " security.SMACK64IPOUT " , label , strlen ( label ) , 0 ) ;
else
r = fremovexattr ( fd , " security.SMACK64IPOUT " ) ;
if ( r < 0 )
return - errno ;
2013-10-11 11:47:31 +04:00
# endif
2014-10-23 20:06:51 +04:00
return r ;
2013-10-11 11:47:31 +04:00
}
2014-10-23 19:49:29 +04:00
int mac_smack_apply_ip_in_fd ( int fd , const char * label ) {
2014-10-23 20:06:51 +04:00
int r = 0 ;
assert ( fd > = 0 ) ;
2013-10-11 11:47:31 +04:00
# ifdef HAVE_SMACK
2014-10-23 19:34:30 +04:00
if ( ! mac_smack_use ( ) )
2013-10-11 11:47:31 +04:00
return 0 ;
2014-10-23 20:06:51 +04:00
if ( label )
r = fsetxattr ( fd , " security.SMACK64IPIN " , label , strlen ( label ) , 0 ) ;
else
r = fremovexattr ( fd , " security.SMACK64IPIN " ) ;
if ( r < 0 )
return - errno ;
2013-10-11 11:47:31 +04:00
# endif
2014-10-23 20:06:51 +04:00
return r ;
2013-10-11 11:47:31 +04:00
}
2014-10-23 12:23:45 +04:00
2014-11-24 14:46:20 +03:00
int mac_smack_apply_pid ( pid_t pid , const char * label ) {
2014-12-04 18:17:18 +03:00
# ifdef HAVE_SMACK
2014-11-24 14:46:20 +03:00
const char * p ;
2014-12-04 18:17:18 +03:00
# endif
int r = 0 ;
2014-11-24 14:46:20 +03:00
assert ( label ) ;
# ifdef HAVE_SMACK
if ( ! mac_smack_use ( ) )
return 0 ;
p = procfs_file_alloca ( pid , " attr/current " ) ;
r = write_string_file ( p , label ) ;
if ( r < 0 )
return r ;
# endif
return r ;
}
2014-10-23 20:34:58 +04:00
int mac_smack_fix ( const char * path , bool ignore_enoent , bool ignore_erofs ) {
2014-10-23 12:23:45 +04:00
# ifdef HAVE_SMACK
2014-10-23 20:34:58 +04:00
struct stat st ;
2014-12-04 18:17:18 +03:00
# endif
int r = 0 ;
2014-10-23 20:06:51 +04:00
assert ( path ) ;
2014-12-04 18:17:18 +03:00
# ifdef HAVE_SMACK
2014-10-23 20:06:51 +04:00
if ( ! mac_smack_use ( ) )
return 0 ;
2014-10-23 12:23:45 +04:00
/*
* Path must be in / dev and must exist
*/
if ( ! path_startswith ( path , " /dev " ) )
return 0 ;
2014-10-23 20:34:58 +04:00
r = lstat ( path , & st ) ;
if ( r > = 0 ) {
const char * label ;
/*
* Label directories and character devices " * " .
* Label symlinks " _ " .
* Don ' t change anything else .
*/
if ( S_ISDIR ( st . st_mode ) )
label = SMACK_STAR_LABEL ;
else if ( S_ISLNK ( st . st_mode ) )
label = SMACK_FLOOR_LABEL ;
else if ( S_ISCHR ( st . st_mode ) )
label = SMACK_STAR_LABEL ;
else
return 0 ;
2014-10-23 12:23:45 +04:00
2014-10-23 20:34:58 +04:00
r = lsetxattr ( path , " security.SMACK64 " , label , strlen ( label ) , 0 ) ;
/* If the FS doesn't support labels, then exit without warning */
2015-03-13 16:08:00 +03:00
if ( r < 0 & & errno = = EOPNOTSUPP )
2014-10-23 20:34:58 +04:00
return 0 ;
}
2014-10-23 12:23:45 +04:00
if ( r < 0 ) {
2014-10-23 20:34:58 +04:00
/* Ignore ENOENT in some cases */
if ( ignore_enoent & & errno = = ENOENT )
return 0 ;
if ( ignore_erofs & & errno = = EROFS )
return 0 ;
2014-12-04 18:17:18 +03:00
r = log_debug_errno ( errno , " Unable to fix SMACK label of %s: %m " , path ) ;
2014-10-23 12:23:45 +04:00
}
# endif
return r ;
}