2004-02-12 19:48:07 -08:00
/*
2005-07-22 12:35:58 -04:00
* Copyright ( C ) 2003 - 2004 Greg Kroah - Hartman < greg @ kroah . com >
2008-09-10 02:40:42 +02:00
* Copyright ( C ) 2004 - 2008 Kay Sievers < kay . sievers @ vrfy . org >
2004-02-12 19:48:07 -08:00
*
2008-09-10 02:40:42 +02: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-12 19:48:07 -08:00
*
2008-09-10 02:40:42 +02: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-12 19:48:07 -08:00
*/
# include <stdlib.h>
# include <string.h>
# include <stdio.h>
2006-01-09 21:18:00 +01:00
# include <stddef.h>
# include <unistd.h>
2004-02-12 19:48:07 -08:00
# include <errno.h>
# include <ctype.h>
2007-04-06 19:18:33 +02:00
# include <fcntl.h>
2004-02-12 19:48:07 -08:00
# include <signal.h>
2005-03-27 00:11:03 +01:00
# include <syslog.h>
2007-03-21 11:55:26 +01:00
# include <getopt.h>
2004-02-12 19:48:07 -08:00
# include "udev.h"
2005-03-12 22:36:32 +01:00
# include "udev_rules.h"
2004-02-12 19:48:07 -08:00
2008-09-06 15:45:31 +02:00
static int import_uevent_var ( struct udev * udev , const char * devpath )
2007-04-06 19:18:33 +02:00
{
2008-09-10 21:50:21 +02:00
char path [ UTIL_PATH_SIZE ] ;
2007-04-06 19:18:33 +02:00
static char value [ 4096 ] ; /* must stay, used with putenv */
ssize_t size ;
int fd ;
char * key ;
char * next ;
int rc = - 1 ;
/* read uevent file */
2008-09-10 18:59:42 +02:00
util_strlcpy ( path , udev_get_sys_path ( udev ) , sizeof ( path ) ) ;
util_strlcat ( path , devpath , sizeof ( path ) ) ;
util_strlcat ( path , " /uevent " , sizeof ( path ) ) ;
2007-04-06 19:18:33 +02:00
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 ' ;
2008-09-06 15:45:31 +02:00
info ( udev , " import into environment: '%s' \n " , key ) ;
2007-04-06 19:18:33 +02:00
putenv ( key ) ;
key = & next [ 1 ] ;
}
rc = 0 ;
out :
return rc ;
}
2008-09-06 15:45:31 +02:00
int udevadm_test ( struct udev * udev , int argc , char * argv [ ] )
2004-02-12 19:48:07 -08:00
{
2007-03-21 11:55:26 +01:00
int force = 0 ;
2007-07-09 03:59:08 +02:00
const char * action = " add " ;
const char * subsystem = NULL ;
const char * devpath = NULL ;
2008-09-06 15:45:31 +02:00
struct udevice * udevice ;
2006-01-09 21:18:00 +01:00
struct sysfs_device * dev ;
2007-06-02 10:14:50 +02:00
struct udev_rules rules = { } ;
2006-01-09 21:18:00 +01:00
int retval ;
int rc = 0 ;
2004-03-22 22:18:34 -08:00
2007-03-21 11:55:26 +01:00
static const struct option options [ ] = {
{ " action " , 1 , NULL , ' a ' } ,
2007-06-02 10:14:50 +02:00
{ " subsystem " , 1 , NULL , ' s ' } ,
2007-03-21 11:55:26 +01:00
{ " force " , 0 , NULL , ' f ' } ,
{ " help " , 0 , NULL , ' h ' } ,
{ }
} ;
2008-09-06 15:45:31 +02:00
info ( udev , " version %s \n " , VERSION ) ;
/* export log priority to executed programs */
if ( udev_get_log_priority ( udev ) > 0 ) {
2007-03-16 01:09:47 +01:00
char priority [ 32 ] ;
2008-09-06 15:45:31 +02:00
sprintf ( priority , " %i " , udev_get_log_priority ( udev ) ) ;
2007-03-16 01:09:47 +01:00
setenv ( " UDEV_LOG " , priority , 1 ) ;
}
2004-11-28 13:41:15 +01:00
2007-03-21 11:55:26 +01:00
while ( 1 ) {
int option ;
2007-06-02 10:14:50 +02:00
option = getopt_long ( argc , argv , " a:s:fh " , options , NULL ) ;
2007-03-21 11:55:26 +01:00
if ( option = = - 1 )
break ;
2008-09-06 15:45:31 +02:00
dbg ( udev , " option '%c' \n " , option ) ;
2007-03-21 11:55:26 +01:00
switch ( option ) {
case ' a ' :
action = optarg ;
break ;
2007-06-02 10:14:50 +02:00
case ' s ' :
subsystem = optarg ;
break ;
2007-03-21 11:55:26 +01:00
case ' f ' :
force = 1 ;
break ;
case ' h ' :
2007-11-08 17:51:59 +01:00
printf ( " Usage: udevadm test OPTIONS <devpath> \n "
2007-06-02 10:14:50 +02: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 11:55:26 +01:00
exit ( 0 ) ;
default :
exit ( 1 ) ;
}
2006-08-21 02:38:20 +02:00
}
2007-03-21 11:55:26 +01:00
devpath = argv [ optind ] ;
2006-08-21 02:38:20 +02:00
if ( devpath = = NULL ) {
fprintf ( stderr , " devpath parameter missing \n " ) ;
rc = 1 ;
goto exit ;
2005-03-29 11:30:51 +02:00
}
2007-05-21 07:02:04 +02: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 " ) ;
2008-09-06 15:45:31 +02:00
udev_rules_init ( udev , & rules , 0 ) ;
2006-08-18 02:33:46 +02:00
/* remove /sys if given */
2008-09-06 15:45:31 +02:00
if ( strncmp ( devpath , udev_get_sys_path ( udev ) , strlen ( udev_get_sys_path ( udev ) ) ) = = 0 )
devpath = & devpath [ strlen ( udev_get_sys_path ( udev ) ) ] ;
2004-02-12 19:48:07 -08:00
2008-09-06 15:45:31 +02:00
dev = sysfs_device_get ( udev , devpath ) ;
2006-01-09 21:18:00 +01:00
if ( dev = = NULL ) {
2006-08-21 02:38:20 +02:00
fprintf ( stderr , " unable to open device '%s' \n " , devpath ) ;
2006-01-09 21:18:00 +01:00
rc = 2 ;
goto exit ;
}
2004-10-18 19:11:51 -07:00
2008-09-06 15:45:31 +02:00
udevice = udev_device_init ( udev ) ;
if ( udevice = = NULL ) {
2006-08-21 02:38:20 +02:00
fprintf ( stderr , " error initializing device \n " ) ;
2006-01-09 21:18:00 +01:00
rc = 3 ;
goto exit ;
2004-10-30 13:44:46 +02:00
}
2004-10-18 19:11:51 -07:00
2007-06-02 10:14:50 +02:00
if ( subsystem ! = NULL )
2008-09-10 18:59:42 +02:00
util_strlcpy ( dev - > subsystem , subsystem , sizeof ( dev - > subsystem ) ) ;
2007-06-02 10:14:50 +02:00
2006-01-09 21:18:00 +01:00
/* override built-in sysfs device */
2008-09-06 15:45:31 +02:00
udevice - > dev = dev ;
2008-09-10 18:59:42 +02:00
util_strlcpy ( udevice - > action , action , sizeof ( udevice - > action ) ) ;
2008-09-06 15:45:31 +02:00
udevice - > devt = udev_device_get_devt ( udevice ) ;
2005-06-22 01:31:24 +02:00
2004-10-18 19:28:39 -07:00
/* simulate node creation with test flag */
2007-03-21 11:55:26 +01:00
if ( ! force )
2008-09-06 15:45:31 +02:00
udevice - > test_run = 1 ;
2006-01-09 21:18:00 +01:00
2008-09-06 15:45:31 +02:00
setenv ( " DEVPATH " , udevice - > dev - > devpath , 1 ) ;
setenv ( " SUBSYSTEM " , udevice - > dev - > subsystem , 1 ) ;
setenv ( " ACTION " , udevice - > action , 1 ) ;
import_uevent_var ( udev , udevice - > dev - > devpath ) ;
2006-01-09 21:18:00 +01:00
2008-09-06 15:45:31 +02:00
info ( udev , " looking at device '%s' from subsystem '%s' \n " , udevice - > dev - > devpath , udevice - > dev - > subsystem ) ;
retval = udev_device_event ( & rules , udevice ) ;
2008-04-21 19:00:54 +02:00
2008-09-06 15:45:31 +02:00
if ( udevice - > event_timeout > = 0 )
info ( udev , " custom event timeout: %i \n " , udevice - > event_timeout ) ;
2008-04-21 19:00:54 +02:00
2008-09-06 15:45:31 +02:00
if ( retval = = 0 & & ! udevice - > ignore_device & & udev_get_run ( udev ) ) {
2006-01-09 21:18:00 +01:00
struct name_entry * name_loop ;
2008-09-06 15:45:31 +02:00
list_for_each_entry ( name_loop , & udevice - > run_list , node ) {
2008-09-10 21:50:21 +02:00
char program [ UTIL_PATH_SIZE ] ;
2006-01-25 02:18:13 +01:00
2008-09-10 18:59:42 +02:00
util_strlcpy ( program , name_loop - > name , sizeof ( program ) ) ;
2008-09-06 15:45:31 +02:00
udev_rules_apply_format ( udevice , program , sizeof ( program ) ) ;
info ( udev , " run: '%s' \n " , program ) ;
2006-01-25 02:18:13 +01:00
}
2006-01-09 21:18:00 +01:00
}
2008-09-06 15:45:31 +02:00
udev_device_cleanup ( udevice ) ;
2006-01-09 21:18:00 +01:00
exit :
udev_rules_cleanup ( & rules ) ;
return rc ;
2004-02-12 19:48:07 -08:00
}