2005-04-28 18:49:41 +04:00
/*
* Copyright ( C ) 2005 Red Hat , Inc . All rights reserved .
*
* This file is part of the device - mapper userspace tools .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "libdevmapper.h"
# include "libdm-event.h"
2005-05-04 01:29:13 +04:00
# include "libmultilog.h"
2005-04-28 18:49:41 +04:00
# include <errno.h>
# include <fcntl.h>
# include <stdio.h>
# include <stdint.h>
# include <stdlib.h>
# include <string.h>
# include <sys/file.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <unistd.h>
2005-06-14 23:06:26 +04:00
# include <dlfcn.h>
2005-04-28 18:49:41 +04:00
static enum event_type events = ALL_ERRORS ; /* All until we can distinguish. */
2005-05-03 20:15:20 +04:00
static char default_dso_name [ ] = " noop " ; /* default DSO is noop */
2005-04-28 18:49:41 +04:00
static int default_reg = 1 ; /* default action is register */
2005-06-09 22:40:49 +04:00
static uint32_t timeout ;
2005-04-28 18:49:41 +04:00
2005-06-14 23:06:26 +04:00
struct event_ops {
int ( * dm_register_for_event ) ( char * dso_name , char * device ,
enum event_type event_types ) ;
int ( * dm_unregister_for_event ) ( char * dso_name , char * device ,
enum event_type event_types ) ;
int ( * dm_get_registered_device ) ( char * * dso_name , char * * device ,
enum event_type * event_types , int next ) ;
int ( * dm_set_event_timeout ) ( char * device , uint32_t time ) ;
int ( * dm_get_event_timeout ) ( char * device , uint32_t * time ) ;
} ;
2005-04-28 18:49:41 +04:00
/* Display help. */
static void print_usage ( char * name )
{
char * cmd = strrchr ( name , ' / ' ) ;
cmd = cmd ? cmd + 1 : name ;
printf ( " Usage:: \n "
" %s [options] <device> \n "
" \n "
" Options: \n "
" -d <dso> Specify the DSO to use. \n "
" -h Print this usage. \n "
" -l List registered devices. \n "
" -r Register for event (default). \n "
2005-06-09 22:40:49 +04:00
" -t <timeout> (un)register for timeout event. \n "
2005-04-28 18:49:41 +04:00
" -u Unregister for event. \n "
" \n " , cmd ) ;
}
/* Parse command line arguments. */
2005-05-04 22:53:28 +04:00
static int parse_argv ( int argc , char * * argv , char * * dso_name_arg ,
char * * device_arg , int * reg , int * list )
2005-04-28 18:49:41 +04:00
{
int c ;
2005-06-09 22:40:49 +04:00
const char * options = " d:hlrt:u " ;
2005-04-28 18:49:41 +04:00
while ( ( c = getopt ( argc , argv , options ) ) ! = - 1 ) {
switch ( c ) {
case ' d ' :
2005-05-04 23:24:03 +04:00
* dso_name_arg = optarg ;
2005-04-28 18:49:41 +04:00
break ;
case ' h ' :
print_usage ( argv [ 0 ] ) ;
exit ( EXIT_SUCCESS ) ;
case ' l ' :
* list = 1 ;
break ;
case ' r ' :
* reg = 1 ;
break ;
2005-06-09 22:40:49 +04:00
case ' t ' :
events = TIMEOUT ;
if ( sscanf ( optarg , " % " SCNu32 , & timeout ) ! = 1 ) {
fprintf ( stderr , " invalid timeout '%s' \n " ,
optarg ) ;
timeout = 0 ;
}
break ;
2005-04-28 18:49:41 +04:00
case ' u ' :
* reg = 0 ;
break ;
default :
fprintf ( stderr , " Unknown option '%c'. \n "
" Try '-h' for help. \n " , c ) ;
return 0 ;
}
}
2005-05-03 20:15:20 +04:00
if ( optind > = argc ) {
if ( ! * list ) {
2005-04-28 18:49:41 +04:00
fprintf ( stderr , " You need to specify a device. \n " ) ;
return 0 ;
}
2005-05-04 22:53:28 +04:00
} else
2005-05-04 23:24:03 +04:00
* device_arg = argv [ optind ] ;
2005-04-28 18:49:41 +04:00
return 1 ;
}
2005-06-14 23:06:26 +04:00
static int lookup_symbol ( void * dl , void * * symbol , const char * name )
{
if ( ( * symbol = dlsym ( dl , name ) ) )
return 1 ;
fprintf ( stderr , " error looking up %s symbol: %s \n " , name , dlerror ( ) ) ;
return 0 ;
}
static int lookup_symbols ( void * dl , struct event_ops * e )
{
return lookup_symbol ( dl , ( void * ) & e - > dm_register_for_event ,
" dm_register_for_event " ) & &
lookup_symbol ( dl , ( void * ) & e - > dm_unregister_for_event ,
" dm_unregister_for_event " ) & &
lookup_symbol ( dl , ( void * ) & e - > dm_get_registered_device ,
" dm_get_registered_device " ) & &
lookup_symbol ( dl , ( void * ) & e - > dm_set_event_timeout ,
" dm_set_event_timeout " ) & &
lookup_symbol ( dl , ( void * ) & e - > dm_get_event_timeout ,
" dm_get_event_timeout " ) ;
}
2005-04-28 18:49:41 +04:00
int main ( int argc , char * * argv )
{
2005-06-14 23:06:26 +04:00
void * dl ;
struct event_ops e ;
2005-04-29 17:41:25 +04:00
int list = 0 , next = 0 , ret , reg = default_reg ;
2005-05-04 22:53:28 +04:00
char * device , * device_arg = NULL , * dso_name , * dso_name_arg = NULL ;
2005-04-28 18:49:41 +04:00
2005-05-04 22:53:28 +04:00
if ( ! parse_argv ( argc , argv , & dso_name_arg , & device_arg , & reg , & list ) )
2005-04-28 18:49:41 +04:00
exit ( EXIT_FAILURE ) ;
2005-04-29 17:41:25 +04:00
2005-05-20 17:53:26 +04:00
if ( device_arg ) {
2005-05-04 22:53:28 +04:00
if ( ! ( device = strdup ( device_arg ) ) )
exit ( EXIT_FAILURE ) ;
} else
device = NULL ;
2005-05-03 20:15:20 +04:00
2005-05-20 17:53:26 +04:00
if ( dso_name_arg ) {
2005-05-04 22:53:28 +04:00
if ( ! ( dso_name = strdup ( dso_name_arg ) ) )
exit ( EXIT_FAILURE ) ;
} else {
if ( ! ( dso_name = strdup ( default_dso_name ) ) )
exit ( EXIT_FAILURE ) ;
}
2005-05-04 01:29:13 +04:00
/* FIXME: use -v/-q options to set this */
2005-05-09 22:44:35 +04:00
multilog_add_type ( standard , NULL ) ;
multilog_init_verbose ( standard , _LOG_DEBUG ) ;
2005-05-04 01:29:13 +04:00
2005-06-14 23:06:26 +04:00
if ( ! ( dl = dlopen ( " libdmevent.so " , RTLD_NOW ) ) ) {
fprintf ( stderr , " Cannot dlopen libdmevent.so: %s \n " , dlerror ( ) ) ;
goto out ;
}
if ( ! ( lookup_symbols ( dl , & e ) ) )
goto out ;
2005-04-28 18:49:41 +04:00
if ( list ) {
2005-06-09 22:40:49 +04:00
while ( 1 ) {
2005-06-14 23:06:26 +04:00
if ( ( ret = e . dm_get_registered_device ( & dso_name ,
& device ,
& events , next ) ) )
2005-06-09 22:40:49 +04:00
break ;
printf ( " %s %s 0x%x " , dso_name , device , events ) ;
if ( events & TIMEOUT ) {
2005-06-14 23:06:26 +04:00
if ( ( ret = e . dm_get_event_timeout ( device ,
& timeout ) ) ) {
2005-06-09 22:40:49 +04:00
ret = EXIT_FAILURE ;
goto out ;
}
printf ( " % " PRIu32 " \n " , timeout ) ;
} else
printf ( " \n " ) ;
if ( device_arg )
break ;
next = 1 ;
}
2005-04-28 18:49:41 +04:00
2005-05-03 20:15:20 +04:00
ret = ( ret & & device_arg ) ? EXIT_FAILURE : EXIT_SUCCESS ;
goto out ;
}
2005-05-03 17:50:42 +04:00
2005-06-14 23:06:26 +04:00
if ( ( ret = reg ? e . dm_register_for_event ( dso_name , device , events ) :
e . dm_unregister_for_event ( dso_name , device , events ) ) ) {
2005-04-28 18:49:41 +04:00
fprintf ( stderr , " Failed to %sregister %s: %s \n " ,
reg ? " " : " un " , device , strerror ( - ret ) ) ;
2005-05-03 20:15:20 +04:00
ret = EXIT_FAILURE ;
} else {
2005-06-09 22:40:49 +04:00
if ( reg & & ( events & TIMEOUT ) & &
2005-06-14 23:06:26 +04:00
( ( ret = e . dm_set_event_timeout ( device , timeout ) ) ) ) {
2005-06-09 22:40:49 +04:00
fprintf ( stderr , " Failed to set timeout for %s: %s \n " ,
device , strerror ( - ret ) ) ;
ret = EXIT_FAILURE ;
} else {
printf ( " %s %sregistered successfully. \n " ,
device , reg ? " " : " un " ) ;
ret = EXIT_SUCCESS ;
}
2005-04-28 18:49:41 +04:00
}
2005-05-03 20:15:20 +04:00
out :
2005-05-09 22:44:35 +04:00
multilog_del_type ( standard ) ;
2005-05-04 15:52:07 +04:00
2005-05-04 22:53:28 +04:00
if ( device )
free ( device ) ;
if ( dso_name )
free ( dso_name ) ;
2005-04-28 18:49:41 +04:00
2005-05-03 20:15:20 +04:00
exit ( ret ) ;
2005-04-28 18:49:41 +04:00
}
/*
* Overrides for Emacs so that we follow Linus ' s tabbing style .
* Emacs will notice this stuff at the end of the file and automatically
* adjust the settings for this buffer only . This must remain at the end
* of the file .
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* Local variables :
* c - file - style : " linux "
* End :
*/