2006-03-17 03:52:15 +03:00
/*
2008-09-30 15:43:35 +04:00
* Copyright ( C ) 2008 Kay Sievers < kayi . sievers @ vrfy . org >
2006-03-17 03:52:15 +03:00
*
2008-09-10 04:40:42 +04:00
* This program 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 .
2006-03-17 03:52:15 +03:00
*
2008-09-10 04:40:42 +04:00
* This program 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 this program . If not , see < http : //www.gnu.org/licenses/>.
2006-03-17 03:52:15 +03:00
*/
# include <stdlib.h>
# include <stddef.h>
# include <string.h>
# include <stdio.h>
# include <unistd.h>
2006-09-03 05:04:20 +04:00
# include <getopt.h>
2006-03-17 03:52:15 +03:00
# include <errno.h>
# include <dirent.h>
# include <fcntl.h>
# include <syslog.h>
2006-09-03 05:04:20 +04:00
# include <fnmatch.h>
2006-03-17 03:52:15 +03:00
# include <sys/stat.h>
# include <sys/types.h>
2008-03-27 02:58:20 +03:00
# include <sys/socket.h>
# include <sys/un.h>
2006-03-17 03:52:15 +03:00
# include "udev.h"
2006-09-19 19:51:31 +04:00
static int verbose ;
static int dry_run ;
2008-09-30 15:43:35 +04:00
static void exec_list ( struct udev_enumerate * udev_enumerate , const char * action )
2006-09-19 19:51:31 +04:00
{
2008-09-30 15:43:35 +04:00
struct udev * udev = udev_enumerate_get_udev ( udev_enumerate ) ;
struct udev_list_entry * entry ;
2008-03-27 02:58:20 +03:00
2008-09-30 15:43:35 +04:00
udev_list_entry_foreach ( entry , udev_enumerate_get_list_entry ( udev_enumerate ) ) {
char filename [ UTIL_PATH_SIZE ] ;
2006-09-03 05:04:20 +04:00
int fd ;
2008-09-30 15:43:35 +04:00
if ( verbose )
printf ( " %s \n " , udev_list_entry_get_name ( entry ) ) ;
if ( dry_run )
continue ;
util_strlcpy ( filename , udev_list_entry_get_name ( entry ) , sizeof ( filename ) ) ;
util_strlcat ( filename , " /uevent " , sizeof ( filename ) ) ;
fd = open ( filename , O_WRONLY ) ;
if ( fd < 0 ) {
dbg ( udev , " error on opening %s: %m \n " , filename ) ;
continue ;
2006-03-17 03:52:15 +03:00
}
2008-09-30 15:43:35 +04:00
if ( write ( fd , action , strlen ( action ) ) < 0 )
info ( udev , " error writing '%s' to '%s': %m \n " , action , filename ) ;
close ( fd ) ;
2006-03-17 03:52:15 +03:00
}
}
2008-10-01 16:41:43 +04:00
static int scan_failed ( struct udev_enumerate * udev_enumerate )
2006-08-20 20:22:44 +04:00
{
2008-09-30 15:43:35 +04:00
struct udev * udev = udev_enumerate_get_udev ( udev_enumerate ) ;
2008-10-01 16:41:43 +04:00
struct udev_queue * udev_queue ;
struct udev_list_entry * list_entry ;
2006-08-20 20:22:44 +04:00
2008-10-01 16:41:43 +04:00
udev_queue = udev_queue_new ( udev ) ;
if ( udev_queue = = NULL )
return - 1 ;
2008-10-02 13:54:33 +04:00
udev_list_entry_foreach ( list_entry , udev_queue_get_failed_list_entry ( udev_queue ) )
udev_enumerate_add_syspath ( udev_enumerate , udev_list_entry_get_name ( list_entry ) ) ;
2008-09-30 15:43:35 +04:00
return 0 ;
2006-08-20 20:22:44 +04:00
}
2008-09-06 17:45:31 +04:00
int udevadm_trigger ( struct udev * udev , int argc , char * argv [ ] )
2006-03-17 03:52:15 +03:00
{
2006-09-03 16:12:51 +04:00
static const struct option options [ ] = {
2006-09-03 05:04:20 +04:00
{ " verbose " , 0 , NULL , ' v ' } ,
{ " dry-run " , 0 , NULL , ' n ' } ,
2008-09-30 18:25:46 +04:00
{ " type " , 1 , NULL , ' t ' } ,
2006-09-03 05:04:20 +04:00
{ " retry-failed " , 0 , NULL , ' F ' } ,
2007-07-09 05:59:08 +04:00
{ " action " , 1 , NULL , ' c ' } ,
2006-09-03 05:04:20 +04:00
{ " subsystem-match " , 1 , NULL , ' s ' } ,
{ " subsystem-nomatch " , 1 , NULL , ' S ' } ,
{ " attr-match " , 1 , NULL , ' a ' } ,
{ " attr-nomatch " , 1 , NULL , ' A ' } ,
2008-09-30 15:43:35 +04:00
{ " help " , 0 , NULL , ' h ' } ,
2006-09-03 05:04:20 +04:00
{ }
} ;
2008-09-30 15:43:35 +04:00
enum {
TYPE_DEVICES ,
TYPE_SUBSYSTEMS ,
TYPE_FAILED ,
} device_type = TYPE_DEVICES ;
const char * action = " add " ;
struct udev_enumerate * udev_enumerate ;
int rc = 0 ;
2006-03-17 03:52:15 +03:00
2008-09-06 17:45:31 +04:00
dbg ( udev , " version %s \n " , VERSION ) ;
2008-09-30 15:43:35 +04:00
udev_enumerate = udev_enumerate_new ( udev ) ;
if ( udev_enumerate = = NULL ) {
rc = 1 ;
goto exit ;
}
2006-03-17 03:52:15 +03:00
2006-09-03 05:04:20 +04:00
while ( 1 ) {
2008-09-30 15:43:35 +04:00
int option ;
char attr [ UTIL_PATH_SIZE ] ;
char * val ;
option = getopt_long ( argc , argv , " vnFo:t:hce::s:S:a:A: " , options , NULL ) ;
2006-09-03 05:04:20 +04:00
if ( option = = - 1 )
break ;
2006-03-17 03:52:15 +03:00
2006-09-03 05:04:20 +04:00
switch ( option ) {
case ' v ' :
2006-03-17 03:52:15 +03:00
verbose = 1 ;
2006-09-03 05:04:20 +04:00
break ;
case ' n ' :
2006-03-17 03:52:15 +03:00
dry_run = 1 ;
2006-09-03 05:04:20 +04:00
break ;
2008-09-30 15:43:35 +04:00
case ' t ' :
if ( strcmp ( optarg , " devices " ) = = 0 ) {
device_type = TYPE_DEVICES ;
} else if ( strcmp ( optarg , " subsystems " ) = = 0 ) {
device_type = TYPE_SUBSYSTEMS ;
} else if ( strcmp ( optarg , " failed " ) = = 0 ) {
device_type = TYPE_FAILED ;
} else {
fprintf ( stderr , " unknown type --type=%s \n " , optarg ) ;
err ( udev , " unknown type --type=%s \n " , optarg ) ;
rc = 2 ;
goto exit ;
}
2006-09-03 05:04:20 +04:00
break ;
2008-09-30 15:43:35 +04:00
case ' F ' :
device_type = TYPE_FAILED ;
2008-03-27 02:58:20 +03:00
break ;
2007-07-09 05:59:08 +04:00
case ' c ' :
action = optarg ;
break ;
2006-09-03 05:04:20 +04:00
case ' s ' :
2008-09-30 15:43:35 +04:00
udev_enumerate_add_match_subsystem ( udev_enumerate , optarg ) ;
2006-09-03 05:04:20 +04:00
break ;
case ' S ' :
2008-09-30 15:43:35 +04:00
udev_enumerate_add_nomatch_subsystem ( udev_enumerate , optarg ) ;
2006-09-03 05:04:20 +04:00
break ;
case ' a ' :
2008-09-30 15:43:35 +04:00
util_strlcpy ( attr , optarg , sizeof ( attr ) ) ;
val = strchr ( attr , ' = ' ) ;
if ( val ! = NULL ) {
val [ 0 ] = 0 ;
val = & val [ 1 ] ;
}
udev_enumerate_add_match_attr ( udev_enumerate , attr , val ) ;
2006-09-03 05:04:20 +04:00
break ;
case ' A ' :
2008-09-30 15:43:35 +04:00
util_strlcpy ( attr , optarg , sizeof ( attr ) ) ;
val = strchr ( attr , ' = ' ) ;
if ( val ! = NULL ) {
val [ 0 ] = 0 ;
val = & val [ 1 ] ;
}
udev_enumerate_add_nomatch_attr ( udev_enumerate , attr , val ) ;
2006-09-03 05:04:20 +04:00
break ;
case ' h ' :
2007-11-08 19:51:59 +03:00
printf ( " Usage: udevadm trigger OPTIONS \n "
2006-09-03 07:45:51 +04:00
" --verbose print the list of devices while running \n "
" --dry-run do not actually trigger the events \n "
2008-09-30 15:43:35 +04:00
" --type= type of events to trigger \n "
" devices sys devices \n "
" subsystems sys subsystems and drivers \n "
" failed trigger only the events which have been \n "
" marked as failed during a previous run \n "
2006-09-03 07:45:51 +04:00
" --subsystem-match=<subsystem> trigger devices from a matching subystem \n "
" --subsystem-nomatch=<subsystem> exclude devices from a matching subystem \n "
2008-09-30 15:43:35 +04:00
" --attr-match=<file[=<value>]> trigger devices with a matching attribute \n "
" --attr-nomatch=<file[=<value>]> exclude devices with a matching attribute \n "
2006-09-03 07:45:51 +04:00
" --help print this text \n "
2006-09-03 05:04:20 +04:00
" \n " ) ;
goto exit ;
default :
2006-08-20 20:22:44 +04:00
goto exit ;
2006-03-17 03:52:15 +03:00
}
}
2008-09-30 15:43:35 +04:00
switch ( device_type ) {
case TYPE_FAILED :
2008-10-01 16:41:43 +04:00
scan_failed ( udev_enumerate ) ;
2008-09-30 15:43:35 +04:00
exec_list ( udev_enumerate , action ) ;
goto exit ;
case TYPE_SUBSYSTEMS :
udev_enumerate_scan_subsystems ( udev_enumerate ) ;
exec_list ( udev_enumerate , action ) ;
goto exit ;
case TYPE_DEVICES :
udev_enumerate_scan_devices ( udev_enumerate ) ;
exec_list ( udev_enumerate , action ) ;
goto exit ;
default :
2008-04-22 05:01:29 +04:00
goto exit ;
2008-03-27 02:58:20 +03:00
}
2006-03-17 03:52:15 +03:00
exit :
2008-09-30 15:43:35 +04:00
udev_enumerate_unref ( udev_enumerate ) ;
return rc ;
2006-03-17 03:52:15 +03:00
}