2005-06-06 21:12:08 +04: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 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
*/
# include "tools.h"
2005-11-29 21:20:23 +03:00
# include "lv_alloc.h"
2005-06-06 21:12:08 +04:00
struct lvconvert_params {
2005-08-15 18:10:28 +04:00
const char * lv_name ;
uint32_t mirrors ;
2005-11-29 21:20:23 +03:00
sign_t mirrors_sign ;
uint32_t region_size ;
2005-08-15 18:10:28 +04:00
alloc_policy_t alloc ;
int pv_count ;
char * * pvs ;
2005-06-06 21:12:08 +04:00
struct list * pvh ;
} ;
2005-08-15 18:10:28 +04:00
static int _read_params ( struct lvconvert_params * lp , struct cmd_context * cmd ,
int argc , char * * argv )
{
memset ( lp , 0 , sizeof ( * lp ) ) ;
lp - > alloc = ALLOC_INHERIT ;
if ( arg_count ( cmd , alloc_ARG ) )
lp - > alloc = ( alloc_policy_t ) arg_uint_value ( cmd , alloc_ARG ,
lp - > alloc ) ;
if ( ! arg_count ( cmd , mirrors_ARG ) ) {
log_error ( " --mirrors argument required " ) ;
return 0 ;
}
2005-11-29 21:20:23 +03:00
lp - > mirrors = arg_uint_value ( cmd , mirrors_ARG , 0 ) ;
lp - > mirrors_sign = arg_sign_value ( cmd , mirrors_ARG , 0 ) ;
/*
* - - regionsize is only valid when converting an LV into a mirror .
* This is checked when we know the state of the LV being converted .
*/
if ( arg_count ( cmd , regionsize_ARG ) ) {
if ( arg_sign_value ( cmd , regionsize_ARG , 0 ) = = SIGN_MINUS ) {
log_error ( " Negative regionsize is invalid " ) ;
return 0 ;
}
lp - > region_size = 2 * arg_uint_value ( cmd , regionsize_ARG , 0 ) ;
} else
lp - > region_size = 2 * find_config_int ( cmd - > cft - > root ,
" activation/mirror_region_size " ,
DEFAULT_MIRROR_REGION_SIZE ) ;
if ( lp - > region_size & ( lp - > region_size - 1 ) ) {
log_error ( " Region size (% " PRIu32 " ) must be a power of 2 " ,
lp - > region_size ) ;
return 0 ;
}
2005-08-15 18:10:28 +04:00
if ( ! argc ) {
log_error ( " Please give logical volume path " ) ;
return 0 ;
}
lp - > lv_name = argv [ 0 ] ;
argv + + , argc - - ;
lp - > pv_count = argc ;
lp - > pvs = argv ;
return 1 ;
}
2005-06-06 21:12:08 +04:00
static int lvconvert_mirrors ( struct cmd_context * cmd , struct logical_volume * lv ,
2005-08-15 18:10:28 +04:00
struct lvconvert_params * lp )
2005-06-06 21:12:08 +04:00
{
2005-10-28 16:48:50 +04:00
struct lv_segment * seg ;
2005-08-15 18:10:28 +04:00
uint32_t existing_mirrors ;
2005-11-29 21:20:23 +03:00
struct alloc_handle * ah = NULL ;
struct logical_volume * log_lv ;
struct list * parallel_areas ;
struct segment_type * segtype ;
seg = first_seg ( lv ) ;
existing_mirrors = seg - > area_count ;
/* Adjust required number of mirrors */
if ( lp - > mirrors_sign = = SIGN_PLUS )
lp - > mirrors = existing_mirrors + lp - > mirrors ;
else if ( lp - > mirrors_sign = = SIGN_MINUS ) {
if ( lp - > mirrors > = existing_mirrors ) {
log_error ( " Logical volume %s only has % " PRIu32 " mirrors. " ,
lv - > name , existing_mirrors ) ;
return 0 ;
}
lp - > mirrors = existing_mirrors - lp - > mirrors ;
} else
lp - > mirrors + = 1 ;
if ( arg_count ( cmd , regionsize_ARG ) & & ( lv - > status & MIRRORED ) & &
( lp - > region_size ! = seg - > region_size ) ) {
log_error ( " Mirror log region size cannot be changed on "
" an existing mirror. " ) ;
return 0 ;
}
2005-06-06 21:12:08 +04:00
2005-08-15 18:10:28 +04:00
if ( ( lp - > mirrors = = 1 ) ) {
2005-06-06 21:12:08 +04:00
if ( ! ( lv - > status & MIRRORED ) ) {
log_error ( " Logical volume %s is already not mirrored. " ,
lv - > name ) ;
return 1 ;
}
2005-11-29 21:20:23 +03:00
if ( ! remove_mirror_images ( seg , 1 , lp - > pv_count ? lp - > pvh : NULL , 1 ) ) {
2005-06-06 21:12:08 +04:00
stack ;
return 0 ;
}
} else { /* mirrors > 1 */
if ( ( lv - > status & MIRRORED ) ) {
if ( list_size ( & lv - > segments ) ! = 1 ) {
log_error ( " Logical volume %s has multiple "
" mirror segments. " , lv - > name ) ;
return 0 ;
}
2005-08-15 18:10:28 +04:00
if ( lp - > mirrors = = existing_mirrors ) {
2005-06-06 21:12:08 +04:00
log_error ( " Logical volume %s already has % "
PRIu32 " mirror(s). " , lv - > name ,
2005-08-15 18:10:28 +04:00
lp - > mirrors - 1 ) ;
2005-06-06 21:12:08 +04:00
return 1 ;
}
2005-08-15 18:10:28 +04:00
if ( lp - > mirrors > existing_mirrors ) {
2005-06-06 21:12:08 +04:00
/* FIXME Unless anywhere, remove PV of log_lv
* from allocatable_pvs & allocate
* ( mirrors - existing_mirrors ) new areas
*/
/* FIXME Create mirror hierarchy to sync */
log_error ( " Adding mirror images is not "
" supported yet. " ) ;
return 0 ;
} else {
2005-11-29 21:20:23 +03:00
/* Reduce number of mirrors */
if ( ! remove_mirror_images ( seg , lp - > mirrors , lp - > pv_count ? lp - > pvh : NULL , 0 ) ) {
2005-06-06 21:12:08 +04:00
stack ;
return 0 ;
}
}
} else {
2005-11-29 21:20:23 +03:00
/* Make existing LV into mirror set */
2005-06-06 21:12:08 +04:00
/* FIXME Share code with lvcreate */
2005-11-29 21:20:23 +03:00
/* FIXME Why is this restriction here? Fix it! */
list_iterate_items ( seg , & lv - > segments ) {
if ( seg_is_striped ( seg ) & & seg - > area_count > 1 ) {
log_error ( " Mirrors of striped volumes are not yet supported. " ) ;
return 0 ;
}
}
if ( ! ( parallel_areas = build_parallel_areas_from_lv ( cmd , lv ) ) ) {
stack ;
return 0 ;
}
segtype = get_segtype_from_string ( cmd , " striped " ) ;
if ( ! ( ah = allocate_extents ( lv - > vg , NULL , segtype , 1 ,
lp - > mirrors - 1 , 1 ,
lv - > le_count * ( lp - > mirrors - 1 ) ,
NULL , 0 , 0 , lp - > pvh ,
lp - > alloc ,
parallel_areas ) ) ) {
stack ;
return 0 ;
}
lp - > region_size = adjusted_mirror_region_size ( lv - > vg - > extent_size ,
lv - > le_count ,
lp - > region_size ) ;
if ( ! ( log_lv = create_mirror_log ( cmd , lv - > vg , ah ,
lp - > alloc ,
lv - > name ) ) ) {
log_error ( " Failed to create mirror log. " ) ;
return 0 ;
}
if ( ! create_mirror_layers ( ah , 1 , lp - > mirrors , lv ,
segtype , 0 , lp - > region_size ,
log_lv ) ) {
stack ;
return 0 ;
}
2005-06-06 21:12:08 +04:00
}
}
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
if ( ! vg_write ( lv - > vg ) ) {
stack ;
return 0 ;
}
backup ( lv - > vg ) ;
2005-08-15 16:00:04 +04:00
if ( ! suspend_lv ( cmd , lv ) ) {
2005-06-06 21:12:08 +04:00
log_error ( " Failed to lock %s " , lv - > name ) ;
vg_revert ( lv - > vg ) ;
return 0 ;
}
if ( ! vg_commit ( lv - > vg ) ) {
2005-08-15 16:00:04 +04:00
resume_lv ( cmd , lv ) ;
2005-06-06 21:12:08 +04:00
return 0 ;
}
log_very_verbose ( " Updating \" %s \" in kernel " , lv - > name ) ;
2005-08-15 16:00:04 +04:00
if ( ! resume_lv ( cmd , lv ) ) {
2005-06-06 21:12:08 +04:00
log_error ( " Problem reactivating %s " , lv - > name ) ;
return 0 ;
}
log_print ( " Logical volume %s converted. " , lv - > name ) ;
return 1 ;
}
2005-08-15 18:10:28 +04:00
static int lvconvert_single ( struct cmd_context * cmd , struct logical_volume * lv ,
void * handle )
2005-06-06 21:12:08 +04:00
{
struct lvconvert_params * lp = handle ;
if ( lv - > status & LOCKED ) {
log_error ( " Cannot convert locked LV %s " , lv - > name ) ;
return ECMD_FAILED ;
}
if ( lv_is_origin ( lv ) ) {
log_error ( " Can't convert logical volume \" %s \" under snapshot " ,
lv - > name ) ;
return ECMD_FAILED ;
}
if ( lv_is_cow ( lv ) ) {
log_error ( " Can't convert snapshot logical volume \" %s \" " ,
lv - > name ) ;
return ECMD_FAILED ;
}
if ( lv - > status & PVMOVE ) {
log_error ( " Unable to convert pvmove LV %s " , lv - > name ) ;
return ECMD_FAILED ;
}
if ( arg_count ( cmd , mirrors_ARG ) ) {
if ( ! archive ( lv - > vg ) )
return ECMD_FAILED ;
2005-08-15 18:10:28 +04:00
if ( ! lvconvert_mirrors ( cmd , lv , lp ) )
2005-06-06 21:12:08 +04:00
return ECMD_FAILED ;
}
return ECMD_PROCESSED ;
}
int lvconvert ( struct cmd_context * cmd , int argc , char * * argv )
{
2005-08-15 18:10:28 +04:00
const char * vg_name ;
2005-06-06 21:12:08 +04:00
char * st ;
int consistent = 1 ;
struct volume_group * vg ;
struct lv_list * lvl ;
struct lvconvert_params lp ;
int ret = ECMD_FAILED ;
2005-08-15 18:10:28 +04:00
if ( ! _read_params ( & lp , cmd , argc , argv ) ) {
stack ;
2005-06-06 21:12:08 +04:00
return EINVALID_CMD_LINE ;
}
2005-08-15 18:10:28 +04:00
vg_name = extract_vgname ( cmd , lp . lv_name ) ;
2005-06-06 21:12:08 +04:00
if ( ! validate_name ( vg_name ) ) {
log_error ( " Please provide a valid volume group name " ) ;
return EINVALID_CMD_LINE ;
}
2005-08-15 18:10:28 +04:00
if ( ( st = strrchr ( lp . lv_name , ' / ' ) ) )
lp . lv_name = st + 1 ;
2005-06-06 21:12:08 +04:00
log_verbose ( " Checking for existing volume group \" %s \" " , vg_name ) ;
if ( ! lock_vol ( cmd , vg_name , LCK_VG_WRITE ) ) {
log_error ( " Can't get lock for %s " , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! ( vg = vg_read ( cmd , vg_name , & consistent ) ) ) {
log_error ( " Volume group \" %s \" doesn't exist " , vg_name ) ;
goto error ;
}
if ( vg - > status & EXPORTED_VG ) {
log_error ( " Volume group \" %s \" is exported " , vg_name ) ;
goto error ;
}
if ( ! ( vg - > status & LVM_WRITE ) ) {
log_error ( " Volume group \" %s \" is read-only " , vg_name ) ;
goto error ;
}
2005-08-15 18:10:28 +04:00
if ( ! ( lvl = find_lv_in_vg ( vg , lp . lv_name ) ) ) {
2005-06-06 21:12:08 +04:00
log_error ( " Logical volume \" %s \" not found in "
2005-08-15 18:10:28 +04:00
" volume group \" %s \" " , lp . lv_name , vg_name ) ;
2005-06-06 21:12:08 +04:00
goto error ;
}
2005-08-15 18:10:28 +04:00
if ( lp . pv_count ) {
if ( ! ( lp . pvh = create_pv_list ( cmd - > mem , vg , lp . pv_count ,
lp . pvs , 1 ) ) ) {
2005-06-06 21:12:08 +04:00
stack ;
goto error ;
}
} else
lp . pvh = & vg - > pvs ;
2005-08-15 18:10:28 +04:00
ret = lvconvert_single ( cmd , lvl - > lv , & lp ) ;
2005-06-06 21:12:08 +04:00
error :
unlock_vg ( cmd , vg_name ) ;
return ret ;
}