2005-12-02 22:52:06 +03:00
/*
* Copyright ( C ) 2005 Red Hat , Inc . All rights reserved .
*
* This file is part of LVM2 .
*
* 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 "libdevmapper-event.h"
# include "lvm2cmd.h"
# include "lvm-string.h"
# include <errno.h>
# include <signal.h>
# include <string.h>
# include <stdio.h>
# include <stdlib.h>
# include <pthread.h>
# include <unistd.h>
2006-01-27 23:43:52 +03:00
# include <syslog.h> /* FIXME Replace syslog with multilog */
2005-12-02 22:52:06 +03:00
# define ME_IGNORE 0
# define ME_INSYNC 1
# define ME_FAILURE 2
2006-01-27 23:43:52 +03:00
/* FIXME: We may need to lock around operations to these */
static int register_count = 0 ;
static struct dm_pool * mem_pool = NULL ;
2005-12-02 22:52:06 +03:00
static int _get_mirror_event ( char * params )
{
int i , rtn = ME_INSYNC ;
int max_args = 30 ; /* should support at least 8-way mirrors */
char * args [ max_args ] ;
char * dev_status_str ;
char * log_status_str ;
char * sync_str ;
char * p ;
int log_argc , num_devs , num_failures = 0 ;
if ( max_args < = split_words ( params , max_args , args ) ) {
syslog ( LOG_ERR , " Unable to split mirror parameters: Arg list too long " ) ;
return - E2BIG ;
}
/*
* Unused : 0 409600 mirror
* Used : 2 253 : 4 253 : 5 400 / 400 1 AA 3 cluster 253 : 3 A
*/
num_devs = atoi ( args [ 0 ] ) ;
dev_status_str = args [ 3 + num_devs ] ;
log_argc = atoi ( args [ 4 + num_devs ] ) ;
log_status_str = args [ 4 + num_devs + log_argc ] ;
sync_str = args [ 1 + num_devs ] ;
/* Check for bad mirror devices */
for ( i = 0 ; i < num_devs ; i + + ) {
if ( dev_status_str [ i ] = = ' D ' ) {
syslog ( LOG_ERR , " Mirror device, %s, has failed. \n " , args [ i + 1 ] ) ;
num_failures + + ;
}
}
/* Check for bad log device */
if ( log_status_str [ 0 ] = = ' D ' ) {
syslog ( LOG_ERR , " Log device, %s, has failed. \n " ,
args [ 3 + num_devs + log_argc ] ) ;
num_failures + + ;
}
if ( num_failures ) {
rtn = ME_FAILURE ;
goto out ;
}
2006-01-27 23:48:19 +03:00
p = strstr ( sync_str , " / " ) ;
if ( p ) {
p [ 0 ] = ' \0 ' ;
if ( strcmp ( sync_str , p + 1 ) )
2005-12-02 22:52:06 +03:00
rtn = ME_IGNORE ;
2006-01-27 23:48:19 +03:00
p [ 0 ] = ' / ' ;
} else {
/*
* How the hell did we get this ?
* Might mean all our parameters are screwed .
*/
syslog ( LOG_ERR , " Unable to parse sync string. " ) ;
rtn = ME_IGNORE ;
2005-12-02 22:52:06 +03:00
}
out :
return rtn ;
}
2006-05-11 23:45:53 +04:00
static void _temporary_log_fn ( int level , const char * file ,
int line , const char * format )
2005-12-02 22:52:06 +03:00
{
2006-05-11 23:45:53 +04:00
if ( ! strncmp ( format , " WARNING: " , 9 ) & & ( level < 5 ) )
syslog ( LOG_CRIT , " %s " , format ) ;
else
syslog ( LOG_DEBUG , " %s " , format ) ;
2005-12-02 22:52:06 +03:00
}
static int _remove_failed_devices ( const char * device )
{
int r ;
void * handle ;
int cmd_size = 256 ; /* FIXME Use system restriction */
char cmd_str [ cmd_size ] ;
2006-01-27 23:43:52 +03:00
char * vg = NULL , * lv = NULL , * layer = NULL ;
2005-12-02 22:52:06 +03:00
if ( strlen ( device ) > 200 )
return - ENAMETOOLONG ;
2006-01-27 23:43:52 +03:00
if ( ! split_dm_name ( mem_pool , device , & vg , & lv , & layer ) ) {
syslog ( LOG_ERR , " Unable to determine VG name from %s " ,
2005-12-02 22:52:06 +03:00
device ) ;
2006-01-27 23:43:52 +03:00
return - ENOMEM ;
2005-12-02 22:52:06 +03:00
}
/* FIXME Is any sanity-checking required on %s? */
if ( cmd_size < = snprintf ( cmd_str , cmd_size , " vgreduce --removemissing %s " , vg ) ) {
/* this error should be caught above, but doesn't hurt to check again */
syslog ( LOG_ERR , " Unable to form LVM command: Device name too long " ) ;
2006-01-27 23:43:52 +03:00
dm_pool_empty ( mem_pool ) ; /* FIXME: not safe with multiple threads */
2005-12-02 22:52:06 +03:00
return - ENAMETOOLONG ;
}
lvm2_log_fn ( _temporary_log_fn ) ;
handle = lvm2_init ( ) ;
lvm2_log_level ( handle , 1 ) ;
r = lvm2_run ( handle , cmd_str ) ;
2006-01-27 23:43:52 +03:00
dm_pool_empty ( mem_pool ) ; /* FIXME: not safe with multiple threads */
2005-12-02 22:52:06 +03:00
return ( r = = 1 ) ? 0 : - 1 ;
}
void process_event ( const char * device , enum dm_event_type event )
{
struct dm_task * dmt ;
void * next = NULL ;
uint64_t start , length ;
char * target_type = NULL ;
char * params ;
/* FIXME Move inside libdevmapper */
if ( ! ( dmt = dm_task_create ( DM_DEVICE_STATUS ) ) ) {
syslog ( LOG_ERR , " Unable to create dm_task. \n " ) ;
goto fail ;
}
if ( ! dm_task_set_name ( dmt , device ) ) {
syslog ( LOG_ERR , " Unable to set device name. \n " ) ;
goto fail ;
}
if ( ! dm_task_run ( dmt ) ) {
syslog ( LOG_ERR , " Unable to run task. \n " ) ;
goto fail ;
}
do {
next = dm_get_next_target ( dmt , next , & start , & length ,
& target_type , & params ) ;
if ( strcmp ( target_type , " mirror " ) ) {
syslog ( LOG_INFO , " %s has unmirrored portion. \n " , device ) ;
continue ;
}
switch ( _get_mirror_event ( params ) ) {
case ME_INSYNC :
/* FIXME: all we really know is that this
_part_ of the device is in sync
Also , this is not an error
*/
syslog ( LOG_NOTICE , " %s is now in-sync \n " , device ) ;
break ;
case ME_FAILURE :
syslog ( LOG_ERR , " Device failure in %s \n " , device ) ;
if ( _remove_failed_devices ( device ) )
syslog ( LOG_ERR , " Failed to remove faulty devices in %s \n " ,
device ) ;
2006-01-27 23:48:19 +03:00
/* Should check before warning user that device is now linear
2005-12-02 22:52:06 +03:00
else
syslog ( LOG_NOTICE , " %s is now a linear device. \n " ,
device ) ;
2006-01-27 23:48:19 +03:00
*/
2005-12-02 22:52:06 +03:00
break ;
case ME_IGNORE :
break ;
default :
syslog ( LOG_INFO , " Unknown event received. \n " ) ;
}
} while ( next ) ;
fail :
if ( dmt )
dm_task_destroy ( dmt ) ;
}
int register_device ( const char * device )
{
2006-05-11 23:45:53 +04:00
syslog ( LOG_INFO , " Monitoring mirror device, %s for events \n " , device ) ;
2006-01-27 23:43:52 +03:00
/*
* Need some space for allocations . 1024 should be more
* than enough for what we need ( device mapper name splitting )
*/
if ( ! mem_pool )
mem_pool = dm_pool_create ( " mirror_dso " , 1024 ) ;
if ( ! mem_pool )
return 0 ;
register_count + + ;
2005-12-02 22:52:06 +03:00
return 1 ;
}
int unregister_device ( const char * device )
{
2006-01-27 23:43:52 +03:00
if ( ! ( - - register_count ) ) {
dm_pool_destroy ( mem_pool ) ;
mem_pool = NULL ;
}
2005-12-02 22:52:06 +03:00
return 1 ;
}
/*
* 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 :
*/