2005-04-28 02:32:00 +04:00
/*
2007-01-08 18:18:52 +03:00
* Copyright ( C ) 2005 - 2007 Red Hat , Inc . All rights reserved .
2005-04-28 02:32:00 +04:00
*
* 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
*/
/*
* dmeventd - dm event daemon to monitor active mapped devices
*/
2006-05-10 23:38:25 +04:00
# define _GNU_SOURCE
# define _FILE_OFFSET_BITS 64
2007-01-12 00:54:53 +03:00
# include "configure.h"
2005-04-28 02:32:00 +04:00
# include "libdevmapper.h"
2005-12-02 18:39:16 +03:00
# include "libdevmapper-event.h"
# include "dmeventd.h"
//#include "libmultilog.h"
2008-11-04 02:01:21 +03:00
# include "dm-logging.h"
2005-04-28 02:32:00 +04:00
# include <dlfcn.h>
# include <errno.h>
# include <pthread.h>
# include <sys/file.h>
2007-01-16 21:04:15 +03:00
# include <sys/stat.h>
2005-04-28 02:32:00 +04:00
# include <sys/wait.h>
2007-01-16 21:04:15 +03:00
# include <sys/time.h>
# include <sys/resource.h>
2005-04-28 02:32:00 +04:00
# include <unistd.h>
2008-07-09 17:26:07 +04:00
# include <signal.h>
2007-01-12 00:54:53 +03:00
# include <arpa/inet.h> /* for htonl, ntohl */
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
# ifdef linux
2007-01-19 20:22:17 +03:00
# include <malloc.h>
2007-01-22 18:03:57 +03:00
# define OOM_ADJ_FILE " / proc / self / oom_adj"
2007-01-19 20:22:17 +03:00
/* From linux/oom.h */
# define OOM_DISABLE (-17)
# define OOM_ADJUST_MIN (-16)
2005-12-02 18:39:16 +03:00
# endif
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
/* FIXME We use syslog for now, because multilog is not yet implemented */
2007-01-08 18:18:52 +03:00
# include <syslog.h>
2007-01-12 00:54:53 +03:00
static volatile sig_atomic_t _exit_now = 0 ; /* set to '1' when signal is given to exit */
static volatile sig_atomic_t _thread_registries_empty = 1 ; /* registries are empty initially */
2007-01-19 20:22:17 +03:00
static int _debug = 0 ;
2007-01-08 18:18:52 +03:00
2005-04-28 02:32:00 +04:00
/* List (un)link macros. */
2008-11-04 01:14:30 +03:00
# define LINK(x, head) dm_list_add(head, &(x)->list)
2007-01-08 18:18:52 +03:00
# define LINK_DSO(dso) LINK(dso, &_dso_registry)
# define LINK_THREAD(thread) LINK(thread, &_thread_registry)
2005-04-28 02:32:00 +04:00
2008-11-04 01:14:30 +03:00
# define UNLINK(x) dm_list_del(&(x)->list)
2005-04-28 02:32:00 +04:00
# define UNLINK_DSO(x) UNLINK(x)
# define UNLINK_THREAD(x) UNLINK(x)
2005-12-02 18:39:16 +03:00
# define DAEMON_NAME "dmeventd"
2005-04-28 02:32:00 +04:00
2007-01-16 23:27:07 +03:00
/*
2007-01-19 20:22:17 +03:00
Global mutex for thread list access . Has to be held when :
2007-01-16 23:27:07 +03:00
- iterating thread list
- adding or removing elements from thread list
- changing or reading thread_status ' s fields :
processing , status , events
Use _lock_mutex ( ) and _unlock_mutex ( ) to hold / release it
*/
2007-01-08 18:18:52 +03:00
static pthread_mutex_t _global_mutex ;
2007-01-23 20:38:39 +03:00
/*
There are three states a thread can attain ( see struct
thread_status , field int status ) :
- DM_THREAD_RUNNING : thread has started up and is either working or
waiting for events . . . transitions to either SHUTDOWN or DONE
- DM_THREAD_SHUTDOWN : thread is still doing something , but it is
supposed to terminate ( and transition to DONE ) as soon as it
finishes whatever it was doing at the point of flipping state to
SHUTDOWN . . . the thread is still on the thread list
- DM_THREAD_DONE : thread has terminated and has been moved over to
unused thread list , cleanup pending
*/
2007-01-08 18:18:52 +03:00
# define DM_THREAD_RUNNING 0
# define DM_THREAD_SHUTDOWN 1
# define DM_THREAD_DONE 2
2005-04-28 02:32:00 +04:00
2007-01-16 23:27:07 +03:00
# define THREAD_STACK_SIZE (300*1024)
2007-01-16 23:13:04 +03:00
2007-01-19 20:22:17 +03:00
# define DEBUGLOG(fmt, args...) _debuglog(fmt, ## args)
2005-04-28 02:32:00 +04:00
/* Data kept about a DSO. */
struct dso_data {
2008-11-04 01:14:30 +03:00
struct dm_list list ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
char * dso_name ; /* DSO name (eg, "evms", "dmraid", "lvm2"). */
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
void * dso_handle ; /* Opaque handle as returned from dlopen(). */
unsigned int ref_count ; /* Library reference count. */
2005-04-28 02:32:00 +04:00
/*
* Event processing .
*
2007-01-12 00:54:53 +03:00
* The DSO can do whatever appropriate steps if an event
* happens such as changing the mapping in case a mirror
* fails , update the application metadata etc .
*
* This function gets a dm_task that is a result of
* DM_DEVICE_WAITEVENT ioctl ( results equivalent to
* DM_DEVICE_STATUS ) . It should not destroy it .
* The caller must dispose of the task .
2005-04-28 02:32:00 +04:00
*/
2007-01-23 20:38:39 +03:00
void ( * process_event ) ( struct dm_task * dmt , enum dm_event_mask event , void * * user ) ;
2005-04-28 02:32:00 +04:00
/*
* Device registration .
*
* When an application registers a device for an event , the DSO
* can carry out appropriate steps so that a later call to
* the process_event ( ) function is sane ( eg , read metadata
* and activate a mapping ) .
*/
2007-01-12 00:54:53 +03:00
int ( * register_device ) ( const char * device , const char * uuid , int major ,
2007-01-23 20:38:39 +03:00
int minor , void * * user ) ;
2005-04-28 02:32:00 +04:00
/*
* Device unregistration .
*
* In case all devices of a mapping ( eg , RAID10 ) are unregistered
* for events , the DSO can recognize this and carry out appropriate
* steps ( eg , deactivate mapping , metadata update ) .
*/
2007-01-12 00:54:53 +03:00
int ( * unregister_device ) ( const char * device , const char * uuid ,
2007-01-23 20:38:39 +03:00
int major , int minor , void * * user ) ;
2005-04-28 02:32:00 +04:00
} ;
2008-11-04 01:14:30 +03:00
static DM_LIST_INIT ( _dso_registry ) ;
2005-04-28 02:32:00 +04:00
/* Structure to keep parsed register variables from client message. */
2005-04-28 18:02:30 +04:00
struct message_data {
2007-02-02 20:08:51 +03:00
char * id ;
2005-04-28 02:32:00 +04:00
char * dso_name ; /* Name of DSO. */
2007-01-12 00:54:53 +03:00
char * device_uuid ; /* Mapped device path. */
2005-04-28 02:32:00 +04:00
union {
char * str ; /* Events string as fetched from message. */
2007-01-15 21:21:01 +03:00
enum dm_event_mask field ; /* Events bitfield. */
2005-04-28 02:32:00 +04:00
} events ;
2005-12-02 18:39:16 +03:00
union {
char * str ;
uint32_t secs ;
} timeout ;
struct dm_event_daemon_message * msg ; /* Pointer to message buffer. */
2005-04-28 02:32:00 +04:00
} ;
/*
* Housekeeping of thread + device states .
*
* One thread per mapped device which can block on it until an event
* occurs and the event processing function of the DSO gets called .
*/
struct thread_status {
2008-11-04 01:14:30 +03:00
struct dm_list list ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
pthread_t thread ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
struct dso_data * dso_data ; /* DSO this thread accesses. */
struct {
char * uuid ;
char * name ;
int major , minor ;
} device ;
2006-12-20 17:35:02 +03:00
uint32_t event_nr ; /* event number */
int processing ; /* Set when event is being processed */
2007-01-23 20:38:39 +03:00
int status ; /* see DM_THREAD_{RUNNING,SHUTDOWN,DONE}
constants above */
2007-01-15 21:21:01 +03:00
enum dm_event_mask events ; /* bitfield for event filter. */
enum dm_event_mask current_events ; /* bitfield for occured events. */
2007-01-12 00:54:53 +03:00
struct dm_task * current_task ;
2005-12-02 18:39:16 +03:00
time_t next_time ;
uint32_t timeout ;
2008-11-04 01:14:30 +03:00
struct dm_list timeout_list ;
2007-01-23 20:38:39 +03:00
void * dso_private ; /* dso per-thread status variable */
2005-04-28 02:32:00 +04:00
} ;
2008-11-04 01:14:30 +03:00
static DM_LIST_INIT ( _thread_registry ) ;
static DM_LIST_INIT ( _thread_registry_unused ) ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
static int _timeout_running ;
2008-11-04 01:14:30 +03:00
static DM_LIST_INIT ( _timeout_registry ) ;
2007-01-08 18:18:52 +03:00
static pthread_mutex_t _timeout_mutex = PTHREAD_MUTEX_INITIALIZER ;
static pthread_cond_t _timeout_cond = PTHREAD_COND_INITIALIZER ;
2005-12-02 18:39:16 +03:00
2007-01-19 20:22:17 +03:00
static void _debuglog ( const char * fmt , . . . )
{
time_t P ;
va_list ap ;
if ( ! _debug )
return ;
va_start ( ap , fmt ) ;
time ( & P ) ;
2007-01-22 18:03:57 +03:00
fprintf ( stderr , " dmeventd[%p]: %.15s " , ( void * ) pthread_self ( ) , ctime ( & P ) + 4 ) ;
2007-01-19 20:22:17 +03:00
vfprintf ( stderr , fmt , ap ) ;
fprintf ( stderr , " \n " ) ;
va_end ( ap ) ;
}
2005-04-28 02:32:00 +04:00
/* Allocate/free the status structure for a monitoring thread. */
2007-01-15 21:58:40 +03:00
static struct thread_status * _alloc_thread_status ( struct message_data * data ,
struct dso_data * dso_data )
2005-04-28 02:32:00 +04:00
{
2006-01-31 17:50:38 +03:00
struct thread_status * ret = ( typeof ( ret ) ) dm_malloc ( sizeof ( * ret ) ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! ret )
return NULL ;
2007-03-16 17:36:14 +03:00
memset ( ret , 0 , sizeof ( * ret ) ) ;
if ( ! ( ret - > device . uuid = dm_strdup ( data - > device_uuid ) ) ) {
2007-01-15 21:58:40 +03:00
dm_free ( ret ) ;
return NULL ;
2005-04-28 02:32:00 +04:00
}
2007-01-15 21:58:40 +03:00
ret - > current_task = NULL ;
ret - > device . name = NULL ;
ret - > device . major = ret - > device . minor = 0 ;
ret - > dso_data = dso_data ;
ret - > events = data - > events . field ;
ret - > timeout = data - > timeout . secs ;
2008-11-04 01:14:30 +03:00
dm_list_init ( & ret - > timeout_list ) ;
2007-01-15 21:58:40 +03:00
2005-04-28 02:32:00 +04:00
return ret ;
}
2007-01-15 21:58:40 +03:00
static void _free_thread_status ( struct thread_status * thread )
2005-04-28 02:32:00 +04:00
{
2007-04-23 19:06:03 +04:00
if ( thread - > current_task )
dm_task_destroy ( thread - > current_task ) ;
2007-01-12 00:54:53 +03:00
dm_free ( thread - > device . uuid ) ;
dm_free ( thread - > device . name ) ;
2006-01-31 17:50:38 +03:00
dm_free ( thread ) ;
2005-04-28 02:32:00 +04:00
}
/* Allocate/free DSO data. */
2007-01-15 21:58:40 +03:00
static struct dso_data * _alloc_dso_data ( struct message_data * data )
2005-04-28 02:32:00 +04:00
{
2006-01-31 17:50:38 +03:00
struct dso_data * ret = ( typeof ( ret ) ) dm_malloc ( sizeof ( * ret ) ) ;
2005-04-28 02:32:00 +04:00
2006-12-20 17:35:02 +03:00
if ( ! ret )
return NULL ;
2007-03-16 17:36:14 +03:00
memset ( ret , 0 , sizeof ( * ret ) ) ;
if ( ! ( ret - > dso_name = dm_strdup ( data - > dso_name ) ) ) {
2006-12-20 17:35:02 +03:00
dm_free ( ret ) ;
return NULL ;
2005-04-28 02:32:00 +04:00
}
return ret ;
}
2007-01-16 23:13:04 +03:00
/* Create a device monitoring thread. */
static int _pthread_create_smallstack ( pthread_t * t , void * ( * fun ) ( void * ) , void * arg )
{
pthread_attr_t attr ;
pthread_attr_init ( & attr ) ;
2007-01-16 23:27:07 +03:00
/*
* We use a smaller stack since it gets preallocated in its entirety
*/
2007-01-16 23:13:04 +03:00
pthread_attr_setstacksize ( & attr , THREAD_STACK_SIZE ) ;
return pthread_create ( t , & attr , fun , arg ) ;
}
2007-01-15 21:58:40 +03:00
static void _free_dso_data ( struct dso_data * data )
2005-04-28 02:32:00 +04:00
{
2006-01-31 17:50:38 +03:00
dm_free ( data - > dso_name ) ;
dm_free ( data ) ;
2005-04-28 02:32:00 +04:00
}
2005-12-02 18:39:16 +03:00
/*
* Fetch a string off src and duplicate it into * ptr .
2007-01-12 00:54:53 +03:00
* Pay attention to zero - length strings .
2005-12-02 18:39:16 +03:00
*/
2007-01-12 00:54:53 +03:00
/* FIXME? move to libdevmapper to share with the client lib (need to
make delimiter a parameter then ) */
2007-01-17 00:13:07 +03:00
static int _fetch_string ( char * * ptr , char * * src , const int delimiter )
2005-04-28 02:32:00 +04:00
{
2005-12-02 18:39:16 +03:00
int ret = 0 ;
char * p ;
size_t len ;
2005-04-28 02:32:00 +04:00
2005-04-29 18:56:35 +04:00
if ( ( p = strchr ( * src , delimiter ) ) )
2005-04-28 02:32:00 +04:00
* p = 0 ;
2006-01-31 17:50:38 +03:00
if ( ( * ptr = dm_strdup ( * src ) ) ) {
2005-12-02 18:39:16 +03:00
if ( ( len = strlen ( * ptr ) ) )
* src + = len ;
else {
2006-01-31 17:50:38 +03:00
dm_free ( * ptr ) ;
2005-12-02 18:39:16 +03:00
* ptr = NULL ;
}
( * src ) + + ;
ret = 1 ;
}
2005-04-28 02:32:00 +04:00
2005-04-29 18:56:35 +04:00
if ( p )
* p = delimiter ;
2005-04-28 02:32:00 +04:00
return ret ;
}
/* Free message memory. */
2007-01-15 21:58:40 +03:00
static void _free_message ( struct message_data * message_data )
2005-04-28 02:32:00 +04:00
{
2007-02-02 20:08:51 +03:00
if ( message_data - > id )
dm_free ( message_data - > id ) ;
2005-04-28 18:02:30 +04:00
if ( message_data - > dso_name )
2006-01-31 17:50:38 +03:00
dm_free ( message_data - > dso_name ) ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
if ( message_data - > device_uuid )
dm_free ( message_data - > device_uuid ) ;
2007-01-08 18:18:52 +03:00
2005-04-28 02:32:00 +04:00
}
/* Parse a register message from the client. */
2007-01-15 21:58:40 +03:00
static int _parse_message ( struct message_data * message_data )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
int ret = 0 ;
char * p = message_data - > msg - > data ;
struct dm_event_daemon_message * msg = message_data - > msg ;
if ( ! msg - > data )
return 0 ;
2005-04-28 02:32:00 +04:00
/*
* Retrieve application identifier , mapped device
* path and events # string from message .
*/
2007-02-02 20:08:51 +03:00
if ( _fetch_string ( & message_data - > id , & p , ' ' ) & &
_fetch_string ( & message_data - > dso_name , & p , ' ' ) & &
2007-01-15 21:58:40 +03:00
_fetch_string ( & message_data - > device_uuid , & p , ' ' ) & &
_fetch_string ( & message_data - > events . str , & p , ' ' ) & &
_fetch_string ( & message_data - > timeout . str , & p , ' ' ) ) {
2005-04-28 18:02:30 +04:00
if ( message_data - > events . str ) {
2007-01-15 21:21:01 +03:00
enum dm_event_mask i = atoi ( message_data - > events . str ) ;
2005-04-28 18:02:30 +04:00
/*
* Free string representaion of events .
* Not needed an more .
*/
2006-01-31 17:50:38 +03:00
dm_free ( message_data - > events . str ) ;
2005-04-28 18:02:30 +04:00
message_data - > events . field = i ;
}
2005-12-02 18:39:16 +03:00
if ( message_data - > timeout . str ) {
uint32_t secs = atoi ( message_data - > timeout . str ) ;
2006-01-31 17:50:38 +03:00
dm_free ( message_data - > timeout . str ) ;
2005-12-02 18:39:16 +03:00
message_data - > timeout . secs = secs ? secs :
2007-01-12 00:54:53 +03:00
DM_EVENT_DEFAULT_TIMEOUT ;
2005-12-02 18:39:16 +03:00
}
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
ret = 1 ;
2005-04-28 02:32:00 +04:00
}
2007-01-08 18:18:52 +03:00
dm_free ( msg - > data ) ;
msg - > data = NULL ;
msg - > size = 0 ;
return ret ;
2005-04-28 02:32:00 +04:00
} ;
2007-01-16 23:27:07 +03:00
/* Global mutex to lock access to lists et al. See _global_mutex
above . */
2007-01-15 21:58:40 +03:00
static int _lock_mutex ( void )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
return pthread_mutex_lock ( & _global_mutex ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-15 21:58:40 +03:00
static int _unlock_mutex ( void )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
return pthread_mutex_unlock ( & _global_mutex ) ;
2005-04-28 02:32:00 +04:00
}
/* Store pid in pidfile. */
2007-01-15 21:58:40 +03:00
static int _storepid ( int lf )
2005-04-28 02:32:00 +04:00
{
int len ;
char pid [ 8 ] ;
if ( ( len = snprintf ( pid , sizeof ( pid ) , " %u \n " , getpid ( ) ) ) < 0 )
return 0 ;
2006-01-31 17:50:38 +03:00
if ( len > ( int ) sizeof ( pid ) )
len = ( int ) sizeof ( pid ) ;
2005-04-28 02:32:00 +04:00
2006-01-31 17:50:38 +03:00
if ( write ( lf , pid , ( size_t ) len ) ! = len )
2005-04-28 02:32:00 +04:00
return 0 ;
fsync ( lf ) ;
return 1 ;
}
/* Check, if a device exists. */
2007-01-15 21:58:40 +03:00
static int _fill_device_data ( struct thread_status * ts )
2005-04-28 02:32:00 +04:00
{
2007-01-12 00:54:53 +03:00
struct dm_task * dmt ;
struct dm_info dmi ;
2006-02-03 22:44:59 +03:00
2007-01-12 00:54:53 +03:00
if ( ! ts - > device . uuid )
2006-02-03 22:44:59 +03:00
return 0 ;
2007-01-12 00:54:53 +03:00
ts - > device . name = NULL ;
ts - > device . major = ts - > device . minor = 0 ;
2006-02-03 22:44:59 +03:00
2007-01-12 00:54:53 +03:00
dmt = dm_task_create ( DM_DEVICE_INFO ) ;
if ( ! dmt )
2006-02-03 22:44:59 +03:00
return 0 ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
dm_task_set_uuid ( dmt , ts - > device . uuid ) ;
if ( ! dm_task_run ( dmt ) )
goto fail ;
ts - > device . name = dm_strdup ( dm_task_get_name ( dmt ) ) ;
if ( ! ts - > device . name )
goto fail ;
if ( ! dm_task_get_info ( dmt , & dmi ) )
goto fail ;
ts - > device . major = dmi . major ;
ts - > device . minor = dmi . minor ;
dm_task_destroy ( dmt ) ;
return 1 ;
fail :
dm_task_destroy ( dmt ) ;
dm_free ( ts - > device . name ) ;
return 0 ;
2005-04-28 02:32:00 +04:00
}
/*
* Find an existing thread for a device .
*
2006-01-27 23:52:21 +03:00
* Mutex must be held when calling this .
2005-04-28 02:32:00 +04:00
*/
2007-01-15 21:58:40 +03:00
static struct thread_status * _lookup_thread_status ( struct message_data * data )
2005-04-28 02:32:00 +04:00
{
struct thread_status * thread ;
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( thread , & _thread_registry )
2007-01-12 00:54:53 +03:00
if ( ! strcmp ( data - > device_uuid , thread - > device . uuid ) )
return thread ;
2005-04-28 02:32:00 +04:00
return NULL ;
}
/* Cleanup at exit. */
2007-01-15 21:58:40 +03:00
static void _exit_dm_lib ( void )
2005-04-28 02:32:00 +04:00
{
dm_lib_release ( ) ;
dm_lib_exit ( ) ;
}
2007-01-22 18:03:57 +03:00
static void _exit_timeout ( void * unused __attribute ( ( unused ) ) )
2005-12-02 18:39:16 +03:00
{
2007-01-08 18:18:52 +03:00
_timeout_running = 0 ;
pthread_mutex_unlock ( & _timeout_mutex ) ;
2005-12-02 18:39:16 +03:00
}
/* Wake up monitor threads every so often. */
2007-01-22 18:03:57 +03:00
static void * _timeout_thread ( void * unused __attribute ( ( unused ) ) )
2005-12-02 18:39:16 +03:00
{
struct timespec timeout ;
time_t curr_time ;
timeout . tv_nsec = 0 ;
2007-01-15 21:58:40 +03:00
pthread_cleanup_push ( _exit_timeout , NULL ) ;
2007-01-08 18:18:52 +03:00
pthread_mutex_lock ( & _timeout_mutex ) ;
2005-12-02 18:39:16 +03:00
2008-11-04 01:14:30 +03:00
while ( ! dm_list_empty ( & _timeout_registry ) ) {
2005-12-02 18:39:16 +03:00
struct thread_status * thread ;
2007-01-22 18:03:57 +03:00
timeout . tv_sec = 0 ;
2005-12-02 18:39:16 +03:00
curr_time = time ( NULL ) ;
2008-11-04 01:14:30 +03:00
dm_list_iterate_items_gen ( thread , & _timeout_registry , timeout_list ) {
2007-01-22 18:03:57 +03:00
if ( thread - > next_time < = curr_time ) {
2005-12-02 18:39:16 +03:00
thread - > next_time = curr_time + thread - > timeout ;
pthread_kill ( thread - > thread , SIGALRM ) ;
}
2007-01-22 18:03:57 +03:00
if ( thread - > next_time < timeout . tv_sec | | ! timeout . tv_sec )
2005-12-02 18:39:16 +03:00
timeout . tv_sec = thread - > next_time ;
}
2007-01-12 00:54:53 +03:00
pthread_cond_timedwait ( & _timeout_cond , & _timeout_mutex ,
& timeout ) ;
2005-12-02 18:39:16 +03:00
}
pthread_cleanup_pop ( 1 ) ;
return NULL ;
}
2007-01-15 21:58:40 +03:00
static int _register_for_timeout ( struct thread_status * thread )
2005-12-02 18:39:16 +03:00
{
int ret = 0 ;
2007-01-08 18:18:52 +03:00
pthread_mutex_lock ( & _timeout_mutex ) ;
2005-12-02 18:39:16 +03:00
thread - > next_time = time ( NULL ) + thread - > timeout ;
2008-11-04 01:14:30 +03:00
if ( dm_list_empty ( & thread - > timeout_list ) ) {
dm_list_add ( & _timeout_registry , & thread - > timeout_list ) ;
2007-01-08 18:18:52 +03:00
if ( _timeout_running )
pthread_cond_signal ( & _timeout_cond ) ;
2005-12-02 18:39:16 +03:00
}
2007-01-08 18:18:52 +03:00
if ( ! _timeout_running ) {
2005-12-02 18:39:16 +03:00
pthread_t timeout_id ;
2007-01-16 23:13:04 +03:00
if ( ! ( ret = - _pthread_create_smallstack ( & timeout_id , _timeout_thread , NULL ) ) )
2007-01-08 18:18:52 +03:00
_timeout_running = 1 ;
2005-12-02 18:39:16 +03:00
}
2007-01-08 18:18:52 +03:00
pthread_mutex_unlock ( & _timeout_mutex ) ;
2005-12-02 18:39:16 +03:00
return ret ;
}
2007-01-15 21:58:40 +03:00
static void _unregister_for_timeout ( struct thread_status * thread )
2005-12-02 18:39:16 +03:00
{
2007-01-08 18:18:52 +03:00
pthread_mutex_lock ( & _timeout_mutex ) ;
2008-11-04 01:14:30 +03:00
if ( ! dm_list_empty ( & thread - > timeout_list ) ) {
dm_list_del ( & thread - > timeout_list ) ;
dm_list_init ( & thread - > timeout_list ) ;
2005-12-02 18:39:16 +03:00
}
2007-01-08 18:18:52 +03:00
pthread_mutex_unlock ( & _timeout_mutex ) ;
2005-12-02 18:39:16 +03:00
}
2007-01-15 21:58:40 +03:00
static void _no_intr_log ( int level , const char * file , int line ,
2007-01-12 00:54:53 +03:00
const char * f , . . . )
2005-12-02 18:39:16 +03:00
{
va_list ap ;
if ( errno = = EINTR )
return ;
if ( level > _LOG_WARN )
return ;
va_start ( ap , f ) ;
if ( level < _LOG_WARN )
vfprintf ( stderr , f , ap ) ;
else
vprintf ( f , ap ) ;
va_end ( ap ) ;
if ( level < _LOG_WARN )
fprintf ( stderr , " \n " ) ;
else
fprintf ( stdout , " \n " ) ;
}
2007-01-15 21:58:40 +03:00
static sigset_t _unblock_sigalrm ( void )
2005-12-02 18:39:16 +03:00
{
sigset_t set , old ;
sigemptyset ( & set ) ;
sigaddset ( & set , SIGALRM ) ;
pthread_sigmask ( SIG_UNBLOCK , & set , & old ) ;
return old ;
}
2007-01-08 18:18:52 +03:00
# define DM_WAIT_RETRY 0
# define DM_WAIT_INTR 1
# define DM_WAIT_FATAL 2
2005-04-28 02:32:00 +04:00
/* Wait on a device until an event occurs. */
2007-01-15 21:58:40 +03:00
static int _event_wait ( struct thread_status * thread , struct dm_task * * task )
2005-04-28 02:32:00 +04:00
{
2005-12-02 18:39:16 +03:00
sigset_t set ;
2007-01-08 18:18:52 +03:00
int ret = DM_WAIT_RETRY ;
2005-04-28 02:32:00 +04:00
struct dm_task * dmt ;
2005-12-02 18:39:16 +03:00
struct dm_info info ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
* task = 0 ;
2005-04-28 02:32:00 +04:00
if ( ! ( dmt = dm_task_create ( DM_DEVICE_WAITEVENT ) ) )
2007-01-08 18:18:52 +03:00
return DM_WAIT_RETRY ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
thread - > current_task = dmt ;
if ( ! dm_task_set_uuid ( dmt , thread - > device . uuid ) | |
! dm_task_set_event_nr ( dmt , thread - > event_nr ) )
2005-12-02 18:39:16 +03:00
goto out ;
/*
* This is so that you can break out of waiting on an event ,
* either for a timeout event , or to cancel the thread .
*/
2007-01-15 21:58:40 +03:00
set = _unblock_sigalrm ( ) ;
dm_log_init ( _no_intr_log ) ;
2005-12-02 18:39:16 +03:00
errno = 0 ;
2007-01-12 00:54:53 +03:00
if ( dm_task_run ( dmt ) ) {
2005-12-02 18:39:16 +03:00
thread - > current_events | = DM_EVENT_DEVICE_ERROR ;
2007-01-08 18:18:52 +03:00
ret = DM_WAIT_INTR ;
2005-12-02 18:39:16 +03:00
if ( ( ret = dm_task_get_info ( dmt , & info ) ) )
thread - > event_nr = info . event_nr ;
} else if ( thread - > events & DM_EVENT_TIMEOUT & & errno = = EINTR ) {
thread - > current_events | = DM_EVENT_TIMEOUT ;
2007-01-08 18:18:52 +03:00
ret = DM_WAIT_INTR ;
2007-04-24 17:29:02 +04:00
} else if ( thread - > status = = DM_THREAD_SHUTDOWN & & errno = = EINTR ) {
ret = DM_WAIT_FATAL ;
2006-12-20 17:35:02 +03:00
} else {
syslog ( LOG_NOTICE , " dm_task_run failed, errno = %d, %s " ,
errno , strerror ( errno ) ) ;
if ( errno = = ENXIO ) {
2007-01-12 00:54:53 +03:00
syslog ( LOG_ERR , " %s disappeared, detaching " ,
thread - > device . name ) ;
2007-01-08 18:18:52 +03:00
ret = DM_WAIT_FATAL ;
2006-12-20 17:35:02 +03:00
}
2005-04-28 02:32:00 +04:00
}
2005-12-02 18:39:16 +03:00
pthread_sigmask ( SIG_SETMASK , & set , NULL ) ;
dm_log_init ( NULL ) ;
2007-01-12 00:54:53 +03:00
out :
if ( ret = = DM_WAIT_FATAL | | ret = = DM_WAIT_RETRY ) {
dm_task_destroy ( dmt ) ;
thread - > current_task = NULL ;
} else
* task = dmt ;
2005-04-28 02:32:00 +04:00
return ret ;
}
/* Register a device with the DSO. */
2007-01-15 21:58:40 +03:00
static int _do_register_device ( struct thread_status * thread )
2005-04-28 02:32:00 +04:00
{
2007-01-12 00:54:53 +03:00
return thread - > dso_data - > register_device ( thread - > device . name ,
thread - > device . uuid ,
thread - > device . major ,
2007-01-23 20:38:39 +03:00
thread - > device . minor ,
& ( thread - > dso_private ) ) ;
2005-04-28 02:32:00 +04:00
}
/* Unregister a device with the DSO. */
2007-01-15 21:58:40 +03:00
static int _do_unregister_device ( struct thread_status * thread )
2005-04-28 02:32:00 +04:00
{
2007-01-12 00:54:53 +03:00
return thread - > dso_data - > unregister_device ( thread - > device . name ,
thread - > device . uuid ,
thread - > device . major ,
2007-01-23 20:38:39 +03:00
thread - > device . minor ,
& ( thread - > dso_private ) ) ;
2005-04-28 02:32:00 +04:00
}
2005-12-02 18:39:16 +03:00
/* Process an event in the DSO. */
2007-01-15 21:58:40 +03:00
static void _do_process_event ( struct thread_status * thread , struct dm_task * task )
2005-04-28 02:32:00 +04:00
{
2007-01-23 20:38:39 +03:00
thread - > dso_data - > process_event ( task , thread - > current_events , & ( thread - > dso_private ) ) ;
2005-04-28 02:32:00 +04:00
}
/* Thread cleanup handler to unregister device. */
2007-01-15 21:58:40 +03:00
static void _monitor_unregister ( void * arg )
2005-04-28 02:32:00 +04:00
{
2007-01-16 23:27:07 +03:00
struct thread_status * thread = arg , * thread_iter ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! _do_unregister_device ( thread ) )
2007-01-08 18:18:52 +03:00
syslog ( LOG_ERR , " %s: %s unregister failed \n " , __func__ ,
2007-01-12 00:54:53 +03:00
thread - > device . name ) ;
if ( thread - > current_task )
dm_task_destroy ( thread - > current_task ) ;
thread - > current_task = NULL ;
2007-01-16 23:27:07 +03:00
_lock_mutex ( ) ;
if ( thread - > events & DM_EVENT_TIMEOUT ) {
/* _unregister_for_timeout locks another mutex, we
don ' t want to deadlock so we release our mutex for
a bit */
_unlock_mutex ( ) ;
_unregister_for_timeout ( thread ) ;
_lock_mutex ( ) ;
}
/* we may have been relinked to unused registry since we were
called , so check that */
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( thread_iter , & _thread_registry_unused )
2007-01-16 23:27:07 +03:00
if ( thread_iter = = thread ) {
thread - > status = DM_THREAD_DONE ;
_unlock_mutex ( ) ;
return ;
}
thread - > status = DM_THREAD_DONE ;
UNLINK_THREAD ( thread ) ;
LINK ( thread , & _thread_registry_unused ) ;
_unlock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-22 18:03:57 +03:00
static struct dm_task * _get_device_status ( struct thread_status * ts )
{
struct dm_task * dmt = dm_task_create ( DM_DEVICE_STATUS ) ;
if ( ! dmt )
return NULL ;
dm_task_set_uuid ( dmt , ts - > device . uuid ) ;
if ( ! dm_task_run ( dmt ) ) {
dm_task_destroy ( dmt ) ;
return NULL ;
}
return dmt ;
}
2005-04-28 02:32:00 +04:00
/* Device monitoring thread. */
2007-01-15 21:58:40 +03:00
static void * _monitor_thread ( void * arg )
2005-04-28 02:32:00 +04:00
{
struct thread_status * thread = arg ;
2006-12-20 17:35:02 +03:00
int wait_error = 0 ;
2007-01-12 00:54:53 +03:00
struct dm_task * task ;
2005-04-28 02:32:00 +04:00
pthread_setcanceltype ( PTHREAD_CANCEL_DEFERRED , NULL ) ;
2007-01-15 21:58:40 +03:00
pthread_cleanup_push ( _monitor_unregister , thread ) ;
2005-04-28 02:32:00 +04:00
2006-01-27 23:52:21 +03:00
/* Wait for do_process_request() to finish its task. */
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2007-01-08 18:18:52 +03:00
thread - > status = DM_THREAD_RUNNING ;
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
/* Loop forever awaiting/analyzing device events. */
while ( 1 ) {
thread - > current_events = 0 ;
2007-01-15 21:58:40 +03:00
wait_error = _event_wait ( thread , & task ) ;
2007-01-08 18:18:52 +03:00
if ( wait_error = = DM_WAIT_RETRY )
2005-04-28 02:32:00 +04:00
continue ;
2007-01-08 18:18:52 +03:00
if ( wait_error = = DM_WAIT_FATAL )
2006-12-20 17:35:02 +03:00
break ;
2007-01-22 18:03:57 +03:00
/* Timeout occurred, task is not filled properly.
* We get device status here for processing it in DSO .
*/
if ( wait_error = = DM_WAIT_INTR & &
thread - > current_events & DM_EVENT_TIMEOUT ) {
dm_task_destroy ( task ) ;
task = _get_device_status ( thread ) ;
/* FIXME: syslog fail here ? */
if ( ! ( thread - > current_task = task ) )
continue ;
}
2007-01-12 00:54:53 +03:00
/*
* We know that wait succeeded and stored a
* pointer to dm_task with device status into task .
*/
2005-04-28 18:02:30 +04:00
/*
* Check against filter .
*
2007-01-15 21:58:40 +03:00
* If there ' s current events delivered from _event_wait ( ) AND
2005-04-28 18:02:30 +04:00
* the device got registered for those events AND
* those events haven ' t been processed yet , call
* the DSO ' s process_event ( ) handler .
*/
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2007-01-08 18:18:52 +03:00
if ( thread - > status = = DM_THREAD_SHUTDOWN ) {
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2007-01-08 18:18:52 +03:00
break ;
}
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2007-01-08 18:18:52 +03:00
2007-01-12 00:54:53 +03:00
if ( thread - > events & thread - > current_events ) {
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2006-01-27 23:50:01 +03:00
thread - > processing = 1 ;
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2007-01-12 00:54:53 +03:00
2007-01-15 21:58:40 +03:00
_do_process_event ( thread , task ) ;
2007-01-12 00:54:53 +03:00
dm_task_destroy ( task ) ;
thread - > current_task = NULL ;
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2006-01-27 23:50:01 +03:00
thread - > processing = 0 ;
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2007-01-12 00:54:53 +03:00
} else {
dm_task_destroy ( task ) ;
thread - > current_task = NULL ;
2005-04-28 02:32:00 +04:00
}
}
2007-01-12 00:54:53 +03:00
pthread_cleanup_pop ( 1 ) ;
2007-01-08 18:18:52 +03:00
return NULL ;
2005-04-28 02:32:00 +04:00
}
/* Create a device monitoring thread. */
2007-01-15 21:58:40 +03:00
static int _create_thread ( struct thread_status * thread )
2005-04-28 02:32:00 +04:00
{
2007-01-16 23:13:04 +03:00
return _pthread_create_smallstack ( & thread - > thread , _monitor_thread , thread ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-15 21:58:40 +03:00
static int _terminate_thread ( struct thread_status * thread )
2005-04-28 02:32:00 +04:00
{
2005-12-02 18:39:16 +03:00
return pthread_kill ( thread - > thread , SIGALRM ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-16 23:27:07 +03:00
/* DSO reference counting. Call with _global_mutex locked! */
2007-01-15 21:58:40 +03:00
static void _lib_get ( struct dso_data * data )
2005-04-28 02:32:00 +04:00
{
data - > ref_count + + ;
}
2007-01-15 21:58:40 +03:00
static void _lib_put ( struct dso_data * data )
2005-04-28 02:32:00 +04:00
{
if ( ! - - data - > ref_count ) {
dlclose ( data - > dso_handle ) ;
UNLINK_DSO ( data ) ;
2007-01-15 21:58:40 +03:00
_free_dso_data ( data ) ;
2005-04-28 02:32:00 +04:00
}
}
/* Find DSO data. */
2007-01-15 21:58:40 +03:00
static struct dso_data * _lookup_dso ( struct message_data * data )
2005-04-28 02:32:00 +04:00
{
struct dso_data * dso_data , * ret = NULL ;
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( dso_data , & _dso_registry )
2007-01-12 00:54:53 +03:00
if ( ! strcmp ( data - > dso_name , dso_data - > dso_name ) ) {
2007-01-15 21:58:40 +03:00
_lib_get ( dso_data ) ;
2007-01-12 00:54:53 +03:00
ret = dso_data ;
break ;
}
2005-04-28 02:32:00 +04:00
return ret ;
}
/* Lookup DSO symbols we need. */
2007-01-22 18:03:57 +03:00
static int _lookup_symbol ( void * dl , void * * symbol , const char * name )
2005-04-28 02:32:00 +04:00
{
if ( ( * symbol = dlsym ( dl , name ) ) )
return 1 ;
return 0 ;
}
static int lookup_symbols ( void * dl , struct dso_data * data )
{
2007-01-22 18:03:57 +03:00
return _lookup_symbol ( dl , ( void * ) & data - > process_event ,
2005-04-28 02:32:00 +04:00
" process_event " ) & &
2007-01-22 18:03:57 +03:00
_lookup_symbol ( dl , ( void * ) & data - > register_device ,
2007-01-12 00:54:53 +03:00
" register_device " ) & &
2007-01-22 18:03:57 +03:00
_lookup_symbol ( dl , ( void * ) & data - > unregister_device ,
2007-01-12 00:54:53 +03:00
" unregister_device " ) ;
2005-04-28 02:32:00 +04:00
}
/* Load an application specific DSO. */
2007-01-15 21:58:40 +03:00
static struct dso_data * _load_dso ( struct message_data * data )
2005-04-28 02:32:00 +04:00
{
void * dl ;
struct dso_data * ret = NULL ;
2007-01-12 00:54:53 +03:00
if ( ! ( dl = dlopen ( data - > dso_name , RTLD_NOW ) ) ) {
2007-01-08 18:18:52 +03:00
const char * dlerr = dlerror ( ) ;
2007-01-12 00:54:53 +03:00
syslog ( LOG_ERR , " dmeventd %s dlopen failed: %s " , data - > dso_name ,
dlerr ) ;
data - > msg - > size =
2007-02-02 20:08:51 +03:00
dm_asprintf ( & ( data - > msg - > data ) , " %s %s dlopen failed: %s " ,
data - > id , data - > dso_name , dlerr ) ;
2006-02-03 22:41:34 +03:00
return NULL ;
2005-12-02 18:39:16 +03:00
}
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! ( ret = _alloc_dso_data ( data ) ) ) {
2006-02-03 22:41:34 +03:00
dlclose ( dl ) ;
return NULL ;
}
2005-04-28 02:32:00 +04:00
2006-02-03 22:41:34 +03:00
if ( ! ( lookup_symbols ( dl , ret ) ) ) {
2007-01-15 21:58:40 +03:00
_free_dso_data ( ret ) ;
2006-02-03 22:41:34 +03:00
dlclose ( dl ) ;
return NULL ;
}
2005-04-28 02:32:00 +04:00
/*
* Keep handle to close the library once
* we ' ve got no references to it any more .
*/
ret - > dso_handle = dl ;
2007-01-15 21:58:40 +03:00
_lib_get ( ret ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
LINK_DSO ( ret ) ;
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
return ret ;
}
2005-12-02 18:39:16 +03:00
/* Return success on daemon active check. */
2007-01-15 21:58:40 +03:00
static int _active ( struct message_data * message_data )
2005-12-02 18:39:16 +03:00
{
return 0 ;
}
2005-04-28 02:32:00 +04:00
/*
* Register for an event .
*
2005-12-02 18:39:16 +03:00
* Only one caller at a time here , because we use
* a FIFO and lock it against multiple accesses .
2005-04-28 02:32:00 +04:00
*/
2007-01-15 21:58:40 +03:00
static int _register_for_event ( struct message_data * message_data )
2005-04-28 02:32:00 +04:00
{
int ret = 0 ;
2005-12-02 18:39:16 +03:00
struct thread_status * thread , * thread_new = NULL ;
2005-04-28 02:32:00 +04:00
struct dso_data * dso_data ;
2007-01-15 21:58:40 +03:00
if ( ! ( dso_data = _lookup_dso ( message_data ) ) & &
! ( dso_data = _load_dso ( message_data ) ) ) {
2005-04-28 02:32:00 +04:00
stack ;
2005-12-02 18:39:16 +03:00
# ifdef ELIBACC
2005-04-28 02:32:00 +04:00
ret = - ELIBACC ;
2005-12-02 18:39:16 +03:00
# else
ret = - ENODEV ;
# endif
2005-04-28 02:32:00 +04:00
goto out ;
}
2007-01-12 00:54:53 +03:00
2005-04-28 02:32:00 +04:00
/* Preallocate thread status struct to avoid deadlock. */
2007-01-15 21:58:40 +03:00
if ( ! ( thread_new = _alloc_thread_status ( message_data , dso_data ) ) ) {
2005-04-28 02:32:00 +04:00
stack ;
ret = - ENOMEM ;
goto out ;
}
2007-01-15 21:58:40 +03:00
if ( ! _fill_device_data ( thread_new ) ) {
2007-01-12 00:54:53 +03:00
stack ;
ret = - ENODEV ;
goto out ;
}
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 22:19:31 +03:00
/* If creation of timeout thread fails (as it may), we fail
here completely . The client is responsible for either
retrying later or trying to register without timeout
events . However , if timeout thread cannot be started , it
usually means we are so starved on resources that we are
almost as good as dead already . . . */
2007-01-15 22:47:49 +03:00
if ( thread_new - > events & DM_EVENT_TIMEOUT ) {
2007-01-22 18:03:57 +03:00
ret = - _register_for_timeout ( thread_new ) ;
2007-01-19 20:22:17 +03:00
if ( ret ) {
_unlock_mutex ( ) ;
goto out ;
}
}
2007-01-15 22:19:31 +03:00
2007-01-15 21:58:40 +03:00
if ( ! ( thread = _lookup_thread_status ( message_data ) ) ) {
_unlock_mutex ( ) ;
2005-12-02 18:39:16 +03:00
2007-01-15 21:58:40 +03:00
if ( ! ( ret = _do_register_device ( thread_new ) ) )
2005-12-02 18:39:16 +03:00
goto out ;
2005-04-28 02:32:00 +04:00
thread = thread_new ;
thread_new = NULL ;
/* Try to create the monitoring thread for this device. */
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
if ( ( ret = - _create_thread ( thread ) ) ) {
_unlock_mutex ( ) ;
_do_unregister_device ( thread ) ;
_free_thread_status ( thread ) ;
2005-04-28 02:32:00 +04:00
goto out ;
} else
LINK_THREAD ( thread ) ;
}
/* Or event # into events bitfield. */
2005-04-28 18:02:30 +04:00
thread - > events | = message_data - > events . field ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
out :
2005-04-28 02:32:00 +04:00
/*
* Deallocate thread status after releasing
* the lock in case we haven ' t used it .
*/
if ( thread_new )
2007-01-15 21:58:40 +03:00
_free_thread_status ( thread_new ) ;
2005-04-28 02:32:00 +04:00
return ret ;
}
/*
* Unregister for an event .
*
* Only one caller at a time here as with register_for_event ( ) .
*/
2007-01-15 21:58:40 +03:00
static int _unregister_for_event ( struct message_data * message_data )
2005-04-28 02:32:00 +04:00
{
int ret = 0 ;
struct thread_status * thread ;
/*
* Clear event in bitfield and deactivate
* monitoring thread in case bitfield is 0.
*/
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! ( thread = _lookup_thread_status ( message_data ) ) ) {
_unlock_mutex ( ) ;
2005-12-02 18:39:16 +03:00
ret = - ENODEV ;
2005-04-28 02:32:00 +04:00
goto out ;
}
2007-01-16 23:27:07 +03:00
if ( thread - > status = = DM_THREAD_DONE ) {
/* the thread has terminated while we were not
watching */
_unlock_mutex ( ) ;
return 0 ;
}
2005-04-28 18:02:30 +04:00
thread - > events & = ~ message_data - > events . field ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
if ( ! ( thread - > events & DM_EVENT_TIMEOUT ) )
2007-01-15 21:58:40 +03:00
_unregister_for_timeout ( thread ) ;
2005-04-28 02:32:00 +04:00
/*
2005-12-02 18:39:16 +03:00
* In case there ' s no events to monitor on this device - >
* unlink and terminate its monitoring thread .
2005-04-28 02:32:00 +04:00
*/
if ( ! thread - > events ) {
2006-01-27 23:50:01 +03:00
UNLINK_THREAD ( thread ) ;
2007-01-08 18:18:52 +03:00
LINK ( thread , & _thread_registry_unused ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-04-28 02:32:00 +04:00
2007-01-12 00:54:53 +03:00
out :
2005-04-28 02:32:00 +04:00
return ret ;
}
2005-04-28 18:02:30 +04:00
/*
2005-12-02 18:39:16 +03:00
* Get registered device .
2005-04-28 18:02:30 +04:00
*
* Only one caller at a time here as with register_for_event ( ) .
*/
2007-01-15 21:58:40 +03:00
static int _registered_device ( struct message_data * message_data ,
2005-04-29 17:41:25 +04:00
struct thread_status * thread )
2005-04-28 18:02:30 +04:00
{
2005-12-02 18:39:16 +03:00
struct dm_event_daemon_message * msg = message_data - > msg ;
2005-04-28 18:02:30 +04:00
2007-02-02 20:08:51 +03:00
const char * fmt = " %s %s %s %u " ;
const char * id = message_data - > id ;
2007-01-08 18:18:52 +03:00
const char * dso = thread - > dso_data - > dso_name ;
2007-01-12 00:54:53 +03:00
const char * dev = thread - > device . uuid ;
unsigned events = ( ( thread - > status = = DM_THREAD_RUNNING )
& & ( thread - > events ) ) ? thread - > events : thread - >
events | DM_EVENT_REGISTRATION_PENDING ;
2007-01-08 18:18:52 +03:00
if ( msg - > data )
dm_free ( msg - > data ) ;
2007-02-02 20:08:51 +03:00
msg - > size = dm_asprintf ( & ( msg - > data ) , fmt , id , dso , dev , events ) ;
2005-04-29 17:41:25 +04:00
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-04-28 18:02:30 +04:00
return 0 ;
}
2007-01-15 21:58:40 +03:00
static int _want_registered_device ( char * dso_name , char * device_uuid ,
2005-12-02 18:39:16 +03:00
struct thread_status * thread )
2005-04-28 18:02:30 +04:00
{
2005-12-02 18:39:16 +03:00
/* If DSO names and device paths are equal. */
2007-01-12 00:54:53 +03:00
if ( dso_name & & device_uuid )
2005-12-02 18:39:16 +03:00
return ! strcmp ( dso_name , thread - > dso_data - > dso_name ) & &
2007-01-17 02:03:13 +03:00
! strcmp ( device_uuid , thread - > device . uuid ) & &
( thread - > status = = DM_THREAD_RUNNING | |
( thread - > events & DM_EVENT_REGISTRATION_PENDING ) ) ;
2005-12-02 18:39:16 +03:00
/* If DSO names are equal. */
if ( dso_name )
2007-01-17 02:03:13 +03:00
return ! strcmp ( dso_name , thread - > dso_data - > dso_name ) & &
( thread - > status = = DM_THREAD_RUNNING | |
( thread - > events & DM_EVENT_REGISTRATION_PENDING ) ) ;
2007-01-12 00:54:53 +03:00
2005-12-02 18:39:16 +03:00
/* If device paths are equal. */
2007-01-12 00:54:53 +03:00
if ( device_uuid )
2007-01-17 02:03:13 +03:00
return ! strcmp ( device_uuid , thread - > device . uuid ) & &
( thread - > status = = DM_THREAD_RUNNING | |
( thread - > events & DM_EVENT_REGISTRATION_PENDING ) ) ;
2005-04-28 18:02:30 +04:00
2005-12-02 18:39:16 +03:00
return 1 ;
}
2005-04-28 18:02:30 +04:00
2007-01-15 21:58:40 +03:00
static int _get_registered_dev ( struct message_data * message_data , int next )
2005-12-02 18:39:16 +03:00
{
2007-01-17 02:03:13 +03:00
struct thread_status * thread , * hit = NULL ;
2005-05-02 15:02:19 +04:00
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2005-04-29 17:41:25 +04:00
2005-12-02 18:39:16 +03:00
/* Iterate list of threads checking if we want a particular one. */
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( thread , & _thread_registry )
2007-01-17 02:03:13 +03:00
if ( _want_registered_device ( message_data - > dso_name ,
message_data - > device_uuid ,
thread ) ) {
hit = thread ;
break ;
}
2005-04-28 18:02:30 +04:00
2005-04-29 17:41:25 +04:00
/*
* If we got a registered device and want the next one - >
2005-12-02 18:39:16 +03:00
* fetch next conforming element off the list .
2005-04-29 17:41:25 +04:00
*/
2007-01-17 02:03:13 +03:00
if ( hit & & ! next ) {
_unlock_mutex ( ) ;
return _registered_device ( message_data , hit ) ;
}
if ( ! hit )
2006-12-20 17:35:02 +03:00
goto out ;
2005-04-28 18:02:30 +04:00
2007-01-23 20:38:39 +03:00
thread = hit ;
2007-01-17 02:03:13 +03:00
2007-01-23 20:38:39 +03:00
while ( 1 ) {
2008-11-04 01:14:30 +03:00
if ( dm_list_end ( & _thread_registry , & thread - > list ) )
2006-12-20 17:35:02 +03:00
goto out ;
2007-01-12 00:54:53 +03:00
2008-11-04 01:14:30 +03:00
thread = dm_list_item ( thread - > list . n , struct thread_status ) ;
2007-01-23 20:38:39 +03:00
if ( _want_registered_device ( message_data - > dso_name , NULL , thread ) ) {
hit = thread ;
break ;
}
}
2006-12-20 17:35:02 +03:00
2007-01-17 02:03:13 +03:00
_unlock_mutex ( ) ;
2007-01-23 20:38:39 +03:00
return _registered_device ( message_data , hit ) ;
2005-04-28 18:02:30 +04:00
2007-01-12 00:54:53 +03:00
out :
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2007-01-23 20:38:39 +03:00
2005-12-02 18:39:16 +03:00
return - ENOENT ;
}
2007-01-15 21:58:40 +03:00
static int _get_registered_device ( struct message_data * message_data )
2005-12-02 18:39:16 +03:00
{
2007-01-15 21:58:40 +03:00
return _get_registered_dev ( message_data , 0 ) ;
2005-12-02 18:39:16 +03:00
}
2007-01-15 21:58:40 +03:00
static int _get_next_registered_device ( struct message_data * message_data )
2005-12-02 18:39:16 +03:00
{
2007-01-15 21:58:40 +03:00
return _get_registered_dev ( message_data , 1 ) ;
2005-04-28 18:02:30 +04:00
}
2007-01-15 21:58:40 +03:00
static int _set_timeout ( struct message_data * message_data )
2005-12-02 18:39:16 +03:00
{
struct thread_status * thread ;
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
if ( ( thread = _lookup_thread_status ( message_data ) ) )
2007-01-12 00:54:53 +03:00
thread - > timeout = message_data - > timeout . secs ;
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-12-02 18:39:16 +03:00
return thread ? 0 : - ENODEV ;
}
2007-01-15 21:58:40 +03:00
static int _get_timeout ( struct message_data * message_data )
2005-12-02 18:39:16 +03:00
{
struct thread_status * thread ;
struct dm_event_daemon_message * msg = message_data - > msg ;
2007-01-08 18:18:52 +03:00
if ( msg - > data )
dm_free ( msg - > data ) ;
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
if ( ( thread = _lookup_thread_status ( message_data ) ) ) {
2007-01-12 00:54:53 +03:00
msg - > size =
2007-02-02 20:08:51 +03:00
dm_asprintf ( & ( msg - > data ) , " %s % " PRIu32 , message_data - > id ,
thread - > timeout ) ;
2007-01-08 18:18:52 +03:00
} else {
msg - > data = NULL ;
msg - > size = 0 ;
}
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2005-12-02 18:39:16 +03:00
return thread ? 0 : - ENODEV ;
}
2005-04-28 02:32:00 +04:00
/* Initialize a fifos structure with path names. */
2007-01-15 21:58:40 +03:00
static void _init_fifos ( struct dm_event_fifos * fifos )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
memset ( fifos , 0 , sizeof ( * fifos ) ) ;
2005-12-02 18:39:16 +03:00
2007-01-08 18:18:52 +03:00
fifos - > client_path = DM_EVENT_FIFO_CLIENT ;
fifos - > server_path = DM_EVENT_FIFO_SERVER ;
2005-04-28 02:32:00 +04:00
}
/* Open fifos used for client communication. */
2007-01-15 21:58:40 +03:00
static int _open_fifos ( struct dm_event_fifos * fifos )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
/* Create fifos */
if ( ( ( mkfifo ( fifos - > client_path , 0600 ) = = - 1 ) & & errno ! = EEXIST ) | |
( ( mkfifo ( fifos - > server_path , 0600 ) = = - 1 ) & & errno ! = EEXIST ) ) {
syslog ( LOG_ERR , " %s: Failed to create a fifo. \n " , __func__ ) ;
2005-04-28 02:32:00 +04:00
stack ;
2007-01-08 18:18:52 +03:00
return - errno ;
}
2007-01-12 00:54:53 +03:00
struct stat st ;
/* Warn about wrong permissions if applicable */
if ( ( ! stat ( fifos - > client_path , & st ) ) & & ( st . st_mode & 0777 ) ! = 0600 )
syslog ( LOG_WARNING , " Fixing wrong permissions on %s " ,
fifos - > client_path ) ;
if ( ( ! stat ( fifos - > server_path , & st ) ) & & ( st . st_mode & 0777 ) ! = 0600 )
syslog ( LOG_WARNING , " Fixing wrong permissions on %s " ,
fifos - > server_path ) ;
2007-01-08 18:18:52 +03:00
/* If they were already there, make sure permissions are ok. */
if ( chmod ( fifos - > client_path , 0600 ) ) {
syslog ( LOG_ERR , " Unable to set correct file permissions on %s " ,
2007-01-12 00:54:53 +03:00
fifos - > client_path ) ;
2007-01-08 18:18:52 +03:00
return - errno ;
}
if ( chmod ( fifos - > server_path , 0600 ) ) {
syslog ( LOG_ERR , " Unable to set correct file permissions on %s " ,
2007-01-12 00:54:53 +03:00
fifos - > server_path ) ;
2007-01-08 18:18:52 +03:00
return - errno ;
}
/* Need to open read+write or we will block or fail */
if ( ( fifos - > server = open ( fifos - > server_path , O_RDWR ) ) < 0 ) {
stack ;
return - errno ;
2005-04-28 02:32:00 +04:00
}
/* Need to open read+write for select() to work. */
2007-01-08 18:18:52 +03:00
if ( ( fifos - > client = open ( fifos - > client_path , O_RDWR ) ) < 0 ) {
2005-04-28 02:32:00 +04:00
stack ;
close ( fifos - > server ) ;
2007-01-08 18:18:52 +03:00
return - errno ;
2005-04-28 02:32:00 +04:00
}
2005-12-02 18:39:16 +03:00
return 0 ;
2005-04-28 02:32:00 +04:00
}
/*
* Read message from client making sure that data is available
2006-03-10 00:33:59 +03:00
* and a complete message is read . Must not block indefinitely .
2005-04-28 02:32:00 +04:00
*/
2007-01-15 21:58:40 +03:00
static int _client_read ( struct dm_event_fifos * fifos ,
2007-01-12 00:54:53 +03:00
struct dm_event_daemon_message * msg )
2005-04-28 02:32:00 +04:00
{
2006-03-10 00:33:59 +03:00
struct timeval t ;
2006-01-31 17:50:38 +03:00
unsigned bytes = 0 ;
int ret = 0 ;
2005-04-28 02:32:00 +04:00
fd_set fds ;
2007-01-08 18:18:52 +03:00
int header = 1 ;
size_t size = 2 * sizeof ( uint32_t ) ; /* status + size */
char * buf = alloca ( size ) ;
msg - > data = NULL ;
2005-04-28 02:32:00 +04:00
errno = 0 ;
2007-01-08 18:18:52 +03:00
while ( bytes < size & & errno ! = EOF ) {
2006-03-10 00:33:59 +03:00
/* Watch client read FIFO for input. */
FD_ZERO ( & fds ) ;
FD_SET ( fifos - > client , & fds ) ;
t . tv_sec = 1 ;
t . tv_usec = 0 ;
2007-01-12 00:54:53 +03:00
ret = select ( fifos - > client + 1 , & fds , NULL , NULL , & t ) ;
2006-03-10 00:33:59 +03:00
2007-01-12 00:54:53 +03:00
if ( ! ret & & ! bytes ) /* nothing to read */
2006-03-10 00:33:59 +03:00
return 0 ;
2007-01-12 00:54:53 +03:00
if ( ! ret ) /* trying to finish read */
2006-03-10 00:33:59 +03:00
continue ;
2007-01-12 00:54:53 +03:00
if ( ret < 0 ) /* error */
2006-03-10 00:33:59 +03:00
return 0 ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
ret = read ( fifos - > client , buf + bytes , size - bytes ) ;
2005-04-28 02:32:00 +04:00
bytes + = ret > 0 ? ret : 0 ;
2007-01-12 00:54:53 +03:00
if ( bytes = = 2 * sizeof ( uint32_t ) & & header ) {
msg - > cmd = ntohl ( * ( ( uint32_t * ) buf ) ) ;
msg - > size = ntohl ( * ( ( uint32_t * ) buf + 1 ) ) ;
2007-01-08 18:18:52 +03:00
buf = msg - > data = dm_malloc ( msg - > size ) ;
size = msg - > size ;
bytes = 0 ;
header = 0 ;
}
2005-04-28 02:32:00 +04:00
}
2007-01-08 18:18:52 +03:00
if ( bytes ! = size ) {
if ( msg - > data )
dm_free ( msg - > data ) ;
msg - > data = NULL ;
2007-01-12 00:54:53 +03:00
msg - > size = 0 ;
2007-01-08 18:18:52 +03:00
}
return bytes = = size ;
2005-04-28 02:32:00 +04:00
}
/*
* Write a message to the client making sure that it is ready to write .
*/
2007-01-15 21:58:40 +03:00
static int _client_write ( struct dm_event_fifos * fifos ,
2007-01-12 00:54:53 +03:00
struct dm_event_daemon_message * msg )
2005-04-28 02:32:00 +04:00
{
2006-01-31 17:50:38 +03:00
unsigned bytes = 0 ;
int ret = 0 ;
2005-04-28 02:32:00 +04:00
fd_set fds ;
2007-01-12 00:54:53 +03:00
size_t size = 2 * sizeof ( uint32_t ) + msg - > size ;
2007-01-08 18:18:52 +03:00
char * buf = alloca ( size ) ;
* ( ( uint32_t * ) buf ) = htonl ( msg - > cmd ) ;
* ( ( uint32_t * ) buf + 1 ) = htonl ( msg - > size ) ;
2007-01-12 00:54:53 +03:00
if ( msg - > data )
memcpy ( buf + 2 * sizeof ( uint32_t ) , msg - > data , msg - > size ) ;
2007-01-08 18:18:52 +03:00
2005-04-29 17:41:25 +04:00
errno = 0 ;
2007-01-08 18:18:52 +03:00
while ( bytes < size & & errno ! = EIO ) {
2005-04-29 17:41:25 +04:00
do {
/* Watch client write FIFO to be ready for output. */
FD_ZERO ( & fds ) ;
FD_SET ( fifos - > server , & fds ) ;
2007-01-12 00:54:53 +03:00
} while ( select ( fifos - > server + 1 , NULL , & fds , NULL , NULL ) ! =
1 ) ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
ret = write ( fifos - > server , buf + bytes , size - bytes ) ;
2005-04-29 17:41:25 +04:00
bytes + = ret > 0 ? ret : 0 ;
}
2007-01-08 18:18:52 +03:00
return bytes = = size ;
2005-04-28 02:32:00 +04:00
}
2005-12-02 18:39:16 +03:00
/*
* Handle a client request .
*
* We put the request handling functions into
* a list because of the growing number .
*/
2007-01-15 21:58:40 +03:00
static int _handle_request ( struct dm_event_daemon_message * msg ,
2005-12-02 18:39:16 +03:00
struct message_data * message_data )
{
static struct {
unsigned int cmd ;
2007-01-12 00:54:53 +03:00
int ( * f ) ( struct message_data * ) ;
2005-12-02 18:39:16 +03:00
} requests [ ] = {
2007-01-15 21:58:40 +03:00
{ DM_EVENT_CMD_REGISTER_FOR_EVENT , _register_for_event } ,
{ DM_EVENT_CMD_UNREGISTER_FOR_EVENT , _unregister_for_event } ,
{ DM_EVENT_CMD_GET_REGISTERED_DEVICE , _get_registered_device } ,
2007-01-12 00:54:53 +03:00
{ DM_EVENT_CMD_GET_NEXT_REGISTERED_DEVICE ,
2007-01-15 21:58:40 +03:00
_get_next_registered_device } ,
{ DM_EVENT_CMD_SET_TIMEOUT , _set_timeout } ,
{ DM_EVENT_CMD_GET_TIMEOUT , _get_timeout } ,
{ DM_EVENT_CMD_ACTIVE , _active } ,
2005-12-02 18:39:16 +03:00
} , * req ;
2006-12-20 17:35:02 +03:00
for ( req = requests ; req < requests + sizeof ( requests ) ; req + + )
2007-01-08 18:18:52 +03:00
if ( req - > cmd = = msg - > cmd )
2005-12-02 18:39:16 +03:00
return req - > f ( message_data ) ;
return - EINVAL ;
}
2005-04-28 02:32:00 +04:00
/* Process a request passed from the communication thread. */
2007-01-15 21:58:40 +03:00
static int _do_process_request ( struct dm_event_daemon_message * msg )
2005-04-28 02:32:00 +04:00
{
int ret ;
2007-02-02 20:08:51 +03:00
char * answer ;
2005-04-28 18:02:30 +04:00
static struct message_data message_data ;
/* Parse the message. */
2005-12-02 18:39:16 +03:00
memset ( & message_data , 0 , sizeof ( message_data ) ) ;
2005-04-28 18:02:30 +04:00
message_data . msg = msg ;
2007-02-02 20:08:51 +03:00
if ( msg - > cmd = = DM_EVENT_CMD_HELLO ) {
ret = 0 ;
2007-04-23 19:06:03 +04:00
answer = msg - > data ;
2007-02-02 20:08:51 +03:00
if ( answer ) {
msg - > size = dm_asprintf ( & ( msg - > data ) , " %s HELLO " , answer ) ;
dm_free ( answer ) ;
} else {
msg - > size = 0 ;
msg - > data = NULL ;
}
} else if ( msg - > cmd ! = DM_EVENT_CMD_ACTIVE & & ! _parse_message ( & message_data ) ) {
2005-04-28 18:02:30 +04:00
stack ;
2005-04-30 02:12:09 +04:00
ret = - EINVAL ;
2006-12-20 17:35:02 +03:00
} else
2007-01-15 21:58:40 +03:00
ret = _handle_request ( msg , & message_data ) ;
2005-04-28 02:32:00 +04:00
2007-02-02 20:08:51 +03:00
msg - > cmd = ret ;
if ( ! msg - > data )
msg - > size = dm_asprintf ( & ( msg - > data ) , " %s %s " , message_data . id , strerror ( - ret ) ) ;
2007-01-15 21:58:40 +03:00
_free_message ( & message_data ) ;
2005-12-02 18:39:16 +03:00
2005-04-28 02:32:00 +04:00
return ret ;
}
2005-04-28 18:02:30 +04:00
/* Only one caller at a time. */
2007-01-15 21:58:40 +03:00
static void _process_request ( struct dm_event_fifos * fifos )
2005-04-28 02:32:00 +04:00
{
2005-12-02 18:39:16 +03:00
struct dm_event_daemon_message msg ;
2006-03-10 00:33:59 +03:00
memset ( & msg , 0 , sizeof ( msg ) ) ;
/*
2007-01-12 00:54:53 +03:00
* Read the request from the client ( client_read , client_write
* give true on success and false on failure ) .
2006-03-10 00:33:59 +03:00
*/
2007-01-15 21:58:40 +03:00
if ( ! _client_read ( fifos , & msg ) )
2005-04-28 02:32:00 +04:00
return ;
2007-02-02 20:08:51 +03:00
/* _do_process_request fills in msg (if memory allows for
data , otherwise just cmd and size = 0 ) */
_do_process_request ( & msg ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! _client_write ( fifos , & msg ) )
2005-04-28 02:32:00 +04:00
stack ;
2007-01-08 18:18:52 +03:00
if ( msg . data )
dm_free ( msg . data ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-15 21:58:40 +03:00
static void _cleanup_unused_threads ( void )
2006-01-27 23:50:01 +03:00
{
int ret ;
2008-11-04 01:14:30 +03:00
struct dm_list * l ;
2006-01-27 23:50:01 +03:00
struct thread_status * thread ;
2007-01-15 21:58:40 +03:00
_lock_mutex ( ) ;
2008-11-04 01:14:30 +03:00
while ( ( l = dm_list_first ( & _thread_registry_unused ) ) ) {
thread = dm_list_item ( l , struct thread_status ) ;
2007-01-16 01:37:40 +03:00
if ( thread - > processing )
break ; /* cleanup on the next round */
2007-01-08 18:18:52 +03:00
if ( thread - > status = = DM_THREAD_RUNNING ) {
thread - > status = DM_THREAD_SHUTDOWN ;
2007-01-16 01:37:40 +03:00
break ;
2007-01-19 20:22:17 +03:00
}
2007-01-16 01:37:40 +03:00
if ( thread - > status = = DM_THREAD_SHUTDOWN ) {
2007-01-08 18:18:52 +03:00
if ( ! thread - > events ) {
/* turn codes negative -- should we be returning this? */
2007-01-15 21:58:40 +03:00
ret = _terminate_thread ( thread ) ;
2007-01-08 18:18:52 +03:00
if ( ret = = ESRCH ) {
thread - > status = DM_THREAD_DONE ;
} else if ( ret ) {
2007-01-12 00:54:53 +03:00
syslog ( LOG_ERR ,
" Unable to terminate thread: %s \n " ,
2007-01-08 18:18:52 +03:00
strerror ( - ret ) ) ;
stack ;
}
2007-01-16 01:37:40 +03:00
break ;
2007-01-19 20:22:17 +03:00
}
2007-01-16 01:37:40 +03:00
2008-11-04 01:14:30 +03:00
dm_list_del ( l ) ;
2007-01-16 01:37:40 +03:00
syslog ( LOG_ERR ,
" thread can't be on unused list unless !thread->events " ) ;
thread - > status = DM_THREAD_RUNNING ;
LINK_THREAD ( thread ) ;
continue ;
}
if ( thread - > status = = DM_THREAD_DONE ) {
2008-11-04 01:14:30 +03:00
dm_list_del ( l ) ;
2007-01-08 18:18:52 +03:00
pthread_join ( thread - > thread , NULL ) ;
2007-01-15 21:58:40 +03:00
_lib_put ( thread - > dso_data ) ;
_free_thread_status ( thread ) ;
2006-01-27 23:50:01 +03:00
}
}
2007-01-16 01:37:40 +03:00
2007-01-15 21:58:40 +03:00
_unlock_mutex ( ) ;
2006-01-27 23:50:01 +03:00
}
2007-01-22 18:03:57 +03:00
static void _sig_alarm ( int signum __attribute ( ( unused ) ) )
2005-04-28 02:32:00 +04:00
{
2005-12-02 18:39:16 +03:00
pthread_testcancel ( ) ;
}
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
/* Init thread signal handling. */
2007-01-15 21:58:40 +03:00
static void _init_thread_signals ( void )
2005-12-02 18:39:16 +03:00
{
2006-05-11 23:08:02 +04:00
sigset_t my_sigset ;
2005-12-02 18:39:16 +03:00
struct sigaction act ;
2007-01-12 00:54:53 +03:00
2005-12-02 18:39:16 +03:00
memset ( & act , 0 , sizeof ( act ) ) ;
2007-01-15 21:58:40 +03:00
act . sa_handler = _sig_alarm ;
2005-12-02 18:39:16 +03:00
sigaction ( SIGALRM , & act , NULL ) ;
2006-05-11 23:08:02 +04:00
sigfillset ( & my_sigset ) ;
2007-01-08 18:18:52 +03:00
/* These are used for exiting */
sigdelset ( & my_sigset , SIGTERM ) ;
sigdelset ( & my_sigset , SIGINT ) ;
sigdelset ( & my_sigset , SIGHUP ) ;
sigdelset ( & my_sigset , SIGQUIT ) ;
2006-05-11 23:08:02 +04:00
pthread_sigmask ( SIG_BLOCK , & my_sigset , NULL ) ;
2005-04-28 02:32:00 +04:00
}
2007-01-08 18:18:52 +03:00
/*
* exit_handler
* @ sig
*
* Set the global variable which the process should
* be watching to determine when to exit .
*/
2007-01-22 18:03:57 +03:00
static void _exit_handler ( int sig __attribute ( ( unused ) ) )
2005-04-28 02:32:00 +04:00
{
2007-01-08 18:18:52 +03:00
/*
* We exit when ' _exit_now ' is set .
* That is , when a signal has been received .
*
* We can not simply set ' _exit_now ' unless all
* threads are done processing .
*/
if ( ! _thread_registries_empty ) {
syslog ( LOG_ERR , " There are still devices being monitored. " ) ;
syslog ( LOG_ERR , " Refusing to exit. " ) ;
} else
_exit_now = 1 ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
}
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
static int _lock_pidfile ( void )
2005-12-02 18:39:16 +03:00
{
int lf ;
2007-01-12 00:54:53 +03:00
char pidfile [ ] = DMEVENTD_PIDFILE ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
if ( ( lf = open ( pidfile , O_CREAT | O_RDWR , 0644 ) ) < 0 )
2007-01-08 18:18:52 +03:00
exit ( EXIT_OPEN_PID_FAILURE ) ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
if ( flock ( lf , LOCK_EX | LOCK_NB ) < 0 )
2007-01-08 18:18:52 +03:00
exit ( EXIT_LOCKFILE_INUSE ) ;
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ! _storepid ( lf ) )
2007-01-08 18:18:52 +03:00
exit ( EXIT_FAILURE ) ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
return 0 ;
2005-04-28 02:32:00 +04:00
}
2007-01-22 18:03:57 +03:00
# ifdef linux
2007-01-19 20:22:17 +03:00
/*
* Protection against OOM killer if kernel supports it
*/
2007-01-16 23:13:04 +03:00
static int _set_oom_adj ( int val )
{
FILE * fp ;
2007-01-17 17:45:10 +03:00
struct stat st ;
2007-01-19 20:22:17 +03:00
if ( stat ( OOM_ADJ_FILE , & st ) = = - 1 ) {
if ( errno = = ENOENT )
DEBUGLOG ( OOM_ADJ_FILE " not found " ) ;
else
perror ( OOM_ADJ_FILE " : stat failed " ) ;
return 1 ;
}
2007-01-16 23:13:04 +03:00
2007-01-19 20:22:17 +03:00
if ( ! ( fp = fopen ( OOM_ADJ_FILE , " w " ) ) ) {
perror ( OOM_ADJ_FILE " : fopen failed " ) ;
return 0 ;
}
2007-01-16 23:13:04 +03:00
fprintf ( fp , " %i " , val ) ;
2007-07-24 18:16:48 +04:00
if ( dm_fclose ( fp ) )
2007-01-25 17:16:20 +03:00
perror ( OOM_ADJ_FILE " : fclose failed " ) ;
2007-01-16 23:13:04 +03:00
2007-01-19 20:22:17 +03:00
return 1 ;
2007-01-16 23:13:04 +03:00
}
2007-01-22 18:03:57 +03:00
# endif
2007-01-16 23:13:04 +03:00
2007-01-15 21:58:40 +03:00
static void _daemonize ( void )
2005-04-28 02:32:00 +04:00
{
2007-01-19 20:22:17 +03:00
int child_status ;
2007-01-08 18:18:52 +03:00
int fd ;
2007-01-19 20:22:17 +03:00
pid_t pid ;
2007-01-08 18:18:52 +03:00
struct rlimit rlim ;
struct timeval tval ;
sigset_t my_sigset ;
sigemptyset ( & my_sigset ) ;
if ( sigprocmask ( SIG_SETMASK , & my_sigset , NULL ) < 0 ) {
2007-01-25 17:16:20 +03:00
fprintf ( stderr , " Unable to restore signals. \n " ) ;
2007-01-08 18:18:52 +03:00
exit ( EXIT_FAILURE ) ;
}
2007-01-15 21:58:40 +03:00
signal ( SIGTERM , & _exit_handler ) ;
2005-05-02 15:02:19 +04:00
2007-01-19 20:22:17 +03:00
switch ( pid = fork ( ) ) {
case - 1 :
perror ( " fork failed: " ) ;
2007-01-08 18:18:52 +03:00
exit ( EXIT_FAILURE ) ;
2005-04-28 02:32:00 +04:00
2007-01-19 20:22:17 +03:00
case 0 : /* Child */
break ;
default :
2007-01-08 18:18:52 +03:00
/* Wait for response from child */
2007-01-19 20:22:17 +03:00
while ( ! waitpid ( pid , & child_status , WNOHANG ) & & ! _exit_now ) {
2007-01-08 18:18:52 +03:00
tval . tv_sec = 0 ;
2007-01-12 00:54:53 +03:00
tval . tv_usec = 250000 ; /* .25 sec */
2007-01-08 18:18:52 +03:00
select ( 0 , NULL , NULL , NULL , & tval ) ;
}
2007-01-12 00:54:53 +03:00
if ( _exit_now ) /* Child has signaled it is ok - we can exit now */
2007-01-08 18:18:52 +03:00
exit ( EXIT_SUCCESS ) ;
/* Problem with child. Determine what it is by exit code */
2007-01-19 20:22:17 +03:00
switch ( WEXITSTATUS ( child_status ) ) {
2007-01-08 18:18:52 +03:00
case EXIT_LOCKFILE_INUSE :
2007-01-19 21:08:36 +03:00
fprintf ( stderr , " Another dmeventd daemon is already running \n " ) ;
2007-01-08 18:18:52 +03:00
break ;
case EXIT_DESC_CLOSE_FAILURE :
case EXIT_DESC_OPEN_FAILURE :
case EXIT_OPEN_PID_FAILURE :
case EXIT_FIFO_FAILURE :
case EXIT_CHDIR_FAILURE :
default :
2007-01-19 21:08:36 +03:00
fprintf ( stderr , " Child exited with code %d \n " , WEXITSTATUS ( child_status ) ) ;
2007-01-08 18:18:52 +03:00
break ;
}
2007-01-19 21:08:36 +03:00
exit ( WEXITSTATUS ( child_status ) ) ;
2007-01-08 18:18:52 +03:00
}
if ( chdir ( " / " ) )
exit ( EXIT_CHDIR_FAILURE ) ;
if ( getrlimit ( RLIMIT_NOFILE , & rlim ) < 0 )
2007-01-12 00:54:53 +03:00
fd = 256 ; /* just have to guess */
2007-01-08 18:18:52 +03:00
else
fd = rlim . rlim_cur ;
for ( - - fd ; fd > = 0 ; fd - - )
close ( fd ) ;
if ( ( open ( " /dev/null " , O_RDONLY ) < 0 ) | |
( open ( " /dev/null " , O_WRONLY ) < 0 ) | |
( open ( " /dev/null " , O_WRONLY ) < 0 ) )
exit ( EXIT_DESC_OPEN_FAILURE ) ;
2007-01-19 20:22:17 +03:00
setsid ( ) ;
}
2007-01-08 18:18:52 +03:00
2007-01-19 20:22:17 +03:00
static void usage ( char * prog , FILE * file )
{
fprintf ( file , " Usage: \n " ) ;
fprintf ( file , " %s [Vhd] \n " , prog ) ;
fprintf ( file , " \n " ) ;
fprintf ( file , " -V Show version of dmeventd \n " ) ;
fprintf ( file , " -h Show this help information \n " ) ;
fprintf ( file , " -d Don't fork, run in the foreground \n " ) ;
fprintf ( file , " \n " ) ;
2007-01-08 18:18:52 +03:00
}
int main ( int argc , char * argv [ ] )
{
int ret ;
2007-01-19 20:22:17 +03:00
signed char opt ;
2007-01-08 18:18:52 +03:00
struct dm_event_fifos fifos ;
//struct sys_log logdata = {DAEMON_NAME, LOG_DAEMON};
2007-01-19 20:22:17 +03:00
opterr = 0 ;
optind = 0 ;
while ( ( opt = getopt ( argc , argv , " ?hVd " ) ) ! = EOF ) {
switch ( opt ) {
case ' h ' :
usage ( argv [ 0 ] , stdout ) ;
exit ( 0 ) ;
case ' ? ' :
usage ( argv [ 0 ] , stderr ) ;
exit ( 0 ) ;
case ' d ' :
_debug + + ;
break ;
case ' V ' :
printf ( " dmeventd version: %s \n " , DM_LIB_VERSION ) ;
exit ( 1 ) ;
break ;
}
}
if ( ! _debug )
_daemonize ( ) ;
2005-04-28 02:32:00 +04:00
2007-01-19 20:22:17 +03:00
openlog ( " dmeventd " , LOG_PID , LOG_DAEMON ) ;
_lock_pidfile ( ) ; /* exits if failure */
/* Set the rest of the signals to cause '_exit_now' to be set */
signal ( SIGINT , & _exit_handler ) ;
signal ( SIGHUP , & _exit_handler ) ;
signal ( SIGQUIT , & _exit_handler ) ;
2007-01-22 18:03:57 +03:00
# ifdef linux
2007-01-19 20:22:17 +03:00
if ( ! _set_oom_adj ( OOM_DISABLE ) & & ! _set_oom_adj ( OOM_ADJUST_MIN ) )
2007-01-16 23:13:04 +03:00
syslog ( LOG_ERR , " Failed to set oom_adj to protect against OOM killer " ) ;
2007-01-22 18:03:57 +03:00
# endif
2007-01-16 23:13:04 +03:00
2007-01-15 21:58:40 +03:00
_init_thread_signals ( ) ;
2005-04-29 02:47:52 +04:00
2005-12-02 18:39:16 +03:00
//multilog_clear_logging();
//multilog_add_type(std_syslog, &logdata);
//multilog_init_verbose(std_syslog, _LOG_DEBUG);
//multilog_async(1);
2005-04-29 02:47:52 +04:00
2007-01-15 21:58:40 +03:00
_init_fifos ( & fifos ) ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:35:08 +03:00
pthread_mutex_init ( & _global_mutex , NULL ) ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
# ifdef MCL_CURRENT
if ( mlockall ( MCL_CURRENT | MCL_FUTURE ) = = - 1 )
exit ( EXIT_FAILURE ) ;
# endif
2005-04-28 02:32:00 +04:00
2007-01-15 21:58:40 +03:00
if ( ( ret = _open_fifos ( & fifos ) ) )
2007-01-08 18:18:52 +03:00
exit ( EXIT_FIFO_FAILURE ) ;
2005-04-28 02:32:00 +04:00
2005-12-02 18:39:16 +03:00
/* Signal parent, letting them know we are ready to go. */
2007-01-08 18:18:52 +03:00
kill ( getppid ( ) , SIGTERM ) ;
2007-01-19 20:22:17 +03:00
syslog ( LOG_NOTICE , " dmeventd ready for processing. " ) ;
2005-04-28 02:32:00 +04:00
2007-01-08 18:18:52 +03:00
while ( ! _exit_now ) {
2007-01-15 21:58:40 +03:00
_process_request ( & fifos ) ;
_cleanup_unused_threads ( ) ;
2008-11-04 01:14:30 +03:00
if ( ! dm_list_empty ( & _thread_registry )
| | ! dm_list_empty ( & _thread_registry_unused ) )
2007-01-08 18:18:52 +03:00
_thread_registries_empty = 0 ;
else
_thread_registries_empty = 1 ;
}
2006-01-27 23:50:01 +03:00
2007-01-15 21:58:40 +03:00
_exit_dm_lib ( ) ;
2006-01-27 23:50:01 +03:00
2005-12-02 18:39:16 +03:00
# ifdef MCL_CURRENT
munlockall ( ) ;
# endif
2007-01-08 18:35:08 +03:00
pthread_mutex_destroy ( & _global_mutex ) ;
2005-04-28 02:32:00 +04:00
2007-01-19 20:22:17 +03:00
syslog ( LOG_NOTICE , " dmeventd shutting down. " ) ;
2007-01-08 18:18:52 +03:00
closelog ( ) ;
2007-01-19 20:22:17 +03:00
2005-05-02 15:02:19 +04:00
exit ( EXIT_SUCCESS ) ;
2005-04-28 02:32:00 +04:00
}