2012-11-12 22:36:23 +04:00
/***
This file is part of systemd .
Copyright 2003 - 2004 Greg Kroah - Hartman < greg @ kroah . com >
Copyright 2004 - 2012 Kay Sievers < kay @ vrfy . org >
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/>.
* * */
2005-04-27 07:59:47 +04:00
2004-03-23 09:22:20 +03:00
# include <stdio.h>
# include <stddef.h>
2005-04-27 07:59:47 +04:00
# include <stdlib.h>
# include <string.h>
2004-11-23 09:34:56 +03:00
# include <fcntl.h>
2003-12-03 12:08:46 +03:00
# include <ctype.h>
2004-03-23 09:22:20 +03:00
# include <errno.h>
2004-10-19 10:14:20 +04:00
# include <unistd.h>
2005-08-11 19:32:59 +04:00
# include <syslog.h>
2008-09-01 22:59:09 +04:00
# include <grp.h>
2012-07-16 00:10:46 +04:00
# include <sched.h>
# include <sys/mount.h>
2011-04-20 03:53:03 +04:00
# include <sys/signalfd.h>
2003-04-10 21:53:46 +04:00
2013-02-13 11:50:15 +04:00
# include "missing.h"
2005-04-27 07:59:47 +04:00
# include "udev.h"
2013-10-13 04:28:21 +04:00
# include "udev-util.h"
2005-04-27 07:59:47 +04:00
2011-12-25 23:41:52 +04:00
void udev_main_log ( struct udev * udev , int priority ,
2012-01-10 04:34:15 +04:00
const char * file , int line , const char * fn ,
const char * format , va_list args ) { }
2011-12-25 23:41:52 +04:00
2012-07-16 00:10:46 +04:00
static int fake_filesystems ( void ) {
static const struct fakefs {
const char * src ;
const char * target ;
const char * error ;
} fakefss [ ] = {
{ " test/sys " , " /sys " , " failed to mount test /sys " } ,
{ " test/dev " , " /dev " , " failed to mount test /dev " } ,
{ " test/run " , " /run " , " failed to mount test /run " } ,
{ " test/run " , " /etc/udev/rules.d " , " failed to mount empty /etc/udev/rules.d " } ,
{ " test/run " , " /usr/lib/udev/rules.d " , " failed to mount empty /usr/lib/udev/rules.d " } ,
} ;
unsigned int i ;
int err ;
err = unshare ( CLONE_NEWNS ) ;
if ( err < 0 ) {
err = - errno ;
2012-10-08 22:50:20 +04:00
fprintf ( stderr , " failed to call unshare(): %m \n " ) ;
goto out ;
}
if ( mount ( NULL , " / " , NULL , MS_PRIVATE | MS_REC , NULL ) < 0 ) {
err = - errno ;
fprintf ( stderr , " failed to mount / as private: %m \n " ) ;
goto out ;
2012-07-16 00:10:46 +04:00
}
for ( i = 0 ; i < ELEMENTSOF ( fakefss ) ; i + + ) {
err = mount ( fakefss [ i ] . src , fakefss [ i ] . target , NULL , MS_BIND , NULL ) ;
if ( err < 0 ) {
err = - errno ;
fprintf ( stderr , " %s %m " , fakefss [ i ] . error ) ;
return err ;
}
}
2012-10-08 22:50:20 +04:00
out :
2012-07-16 00:10:46 +04:00
return err ;
}
2013-12-04 01:27:45 +04:00
int main ( int argc , char * argv [ ] ) {
2013-10-13 04:28:21 +04:00
_cleanup_udev_unref_ struct udev * udev = NULL ;
_cleanup_udev_event_unref_ struct udev_event * event = NULL ;
_cleanup_udev_device_unref_ struct udev_device * dev = NULL ;
_cleanup_udev_rules_unref_ struct udev_rules * rules = NULL ;
2012-01-10 04:34:15 +04:00
char syspath [ UTIL_PATH_SIZE ] ;
const char * devpath ;
const char * action ;
sigset_t mask , sigmask_orig ;
2012-07-16 00:10:46 +04:00
int err ;
err = fake_filesystems ( ) ;
if ( err < 0 )
return EXIT_FAILURE ;
2012-01-10 04:34:15 +04:00
udev = udev_new ( ) ;
if ( udev = = NULL )
2013-10-13 04:28:21 +04:00
return EXIT_FAILURE ;
2012-04-08 18:06:20 +04:00
log_debug ( " version %s \n " , VERSION ) ;
2012-05-31 15:34:41 +04:00
label_init ( " /dev " ) ;
2012-01-10 04:34:15 +04:00
sigprocmask ( SIG_SETMASK , NULL , & sigmask_orig ) ;
action = argv [ 1 ] ;
if ( action = = NULL ) {
2012-04-08 18:06:20 +04:00
log_error ( " action missing \n " ) ;
2012-01-10 04:34:15 +04:00
goto out ;
}
devpath = argv [ 2 ] ;
if ( devpath = = NULL ) {
2012-04-08 18:06:20 +04:00
log_error ( " devpath missing \n " ) ;
2012-01-10 04:34:15 +04:00
goto out ;
}
rules = udev_rules_new ( udev , 1 ) ;
2013-01-09 22:06:46 +04:00
strscpyl ( syspath , sizeof ( syspath ) , " /sys " , devpath , NULL ) ;
2012-01-10 04:34:15 +04:00
dev = udev_device_new_from_syspath ( udev , syspath ) ;
if ( dev = = NULL ) {
2012-04-08 18:06:20 +04:00
log_debug ( " unknown device '%s' \n " , devpath ) ;
2012-01-10 04:34:15 +04:00
goto out ;
}
udev_device_set_action ( dev , action ) ;
event = udev_event_new ( dev ) ;
sigfillset ( & mask ) ;
sigprocmask ( SIG_SETMASK , & mask , & sigmask_orig ) ;
event - > fd_signal = signalfd ( - 1 , & mask , SFD_NONBLOCK | SFD_CLOEXEC ) ;
if ( event - > fd_signal < 0 ) {
fprintf ( stderr , " error creating signalfd \n " ) ;
goto out ;
}
/* do what devtmpfs usually provides us */
if ( udev_device_get_devnode ( dev ) ! = NULL ) {
2012-04-09 18:37:54 +04:00
mode_t mode = 0600 ;
2012-01-10 04:34:15 +04:00
2013-02-13 21:13:22 +04:00
if ( streq ( udev_device_get_subsystem ( dev ) , " block " ) )
2012-01-10 04:34:15 +04:00
mode | = S_IFBLK ;
else
mode | = S_IFCHR ;
2013-02-13 21:13:22 +04:00
if ( ! streq ( action , " remove " ) ) {
2012-05-31 14:40:20 +04:00
mkdir_parents_label ( udev_device_get_devnode ( dev ) , 0755 ) ;
2012-01-10 04:34:15 +04:00
mknod ( udev_device_get_devnode ( dev ) , mode , udev_device_get_devnum ( dev ) ) ;
} else {
unlink ( udev_device_get_devnode ( dev ) ) ;
util_delete_path ( udev , udev_device_get_devnode ( dev ) ) ;
}
}
err = udev_event_execute_rules ( event , rules , & sigmask_orig ) ;
if ( err = = 0 )
udev_event_execute_run ( event , NULL ) ;
2011-04-20 03:53:03 +04:00
out :
2012-01-10 04:34:15 +04:00
if ( event ! = NULL & & event - > fd_signal > = 0 )
close ( event - > fd_signal ) ;
2012-04-17 18:05:03 +04:00
label_finish ( ) ;
2013-10-13 04:28:21 +04:00
return err ? EXIT_FAILURE : EXIT_SUCCESS ;
2005-04-27 07:59:47 +04:00
}