2001-10-22 14:14:00 +00:00
/*
2004-03-30 19:35:44 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2006-11-04 03:34:10 +00:00
* Copyright ( C ) 2004 - 2006 Red Hat , Inc . All rights reserved .
2001-10-22 14:14:00 +00:00
*
2004-03-30 19:35:44 +00:00
* 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 General Public License v .2 .
*
* You should have received a copy of the GNU 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
2001-10-22 14:14:00 +00:00
*/
2002-11-18 14:01:16 +00:00
# include "lib.h"
2001-10-22 14:14:00 +00:00
# include "config.h"
# include "dev-cache.h"
# include "filter-persistent.h"
2006-11-04 03:34:10 +00:00
# include "lvm-file.h"
2001-10-22 14:14:00 +00:00
# include <sys/stat.h>
# include <fcntl.h>
# include <unistd.h>
struct pfilter {
char * file ;
2005-10-16 23:03:59 +00:00
struct dm_hash_table * devices ;
2001-10-23 12:24:55 +00:00
struct dev_filter * real ;
2006-11-04 03:34:10 +00:00
time_t ctime ;
2001-10-22 14:14:00 +00:00
} ;
/*
2006-11-04 03:34:10 +00:00
* The hash table holds one of these two states
* against each entry .
2001-10-22 14:14:00 +00:00
*/
2001-10-23 12:24:55 +00:00
# define PF_BAD_DEVICE ((void *) 1)
# define PF_GOOD_DEVICE ((void *) 2)
2001-10-22 14:14:00 +00:00
2001-10-22 14:39:12 +00:00
static int _init_hash ( struct pfilter * pf )
2001-10-22 14:14:00 +00:00
{
if ( pf - > devices )
2005-10-16 23:03:59 +00:00
dm_hash_destroy ( pf - > devices ) ;
2001-10-22 14:14:00 +00:00
2005-10-16 23:03:59 +00:00
if ( ! ( pf - > devices = dm_hash_create ( 128 ) ) ) {
2003-01-08 16:41:22 +00:00
stack ;
return 0 ;
}
return 1 ;
2001-10-22 14:14:00 +00:00
}
2001-10-23 12:24:55 +00:00
int persistent_filter_wipe ( struct dev_filter * f )
2001-10-22 14:14:00 +00:00
{
2001-10-23 12:24:55 +00:00
struct pfilter * pf = ( struct pfilter * ) f - > private ;
2001-10-22 14:14:00 +00:00
2005-03-21 22:40:35 +00:00
log_verbose ( " Wiping cache of LVM-capable devices " ) ;
2005-10-16 23:03:59 +00:00
dm_hash_wipe ( pf - > devices ) ;
2005-03-21 22:40:35 +00:00
2002-11-18 14:01:16 +00:00
/* Trigger complete device scan */
dev_cache_scan ( 1 ) ;
2001-10-24 17:53:50 +00:00
return 1 ;
2001-10-23 12:24:55 +00:00
}
2001-10-22 14:14:00 +00:00
2004-03-08 18:28:45 +00:00
static int _read_array ( struct pfilter * pf , struct config_tree * cft ,
2001-10-23 12:24:55 +00:00
const char * path , void * data )
{
2004-05-04 18:28:15 +00:00
const struct config_node * cn ;
2001-10-23 12:24:55 +00:00
struct config_value * cv ;
2001-10-22 14:14:00 +00:00
2004-03-08 18:28:45 +00:00
if ( ! ( cn = find_config_node ( cft - > root , path ) ) ) {
2001-10-24 17:53:50 +00:00
log_very_verbose ( " Couldn't find %s array in '%s' " ,
2002-04-24 18:20:51 +00:00
path , pf - > file ) ;
2001-10-23 12:24:55 +00:00
return 0 ;
2001-10-22 14:14:00 +00:00
}
/*
* iterate through the array , adding
* devices as we go .
*/
for ( cv = cn - > v ; cv ; cv = cv - > next ) {
if ( cv - > type ! = CFG_STRING ) {
2001-10-24 17:53:50 +00:00
log_verbose ( " Devices array contains a value "
2002-04-24 18:20:51 +00:00
" which is not a string ... ignoring " ) ;
2001-10-22 14:14:00 +00:00
continue ;
}
2005-10-16 23:03:59 +00:00
if ( ! dm_hash_insert ( pf - > devices , cv - > v . str , data ) )
2001-10-23 18:20:27 +00:00
log_verbose ( " Couldn't add '%s' to filter ... ignoring " ,
2002-04-24 18:20:51 +00:00
cv - > v . str ) ;
2002-11-18 14:01:16 +00:00
/* Populate dev_cache ourselves */
dev_cache_get ( cv - > v . str , NULL ) ;
2001-10-22 14:14:00 +00:00
}
2001-10-23 12:24:55 +00:00
return 1 ;
}
2006-11-04 03:34:10 +00:00
int persistent_filter_load ( struct dev_filter * f , struct config_tree * * cft_out )
2001-10-23 12:24:55 +00:00
{
struct pfilter * pf = ( struct pfilter * ) f - > private ;
2004-03-08 18:28:45 +00:00
struct config_tree * cft ;
2006-11-04 03:34:10 +00:00
struct stat info ;
int r = 0 ;
2001-10-23 12:24:55 +00:00
2006-11-04 03:34:10 +00:00
if ( ! stat ( pf - > file , & info ) )
pf - > ctime = info . st_ctime ;
else {
log_very_verbose ( " %s: stat failed: %s " , pf - > file ,
strerror ( errno ) ) ;
return_0 ;
2001-10-23 12:24:55 +00:00
}
2006-11-04 03:34:10 +00:00
if ( ! ( cft = create_config_tree ( pf - > file , 1 ) ) )
return_0 ;
if ( ! read_config_file ( cft ) )
goto_out ;
2001-10-23 12:24:55 +00:00
2004-03-08 18:28:45 +00:00
_read_array ( pf , cft , " persistent_filter_cache/valid_devices " ,
2001-10-25 14:41:28 +00:00
PF_GOOD_DEVICE ) ;
2002-11-18 14:01:16 +00:00
/* We don't gain anything by holding invalid devices */
2004-03-08 18:28:45 +00:00
/* _read_array(pf, cft, "persistent_filter_cache/invalid_devices",
2002-11-18 14:01:16 +00:00
PF_BAD_DEVICE ) ; */
/* Did we find anything? */
2005-10-16 23:03:59 +00:00
if ( dm_hash_get_num_entries ( pf - > devices ) ) {
2002-11-18 14:01:16 +00:00
/* We populated dev_cache ourselves */
dev_cache_scan ( 0 ) ;
2001-10-23 12:24:55 +00:00
r = 1 ;
2002-11-18 14:01:16 +00:00
}
2001-10-22 14:14:00 +00:00
2003-01-06 21:09:04 +00:00
log_very_verbose ( " Loaded persistent filter cache from %s " , pf - > file ) ;
2002-04-24 18:20:51 +00:00
out :
2006-11-04 03:34:10 +00:00
if ( r & & cft_out )
* cft_out = cft ;
else
destroy_config_tree ( cft ) ;
2001-10-22 14:14:00 +00:00
return r ;
}
2002-11-18 14:01:16 +00:00
static void _write_array ( struct pfilter * pf , FILE * fp , const char * path ,
2001-10-23 12:24:55 +00:00
void * data )
2001-10-22 14:14:00 +00:00
{
2001-10-23 12:24:55 +00:00
void * d ;
2001-10-22 14:14:00 +00:00
int first = 1 ;
2005-10-16 23:03:59 +00:00
struct dm_hash_node * n ;
2001-10-22 14:14:00 +00:00
2005-10-16 23:03:59 +00:00
for ( n = dm_hash_get_first ( pf - > devices ) ; n ;
n = dm_hash_get_next ( pf - > devices , n ) ) {
d = dm_hash_get_data ( pf - > devices , n ) ;
2001-10-23 12:24:55 +00:00
if ( d ! = data )
continue ;
2001-10-22 14:14:00 +00:00
if ( ! first )
fprintf ( fp , " , \n " ) ;
2001-10-24 17:53:50 +00:00
else {
fprintf ( fp , " \t %s=[ \n " , path ) ;
2001-10-22 14:14:00 +00:00
first = 0 ;
2001-10-24 17:53:50 +00:00
}
2001-10-22 14:14:00 +00:00
2005-10-16 23:03:59 +00:00
fprintf ( fp , " \t \t \" %s \" " , dm_hash_get_key ( pf - > devices , n ) ) ;
2001-10-22 14:14:00 +00:00
}
2001-10-24 17:53:50 +00:00
if ( ! first )
fprintf ( fp , " \n \t ] \n " ) ;
return ;
2001-10-22 14:14:00 +00:00
}
2001-10-23 12:24:55 +00:00
int persistent_filter_dump ( struct dev_filter * f )
2001-10-22 14:14:00 +00:00
{
2001-10-23 12:24:55 +00:00
struct pfilter * pf = ( struct pfilter * ) f - > private ;
2006-11-04 03:34:10 +00:00
char * tmp_file ;
struct stat info , info2 ;
struct config_tree * cft = NULL ;
2001-10-25 15:07:26 +00:00
FILE * fp ;
2006-11-04 03:34:10 +00:00
int lockfd ;
int r = 0 ;
2002-01-24 23:17:16 +00:00
2005-10-16 23:03:59 +00:00
if ( ! dm_hash_get_num_entries ( pf - > devices ) ) {
2002-01-24 23:17:16 +00:00
log_very_verbose ( " Internal persistent device cache empty "
" - not writing to %s " , pf - > file ) ;
return 0 ;
}
2002-11-18 14:01:16 +00:00
if ( ! dev_cache_has_scanned ( ) ) {
log_very_verbose ( " Device cache incomplete - not writing "
" to %s " , pf - > file ) ;
return 0 ;
}
2001-10-23 12:24:55 +00:00
log_very_verbose ( " Dumping persistent device cache to %s " , pf - > file ) ;
2001-10-22 14:14:00 +00:00
2006-11-04 03:34:10 +00:00
while ( 1 ) {
if ( ( lockfd = fcntl_lock_file ( pf - > file , F_WRLCK , 0 ) ) < 0 )
return_0 ;
/*
* Ensure we locked the file we expected
*/
if ( fstat ( lockfd , & info ) ) {
log_sys_error ( " fstat " , pf - > file ) ;
goto out ;
}
if ( stat ( pf - > file , & info2 ) ) {
log_sys_error ( " stat " , pf - > file ) ;
goto out ;
}
if ( ! memcmp ( & info . st_ino , & info2 . st_ino , sizeof ( ino_t ) ) )
break ;
fcntl_unlock_file ( lockfd ) ;
}
/*
* If file contents changed since we loaded it , merge new contents
*/
if ( info . st_ctime ! = pf - > ctime )
/* Keep cft open to avoid losing lock */
persistent_filter_load ( f , & cft ) ;
tmp_file = alloca ( strlen ( pf - > file ) + 5 ) ;
sprintf ( tmp_file , " %s.tmp " , pf - > file ) ;
if ( ! ( fp = fopen ( tmp_file , " w " ) ) ) {
/* EACCES has been reported over NFS */
if ( errno ! = EROFS & & errno ! = EACCES )
log_sys_error ( " fopen " , tmp_file ) ;
goto out ;
2001-10-23 12:24:55 +00:00
}
2001-10-22 14:14:00 +00:00
2001-10-23 12:24:55 +00:00
fprintf ( fp , " # This file is automatically maintained by lvm. \n \n " ) ;
2001-10-24 17:53:50 +00:00
fprintf ( fp , " persistent_filter_cache { \n " ) ;
2001-10-22 14:14:00 +00:00
2001-10-23 12:24:55 +00:00
_write_array ( pf , fp , " valid_devices " , PF_GOOD_DEVICE ) ;
2002-11-18 14:01:16 +00:00
/* We don't gain anything by remembering invalid devices */
/* _write_array(pf, fp, "invalid_devices", PF_BAD_DEVICE); */
2001-10-22 14:14:00 +00:00
2001-10-24 17:53:50 +00:00
fprintf ( fp , " } \n " ) ;
2007-01-25 14:37:48 +00:00
if ( fclose ( fp ) ) {
log_sys_error ( " fclose " , tmp_file ) ;
goto out ;
}
2006-11-04 03:34:10 +00:00
if ( rename ( tmp_file , pf - > file ) )
log_error ( " %s: rename to %s failed: %s " , tmp_file , pf - > file ,
strerror ( errno ) ) ;
r = 1 ;
out :
fcntl_unlock_file ( lockfd ) ;
if ( cft )
destroy_config_tree ( cft ) ;
return r ;
2001-10-22 14:14:00 +00:00
}
2001-10-23 12:24:55 +00:00
static int _lookup_p ( struct dev_filter * f , struct device * dev )
2001-10-22 14:14:00 +00:00
{
struct pfilter * pf = ( struct pfilter * ) f - > private ;
2005-10-16 23:03:59 +00:00
void * l = dm_hash_lookup ( pf - > devices , dev_name ( dev ) ) ;
2001-10-25 14:51:51 +00:00
struct str_list * sl ;
2001-10-22 14:14:00 +00:00
2001-10-23 12:24:55 +00:00
if ( ! l ) {
l = pf - > real - > passes_filter ( pf - > real , dev ) ?
2002-04-24 18:20:51 +00:00
PF_GOOD_DEVICE : PF_BAD_DEVICE ;
2001-10-22 14:14:00 +00:00
2005-06-01 16:51:55 +00:00
list_iterate_items ( sl , & dev - > aliases )
2005-10-16 23:03:59 +00:00
dm_hash_insert ( pf - > devices , sl - > str , l ) ;
2005-06-01 16:51:55 +00:00
2004-11-24 20:36:52 +00:00
} else if ( l = = PF_BAD_DEVICE )
log_debug ( " %s: Skipping (cached) " , dev_name ( dev ) ) ;
2001-10-22 14:14:00 +00:00
2004-11-24 20:36:52 +00:00
return ( l = = PF_BAD_DEVICE ) ? 0 : 1 ;
2001-10-22 14:14:00 +00:00
}
2006-04-19 15:33:07 +00:00
static void _persistent_destroy ( struct dev_filter * f )
2001-10-22 14:14:00 +00:00
{
struct pfilter * pf = ( struct pfilter * ) f - > private ;
2005-10-16 23:03:59 +00:00
dm_hash_destroy ( pf - > devices ) ;
dm_free ( pf - > file ) ;
2001-10-23 12:24:55 +00:00
pf - > real - > destroy ( pf - > real ) ;
2005-10-16 23:03:59 +00:00
dm_free ( pf ) ;
dm_free ( f ) ;
2001-10-22 14:14:00 +00:00
}
2001-10-23 12:24:55 +00:00
struct dev_filter * persistent_filter_create ( struct dev_filter * real ,
const char * file )
2001-10-22 14:14:00 +00:00
{
struct pfilter * pf ;
struct dev_filter * f = NULL ;
2005-10-16 23:03:59 +00:00
if ( ! ( pf = dm_malloc ( sizeof ( * pf ) ) ) ) {
2001-10-22 14:14:00 +00:00
stack ;
return NULL ;
}
memset ( pf , 0 , sizeof ( * pf ) ) ;
2005-10-16 23:03:59 +00:00
if ( ! ( pf - > file = dm_malloc ( strlen ( file ) + 1 ) ) ) {
2001-10-22 14:14:00 +00:00
stack ;
goto bad ;
}
strcpy ( pf - > file , file ) ;
2001-10-23 12:24:55 +00:00
pf - > real = real ;
2001-10-22 14:14:00 +00:00
if ( ! ( _init_hash ( pf ) ) ) {
2001-10-23 18:20:27 +00:00
log_error ( " Couldn't create hash table for persistent filter. " ) ;
2001-10-22 14:14:00 +00:00
goto bad ;
}
2005-10-16 23:03:59 +00:00
if ( ! ( f = dm_malloc ( sizeof ( * f ) ) ) ) {
2001-10-22 14:14:00 +00:00
stack ;
goto bad ;
}
2001-10-23 12:24:55 +00:00
f - > passes_filter = _lookup_p ;
2006-04-19 15:33:07 +00:00
f - > destroy = _persistent_destroy ;
2001-10-22 14:14:00 +00:00
f - > private = pf ;
return f ;
2002-04-24 18:20:51 +00:00
bad :
2005-10-16 23:03:59 +00:00
dm_free ( pf - > file ) ;
2001-10-22 14:14:00 +00:00
if ( pf - > devices )
2005-10-16 23:03:59 +00:00
dm_hash_destroy ( pf - > devices ) ;
dm_free ( pf ) ;
dm_free ( f ) ;
2001-10-22 14:14:00 +00:00
return NULL ;
}