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 >
2006-08-21 04:38:20 +04:00
* Copyright ( C ) 2004 - 2006 Kay Sievers < kay . sievers @ vrfy . org >
2004-02-13 06:48:07 +03: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 version 2 of the License .
*
* 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 , write to the Free Software Foundation , Inc . ,
2006-08-28 02:29:11 +04:00
* 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
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>
2004-02-13 06:48:07 +03:00
# include "udev.h"
2005-03-13 00:36:32 +03:00
# include "udev_rules.h"
2004-02-13 06:48:07 +03:00
2007-04-06 21:18:33 +04:00
static int import_uevent_var ( const char * devpath )
{
char path [ PATH_SIZE ] ;
static char value [ 4096 ] ; /* must stay, used with putenv */
ssize_t size ;
int fd ;
char * key ;
char * next ;
int rc = - 1 ;
/* read uevent file */
strlcpy ( path , sysfs_path , sizeof ( path ) ) ;
strlcat ( path , devpath , sizeof ( path ) ) ;
strlcat ( path , " /uevent " , sizeof ( path ) ) ;
fd = open ( path , O_RDONLY ) ;
if ( fd < 0 )
goto out ;
size = read ( fd , value , sizeof ( value ) ) ;
close ( fd ) ;
if ( size < 0 )
goto out ;
value [ size ] = ' \0 ' ;
/* import keys into environment */
key = value ;
while ( key [ 0 ] ! = ' \0 ' ) {
next = strchr ( key , ' \n ' ) ;
if ( next = = NULL )
goto out ;
next [ 0 ] = ' \0 ' ;
info ( " import into environment: '%s' " , key ) ;
putenv ( key ) ;
key = & next [ 1 ] ;
}
rc = 0 ;
out :
return rc ;
}
2007-11-08 19:51:59 +03:00
int udevtest ( int argc , char * argv [ ] , char * envp [ ] )
2004-02-13 06:48:07 +03:00
{
2007-03-21 13:55:26 +03:00
int force = 0 ;
2007-07-09 05:59:08 +04:00
const char * action = " add " ;
const char * subsystem = NULL ;
const char * devpath = NULL ;
2006-01-09 23:18:00 +03:00
struct udevice * udev ;
struct sysfs_device * dev ;
2007-06-02 12:14:50 +04:00
struct udev_rules rules = { } ;
2006-01-09 23:18:00 +03:00
int retval ;
int rc = 0 ;
2004-03-23 09:18:34 +03:00
2007-03-21 13:55:26 +03:00
static const struct option options [ ] = {
{ " action " , 1 , NULL , ' a ' } ,
2007-06-02 12:14:50 +04:00
{ " subsystem " , 1 , NULL , ' s ' } ,
2007-03-21 13:55:26 +03:00
{ " force " , 0 , NULL , ' f ' } ,
{ " help " , 0 , NULL , ' h ' } ,
{ }
} ;
2004-03-23 09:18:34 +03:00
info ( " version %s " , UDEV_VERSION ) ;
2006-01-09 23:18:00 +03:00
udev_config_init ( ) ;
2007-03-16 03:09:47 +03:00
if ( udev_log_priority < LOG_INFO ) {
char priority [ 32 ] ;
2005-03-27 03:11:03 +04:00
udev_log_priority = LOG_INFO ;
2007-03-16 03:09:47 +03:00
sprintf ( priority , " %i " , udev_log_priority ) ;
setenv ( " UDEV_LOG " , priority , 1 ) ;
}
2004-11-28 15:41:15 +03:00
2007-03-21 13:55:26 +03:00
while ( 1 ) {
int option ;
2007-06-02 12:14:50 +04:00
option = getopt_long ( argc , argv , " a:s:fh " , options , NULL ) ;
2007-03-21 13:55:26 +03:00
if ( option = = - 1 )
break ;
dbg ( " option '%c' " , option ) ;
switch ( option ) {
case ' a ' :
action = optarg ;
break ;
2007-06-02 12:14:50 +04:00
case ' s ' :
subsystem = optarg ;
break ;
2007-03-21 13:55:26 +03:00
case ' f ' :
force = 1 ;
break ;
case ' h ' :
2007-11-08 19:51:59 +03:00
printf ( " Usage: udevadm test OPTIONS <devpath> \n "
2007-06-02 12:14:50 +04:00
" --action=<string> set action string \n "
" --subsystem=<string> set subsystem string \n "
" --force don't skip node/link creation \n "
" --help print this help text \n \n " ) ;
2007-03-21 13:55:26 +03:00
exit ( 0 ) ;
default :
exit ( 1 ) ;
}
2006-08-21 04:38:20 +04:00
}
2007-03-21 13:55:26 +03:00
devpath = argv [ optind ] ;
2006-08-21 04:38:20 +04:00
if ( devpath = = NULL ) {
fprintf ( stderr , " devpath parameter missing \n " ) ;
rc = 1 ;
goto exit ;
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 " ) ;
2006-08-18 04:33:46 +04:00
sysfs_init ( ) ;
2006-08-21 04:38:20 +04:00
udev_rules_init ( & rules , 0 ) ;
2006-08-18 04:33:46 +04:00
/* remove /sys if given */
2006-08-21 04:38:20 +04:00
if ( strncmp ( devpath , sysfs_path , strlen ( sysfs_path ) ) = = 0 )
devpath = & devpath [ strlen ( sysfs_path ) ] ;
2004-02-13 06:48:07 +03:00
2006-01-09 23:18:00 +03:00
dev = sysfs_device_get ( devpath ) ;
if ( dev = = NULL ) {
2006-08-21 04:38:20 +04:00
fprintf ( stderr , " unable to open device '%s' \n " , devpath ) ;
2006-01-09 23:18:00 +03:00
rc = 2 ;
goto exit ;
}
2004-10-19 06:11:51 +04:00
2007-03-17 12:08:25 +03:00
udev = udev_device_init ( NULL ) ;
2006-01-09 23:18:00 +03:00
if ( udev = = NULL ) {
2006-08-21 04:38:20 +04:00
fprintf ( stderr , " error initializing device \n " ) ;
2006-01-09 23:18:00 +03:00
rc = 3 ;
goto exit ;
2004-10-30 15:44:46 +04:00
}
2004-10-19 06:11:51 +04:00
2007-06-02 12:14:50 +04:00
if ( subsystem ! = NULL )
strlcpy ( dev - > subsystem , subsystem , sizeof ( dev - > subsystem ) ) ;
2006-01-09 23:18:00 +03:00
/* override built-in sysfs device */
udev - > dev = dev ;
2007-06-02 12:14:50 +04:00
strlcpy ( udev - > action , action , sizeof ( udev - > action ) ) ;
2006-01-09 23:18:00 +03:00
udev - > devt = udev_device_get_devt ( udev ) ;
2005-06-22 03:31:24 +04:00
2004-10-19 06:28:39 +04:00
/* simulate node creation with test flag */
2007-03-21 13:55:26 +03:00
if ( ! force )
udev - > test_run = 1 ;
2006-01-09 23:18:00 +03:00
setenv ( " DEVPATH " , udev - > dev - > devpath , 1 ) ;
setenv ( " SUBSYSTEM " , udev - > dev - > subsystem , 1 ) ;
2007-03-31 04:42:14 +04:00
setenv ( " ACTION " , udev - > action , 1 ) ;
2007-04-06 21:18:33 +04:00
import_uevent_var ( udev - > dev - > devpath ) ;
2006-01-09 23:18:00 +03:00
info ( " looking at device '%s' from subsystem '%s' " , udev - > dev - > devpath , udev - > dev - > subsystem ) ;
retval = udev_device_event ( & rules , udev ) ;
2006-01-25 04:18:13 +03:00
if ( retval = = 0 & & ! udev - > ignore_device & & udev_run ) {
2006-01-09 23:18:00 +03:00
struct name_entry * name_loop ;
2006-01-25 04:18:13 +03:00
list_for_each_entry ( name_loop , & udev - > run_list , node ) {
char program [ PATH_SIZE ] ;
strlcpy ( program , name_loop - > name , sizeof ( program ) ) ;
udev_rules_apply_format ( udev , program , sizeof ( program ) ) ;
info ( " run: '%s' " , program ) ;
}
2006-01-09 23:18:00 +03:00
}
2007-06-22 03:27:02 +04:00
udev_device_cleanup ( udev ) ;
2006-01-09 23:18:00 +03:00
exit :
udev_rules_cleanup ( & rules ) ;
sysfs_cleanup ( ) ;
return rc ;
2004-02-13 06:48:07 +03:00
}