2004-02-13 06:48:07 +03:00
/*
2005-07-22 20:35:58 +04:00
* Copyright ( C ) 2003 - 2004 Greg Kroah - Hartman < greg @ kroah . com >
2008-09-10 04:40:42 +04:00
* Copyright ( C ) 2004 - 2008 Kay Sievers < kay . sievers @ vrfy . org >
2004-02-13 06:48:07 +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 .
2004-02-13 06:48:07 +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/>.
2004-02-13 06:48:07 +03:00
*/
# include <stdlib.h>
# include <string.h>
# include <stdio.h>
2006-01-09 23:18:00 +03:00
# include <stddef.h>
# include <unistd.h>
2004-02-13 06:48:07 +03:00
# include <errno.h>
# include <ctype.h>
2007-04-06 21:18:33 +04:00
# include <fcntl.h>
2004-02-13 06:48:07 +03:00
# include <signal.h>
2005-03-27 03:11:03 +04:00
# include <syslog.h>
2007-03-21 13:55:26 +03:00
# include <getopt.h>
2011-04-20 03:53:03 +04:00
# include <sys/signalfd.h>
2004-02-13 06:48:07 +03:00
# include "udev.h"
2008-09-06 17:45:31 +04:00
int udevadm_test ( struct udev * udev , int argc , char * argv [ ] )
2004-02-13 06:48:07 +03:00
{
2011-04-20 03:53:03 +04:00
int resolve_names = 1 ;
2008-10-16 19:16:58 +04:00
char filename [ UTIL_PATH_SIZE ] ;
2007-07-09 05:59:08 +04:00
const char * action = " add " ;
2008-10-16 19:16:58 +04:00
const char * syspath = NULL ;
2011-04-20 03:53:03 +04:00
struct udev_event * event = NULL ;
struct udev_device * dev = NULL ;
2008-10-18 17:02:01 +04:00
struct udev_rules * rules = NULL ;
2009-07-03 00:56:19 +04:00
struct udev_list_entry * entry ;
2011-04-20 03:53:03 +04:00
sigset_t mask , sigmask_orig ;
2008-10-16 19:16:58 +04:00
int err ;
2006-01-09 23:18:00 +03:00
int rc = 0 ;
2004-03-23 09:18:34 +03:00
2007-03-21 13:55:26 +03:00
static const struct option options [ ] = {
2008-10-02 18:49:05 +04:00
{ " action " , required_argument , NULL , ' a ' } ,
2011-04-20 03:53:03 +04:00
{ " resolve-names " , required_argument , NULL , ' N ' } ,
2008-10-02 18:49:05 +04:00
{ " help " , no_argument , NULL , ' h ' } ,
2007-03-21 13:55:26 +03:00
{ }
} ;
2008-09-06 17:45:31 +04:00
info ( udev , " version %s \n " , VERSION ) ;
2010-05-27 17:11:00 +04:00
for ( ; ; ) {
2007-03-21 13:55:26 +03:00
int option ;
2011-04-20 03:53:03 +04:00
option = getopt_long ( argc , argv , " a:s:N:fh " , options , NULL ) ;
2007-03-21 13:55:26 +03:00
if ( option = = - 1 )
break ;
2008-09-06 17:45:31 +04:00
dbg ( udev , " option '%c' \n " , option ) ;
2007-03-21 13:55:26 +03:00
switch ( option ) {
case ' a ' :
action = optarg ;
break ;
2011-04-20 03:53:03 +04:00
case ' N ' :
if ( strcmp ( optarg , " early " ) = = 0 ) {
resolve_names = 1 ;
} else if ( strcmp ( optarg , " late " ) = = 0 ) {
resolve_names = 0 ;
} else if ( strcmp ( optarg , " never " ) = = 0 ) {
resolve_names = - 1 ;
} else {
fprintf ( stderr , " resolve-names must be early, late or never \n " ) ;
err ( udev , " resolve-names must be early, late or never \n " ) ;
exit ( EXIT_FAILURE ) ;
}
break ;
2007-03-21 13:55:26 +03:00
case ' h ' :
2008-10-16 19:16:58 +04:00
printf ( " Usage: udevadm test OPTIONS <syspath> \n "
2007-06-02 12:14:50 +04:00
" --action=<string> set action string \n "
2009-03-07 04:32:40 +03:00
" --help \n \n " ) ;
2011-04-20 03:53:03 +04:00
exit ( EXIT_SUCCESS ) ;
2007-03-21 13:55:26 +03:00
default :
2011-04-20 03:53:03 +04:00
exit ( EXIT_FAILURE ) ;
2007-03-21 13:55:26 +03:00
}
2006-08-21 04:38:20 +04:00
}
2008-10-16 19:16:58 +04:00
syspath = argv [ optind ] ;
2006-08-21 04:38:20 +04:00
2008-10-16 19:16:58 +04:00
if ( syspath = = NULL ) {
fprintf ( stderr , " syspath parameter missing \n " ) ;
2011-04-20 03:53:03 +04:00
rc = 2 ;
goto out ;
2005-03-29 13:30:51 +04:00
}
2007-05-21 09:02:04 +04:00
printf ( " This program is for debugging only, it does not run any program, \n "
" specified by a RUN key. It may show incorrect results, because \n "
" some values may be different, or not available at a simulation run. \n "
" \n " ) ;
2011-04-20 03:53:03 +04:00
sigprocmask ( SIG_SETMASK , NULL , & sigmask_orig ) ;
rules = udev_rules_new ( udev , resolve_names ) ;
2008-10-18 17:02:01 +04:00
if ( rules = = NULL ) {
fprintf ( stderr , " error reading rules \n " ) ;
2011-04-20 03:53:03 +04:00
rc = 3 ;
goto out ;
2008-10-18 17:02:01 +04:00
}
2006-08-18 04:33:46 +04:00
2008-10-16 19:16:58 +04:00
/* add /sys if needed */
2009-05-20 19:57:52 +04:00
if ( strncmp ( syspath , udev_get_sys_path ( udev ) , strlen ( udev_get_sys_path ( udev ) ) ) ! = 0 )
util_strscpyl ( filename , sizeof ( filename ) , udev_get_sys_path ( udev ) , syspath , NULL ) ;
else
util_strscpy ( filename , sizeof ( filename ) , syspath ) ;
2009-03-18 20:42:16 +03:00
util_remove_trailing_chars ( filename , ' / ' ) ;
2004-02-13 06:48:07 +03:00
2009-03-18 20:42:16 +03:00
dev = udev_device_new_from_syspath ( udev , filename ) ;
2006-01-09 23:18:00 +03:00
if ( dev = = NULL ) {
2009-03-18 20:42:16 +03:00
fprintf ( stderr , " unable to open device '%s' \n " , filename ) ;
2011-04-20 03:53:03 +04:00
rc = 4 ;
goto out ;
2006-01-09 23:18:00 +03:00
}
2004-10-19 06:11:51 +04:00
2008-10-16 19:16:58 +04:00
/* skip reading of db, but read kernel parameters */
udev_device_set_info_loaded ( dev ) ;
udev_device_read_uevent_file ( dev ) ;
2004-10-19 06:11:51 +04:00
2008-10-16 19:16:58 +04:00
udev_device_set_action ( dev , action ) ;
event = udev_event_new ( dev ) ;
2008-04-21 21:00:54 +04:00
2011-04-20 03:53:03 +04:00
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 " ) ;
rc = 5 ;
goto out ;
}
err = udev_event_execute_rules ( event , rules , & sigmask_orig ) ;
2008-04-21 21:00:54 +04:00
2009-07-03 00:56:19 +04:00
udev_list_entry_foreach ( entry , udev_device_get_properties_list_entry ( dev ) )
2011-04-18 00:09:08 +04:00
printf ( " %s=%s \n " , udev_list_entry_get_name ( entry ) , udev_list_entry_get_value ( entry ) ) ;
2006-01-09 23:18:00 +03:00
2011-04-20 03:53:03 +04:00
if ( err = = 0 ) {
2008-10-16 19:16:58 +04:00
udev_list_entry_foreach ( entry , udev_list_get_entry ( & event - > run_list ) ) {
2008-09-10 23:50:21 +04:00
char program [ UTIL_PATH_SIZE ] ;
2006-01-25 04:18:13 +03:00
2009-05-20 19:57:52 +04:00
udev_event_apply_format ( event , udev_list_entry_get_name ( entry ) , program , sizeof ( program ) ) ;
2011-04-18 00:09:08 +04:00
printf ( " run: '%s' \n " , program ) ;
2006-01-25 04:18:13 +03:00
}
2011-04-20 03:53:03 +04:00
}
out :
if ( event ! = NULL & & event - > fd_signal > = 0 )
close ( event - > fd_signal ) ;
2008-10-16 19:16:58 +04:00
udev_event_unref ( event ) ;
udev_device_unref ( dev ) ;
2008-10-18 17:02:01 +04:00
udev_rules_unref ( rules ) ;
2006-01-09 23:18:00 +03:00
return rc ;
2004-02-13 06:48:07 +03:00
}