2007-01-08 15:18:52 +00:00
/*
2005-12-02 15:39:16 +00: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 "lib.h"
# include "libdevmapper-event.h"
//#include "libmultilog.h"
# include "dmeventd.h"
# 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>
# include <sys/wait.h>
2007-01-11 21:54:53 +00:00
# include <arpa/inet.h> /* for htonl, ntohl */
struct dm_event_handler {
const char * dso ;
const char * device ;
const char * uuid ;
int major ;
int minor ;
enum dm_event_type events ;
} ;
static void dm_event_handler_clear_device ( struct dm_event_handler * h )
{
h - > device = h - > uuid = NULL ;
h - > major = h - > minor = 0 ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
struct dm_event_handler * dm_event_handler_create ( void )
{
struct dm_event_handler * ret = 0 ;
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
if ( ! ( ret = dm_malloc ( sizeof ( * ret ) ) ) )
return NULL ;
ret - > dso = ret - > device = ret - > uuid = NULL ;
ret - > major = ret - > minor = 0 ;
ret - > events = 0 ;
return ret ;
}
void dm_event_handler_destroy ( struct dm_event_handler * h )
2005-12-02 15:39:16 +00:00
{
2007-01-11 21:54:53 +00:00
dm_free ( h ) ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_dso ( struct dm_event_handler * h , const char * path )
{
h - > dso = path ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_name ( struct dm_event_handler * h , const char * name )
{
dm_event_handler_clear_device ( h ) ;
h - > device = name ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_uuid ( struct dm_event_handler * h , const char * uuid )
{
dm_event_handler_clear_device ( h ) ;
h - > uuid = uuid ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_major ( struct dm_event_handler * h , int major )
{
int minor = h - > minor ;
dm_event_handler_clear_device ( h ) ;
h - > major = major ;
h - > minor = minor ;
2005-12-02 15:39:16 +00:00
}
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_minor ( struct dm_event_handler * h , int minor )
2005-12-02 15:39:16 +00:00
{
2007-01-11 21:54:53 +00:00
int major = h - > major ;
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
dm_event_handler_clear_device ( h ) ;
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
h - > major = major ;
h - > minor = minor ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
void dm_event_handler_set_events ( struct dm_event_handler * h ,
enum dm_event_type event )
{
h - > events = event ;
}
const char * dm_event_handler_get_dso ( const struct dm_event_handler * h )
{
return h - > dso ;
}
const char * dm_event_handler_get_name ( const struct dm_event_handler * h )
{
return h - > device ;
}
const char * dm_event_handler_get_uuid ( const struct dm_event_handler * h )
{
return h - > uuid ;
}
int dm_event_handler_get_major ( const struct dm_event_handler * h )
{
return h - > major ;
}
int dm_event_handler_get_minor ( const struct dm_event_handler * h )
{
return h - > minor ;
}
enum dm_event_type dm_event_handler_get_events ( const struct dm_event_handler * h )
{
return h - > events ;
2005-12-02 15:39:16 +00:00
}
2005-12-19 22:56:47 +00:00
/*
* daemon_read
* @ fifos
* @ msg
*
* Read message from daemon .
*
* Returns : 0 on failure , 1 on success
*/
2007-01-11 21:54:53 +00:00
static int daemon_read ( struct dm_event_fifos * fifos ,
struct dm_event_daemon_message * msg )
2005-12-02 15:39:16 +00:00
{
2006-01-31 14:50:38 +00:00
unsigned bytes = 0 ;
2007-01-08 15:18:52 +00:00
int ret , i ;
2005-12-02 15:39:16 +00:00
fd_set fds ;
2007-01-11 21:54:53 +00:00
struct timeval tval = { 0 , 0 } ;
size_t size = 2 * sizeof ( uint32_t ) ; /* status + size */
2007-01-08 15:18:52 +00:00
char * buf = alloca ( size ) ;
int header = 1 ;
2005-12-02 15:39:16 +00:00
2007-01-08 15:18:52 +00:00
while ( bytes < size ) {
for ( i = 0 , ret = 0 ; ( i < 20 ) & & ( ret < 1 ) ; i + + ) {
2005-12-02 15:39:16 +00:00
/* Watch daemon read FIFO for input. */
FD_ZERO ( & fds ) ;
FD_SET ( fifos - > server , & fds ) ;
2007-01-08 15:18:52 +00:00
tval . tv_sec = 1 ;
2007-01-11 21:54:53 +00:00
ret = select ( fifos - > server + 1 , & fds , NULL , NULL ,
& tval ) ;
2005-12-19 22:56:47 +00:00
if ( ret < 0 & & errno ! = EINTR ) {
2007-01-08 15:18:52 +00:00
log_error ( " Unable to read from event server " ) ;
2005-12-19 22:56:47 +00:00
return 0 ;
}
2007-01-08 15:18:52 +00:00
}
if ( ret < 1 ) {
log_error ( " Unable to read from event server. " ) ;
return 0 ;
}
2005-12-02 15:39:16 +00:00
2007-01-08 15:18:52 +00:00
ret = read ( fifos - > server , buf + bytes , size ) ;
2005-12-19 22:56:47 +00:00
if ( ret < 0 ) {
if ( ( errno = = EINTR ) | | ( errno = = EAGAIN ) )
continue ;
else {
2007-01-08 15:18:52 +00:00
log_error ( " Unable to read from event server. " ) ;
2005-12-19 22:56:47 +00:00
return 0 ;
}
}
bytes + = ret ;
2007-01-11 21:54:53 +00:00
if ( bytes = = 2 * sizeof ( uint32_t ) & & header ) {
2007-01-08 15:18:52 +00:00
msg - > cmd = ntohl ( * ( ( uint32_t * ) buf ) ) ;
msg - > size = ntohl ( * ( ( uint32_t * ) buf + 1 ) ) ;
buf = msg - > data = dm_malloc ( msg - > size ) ;
size = msg - > size ;
bytes = 0 ;
header = 0 ;
}
2005-12-02 15:39:16 +00:00
}
2007-01-08 15:18:52 +00:00
if ( bytes ! = size ) {
if ( msg - > data )
dm_free ( msg - > data ) ;
msg - > data = NULL ;
}
return bytes = = size ;
2005-12-02 15:39:16 +00:00
}
/* Write message to daemon. */
2007-01-11 21:54:53 +00:00
static int daemon_write ( struct dm_event_fifos * fifos ,
struct dm_event_daemon_message * msg )
2005-12-02 15:39:16 +00:00
{
2006-01-31 14:50:38 +00:00
unsigned bytes = 0 ;
int ret = 0 ;
2005-12-02 15:39:16 +00:00
fd_set fds ;
2007-01-11 21:54:53 +00:00
size_t size = 2 * sizeof ( uint32_t ) + msg - > size ;
2007-01-08 15:18:52 +00:00
char * buf = alloca ( size ) ;
* ( ( uint32_t * ) buf ) = htonl ( msg - > cmd ) ;
* ( ( uint32_t * ) buf + 1 ) = htonl ( msg - > size ) ;
2007-01-11 21:54:53 +00:00
memcpy ( buf + 2 * sizeof ( uint32_t ) , msg - > data , msg - > size ) ;
2007-01-08 15:18:52 +00:00
while ( bytes < size ) {
2005-12-02 15:39:16 +00:00
do {
/* Watch daemon write FIFO to be ready for output. */
FD_ZERO ( & fds ) ;
FD_SET ( fifos - > client , & fds ) ;
2007-01-11 21:54:53 +00:00
ret = select ( fifos - > client + 1 , NULL , & fds , NULL , NULL ) ;
2005-12-19 22:56:47 +00:00
if ( ( ret < 0 ) & & ( errno ! = EINTR ) ) {
2007-01-08 15:18:52 +00:00
log_error ( " Unable to talk to event daemon " ) ;
2005-12-19 22:56:47 +00:00
return 0 ;
}
} while ( ret < 1 ) ;
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
ret = write ( fifos - > client , ( ( char * ) buf ) + bytes ,
size - bytes ) ;
2005-12-19 22:56:47 +00:00
if ( ret < 0 ) {
if ( ( errno = = EINTR ) | | ( errno = = EAGAIN ) )
continue ;
else {
2007-01-08 15:18:52 +00:00
log_error ( " Unable to talk to event daemon " ) ;
2005-12-19 22:56:47 +00:00
return 0 ;
}
}
bytes + = ret ;
2005-12-02 15:39:16 +00:00
}
2007-01-08 15:18:52 +00:00
return bytes = = size ;
2005-12-02 15:39:16 +00:00
}
2007-01-11 21:54:53 +00:00
static int daemon_talk ( struct dm_event_fifos * fifos ,
struct dm_event_daemon_message * msg , int cmd ,
const char * dso_name , const char * device ,
2005-12-02 15:39:16 +00:00
enum dm_event_type events , uint32_t timeout )
{
2007-01-08 15:18:52 +00:00
const char * dso = dso_name ? dso_name : " " ;
const char * dev = device ? device : " " ;
2007-01-11 21:54:53 +00:00
const char * fmt = " %s %s %u % " PRIu32 ;
2005-12-02 15:39:16 +00:00
memset ( msg , 0 , sizeof ( * msg ) ) ;
/*
* Set command and pack the arguments
* into ASCII message string .
*/
2007-01-08 15:18:52 +00:00
msg - > cmd = cmd ;
2007-01-11 21:54:53 +00:00
msg - > size = dm_saprintf ( & ( msg - > data ) , fmt , dso , dev , events , timeout ) ;
2005-12-02 15:39:16 +00:00
/*
* Write command and message to and
* read status return code from daemon .
*/
if ( ! daemon_write ( fifos , msg ) ) {
stack ;
return - EIO ;
}
if ( ! daemon_read ( fifos , msg ) ) {
stack ;
return - EIO ;
}
2007-01-08 15:18:52 +00:00
return ( int32_t ) msg - > cmd ;
2005-12-02 15:39:16 +00:00
}
/*
* start_daemon
*
* This function forks off a process ( dmeventd ) that will handle
2007-01-08 15:18:52 +00:00
* the events . I am currently test opening one of the fifos to
* ensure that the daemon is running and listening . . . I thought
* this would be less expensive than fork / exec ' ing every time .
* Perhaps there is an even quicker / better way ( no , checking the
* lock file is _not_ a better way ) .
2005-12-02 15:39:16 +00:00
*
* Returns : 1 on success , 0 otherwise
*/
2007-01-08 15:18:52 +00:00
static int start_daemon ( struct dm_event_fifos * fifos )
2005-12-02 15:39:16 +00:00
{
2007-01-08 15:18:52 +00:00
int pid , ret = 0 ;
int status ;
struct stat statbuf ;
if ( stat ( fifos - > client_path , & statbuf ) )
goto start_server ;
if ( ! S_ISFIFO ( statbuf . st_mode ) ) {
log_error ( " %s is not a fifo. " , fifos - > client_path ) ;
2005-12-02 15:39:16 +00:00
return 0 ;
}
2007-01-08 15:18:52 +00:00
/* Anyone listening? If not, errno will be ENXIO */
fifos - > client = open ( fifos - > client_path , O_WRONLY | O_NONBLOCK ) ;
if ( fifos - > client > = 0 ) {
/* server is running and listening */
close ( fifos - > client ) ;
return 1 ;
} else if ( errno ! = ENXIO ) {
/* problem */
log_error ( " %s: Can't open client fifo %s: %s " ,
__func__ , fifos - > client_path , strerror ( errno ) ) ;
stack ;
return 0 ;
2006-01-27 20:45:17 +00:00
}
2007-01-08 15:18:52 +00:00
2007-01-11 21:54:53 +00:00
start_server :
2007-01-08 15:18:52 +00:00
/* server is not running */
2005-12-02 15:39:16 +00:00
pid = fork ( ) ;
if ( pid < 0 )
2007-01-08 15:18:52 +00:00
log_error ( " Unable to fork. " ) ;
2005-12-02 15:39:16 +00:00
2007-01-08 15:18:52 +00:00
else if ( ! pid ) {
2007-01-11 21:54:53 +00:00
/* FIXME configure path (cf. lvm2 modprobe) */
execvp ( " dmeventd " , NULL ) ;
2005-12-02 15:39:16 +00:00
exit ( EXIT_FAILURE ) ;
2007-01-08 15:18:52 +00:00
} else {
if ( waitpid ( pid , & status , 0 ) < 0 )
2007-01-11 21:54:53 +00:00
log_error ( " Unable to start dmeventd: %s " ,
strerror ( errno ) ) ;
2007-01-08 15:18:52 +00:00
else if ( WEXITSTATUS ( status ) )
log_error ( " Unable to start dmeventd. " ) ;
else
ret = 1 ;
2005-12-02 15:39:16 +00:00
}
2005-12-19 22:56:47 +00:00
return ret ;
2005-12-02 15:39:16 +00:00
}
/* Initialize client. */
static int init_client ( struct dm_event_fifos * fifos )
{
2007-01-11 21:54:53 +00:00
/* FIXME? Is fifo the most suitable method? Why not share
comms / daemon code with something else e . g . multipath ? */
2005-12-02 15:39:16 +00:00
/* init fifos */
memset ( fifos , 0 , sizeof ( * fifos ) ) ;
2005-12-19 22:56:47 +00:00
fifos - > client_path = DM_EVENT_FIFO_CLIENT ;
fifos - > server_path = DM_EVENT_FIFO_SERVER ;
2005-12-02 15:39:16 +00:00
2007-01-08 15:18:52 +00:00
if ( ! start_daemon ( fifos ) ) {
stack ;
2005-12-02 15:39:16 +00:00
return 0 ;
}
2007-01-08 15:18:52 +00:00
/* Open the fifo used to read from the daemon. */
2005-12-02 15:39:16 +00:00
if ( ( fifos - > server = open ( fifos - > server_path , O_RDWR ) ) < 0 ) {
2007-01-08 15:18:52 +00:00
log_error ( " %s: open server fifo %s " ,
2007-01-11 21:54:53 +00:00
__func__ , fifos - > server_path ) ;
2005-12-02 15:39:16 +00:00
stack ;
return 0 ;
}
/* Lock out anyone else trying to do communication with the daemon. */
2007-01-11 21:54:53 +00:00
if ( flock ( fifos - > server , LOCK_EX ) < 0 ) {
2007-01-08 15:18:52 +00:00
log_error ( " %s: flock %s " , __func__ , fifos - > server_path ) ;
2005-12-02 15:39:16 +00:00
close ( fifos - > server ) ;
return 0 ;
}
2007-01-11 21:54:53 +00:00
/* if ((fifos->client = open(fifos->client_path, O_WRONLY | O_NONBLOCK)) < 0) {*/
2007-01-08 15:18:52 +00:00
if ( ( fifos - > client = open ( fifos - > client_path , O_RDWR | O_NONBLOCK ) ) < 0 ) {
log_error ( " %s: Can't open client fifo %s: %s " ,
__func__ , fifos - > client_path , strerror ( errno ) ) ;
close ( fifos - > server ) ;
stack ;
return 0 ;
2005-12-02 15:39:16 +00:00
}
2007-01-11 21:54:53 +00:00
2005-12-02 15:39:16 +00:00
return 1 ;
}
static void dtr_client ( struct dm_event_fifos * fifos )
{
if ( flock ( fifos - > server , LOCK_UN ) )
2007-01-08 15:18:52 +00:00
log_error ( " flock unlock %s " , fifos - > server_path ) ;
2005-12-02 15:39:16 +00:00
close ( fifos - > client ) ;
close ( fifos - > server ) ;
}
2007-01-11 21:54:53 +00:00
/* Get uuid of a device, if it exists (otherwise NULL). */
static struct dm_task * get_device_info ( const struct dm_event_handler * h )
2005-12-02 15:39:16 +00:00
{
2007-01-11 21:54:53 +00:00
struct dm_task * dmt = dm_task_create ( DM_DEVICE_INFO ) ;
struct dm_task * ret ;
if ( ! dmt )
return NULL ;
if ( h - > uuid )
dm_task_set_uuid ( dmt , h - > uuid ) ;
else if ( h - > device )
dm_task_set_name ( dmt , h - > device ) ;
else if ( h - > major & & h - > minor ) {
dm_task_set_major ( dmt , h - > major ) ;
dm_task_set_minor ( dmt , h - > minor ) ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
if ( ! dm_task_run ( dmt ) )
ret = NULL ;
else
ret = dmt ;
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
return ret ;
2005-12-02 15:39:16 +00:00
}
/* Handle the event (de)registration call and return negative error codes. */
static int do_event ( int cmd , struct dm_event_daemon_message * msg ,
2007-01-11 21:54:53 +00:00
const char * dso_name , const char * device ,
enum dm_event_type events , uint32_t timeout )
2005-12-02 15:39:16 +00:00
{
int ret ;
struct dm_event_fifos fifos ;
if ( ! init_client ( & fifos ) ) {
stack ;
return - ESRCH ;
}
ret = daemon_talk ( & fifos , msg , cmd , dso_name , device , events , timeout ) ;
/* what is the opposite of init? */
dtr_client ( & fifos ) ;
2007-01-11 21:54:53 +00:00
2005-12-02 15:39:16 +00:00
return ret ;
}
/* External library interface. */
2007-01-11 21:54:53 +00:00
int dm_event_register ( const struct dm_event_handler * h )
2005-12-02 15:39:16 +00:00
{
2007-01-08 15:18:52 +00:00
int ret , err ;
2007-01-11 21:54:53 +00:00
const char * uuid ;
struct dm_task * dmt ;
2005-12-02 15:39:16 +00:00
struct dm_event_daemon_message msg ;
2007-01-11 21:54:53 +00:00
if ( ! ( dmt = get_device_info ( h ) ) ) {
log_error ( " %s: device not found " , h - > device ) ;
2006-01-27 20:01:45 +00:00
return 0 ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
uuid = dm_task_get_uuid ( dmt ) ;
2007-01-08 15:18:52 +00:00
if ( ( err = do_event ( DM_EVENT_CMD_REGISTER_FOR_EVENT , & msg ,
2007-01-11 21:54:53 +00:00
h - > dso , uuid , h - > events , 0 ) ) < 0 ) {
log_error ( " %s: event registration failed: %s " ,
dm_task_get_name ( dmt ) ,
2007-01-08 15:18:52 +00:00
msg . data ? msg . data : strerror ( - err ) ) ;
ret = 0 ;
} else
ret = 1 ;
2006-01-27 20:01:45 +00:00
2007-01-08 15:18:52 +00:00
if ( msg . data )
dm_free ( msg . data ) ;
2007-01-11 21:54:53 +00:00
dm_task_destroy ( dmt ) ;
2007-01-08 15:18:52 +00:00
return ret ;
2005-12-02 15:39:16 +00:00
}
2007-01-11 21:54:53 +00:00
int dm_event_unregister ( const struct dm_event_handler * h )
2005-12-02 15:39:16 +00:00
{
2007-01-08 15:18:52 +00:00
int ret , err ;
2007-01-11 21:54:53 +00:00
const char * uuid ;
struct dm_task * dmt ;
2005-12-02 15:39:16 +00:00
struct dm_event_daemon_message msg ;
2007-01-11 21:54:53 +00:00
if ( ! ( dmt = get_device_info ( h ) ) ) {
log_error ( " %s: device not found " , dm_task_get_name ( dmt ) ) ;
2006-01-27 20:01:45 +00:00
return 0 ;
}
2005-12-02 15:39:16 +00:00
2007-01-11 21:54:53 +00:00
uuid = dm_task_get_uuid ( dmt ) ;
2007-01-08 15:18:52 +00:00
if ( ( err = do_event ( DM_EVENT_CMD_UNREGISTER_FOR_EVENT , & msg ,
2007-01-11 21:54:53 +00:00
h - > dso , uuid , h - > events , 0 ) ) < 0 ) {
log_error ( " %s: event deregistration failed: %s " ,
dm_task_get_name ( dmt ) ,
2007-01-08 15:18:52 +00:00
msg . data ? msg . data : strerror ( - err ) ) ;
ret = 0 ;
} else
ret = 1 ;
2006-01-27 20:01:45 +00:00
2007-01-08 15:18:52 +00:00
if ( msg . data )
dm_free ( msg . data ) ;
2007-01-11 21:54:53 +00:00
dm_task_destroy ( dmt ) ;
return ret ;
}
#if 0 /* left out for now */
/* Fetch a string off src and duplicate it into *dest. */
/* FIXME: move to seperate module to share with the daemon. */
static const char delimiter = ' ' ;
static char * fetch_string ( char * * src )
{
char * p , * ret ;
if ( ( p = strchr ( * src , delimiter ) ) )
* p = 0 ;
if ( ( ret = dm_strdup ( * src ) ) )
* src + = strlen ( ret ) + 1 ;
if ( p )
* p = delimiter ;
2007-01-08 15:18:52 +00:00
return ret ;
2005-12-02 15:39:16 +00:00
}
2007-01-11 21:54:53 +00:00
/* Parse a device message from the daemon. */
static int parse_message ( struct dm_event_daemon_message * msg , char * * dso_name ,
char * * device , enum dm_event_type * events )
{
char * p = msg - > data ;
if ( ( * dso_name = fetch_string ( & p ) ) & & ( * device = fetch_string ( & p ) ) ) {
* events = atoi ( p ) ;
return 0 ;
}
return - ENOMEM ;
}
2007-01-08 15:18:52 +00:00
/*
* dm_event_get_registered_device
* @ dso_name
* @ device_path
* @ events
* @ next
*
* FIXME : This function sucks .
*
* Returns : 1 if device found , 0 otherwise ( even on error )
*/
2005-12-02 15:39:16 +00:00
int dm_event_get_registered_device ( char * * dso_name , char * * device_path ,
2007-01-11 21:54:53 +00:00
enum dm_event_type * events , int next )
2005-12-02 15:39:16 +00:00
{
int ret ;
char * dso_name_arg = NULL , * device_path_arg = NULL ;
struct dm_event_daemon_message msg ;
if ( ! ( ret = do_event ( next ? DM_EVENT_CMD_GET_NEXT_REGISTERED_DEVICE :
2007-01-11 21:54:53 +00:00
DM_EVENT_CMD_GET_REGISTERED_DEVICE ,
2007-01-08 15:18:52 +00:00
& msg , * dso_name , * device_path , * events , 0 ) ) ) {
ret = ! parse_message ( & msg , & dso_name_arg , & device_path_arg ,
events ) ;
2007-01-11 21:54:53 +00:00
} else /* FIXME: Make sure this is ENOENT */
2007-01-08 15:18:52 +00:00
ret = 0 ;
if ( msg . data )
dm_free ( msg . data ) ;
2005-12-02 15:39:16 +00:00
2007-01-08 15:18:52 +00:00
if ( next ) {
2005-12-02 15:39:16 +00:00
if ( * dso_name )
2006-01-31 14:50:38 +00:00
dm_free ( * dso_name ) ;
2005-12-02 15:39:16 +00:00
if ( * device_path )
2006-01-31 14:50:38 +00:00
dm_free ( * device_path ) ;
2005-12-02 15:39:16 +00:00
* dso_name = dso_name_arg ;
* device_path = device_path_arg ;
} else {
if ( ! ( * dso_name ) )
* dso_name = dso_name_arg ;
if ( ! ( * device_path ) )
* device_path = device_path_arg ;
}
return ret ;
}
2007-01-08 15:18:52 +00:00
int dm_event_set_timeout ( const char * device_path , uint32_t timeout )
2005-12-02 15:39:16 +00:00
{
struct dm_event_daemon_message msg ;
if ( ! device_exists ( device_path ) )
return - ENODEV ;
return do_event ( DM_EVENT_CMD_SET_TIMEOUT , & msg ,
NULL , device_path , 0 , timeout ) ;
}
2007-01-08 15:18:52 +00:00
int dm_event_get_timeout ( const char * device_path , uint32_t * timeout )
2005-12-02 15:39:16 +00:00
{
int ret ;
struct dm_event_daemon_message msg ;
if ( ! device_exists ( device_path ) )
return - ENODEV ;
2007-01-11 21:54:53 +00:00
if ( ! ( ret = do_event ( DM_EVENT_CMD_GET_TIMEOUT , & msg , NULL , device_path ,
0 , 0 ) ) )
2007-01-08 15:18:52 +00:00
* timeout = atoi ( msg . data ) ;
if ( msg . data )
dm_free ( msg . data ) ;
2005-12-02 15:39:16 +00:00
return ret ;
}
2007-01-11 21:54:53 +00:00
# endif