2014-06-24 15:30:11 +04:00
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd .
Copyright 2014 Kay Sievers
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/>.
* * */
# include <errno.h>
# include <sys/stat.h>
# include <stdlib.h>
# include <string.h>
# include <assert.h>
# include <unistd.h>
# include "base-filesystem.h"
# include "log.h"
# include "macro.h"
2014-07-01 13:42:58 +04:00
# include "strv.h"
2014-06-24 15:30:11 +04:00
# include "util.h"
# include "label.h"
# include "mkdir.h"
typedef struct BaseFilesystem {
const char * dir ;
mode_t mode ;
const char * target ;
2014-07-01 14:12:40 +04:00
const char * exists ;
2014-06-24 15:30:11 +04:00
} BaseFilesystem ;
static const BaseFilesystem table [ ] = {
2014-07-07 19:45:53 +04:00
{ " bin " , 0 , " usr/bin \0 " , NULL } ,
{ " lib " , 0 , " usr/lib \0 " , NULL } ,
{ " root " , 0755 , NULL , NULL } ,
{ " sbin " , 0 , " usr/sbin \0 " , NULL } ,
2014-07-01 13:42:58 +04:00
# if defined(__i386__) || defined(__x86_64__)
2014-07-07 19:45:53 +04:00
{ " lib64 " , 0 , " usr/lib/x86_64-linux-gnu \0 "
" usr/lib64 \0 " , " ld-linux-x86-64.so.2 " } ,
2014-07-01 13:42:58 +04:00
# endif
2014-06-24 15:30:11 +04:00
} ;
int base_filesystem_create ( const char * root ) {
_cleanup_close_ int fd = - 1 ;
unsigned i ;
int r ;
fd = open ( root , O_RDONLY | O_NONBLOCK | O_DIRECTORY | O_CLOEXEC | O_NOFOLLOW ) ;
if ( fd < 0 )
return - errno ;
for ( i = 0 ; i < ELEMENTSOF ( table ) ; i + + ) {
if ( table [ i ] . target ) {
2014-07-01 13:42:58 +04:00
const char * target = NULL ;
const char * s ;
2014-07-01 14:25:38 +04:00
if ( faccessat ( fd , table [ i ] . dir , F_OK , AT_SYMLINK_NOFOLLOW ) > = 0 )
continue ;
2014-07-01 13:42:58 +04:00
/* check if one of the targets exists */
NULSTR_FOREACH ( s , table [ i ] . target ) {
if ( faccessat ( fd , s , F_OK , AT_SYMLINK_NOFOLLOW ) < 0 )
continue ;
2014-07-01 14:12:40 +04:00
/* check if a specific file exists at the target path */
if ( table [ i ] . exists ) {
_cleanup_free_ char * p = NULL ;
p = strjoin ( s , " / " , table [ i ] . exists , NULL ) ;
if ( ! p )
return log_oom ( ) ;
if ( faccessat ( fd , p , F_OK , AT_SYMLINK_NOFOLLOW ) < 0 )
continue ;
}
2014-07-01 13:42:58 +04:00
target = s ;
break ;
}
if ( ! target )
2014-06-24 15:30:11 +04:00
continue ;
2014-07-01 13:42:58 +04:00
r = symlinkat ( target , fd , table [ i ] . dir ) ;
2014-06-24 15:30:11 +04:00
if ( r < 0 & & errno ! = EEXIST ) {
log_error ( " Failed to create symlink at %s/%s: %m " , root , table [ i ] . dir ) ;
return - errno ;
}
continue ;
}
RUN_WITH_UMASK ( 0000 )
r = mkdirat ( fd , table [ i ] . dir , table [ i ] . mode ) ;
if ( r < 0 & & errno ! = EEXIST ) {
log_error ( " Failed to create directory at %s/%s: %m " , root , table [ i ] . dir ) ;
return - errno ;
}
}
return 0 ;
}