2010-03-31 18:29:55 +04:00
/*-*- Mode: C; c-basic-offset: 8 -*-*/
/***
This file is part of systemd .
Copyright 2010 Lennart Poettering
systemd is free software ; you can redistribute it and / or modify it
under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 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
General Public License for more details .
You should have received a copy of the GNU General Public License
along with systemd ; If not , see < http : //www.gnu.org/licenses/>.
* * */
# include <sys/mount.h>
# include <errno.h>
# include <sys/stat.h>
# include <stdlib.h>
# include <string.h>
# include <libgen.h>
# include <assert.h>
# include "mount-setup.h"
# include "log.h"
2010-04-08 03:43:07 +04:00
# include "macro.h"
# include "util.h"
2010-03-31 18:29:55 +04:00
2010-04-12 23:58:01 +04:00
typedef struct MountPoint {
const char * what ;
const char * where ;
const char * type ;
const char * options ;
unsigned long flags ;
2010-04-17 01:22:32 +04:00
bool fatal ;
2010-04-12 23:58:01 +04:00
} MountPoint ;
static const MountPoint mount_table [ ] = {
2010-04-17 01:35:30 +04:00
{ " proc " , " /proc " , " proc " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , true } ,
{ " sysfs " , " /sys " , " sysfs " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , true } ,
{ " devtmps " , " /dev " , " devtmpfs " , " mode=755 " , MS_NOSUID , true } ,
{ " tmpfs " , " /dev/shm " , " tmpfs " , " mode=1777 " , MS_NOSUID | MS_NOEXEC | MS_NODEV , true } ,
{ " devpts " , " /dev/pts " , " devpts " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , false } ,
{ " cgroup " , " /cgroup/debug " , " cgroup " , " debug " , MS_NOSUID | MS_NOEXEC | MS_NODEV , true } ,
{ " debugfs " , " /sys/kernel/debug " , " debugfs " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , false } ,
{ " binfmt_misc " , " /proc/sys/fs/binfmt_misc " , " binfmt_misc " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , false } ,
2010-04-21 16:04:32 +04:00
{ " mqueue " , " /dev/mqueue " , " mqueue " , NULL , MS_NOSUID | MS_NOEXEC | MS_NODEV , false } ,
2010-03-31 18:29:55 +04:00
} ;
2010-04-10 19:42:00 +04:00
bool mount_point_is_api ( const char * path ) {
unsigned i ;
/* Checks if this mount point is considered "API", and hence
* should be ignored */
2010-04-12 23:58:01 +04:00
for ( i = 0 ; i < ELEMENTSOF ( mount_table ) ; i + + )
if ( path_startswith ( path , mount_table [ i ] . where ) )
2010-04-10 19:42:00 +04:00
return true ;
2010-04-17 01:35:44 +04:00
return path_startswith ( path , " /cgroup/ " ) ;
2010-04-10 19:42:00 +04:00
}
2010-04-12 23:58:01 +04:00
static int mount_one ( const MountPoint * p ) {
2010-03-31 18:29:55 +04:00
int r ;
2010-04-12 23:58:01 +04:00
assert ( p ) ;
2010-03-31 18:29:55 +04:00
2010-04-17 01:24:39 +04:00
if ( ( r = path_is_mount_point ( p - > where ) ) < 0 )
2010-03-31 18:29:55 +04:00
return r ;
if ( r > 0 )
return 0 ;
2010-04-06 23:55:58 +04:00
/* The access mode here doesn't really matter too much, since
* the mounted file system will take precedence anyway . */
2010-04-12 23:58:01 +04:00
mkdir_p ( p - > where , 0755 ) ;
2010-04-06 23:55:58 +04:00
2010-03-31 18:29:55 +04:00
log_debug ( " Mounting %s to %s of type %s with options %s. " ,
2010-04-12 23:58:01 +04:00
p - > what ,
p - > where ,
p - > type ,
strna ( p - > options ) ) ;
if ( mount ( p - > what ,
p - > where ,
p - > type ,
p - > flags ,
p - > options ) < 0 ) {
log_error ( " Failed to mount %s: %s " , p - > where , strerror ( errno ) ) ;
2010-04-17 01:22:32 +04:00
return p - > fatal ? - errno : 0 ;
2010-03-31 18:29:55 +04:00
}
return 0 ;
}
2010-04-17 01:22:32 +04:00
static int mount_cgroup_controllers ( void ) {
int r ;
FILE * f ;
char buf [ 256 ] ;
/* Mount all available cgroup controllers. */
if ( ! ( f = fopen ( " /proc/cgroups " , " re " ) ) )
return - ENOENT ;
/* Ignore the header line */
fgets ( buf , sizeof ( buf ) , f ) ;
for ( ; ; ) {
MountPoint p ;
char * controller , * where ;
if ( fscanf ( f , " %ms %*i %*i %*i " , & controller ) ! = 1 ) {
if ( feof ( f ) )
break ;
log_error ( " Failed to parse /proc/cgroups. " ) ;
r = - EIO ;
goto finish ;
}
if ( asprintf ( & where , " /cgroup/%s " , controller ) < 0 ) {
free ( controller ) ;
r = - ENOMEM ;
goto finish ;
}
zero ( p ) ;
p . what = " cgroup " ;
p . where = where ;
p . type = " cgroup " ;
p . options = controller ;
p . flags = MS_NOSUID | MS_NOEXEC | MS_NODEV ;
p . fatal = false ;
r = mount_one ( & p ) ;
free ( controller ) ;
free ( where ) ;
if ( r < 0 )
goto finish ;
}
r = 0 ;
finish :
fclose ( f ) ;
return r ;
}
2010-03-31 18:29:55 +04:00
int mount_setup ( void ) {
int r ;
2010-04-10 19:42:00 +04:00
unsigned i ;
2010-03-31 18:29:55 +04:00
2010-04-12 23:58:01 +04:00
for ( i = 0 ; i < ELEMENTSOF ( mount_table ) ; i + + )
if ( ( r = mount_one ( mount_table + i ) ) < 0 )
2010-03-31 18:29:55 +04:00
return r ;
2010-04-17 01:22:32 +04:00
return mount_cgroup_controllers ( ) ;
2010-03-31 18:29:55 +04:00
}