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"
struct lvconvert_params {
2005-08-15 18:10:28 +04:00
const char * lv_name ;
uint32_t mirrors ;
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 ;
}
lp - > mirrors = arg_uint_value ( cmd , mirrors_ARG , 0 ) + 1 ;
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-06-06 21:12:08 +04:00
// struct alloc_handle *ah = NULL;
// struct logical_volume *log_lv;
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 ;
}
/* FIXME If allocatable_pvs supplied only remove those */
if ( ! remove_all_mirror_images ( lv ) ) {
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-10-28 16:48:50 +04:00
seg = first_seg ( lv ) ;
existing_mirrors = seg - > area_count ;
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-10-28 16:48:50 +04:00
if ( ! remove_mirror_images ( seg , lp - > mirrors ) ) {
2005-06-06 21:12:08 +04:00
stack ;
return 0 ;
}
}
} else {
/* FIXME Share code with lvcreate */
/* region size, log_name, create log_lv, zero it */
// Allocate (mirrors) new areas & log - replace mirrored_pv with mirrored_lv
// Restructure as mirror - add existing param to create_mirror_layers
log_error ( " Adding mirror images is not supported yet. " ) ;
return 0 ;
}
}
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 ;
}