2002-11-18 16:53:58 +03:00
/*
2004-03-30 23:35:44 +04:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 Red Hat , Inc . All rights reserved .
2002-11-18 16:53:58 +03:00
*
2004-03-30 23:35:44 +04: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
2002-11-18 16:53:58 +03:00
*
*/
# include "lib.h"
2003-07-05 02:34:56 +04:00
# include "lvmcache.h"
2002-11-18 16:53:58 +03:00
# include "hash.h"
# include "toolcontext.h"
# include "dev-cache.h"
# include "metadata.h"
2003-01-07 00:10:43 +03:00
# include "filter.h"
2003-07-05 02:34:56 +04:00
# include "memlock.h"
2003-10-16 00:04:29 +04:00
# include "str_list.h"
2003-01-07 00:10:43 +03:00
2002-11-18 16:53:58 +03:00
static struct hash_table * _pvid_hash = NULL ;
static struct hash_table * _vgid_hash = NULL ;
static struct hash_table * _vgname_hash = NULL ;
2003-07-05 02:34:56 +04:00
static struct hash_table * _lock_hash = NULL ;
2002-11-18 16:53:58 +03:00
static struct list _vginfos ;
2003-07-05 02:34:56 +04:00
static int _has_scanned = 0 ;
static int _vgs_locked = 0 ;
2002-11-18 16:53:58 +03:00
2003-07-05 02:34:56 +04:00
int lvmcache_init ( void )
2002-11-18 16:53:58 +03:00
{
list_init ( & _vginfos ) ;
if ( ! ( _vgname_hash = hash_create ( 128 ) ) )
return 0 ;
if ( ! ( _vgid_hash = hash_create ( 128 ) ) )
return 0 ;
if ( ! ( _pvid_hash = hash_create ( 128 ) ) )
return 0 ;
2003-07-05 02:34:56 +04:00
if ( ! ( _lock_hash = hash_create ( 128 ) ) )
return 0 ;
2002-11-18 16:53:58 +03:00
return 1 ;
}
2003-07-05 02:34:56 +04:00
void lvmcache_lock_vgname ( const char * vgname , int read_only )
2002-11-18 16:53:58 +03:00
{
2003-07-05 02:34:56 +04:00
if ( ! _lock_hash & & ! lvmcache_init ( ) ) {
log_error ( " Internal cache initialisation failed " ) ;
return ;
}
if ( ! hash_insert ( _lock_hash , vgname , ( void * ) 1 ) )
log_error ( " Cache locking failure for %s " , vgname ) ;
_vgs_locked + + ;
}
static int _vgname_is_locked ( const char * vgname ) __attribute__ ( ( unused ) ) ;
static int _vgname_is_locked ( const char * vgname )
{
if ( ! _lock_hash )
return 0 ;
return hash_lookup ( _lock_hash , vgname ) ? 1 : 0 ;
}
void lvmcache_unlock_vgname ( const char * vgname )
{
/* FIXME: Clear all CACHE_LOCKED flags in this vg */
hash_remove ( _lock_hash , vgname ) ;
/* FIXME Do this per-VG */
if ( ! - - _vgs_locked )
dev_close_all ( ) ;
}
int vgs_locked ( void )
{
return _vgs_locked ;
}
struct lvmcache_vginfo * vginfo_from_vgname ( const char * vgname )
{
struct lvmcache_vginfo * vginfo ;
2002-11-18 16:53:58 +03:00
if ( ! _vgname_hash )
return NULL ;
if ( ! ( vginfo = hash_lookup ( _vgname_hash , vgname ) ) )
return NULL ;
return vginfo ;
}
2002-12-20 02:25:55 +03:00
const struct format_type * fmt_from_vgname ( const char * vgname )
2002-11-18 16:53:58 +03:00
{
2003-07-05 02:34:56 +04:00
struct lvmcache_vginfo * vginfo ;
2002-11-18 16:53:58 +03:00
if ( ! ( vginfo = vginfo_from_vgname ( vgname ) ) )
return NULL ;
return vginfo - > fmt ;
}
2003-07-05 02:34:56 +04:00
struct lvmcache_vginfo * vginfo_from_vgid ( const char * vgid )
2002-11-18 16:53:58 +03:00
{
2003-07-05 02:34:56 +04:00
struct lvmcache_vginfo * vginfo ;
2003-01-10 22:14:01 +03:00
char id [ ID_LEN + 1 ] ;
2002-11-18 16:53:58 +03:00
if ( ! _vgid_hash | | ! vgid )
return NULL ;
2003-01-10 22:14:01 +03:00
/* vgid not necessarily NULL-terminated */
strncpy ( & id [ 0 ] , vgid , ID_LEN ) ;
id [ ID_LEN ] = ' \0 ' ;
if ( ! ( vginfo = hash_lookup ( _vgid_hash , id ) ) )
2002-11-18 16:53:58 +03:00
return NULL ;
return vginfo ;
}
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info_from_pvid ( const char * pvid )
2002-11-18 16:53:58 +03:00
{
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info ;
2003-01-10 22:14:01 +03:00
char id [ ID_LEN + 1 ] ;
2002-11-18 16:53:58 +03:00
if ( ! _pvid_hash | | ! pvid )
return NULL ;
2003-01-10 22:14:01 +03:00
strncpy ( & id [ 0 ] , pvid , ID_LEN ) ;
id [ ID_LEN ] = ' \0 ' ;
if ( ! ( info = hash_lookup ( _pvid_hash , id ) ) )
2002-11-18 16:53:58 +03:00
return NULL ;
return info ;
}
2003-07-05 02:34:56 +04:00
static void _rescan_entry ( struct lvmcache_info * info )
2002-11-18 16:53:58 +03:00
{
struct label * label ;
if ( info - > status & CACHE_INVALID )
label_read ( info - > dev , & label ) ;
}
2002-12-20 02:25:55 +03:00
static int _scan_invalid ( void )
2002-11-18 16:53:58 +03:00
{
hash_iter ( _pvid_hash , ( iterate_fn ) _rescan_entry ) ;
return 1 ;
}
2003-07-05 02:34:56 +04:00
int lvmcache_label_scan ( struct cmd_context * cmd , int full_scan )
2002-11-18 16:53:58 +03:00
{
struct label * label ;
struct dev_iter * iter ;
struct device * dev ;
struct list * fmth ;
struct format_type * fmt ;
static int _scanning_in_progress = 0 ;
int r = 0 ;
/* Avoid recursion when a PVID can't be found! */
if ( _scanning_in_progress )
return 0 ;
_scanning_in_progress = 1 ;
2003-07-05 02:34:56 +04:00
if ( ! _vgname_hash & & ! lvmcache_init ( ) ) {
2002-11-18 16:53:58 +03:00
log_error ( " Internal cache initialisation failed " ) ;
goto out ;
}
if ( _has_scanned & & ! full_scan ) {
2002-12-20 02:25:55 +03:00
r = _scan_invalid ( ) ;
2002-11-18 16:53:58 +03:00
goto out ;
}
if ( ! ( iter = dev_iter_create ( cmd - > filter ) ) ) {
log_error ( " dev_iter creation failed " ) ;
goto out ;
}
while ( ( dev = dev_iter_get ( iter ) ) )
label_read ( dev , & label ) ;
dev_iter_destroy ( iter ) ;
_has_scanned = 1 ;
/* Perform any format-specific scanning e.g. text files */
list_iterate ( fmth , & cmd - > formats ) {
fmt = list_item ( fmth , struct format_type ) ;
if ( fmt - > ops - > scan & & ! fmt - > ops - > scan ( fmt ) )
goto out ;
}
r = 1 ;
out :
_scanning_in_progress = 0 ;
return r ;
}
2003-07-05 02:34:56 +04:00
struct list * lvmcache_get_vgnames ( struct cmd_context * cmd , int full_scan )
2002-11-18 16:53:58 +03:00
{
2003-10-16 00:04:29 +04:00
struct list * vgnames ;
struct lvmcache_vginfo * vgi ;
2002-11-18 16:53:58 +03:00
2003-07-05 02:34:56 +04:00
lvmcache_label_scan ( cmd , full_scan ) ;
2002-11-18 16:53:58 +03:00
2003-10-16 00:04:29 +04:00
if ( ! ( vgnames = str_list_create ( cmd - > mem ) ) ) {
2002-11-18 16:53:58 +03:00
log_error ( " vgnames list allocation failed " ) ;
return NULL ;
}
2003-10-16 00:04:29 +04:00
list_iterate_items ( vgi , & _vginfos ) {
if ( ! str_list_add ( cmd - > mem , vgnames ,
pool_strdup ( cmd - > mem , vgi - > vgname ) ) ) {
2002-11-18 16:53:58 +03:00
log_error ( " strlist allocation failed " ) ;
return NULL ;
}
}
return vgnames ;
}
struct device * device_from_pvid ( struct cmd_context * cmd , struct id * pvid )
{
struct label * label ;
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info ;
2002-11-18 16:53:58 +03:00
/* Already cached ? */
if ( ( info = info_from_pvid ( ( char * ) pvid ) ) ) {
if ( label_read ( info - > dev , & label ) ) {
2003-07-05 02:34:56 +04:00
info = ( struct lvmcache_info * ) label - > info ;
2002-11-18 16:53:58 +03:00
if ( id_equal ( pvid , ( struct id * ) & info - > dev - > pvid ) )
return info - > dev ;
}
}
2003-07-05 02:34:56 +04:00
lvmcache_label_scan ( cmd , 0 ) ;
2002-11-18 16:53:58 +03:00
/* Try again */
if ( ( info = info_from_pvid ( ( char * ) pvid ) ) ) {
if ( label_read ( info - > dev , & label ) ) {
2003-07-05 02:34:56 +04:00
info = ( struct lvmcache_info * ) label - > info ;
2002-11-18 16:53:58 +03:00
if ( id_equal ( pvid , ( struct id * ) & info - > dev - > pvid ) )
return info - > dev ;
}
}
2003-07-05 02:34:56 +04:00
if ( memlock ( ) )
return NULL ;
lvmcache_label_scan ( cmd , 1 ) ;
2002-11-18 16:53:58 +03:00
/* Try again */
if ( ( info = info_from_pvid ( ( char * ) pvid ) ) ) {
if ( label_read ( info - > dev , & label ) ) {
2003-07-05 02:34:56 +04:00
info = ( struct lvmcache_info * ) label - > info ;
2002-11-18 16:53:58 +03:00
if ( id_equal ( pvid , ( struct id * ) & info - > dev - > pvid ) )
return info - > dev ;
}
}
return NULL ;
}
2003-07-05 02:34:56 +04:00
static void _drop_vginfo ( struct lvmcache_info * info )
2002-11-18 16:53:58 +03:00
{
if ( ! list_empty ( & info - > list ) ) {
list_del ( & info - > list ) ;
list_init ( & info - > list ) ;
}
if ( info - > vginfo & & list_empty ( & info - > vginfo - > infos ) ) {
hash_remove ( _vgname_hash , info - > vginfo - > vgname ) ;
if ( info - > vginfo - > vgname )
dbg_free ( info - > vginfo - > vgname ) ;
if ( * info - > vginfo - > vgid )
hash_remove ( _vgid_hash , info - > vginfo - > vgid ) ;
list_del ( & info - > vginfo - > list ) ;
dbg_free ( info - > vginfo ) ;
}
info - > vginfo = NULL ;
}
/* Unused
2003-07-05 02:34:56 +04:00
void lvmcache_del ( struct lvmcache_info * info )
2002-11-18 16:53:58 +03:00
{
if ( info - > dev - > pvid [ 0 ] & & _pvid_hash )
hash_remove ( _pvid_hash , info - > dev - > pvid ) ;
_drop_vginfo ( info ) ;
info - > label - > labeller - > ops - > destroy_label ( info - > label - > labeller ,
info - > label ) ;
dbg_free ( info ) ;
return ;
} */
2003-07-05 02:34:56 +04:00
static int _lvmcache_update_pvid ( struct lvmcache_info * info , const char * pvid )
2002-11-18 16:53:58 +03:00
{
if ( ! strcmp ( info - > dev - > pvid , pvid ) )
return 1 ;
if ( * info - > dev - > pvid ) {
hash_remove ( _pvid_hash , info - > dev - > pvid ) ;
}
strncpy ( info - > dev - > pvid , pvid , sizeof ( info - > dev - > pvid ) ) ;
if ( ! hash_insert ( _pvid_hash , pvid , info ) ) {
2003-07-05 02:34:56 +04:00
log_error ( " _lvmcache_update: pvid insertion failed: %s " , pvid ) ;
2002-11-18 16:53:58 +03:00
return 0 ;
}
return 1 ;
}
2003-07-05 02:34:56 +04:00
static int _lvmcache_update_vgid ( struct lvmcache_info * info , const char * vgid )
2002-11-18 16:53:58 +03:00
{
if ( ! vgid | | ! info - > vginfo | | ! strncmp ( info - > vginfo - > vgid , vgid ,
sizeof ( info - > vginfo - > vgid ) ) )
return 1 ;
if ( info - > vginfo & & * info - > vginfo - > vgid )
hash_remove ( _vgid_hash , info - > vginfo - > vgid ) ;
if ( ! vgid )
return 1 ;
strncpy ( info - > vginfo - > vgid , vgid , sizeof ( info - > vginfo - > vgid ) ) ;
info - > vginfo - > vgid [ sizeof ( info - > vginfo - > vgid ) - 1 ] = ' \0 ' ;
2003-07-05 02:34:56 +04:00
if ( ! hash_insert ( _vgid_hash , info - > vginfo - > vgid , info - > vginfo ) ) {
log_error ( " _lvmcache_update: vgid hash insertion failed: %s " ,
info - > vginfo - > vgid ) ;
2002-11-18 16:53:58 +03:00
return 0 ;
}
return 1 ;
}
2003-07-05 02:34:56 +04:00
int lvmcache_update_vgname ( struct lvmcache_info * info , const char * vgname )
2002-11-18 16:53:58 +03:00
{
2003-07-05 02:34:56 +04:00
struct lvmcache_vginfo * vginfo ;
2002-11-18 16:53:58 +03:00
/* If vgname is NULL and we don't already have a vgname,
* assume ORPHAN - we want every entry to have a vginfo
* attached for scanning reasons .
*/
if ( ! vgname & & ! info - > vginfo )
vgname = ORPHAN ;
if ( ! vgname | | ( info - > vginfo & & ! strcmp ( info - > vginfo - > vgname , vgname ) ) )
return 1 ;
/* Remove existing vginfo entry */
_drop_vginfo ( info ) ;
/* Get existing vginfo or create new one */
if ( ! ( vginfo = vginfo_from_vgname ( vgname ) ) ) {
if ( ! ( vginfo = dbg_malloc ( sizeof ( * vginfo ) ) ) ) {
2003-07-05 02:34:56 +04:00
log_error ( " lvmcache_update_vgname: list alloc failed " ) ;
2002-11-18 16:53:58 +03:00
return 0 ;
}
memset ( vginfo , 0 , sizeof ( * vginfo ) ) ;
if ( ! ( vginfo - > vgname = dbg_strdup ( vgname ) ) ) {
dbg_free ( vginfo ) ;
log_error ( " cache vgname alloc failed for %s " , vgname ) ;
return 0 ;
}
list_init ( & vginfo - > infos ) ;
if ( ! hash_insert ( _vgname_hash , vginfo - > vgname , vginfo ) ) {
log_error ( " cache_update: vg hash insertion failed: %s " ,
vginfo - > vgname ) ;
dbg_free ( vginfo - > vgname ) ;
dbg_free ( vginfo ) ;
return 0 ;
}
/* Ensure orphans appear last on list_iterate */
if ( ! * vgname )
list_add ( & _vginfos , & vginfo - > list ) ;
else
list_add_h ( & _vginfos , & vginfo - > list ) ;
}
info - > vginfo = vginfo ;
list_add ( & vginfo - > infos , & info - > list ) ;
/* FIXME Check consistency of list! */
vginfo - > fmt = info - > fmt ;
return 1 ;
}
2003-07-05 02:34:56 +04:00
int lvmcache_update_vg ( struct volume_group * vg )
2002-11-18 16:53:58 +03:00
{
struct list * pvh ;
struct physical_volume * pv ;
2003-07-05 02:34:56 +04:00
struct lvmcache_info * info ;
2002-11-18 16:53:58 +03:00
char pvid_s [ ID_LEN + 1 ] ;
int vgid_updated = 0 ;
pvid_s [ sizeof ( pvid_s ) - 1 ] = ' \0 ' ;
list_iterate ( pvh , & vg - > pvs ) {
pv = list_item ( pvh , struct pv_list ) - > pv ;
strncpy ( pvid_s , ( char * ) & pv - > id , sizeof ( pvid_s ) - 1 ) ;
/* FIXME Could pv->dev->pvid ever be different? */
if ( ( info = info_from_pvid ( pvid_s ) ) ) {
2003-07-05 02:34:56 +04:00
lvmcache_update_vgname ( info , vg - > name ) ;
2002-11-18 16:53:58 +03:00
if ( ! vgid_updated ) {
2003-07-05 02:34:56 +04:00
_lvmcache_update_vgid ( info , ( char * ) & vg - > id ) ;
2002-11-18 16:53:58 +03:00
vgid_updated = 1 ;
}
}
}
return 1 ;
}
2003-07-05 02:34:56 +04:00
struct lvmcache_info * lvmcache_add ( struct labeller * labeller , const char * pvid ,
struct device * dev ,
const char * vgname , const char * vgid )
2002-11-18 16:53:58 +03:00
{
struct label * label ;
2003-07-05 02:34:56 +04:00
struct lvmcache_info * existing , * info ;
2002-11-18 16:53:58 +03:00
char pvid_s [ ID_LEN + 1 ] ;
2003-07-05 02:34:56 +04:00
if ( ! _vgname_hash & & ! lvmcache_init ( ) ) {
2002-11-18 16:53:58 +03:00
log_error ( " Internal cache initialisation failed " ) ;
return NULL ;
}
strncpy ( pvid_s , pvid , sizeof ( pvid_s ) ) ;
pvid_s [ sizeof ( pvid_s ) - 1 ] = ' \0 ' ;
if ( ! ( existing = info_from_pvid ( pvid_s ) ) & &
! ( existing = info_from_pvid ( dev - > pvid ) ) ) {
if ( ! ( label = label_create ( labeller ) ) ) {
stack ;
return NULL ;
}
if ( ! ( info = dbg_malloc ( sizeof ( * info ) ) ) ) {
2003-07-05 02:34:56 +04:00
log_error ( " lvmcache_info allocation failed " ) ;
2002-11-18 16:53:58 +03:00
label_destroy ( label ) ;
return NULL ;
}
memset ( info , 0 , sizeof ( * info ) ) ;
label - > info = info ;
info - > label = label ;
list_init ( & info - > list ) ;
info - > dev = dev ;
} else {
2003-01-07 00:10:43 +03:00
if ( existing - > dev ! = dev ) {
/* Is the existing entry a duplicate pvid e.g. md ? */
if ( MAJOR ( existing - > dev - > dev ) = = md_major ( ) & &
MAJOR ( dev - > dev ) ! = md_major ( ) ) {
log_very_verbose ( " Ignoring duplicate PV %s on "
" %s - using md %s " ,
pvid , dev_name ( dev ) ,
dev_name ( existing - > dev ) ) ;
return NULL ;
} else if ( MAJOR ( existing - > dev - > dev ) ! = md_major ( ) & &
MAJOR ( dev - > dev ) = = md_major ( ) )
log_very_verbose ( " Duplicate PV %s on %s - "
" using md %s " , pvid ,
dev_name ( existing - > dev ) ,
dev_name ( dev ) ) ;
else
log_error ( " Found duplicate PV %s: using %s not "
" %s " , pvid , dev_name ( dev ) ,
dev_name ( existing - > dev ) ) ;
}
2002-11-18 16:53:58 +03:00
info = existing ;
/* Has labeller changed? */
if ( info - > label - > labeller ! = labeller ) {
label_destroy ( info - > label ) ;
if ( ! ( info - > label = label_create ( labeller ) ) ) {
/* FIXME leaves info without label! */
stack ;
return NULL ;
}
info - > label - > info = info ;
}
label = info - > label ;
}
2002-12-20 02:25:55 +03:00
info - > fmt = ( const struct format_type * ) labeller - > private ;
2002-11-18 16:53:58 +03:00
info - > status | = CACHE_INVALID ;
2003-07-05 02:34:56 +04:00
if ( ! _lvmcache_update_pvid ( info , pvid_s ) ) {
2002-11-18 16:53:58 +03:00
if ( ! existing ) {
dbg_free ( info ) ;
label_destroy ( label ) ;
}
return NULL ;
}
2003-07-05 02:34:56 +04:00
if ( ! lvmcache_update_vgname ( info , vgname ) ) {
2002-11-18 16:53:58 +03:00
if ( ! existing ) {
hash_remove ( _pvid_hash , pvid_s ) ;
strcpy ( info - > dev - > pvid , " " ) ;
dbg_free ( info ) ;
label_destroy ( label ) ;
}
return NULL ;
}
2003-07-05 02:34:56 +04:00
if ( ! _lvmcache_update_vgid ( info , vgid ) )
2002-11-18 16:53:58 +03:00
/* Non-critical */
stack ;
return info ;
}
2003-07-05 02:34:56 +04:00
static void _lvmcache_destroy_entry ( struct lvmcache_info * info )
2002-11-18 16:53:58 +03:00
{
if ( ! list_empty ( & info - > list ) )
list_del ( & info - > list ) ;
strcpy ( info - > dev - > pvid , " " ) ;
label_destroy ( info - > label ) ;
dbg_free ( info ) ;
}
2003-07-05 02:34:56 +04:00
static void _lvmcache_destroy_vgnamelist ( struct lvmcache_vginfo * vginfo )
2002-11-18 16:53:58 +03:00
{
if ( vginfo - > vgname )
dbg_free ( vginfo - > vgname ) ;
dbg_free ( vginfo ) ;
}
2003-07-05 02:34:56 +04:00
static void _lvmcache_destroy_lockname ( int present )
{
/* Nothing to do */
}
void lvmcache_destroy ( void )
2002-11-18 16:53:58 +03:00
{
_has_scanned = 0 ;
if ( _vgid_hash ) {
hash_destroy ( _vgid_hash ) ;
_vgid_hash = NULL ;
}
if ( _pvid_hash ) {
2003-07-05 02:34:56 +04:00
hash_iter ( _pvid_hash , ( iterate_fn ) _lvmcache_destroy_entry ) ;
2002-11-18 16:53:58 +03:00
hash_destroy ( _pvid_hash ) ;
_pvid_hash = NULL ;
}
if ( _vgname_hash ) {
2003-07-05 02:34:56 +04:00
hash_iter ( _vgname_hash ,
( iterate_fn ) _lvmcache_destroy_vgnamelist ) ;
2002-11-18 16:53:58 +03:00
hash_destroy ( _vgname_hash ) ;
_vgname_hash = NULL ;
}
2003-07-05 02:34:56 +04:00
if ( _lock_hash ) {
hash_iter ( _lock_hash , ( iterate_fn ) _lvmcache_destroy_lockname ) ;
hash_destroy ( _lock_hash ) ;
_lock_hash = NULL ;
}
2002-11-18 16:53:58 +03:00
list_init ( & _vginfos ) ;
}