2001-09-25 12:49:28 +00:00
/*
2004-03-08 17:19:15 +00:00
* Copyright ( C ) 2001 - 2003 Sistina Software ( UK ) Limited .
2001-09-25 12:49:28 +00:00
*
2001-12-20 11:52:54 +00:00
* This file is released under the GPL .
2001-09-25 12:49:28 +00:00
*/
2001-10-05 21:39:30 +00:00
# include "tools.h"
2001-09-25 12:49:28 +00:00
2001-12-31 21:27:39 +00:00
# include <sys/stat.h>
2002-02-11 20:50:53 +00:00
int process_each_lv_in_vg ( struct cmd_context * cmd , struct volume_group * vg ,
2004-03-08 17:19:15 +00:00
struct list * arg_lvnames , struct list * tags ,
2002-11-18 14:04:08 +00:00
void * handle ,
2002-02-11 21:00:35 +00:00
int ( * process_single ) ( struct cmd_context * cmd ,
2002-11-18 14:04:08 +00:00
struct logical_volume * lv ,
void * handle ) )
2001-11-19 15:20:50 +00:00
{
int ret_max = 0 ;
int ret = 0 ;
2004-03-08 17:19:15 +00:00
int process_all = 0 ;
int process_lv = 0 ;
int tags_supplied = 0 ;
int lvargs_supplied = 0 ;
2001-11-19 15:20:50 +00:00
2003-10-15 20:02:46 +00:00
struct lv_list * lvl ;
2001-11-19 15:20:50 +00:00
if ( vg - > status & EXPORTED_VG ) {
2002-01-30 15:04:48 +00:00
log_error ( " Volume group \" %s \" is exported " , vg - > name ) ;
2001-11-19 15:20:50 +00:00
return ECMD_FAILED ;
}
2002-01-29 17:23:33 +00:00
2004-03-08 17:19:15 +00:00
if ( tags & & ! list_empty ( tags ) )
tags_supplied = 1 ;
if ( arg_lvnames & & ! list_empty ( arg_lvnames ) )
lvargs_supplied = 1 ;
/* Process all LVs in this VG if no restrictions given */
if ( ! tags_supplied & & ! lvargs_supplied )
process_all = 1 ;
/* Or if VG tags match */
if ( ! process_lv & & tags_supplied & &
str_list_match_list ( tags , & vg - > tags ) )
process_all = 1 ;
2003-10-15 20:02:46 +00:00
list_iterate_items ( lvl , & vg - > lvs ) {
2004-03-08 17:19:15 +00:00
/* Should we process this LV? */
if ( process_all )
process_lv = 1 ;
else
process_lv = 0 ;
/* LV tag match? */
if ( ! process_lv & & tags_supplied & &
str_list_match_list ( tags , & lvl - > lv - > tags ) )
process_lv = 1 ;
/* LV name match? */
if ( ! process_lv & & lvargs_supplied & &
str_list_match_item ( arg_lvnames , lvl - > lv - > name ) )
process_lv = 1 ;
if ( ! process_lv )
continue ;
2003-10-15 20:02:46 +00:00
ret = process_single ( cmd , lvl - > lv , handle ) ;
2001-11-19 15:20:50 +00:00
if ( ret > ret_max )
ret_max = ret ;
}
return ret_max ;
}
2002-11-18 14:04:08 +00:00
struct volume_group * recover_vg ( struct cmd_context * cmd , const char * vgname ,
int lock_type )
{
int consistent = 1 ;
lock_type & = ~ LCK_TYPE_MASK ;
lock_type | = LCK_WRITE ;
if ( ! lock_vol ( cmd , vgname , lock_type ) ) {
log_error ( " Can't lock %s for metadata recovery: skipping " ,
vgname ) ;
return NULL ;
}
return vg_read ( cmd , vgname , & consistent ) ;
}
2002-02-11 20:50:53 +00:00
int process_each_lv ( struct cmd_context * cmd , int argc , char * * argv ,
2002-11-18 14:04:08 +00:00
int lock_type , void * handle ,
2002-02-11 21:00:35 +00:00
int ( * process_single ) ( struct cmd_context * cmd ,
2002-11-18 14:04:08 +00:00
struct logical_volume * lv ,
void * handle ) )
2001-11-14 18:38:07 +00:00
{
int opt = 0 ;
int ret_max = 0 ;
int ret = 0 ;
2002-11-18 14:04:08 +00:00
int consistent ;
2001-11-14 18:38:07 +00:00
2004-03-08 17:19:15 +00:00
struct list * slh , * tags_arg ;
struct list * vgnames ; /* VGs to process */
struct str_list * sll ;
2001-11-14 18:38:07 +00:00
struct volume_group * vg ;
2004-03-08 17:19:15 +00:00
struct list tags , lvnames ;
struct list arg_lvnames ; /* Cmdline vgname or vgname/lvname */
char * vglv ;
size_t vglv_sz ;
2001-11-14 18:38:07 +00:00
2002-12-19 23:25:55 +00:00
const char * vgname ;
2001-11-14 18:38:07 +00:00
2004-03-08 17:19:15 +00:00
list_init ( & tags ) ;
list_init ( & arg_lvnames ) ;
2001-11-14 18:38:07 +00:00
if ( argc ) {
2004-03-08 17:19:15 +00:00
struct list arg_vgnames ;
2001-11-14 18:38:07 +00:00
log_verbose ( " Using logical volume(s) on command line " ) ;
2004-03-08 17:19:15 +00:00
list_init ( & arg_vgnames ) ;
2001-11-14 18:38:07 +00:00
for ( ; opt < argc ; opt + + ) {
2004-03-08 17:19:15 +00:00
const char * lv_name = argv [ opt ] ;
2003-07-04 22:34:56 +00:00
char * vgname_def ;
int dev_dir_found = 0 ;
2002-11-18 14:04:08 +00:00
2004-03-08 17:19:15 +00:00
/* Do we have a tag or vgname or lvname? */
2002-11-18 14:04:08 +00:00
vgname = lv_name ;
2004-03-08 17:19:15 +00:00
if ( * vgname = = ' @ ' ) {
if ( ! validate_name ( vgname + 1 ) ) {
log_error ( " Skipping invalid tag %s " ,
vgname ) ;
continue ;
}
if ( ! str_list_add ( cmd - > mem , & tags ,
pool_strdup ( cmd - > mem ,
vgname + 1 ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
}
continue ;
}
/* FIXME Jumbled parsing */
2003-07-04 22:34:56 +00:00
if ( * vgname = = ' / ' ) {
while ( * vgname = = ' / ' )
vgname + + ;
vgname - - ;
}
2002-11-18 14:04:08 +00:00
if ( ! strncmp ( vgname , cmd - > dev_dir ,
2003-07-04 22:34:56 +00:00
strlen ( cmd - > dev_dir ) ) ) {
2002-11-18 14:04:08 +00:00
vgname + = strlen ( cmd - > dev_dir ) ;
2003-07-04 22:34:56 +00:00
dev_dir_found = 1 ;
while ( * vgname = = ' / ' )
vgname + + ;
}
if ( * vgname = = ' / ' ) {
log_error ( " \" %s \" : Invalid path for Logical "
2004-03-08 17:19:15 +00:00
" Volume " , argv [ opt ] ) ;
2003-07-04 22:34:56 +00:00
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
2004-03-08 17:19:15 +00:00
lv_name = vgname ;
2002-11-18 14:04:08 +00:00
if ( strchr ( vgname , ' / ' ) ) {
/* Must be an LV */
2004-03-08 17:19:15 +00:00
lv_name = strchr ( vgname , ' / ' ) ;
while ( * lv_name = = ' / ' )
lv_name + + ;
if ( ! ( vgname = extract_vgname ( cmd , vgname ) ) ) {
2002-11-18 14:04:08 +00:00
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
2004-03-08 17:19:15 +00:00
} else if ( ! dev_dir_found & &
2003-07-04 22:34:56 +00:00
( vgname_def = default_vgname ( cmd ) ) ) {
vgname = vgname_def ;
2004-03-08 17:19:15 +00:00
} else
lv_name = NULL ;
2003-07-04 22:34:56 +00:00
2004-03-08 17:19:15 +00:00
if ( ! str_list_add ( cmd - > mem , & arg_vgnames ,
pool_strdup ( cmd - > mem , vgname ) ) ) {
log_error ( " strlist allocation failed " ) ;
2002-01-29 17:23:33 +00:00
return ECMD_FAILED ;
}
2004-03-08 17:19:15 +00:00
if ( ! lv_name ) {
if ( ! str_list_add ( cmd - > mem , & arg_lvnames ,
pool_strdup ( cmd - > mem , vgname ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
}
} else {
vglv_sz = strlen ( vgname ) + strlen ( lv_name ) + 2 ;
if ( ! ( vglv = pool_alloc ( cmd - > mem , vglv_sz ) ) | |
lvm_snprintf ( vglv , vglv_sz , " %s/%s " , vgname ,
lv_name ) < 0 ) {
log_error ( " vg/lv string alloc failed " ) ;
return ECMD_FAILED ;
}
if ( ! str_list_add ( cmd - > mem , & arg_lvnames ,
vglv ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
}
2001-11-14 18:38:07 +00:00
}
}
2004-03-08 17:19:15 +00:00
vgnames = & arg_vgnames ;
}
if ( ! argc | | ! list_empty ( & tags ) ) {
2002-01-16 14:43:27 +00:00
log_verbose ( " Finding all logical volumes " ) ;
2004-03-08 17:19:15 +00:00
if ( ! ( vgnames = get_vgs ( cmd , 0 ) ) | | list_empty ( vgnames ) ) {
2001-11-14 18:38:07 +00:00
log_error ( " No volume groups found " ) ;
return ECMD_FAILED ;
}
2004-03-08 17:19:15 +00:00
}
list_iterate ( slh , vgnames ) {
vgname = list_item ( slh , struct str_list ) - > str ;
if ( ! vgname | | ! * vgname )
continue ; /* FIXME Unnecessary? */
if ( ! lock_vol ( cmd , vgname , lock_type ) ) {
log_error ( " Can't lock %s: skipping " , vgname ) ;
continue ;
}
if ( lock_type & LCK_WRITE )
consistent = 1 ;
else
consistent = 0 ;
if ( ! ( vg = vg_read ( cmd , vgname , & consistent ) ) | |
! consistent ) {
unlock_vg ( cmd , vgname ) ;
if ( ! vg )
log_error ( " Volume group \" %s \" "
" not found " , vgname ) ;
2002-11-18 14:04:08 +00:00
else
2004-03-08 17:19:15 +00:00
log_error ( " Volume group \" %s \" "
" inconsistent " , vgname ) ;
if ( ! vg | | ! ( vg =
recover_vg ( cmd , vgname ,
lock_type ) ) ) {
2002-11-18 14:04:08 +00:00
unlock_vg ( cmd , vgname ) ;
2004-03-08 17:19:15 +00:00
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
}
tags_arg = & tags ;
list_init ( & lvnames ) ; /* LVs to be processed in this VG */
list_iterate_items ( sll , & arg_lvnames ) {
const char * vg_name = sll - > str ;
const char * lv_name = strchr ( vg_name , ' / ' ) ;
if ( ( ! lv_name & & ! strcmp ( vg_name , vgname ) ) ) {
/* Process all LVs in this VG */
tags_arg = NULL ;
list_init ( & lvnames ) ;
break ;
} else if ( ! strncmp ( vg_name , vgname , strlen ( vgname ) ) & &
strlen ( vgname ) = = lv_name - vg_name ) {
if ( ! str_list_add ( cmd - > mem , & lvnames ,
pool_strdup ( cmd - > mem ,
lv_name + 1 ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
2002-11-18 14:04:08 +00:00
}
2001-11-14 18:38:07 +00:00
}
}
2004-03-08 17:19:15 +00:00
ret = process_each_lv_in_vg ( cmd , vg , & lvnames , tags_arg ,
handle , process_single ) ;
unlock_vg ( cmd , vgname ) ;
if ( ret > ret_max )
ret_max = ret ;
2001-11-14 18:38:07 +00:00
}
return ret_max ;
}
2002-12-12 20:55:49 +00:00
int process_each_segment_in_lv ( struct cmd_context * cmd ,
struct logical_volume * lv ,
void * handle ,
int ( * process_single ) ( struct cmd_context * cmd ,
struct lv_segment * seg ,
void * handle ) )
{
struct lv_segment * seg ;
int ret_max = 0 ;
int ret ;
2003-10-15 20:02:46 +00:00
list_iterate_items ( seg , & lv - > segments ) {
2002-12-12 20:55:49 +00:00
ret = process_single ( cmd , seg , handle ) ;
if ( ret > ret_max )
ret_max = ret ;
}
return ret_max ;
}
2004-03-08 17:19:15 +00:00
static int _process_one_vg ( struct cmd_context * cmd , const char * vg_name ,
struct list * tags , struct list * arg_vgnames ,
int lock_type , int consistent , void * handle ,
int ret_max ,
int ( * process_single ) ( struct cmd_context * cmd ,
const char * vg_name ,
struct volume_group * vg ,
int consistent , void * handle ) )
{
struct volume_group * vg ;
int ret = 0 ;
if ( ! lock_vol ( cmd , vg_name , lock_type ) ) {
log_error ( " Can't lock %s: skipping " , vg_name ) ;
return ret_max ;
}
log_verbose ( " Finding volume group \" %s \" " , vg_name ) ;
vg = vg_read ( cmd , vg_name , & consistent ) ;
if ( ! list_empty ( tags ) ) {
/* Only process if a tag matches or it's on arg_vgnames */
if ( ! str_list_match_item ( arg_vgnames , vg_name ) & &
! str_list_match_list ( tags , & vg - > tags ) ) {
unlock_vg ( cmd , vg_name ) ;
return ret_max ;
}
}
if ( ( ret = process_single ( cmd , vg_name , vg , consistent ,
handle ) ) > ret_max )
ret_max = ret ;
unlock_vg ( cmd , vg_name ) ;
return ret_max ;
}
2002-02-11 21:00:35 +00:00
int process_each_vg ( struct cmd_context * cmd , int argc , char * * argv ,
2002-11-18 14:04:08 +00:00
int lock_type , int consistent , void * handle ,
2002-02-11 21:00:35 +00:00
int ( * process_single ) ( struct cmd_context * cmd ,
2002-11-18 14:04:08 +00:00
const char * vg_name ,
struct volume_group * vg ,
int consistent , void * handle ) )
2001-10-05 21:39:30 +00:00
{
int opt = 0 ;
int ret_max = 0 ;
2004-03-08 17:19:15 +00:00
struct str_list * sl ;
struct list * vgnames ;
struct list arg_vgnames , tags ;
2001-10-05 21:39:30 +00:00
2003-07-04 22:34:56 +00:00
const char * vg_name ;
2002-04-24 18:20:51 +00:00
char * dev_dir = cmd - > dev_dir ;
2002-02-11 15:42:34 +00:00
2004-03-08 17:19:15 +00:00
list_init ( & tags ) ;
list_init ( & arg_vgnames ) ;
2001-10-05 21:39:30 +00:00
if ( argc ) {
log_verbose ( " Using volume group(s) on command line " ) ;
2004-03-08 17:19:15 +00:00
2002-02-11 15:42:34 +00:00
for ( ; opt < argc ; opt + + ) {
vg_name = argv [ opt ] ;
2004-03-08 17:19:15 +00:00
if ( * vg_name = = ' @ ' ) {
if ( ! validate_name ( vg_name + 1 ) ) {
log_error ( " Skipping invalid tag %s " ,
vg_name ) ;
continue ;
}
if ( ! str_list_add ( cmd - > mem , & tags ,
pool_strdup ( cmd - > mem ,
vg_name + 1 ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
}
continue ;
}
2003-07-04 22:34:56 +00:00
if ( * vg_name = = ' / ' ) {
while ( * vg_name = = ' / ' )
vg_name + + ;
vg_name - - ;
}
2002-04-24 18:20:51 +00:00
if ( ! strncmp ( vg_name , dev_dir , strlen ( dev_dir ) ) )
vg_name + = strlen ( dev_dir ) ;
if ( strchr ( vg_name , ' / ' ) ) {
log_error ( " Invalid volume group name: %s " ,
vg_name ) ;
continue ;
}
2004-03-08 17:19:15 +00:00
if ( ! str_list_add ( cmd - > mem , & arg_vgnames ,
pool_strdup ( cmd - > mem , vg_name ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
2002-02-11 15:42:34 +00:00
}
}
2004-03-08 17:19:15 +00:00
vgnames = & arg_vgnames ;
}
if ( ! argc | | ! list_empty ( & tags ) ) {
2002-01-16 14:43:27 +00:00
log_verbose ( " Finding all volume groups " ) ;
2002-11-18 14:04:08 +00:00
if ( ! ( vgnames = get_vgs ( cmd , 0 ) ) | | list_empty ( vgnames ) ) {
2001-10-05 21:39:30 +00:00
log_error ( " No volume groups found " ) ;
return ECMD_FAILED ;
}
2004-03-08 17:19:15 +00:00
}
list_iterate_items ( sl , vgnames ) {
vg_name = sl - > str ;
if ( ! vg_name | | ! * vg_name )
continue ; /* FIXME Unnecessary? */
ret_max = _process_one_vg ( cmd , vg_name , & tags , & arg_vgnames ,
lock_type , consistent , handle ,
ret_max , process_single ) ;
2001-10-05 21:39:30 +00:00
}
return ret_max ;
}
2001-10-08 18:44:22 +00:00
2002-02-11 20:50:53 +00:00
int process_each_pv_in_vg ( struct cmd_context * cmd , struct volume_group * vg ,
2004-03-08 17:19:15 +00:00
struct list * tags , void * handle ,
2002-02-11 21:00:35 +00:00
int ( * process_single ) ( struct cmd_context * cmd ,
struct volume_group * vg ,
2002-11-18 14:04:08 +00:00
struct physical_volume * pv ,
void * handle ) )
2001-11-19 15:20:50 +00:00
{
int ret_max = 0 ;
int ret = 0 ;
2003-10-15 20:02:46 +00:00
struct pv_list * pvl ;
2001-11-19 15:20:50 +00:00
2004-03-08 17:19:15 +00:00
list_iterate_items ( pvl , & vg - > pvs ) {
if ( tags & & ! list_empty ( tags ) & &
! str_list_match_list ( tags , & pvl - > pv - > tags ) )
continue ;
2003-10-15 20:02:46 +00:00
if ( ( ret = process_single ( cmd , vg , pvl - > pv , handle ) ) > ret_max )
2002-01-21 16:05:23 +00:00
ret_max = ret ;
2004-03-08 17:19:15 +00:00
}
2002-01-21 16:05:23 +00:00
return ret_max ;
2001-11-19 15:20:50 +00:00
}
2002-02-11 21:00:35 +00:00
int process_each_pv ( struct cmd_context * cmd , int argc , char * * argv ,
2002-11-18 14:04:08 +00:00
struct volume_group * vg , void * handle ,
2002-02-11 21:00:35 +00:00
int ( * process_single ) ( struct cmd_context * cmd ,
struct volume_group * vg ,
2002-11-18 14:04:08 +00:00
struct physical_volume * pv ,
void * handle ) )
2001-10-11 21:35:55 +00:00
{
int opt = 0 ;
int ret_max = 0 ;
int ret = 0 ;
2002-01-21 14:28:12 +00:00
struct pv_list * pvl ;
2002-12-12 20:55:49 +00:00
struct physical_volume * pv ;
2004-03-08 17:19:15 +00:00
struct list * pvslist , * vgnames ;
struct list tags ;
struct str_list * sll ;
char * tagname ;
int consistent = 1 ;
list_init ( & tags ) ;
2001-10-11 21:35:55 +00:00
if ( argc ) {
log_verbose ( " Using physical volume(s) on command line " ) ;
for ( ; opt < argc ; opt + + ) {
2004-03-08 17:19:15 +00:00
if ( * argv [ opt ] = = ' @ ' ) {
tagname = argv [ opt ] + 1 ;
if ( ! validate_name ( tagname ) ) {
log_error ( " Skipping invalid tag %s " ,
tagname ) ;
continue ;
}
if ( ! str_list_add ( cmd - > mem , & tags ,
pool_strdup ( cmd - > mem ,
tagname ) ) ) {
log_error ( " strlist allocation failed " ) ;
return ECMD_FAILED ;
}
continue ;
}
2002-12-12 20:55:49 +00:00
if ( vg ) {
if ( ! ( pvl = find_pv_in_vg ( vg , argv [ opt ] ) ) ) {
log_error ( " Physical Volume \" %s \" not "
" found in Volume Group "
" \" %s \" " , argv [ opt ] ,
vg - > name ) ;
2003-10-21 22:06:07 +00:00
ret_max = ECMD_FAILED ;
2002-12-12 20:55:49 +00:00
continue ;
}
pv = pvl - > pv ;
} else {
if ( ! ( pv = pv_read ( cmd , argv [ opt ] , NULL , NULL ) ) ) {
log_error ( " Failed to read physical "
" volume \" %s \" " , argv [ opt ] ) ;
2003-10-21 22:06:07 +00:00
ret_max = ECMD_FAILED ;
2002-12-12 20:55:49 +00:00
continue ;
}
2001-10-11 21:35:55 +00:00
}
2002-12-12 20:55:49 +00:00
ret = process_single ( cmd , vg , pv , handle ) ;
2001-10-11 21:35:55 +00:00
if ( ret > ret_max )
ret_max = ret ;
}
2004-03-08 17:19:15 +00:00
if ( ! list_empty ( & tags ) & & ( vgnames = get_vgs ( cmd , 0 ) ) & &
! list_empty ( vgnames ) ) {
list_iterate_items ( sll , vgnames ) {
vg = vg_read ( cmd , sll - > str , & consistent ) ;
if ( ! consistent )
continue ;
ret = process_each_pv_in_vg ( cmd , vg , & tags ,
handle ,
process_single ) ;
if ( ret > ret_max )
ret_max = ret ;
}
}
2001-10-11 21:35:55 +00:00
} else {
2002-12-12 20:55:49 +00:00
if ( vg ) {
log_verbose ( " Using all physical volume(s) in "
" volume group " ) ;
2004-03-08 17:19:15 +00:00
ret = process_each_pv_in_vg ( cmd , vg , NULL , handle ,
process_single ) ;
2003-10-21 22:06:07 +00:00
if ( ret > ret_max )
ret_max = ret ;
2002-12-12 20:55:49 +00:00
} else {
log_verbose ( " Scanning for physical volume names " ) ;
2002-12-19 23:25:55 +00:00
if ( ! ( pvslist = get_pvs ( cmd ) ) )
2002-12-12 20:55:49 +00:00
return ECMD_FAILED ;
2003-10-15 20:02:46 +00:00
list_iterate_items ( pvl , pvslist ) {
ret = process_single ( cmd , NULL , pvl - > pv ,
handle ) ;
2002-12-12 20:55:49 +00:00
if ( ret > ret_max )
ret_max = ret ;
}
}
2001-10-11 21:35:55 +00:00
}
return ret_max ;
}
2002-12-19 23:25:55 +00:00
const char * extract_vgname ( struct cmd_context * cmd , const char * lv_name )
2001-11-14 18:38:07 +00:00
{
2002-12-19 23:25:55 +00:00
const char * vg_name = lv_name ;
2001-11-06 19:02:26 +00:00
char * st ;
2002-04-24 18:20:51 +00:00
char * dev_dir = cmd - > dev_dir ;
2003-07-04 22:34:56 +00:00
int dev_dir_provided = 0 ;
2001-10-29 13:52:23 +00:00
/* Path supplied? */
2001-11-06 19:02:26 +00:00
if ( vg_name & & strchr ( vg_name , ' / ' ) ) {
2001-11-15 17:27:45 +00:00
/* Strip dev_dir (optional) */
2003-07-04 22:34:56 +00:00
if ( * vg_name = = ' / ' ) {
while ( * vg_name = = ' / ' )
vg_name + + ;
vg_name - - ;
}
if ( ! strncmp ( vg_name , dev_dir , strlen ( dev_dir ) ) ) {
2001-11-12 15:10:01 +00:00
vg_name + = strlen ( dev_dir ) ;
2003-07-04 22:34:56 +00:00
dev_dir_provided = 1 ;
while ( * vg_name = = ' / ' )
vg_name + + ;
}
if ( * vg_name = = ' / ' ) {
log_error ( " \" %s \" : Invalid path for Logical "
" Volume " , lv_name ) ;
return 0 ;
}
2004-03-08 17:19:15 +00:00
2003-07-04 22:34:56 +00:00
/* Require exactly one set of consecutive slashes */
if ( ( st = strchr ( vg_name , ' / ' ) ) )
while ( * st = = ' / ' )
st + + ;
2001-10-29 13:52:23 +00:00
2003-07-04 22:34:56 +00:00
if ( ! strchr ( vg_name , ' / ' ) | | strchr ( st , ' / ' ) ) {
2002-01-30 15:04:48 +00:00
log_error ( " \" %s \" : Invalid path for Logical Volume " ,
2001-11-14 18:38:07 +00:00
lv_name ) ;
2001-10-29 13:52:23 +00:00
return 0 ;
}
2002-04-24 18:20:51 +00:00
vg_name = pool_strdup ( cmd - > mem , vg_name ) ;
2001-10-29 13:52:23 +00:00
if ( ! vg_name ) {
log_error ( " Allocation of vg_name failed " ) ;
return 0 ;
}
* strchr ( vg_name , ' / ' ) = ' \0 ' ;
return vg_name ;
}
2001-11-06 19:02:26 +00:00
2002-04-24 18:20:51 +00:00
if ( ! ( vg_name = default_vgname ( cmd ) ) ) {
2001-11-06 19:02:26 +00:00
if ( lv_name )
2002-01-30 15:04:48 +00:00
log_error ( " Path required for Logical Volume \" %s \" " ,
2001-11-14 18:38:07 +00:00
lv_name ) ;
2001-11-06 19:02:26 +00:00
return 0 ;
}
2001-11-14 18:38:07 +00:00
2001-11-06 19:02:26 +00:00
return vg_name ;
}
2002-04-24 18:20:51 +00:00
char * default_vgname ( struct cmd_context * cmd )
2001-11-06 19:02:26 +00:00
{
char * vg_path ;
2002-04-24 18:20:51 +00:00
char * dev_dir = cmd - > dev_dir ;
2001-11-06 19:02:26 +00:00
2001-10-29 13:52:23 +00:00
/* Take default VG from environment? */
2001-11-14 18:38:07 +00:00
vg_path = getenv ( " LVM_VG_NAME " ) ;
2001-11-06 19:02:26 +00:00
if ( ! vg_path )
2001-10-29 13:52:23 +00:00
return 0 ;
2001-11-15 17:27:45 +00:00
/* Strip dev_dir (optional) */
2003-07-04 22:34:56 +00:00
if ( * vg_path = = ' / ' ) {
while ( * vg_path = = ' / ' )
vg_path + + ;
vg_path - - ;
}
2001-11-12 15:10:01 +00:00
if ( ! strncmp ( vg_path , dev_dir , strlen ( dev_dir ) ) )
vg_path + = strlen ( dev_dir ) ;
2001-10-29 13:52:23 +00:00
if ( strchr ( vg_path , ' / ' ) ) {
2002-01-30 15:04:48 +00:00
log_error ( " Environment Volume Group in LVM_VG_NAME invalid: "
" \" %s \" " , vg_path ) ;
2001-10-29 13:52:23 +00:00
return 0 ;
}
2002-04-24 18:20:51 +00:00
return pool_strdup ( cmd - > mem , vg_path ) ;
2001-10-29 13:52:23 +00:00
}
2002-01-21 16:05:23 +00:00
2003-04-24 22:23:24 +00:00
static int _add_alloc_area ( struct pool * mem , struct list * alloc_areas ,
uint32_t start , uint32_t count )
{
struct alloc_area * aa ;
log_debug ( " Adding alloc area: start PE % " PRIu32 " length % " PRIu32 ,
start , count ) ;
/* Ensure no overlap with existing areas */
2003-10-15 20:02:46 +00:00
list_iterate_items ( aa , alloc_areas ) {
2003-04-24 22:23:24 +00:00
if ( ( ( start < aa - > start ) & & ( start + count - 1 > = aa - > start ) ) | |
( ( start > = aa - > start ) & &
( aa - > start + aa - > count - 1 ) > = start ) ) {
log_error ( " Overlapping PE ranges detected (% " PRIu32
" -% " PRIu32 " , % " PRIu32 " -% " PRIu32 " ) " ,
start , start + count - 1 , aa - > start ,
aa - > start + aa - > count - 1 ) ;
return 0 ;
}
}
if ( ! ( aa = pool_alloc ( mem , sizeof ( * aa ) ) ) ) {
log_error ( " Allocation of list failed " ) ;
return 0 ;
}
aa - > start = start ;
aa - > count = count ;
list_add ( alloc_areas , & aa - > list ) ;
return 1 ;
}
static int _parse_pes ( struct pool * mem , char * c , struct list * alloc_areas ,
uint32_t size )
{
char * endptr ;
uint32_t start , end ;
/* Default to whole PV */
if ( ! c ) {
if ( ! _add_alloc_area ( mem , alloc_areas , UINT32_C ( 0 ) , size ) ) {
stack ;
return 0 ;
}
return 1 ;
}
while ( * c ) {
if ( * c ! = ' : ' )
goto error ;
c + + ;
/* Disallow :: and :\0 */
if ( * c = = ' : ' | | ! * c )
goto error ;
/* Default to whole range */
start = UINT32_C ( 0 ) ;
end = size - 1 ;
/* Start extent given? */
if ( isdigit ( * c ) ) {
start = ( uint32_t ) strtoul ( c , & endptr , 10 ) ;
if ( endptr = = c )
goto error ;
c = endptr ;
/* Just one number given? */
if ( ! * c | | * c = = ' : ' )
end = start ;
}
/* Range? */
if ( * c = = ' - ' ) {
c + + ;
if ( isdigit ( * c ) ) {
end = ( uint32_t ) strtoul ( c , & endptr , 10 ) ;
if ( endptr = = c )
goto error ;
c = endptr ;
}
}
if ( * c & & * c ! = ' : ' )
goto error ;
if ( ( start > end ) | | ( end > size - 1 ) ) {
log_error ( " PE range error: start extent % " PRIu32 " to "
" end extent % " PRIu32 , start , end ) ;
return 0 ;
}
if ( ! _add_alloc_area ( mem , alloc_areas , start , end - start + 1 ) ) {
stack ;
return 0 ;
}
}
return 1 ;
error :
log_error ( " Physical extent parsing error at %s " , c ) ;
return 0 ;
}
2004-03-08 17:19:15 +00:00
static void _create_pv_entry ( struct pool * mem , struct pv_list * pvl ,
char * colon , struct list * r )
{
const char * pvname ;
struct pv_list * new_pvl ;
struct list * alloc_areas ;
pvname = dev_name ( pvl - > pv - > dev ) ;
if ( ! ( pvl - > pv - > status & ALLOCATABLE_PV ) ) {
log_error ( " Physical volume %s not allocatable " , pvname ) ;
return ;
}
if ( pvl - > pv - > pe_count = = pvl - > pv - > pe_alloc_count ) {
log_err ( " No free extents on physical volume \" %s \" " ,
pvname ) ;
return ;
}
if ( ! ( new_pvl = pool_alloc ( mem , sizeof ( * new_pvl ) ) ) ) {
log_err ( " Unable to allocate physical volume list. " ) ;
return ;
}
memcpy ( new_pvl , pvl , sizeof ( * new_pvl ) ) ;
if ( ! ( alloc_areas = pool_alloc ( mem , sizeof ( * alloc_areas ) ) ) ) {
log_error ( " Allocation of alloc_areas list failed " ) ;
return ;
}
list_init ( alloc_areas ) ;
/* Specify which physical extents may be used for allocation */
if ( ! _parse_pes ( mem , colon , alloc_areas , pvl - > pv - > pe_count ) ) {
stack ;
return ;
}
new_pvl - > alloc_areas = alloc_areas ;
list_add ( r , & new_pvl - > list ) ;
}
2002-01-21 16:05:23 +00:00
struct list * create_pv_list ( struct pool * mem ,
2002-01-29 17:23:33 +00:00
struct volume_group * vg , int argc , char * * argv )
2002-01-21 16:05:23 +00:00
{
struct list * r ;
2004-03-08 17:19:15 +00:00
struct pv_list * pvl ;
struct list tags , arg_pvnames ;
const char * pvname = NULL ;
char * colon , * tagname ;
2002-01-21 16:05:23 +00:00
int i ;
/* Build up list of PVs */
if ( ! ( r = pool_alloc ( mem , sizeof ( * r ) ) ) ) {
log_error ( " Allocation of list failed " ) ;
return NULL ;
}
list_init ( r ) ;
2004-03-08 17:19:15 +00:00
list_init ( & tags ) ;
list_init ( & arg_pvnames ) ;
2002-01-21 16:05:23 +00:00
for ( i = 0 ; i < argc ; i + + ) {
2004-03-08 17:19:15 +00:00
if ( * argv [ i ] = = ' @ ' ) {
tagname = argv [ i ] + 1 ;
if ( ! validate_name ( tagname ) ) {
log_error ( " Skipping invalid tag %s " , tagname ) ;
continue ;
}
list_iterate_items ( pvl , & vg - > pvs ) {
if ( str_list_match_item ( & pvl - > pv - > tags ,
tagname ) ) {
_create_pv_entry ( mem , pvl , NULL , r ) ;
}
2003-04-24 22:23:24 +00:00
}
2002-01-21 16:05:23 +00:00
continue ;
}
2004-03-08 17:19:15 +00:00
pvname = argv [ i ] ;
2003-04-24 22:23:24 +00:00
2004-03-08 17:19:15 +00:00
if ( ( colon = strchr ( pvname , ' : ' ) ) ) {
if ( ! ( pvname = pool_strndup ( mem , pvname ,
( unsigned ) ( colon -
pvname ) ) ) ) {
log_error ( " Failed to clone PV name " ) ;
return NULL ;
}
2003-04-24 22:23:24 +00:00
}
2004-03-08 17:19:15 +00:00
if ( ! ( pvl = find_pv_in_vg ( vg , pvname ) ) ) {
log_err ( " Physical Volume \" %s \" not found in "
" Volume Group \" %s \" " , pvname , vg - > name ) ;
return NULL ;
}
_create_pv_entry ( mem , pvl , colon , r ) ;
2002-01-21 16:05:23 +00:00
}
2004-03-08 17:19:15 +00:00
if ( list_empty ( r ) )
log_error ( " No specified PVs have space available " ) ;
2002-01-29 17:23:33 +00:00
return list_empty ( r ) ? NULL : r ;
2002-01-21 16:05:23 +00:00
}
2003-04-24 22:23:24 +00:00
2004-03-08 17:19:15 +00:00
2003-04-24 22:23:24 +00:00
struct list * clone_pv_list ( struct pool * mem , struct list * pvsl )
{
2003-10-15 20:02:46 +00:00
struct list * r ;
2003-04-24 22:23:24 +00:00
struct pv_list * pvl , * new_pvl ;
/* Build up list of PVs */
if ( ! ( r = pool_alloc ( mem , sizeof ( * r ) ) ) ) {
log_error ( " Allocation of list failed " ) ;
return NULL ;
}
list_init ( r ) ;
2003-10-15 20:02:46 +00:00
list_iterate_items ( pvl , pvsl ) {
2003-04-24 22:23:24 +00:00
if ( ! ( new_pvl = pool_zalloc ( mem , sizeof ( * new_pvl ) ) ) ) {
log_error ( " Unable to allocate physical volume list. " ) ;
return NULL ;
}
memcpy ( new_pvl , pvl , sizeof ( * new_pvl ) ) ;
list_add ( r , & new_pvl - > list ) ;
}
return r ;
}