2001-10-16 02:04:27 +04:00
/*
2008-01-30 17:00:02 +03:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2009-05-13 17:02:52 +04:00
* Copyright ( C ) 2004 - 2009 Red Hat , Inc . All rights reserved .
2001-10-16 02:04:27 +04:00
*
2004-03-30 23:35:44 +04:00
* This file is part of LVM2 .
2001-10-16 02:04:27 +04:00
*
2004-03-30 23:35:44 +04:00
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-21 00:55:30 +04:00
* of the GNU Lesser General Public License v .2 .1 .
2001-10-16 02:04:27 +04:00
*
2007-08-21 00:55:30 +04:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 23:35:44 +04:00
* along with this program ; if not , write to the Free Software Foundation ,
2016-01-21 13:49:46 +03:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2001-10-16 02:04:27 +04:00
*/
# include "tools.h"
2015-02-13 17:58:51 +03:00
struct vgextend_params {
2016-02-19 00:38:23 +03:00
struct pvcreate_params pp ;
2015-02-13 17:58:51 +03:00
} ;
static int _restore_pv ( struct volume_group * vg , const char * pv_name )
2010-10-13 14:34:31 +04:00
{
struct pv_list * pvl = NULL ;
pvl = find_pv_in_vg ( vg , pv_name ) ;
if ( ! pvl ) {
log_warn ( " WARNING: PV %s not found in VG %s " , pv_name , vg - > name ) ;
return 0 ;
}
if ( ! ( pvl - > pv - > status & MISSING_PV ) ) {
log_warn ( " WARNING: PV %s was not missing in VG %s " , pv_name , vg - > name ) ;
return 0 ;
}
if ( ! pvl - > pv - > dev ) {
log_warn ( " WARNING: The PV %s is still missing. " , pv_name ) ;
return 0 ;
}
pvl - > pv - > status & = ~ MISSING_PV ;
return 1 ;
}
2015-02-13 17:58:51 +03:00
static int _vgextend_restoremissing ( struct cmd_context * cmd __attribute__ ( ( unused ) ) ,
const char * vg_name , struct volume_group * vg ,
struct processing_handle * handle )
{
struct vgextend_params * vp = ( struct vgextend_params * ) handle - > custom_handle ;
2016-02-19 00:38:23 +03:00
struct pvcreate_params * pp = & vp - > pp ;
2015-02-13 17:58:51 +03:00
int fixed = 0 ;
int i ;
2015-12-03 20:15:02 +03:00
if ( ! archive ( vg ) )
return_0 ;
2016-02-15 23:28:09 +03:00
for ( i = 0 ; i < pp - > pv_count ; i + + )
if ( _restore_pv ( vg , pp - > pv_names [ i ] ) )
2015-02-13 17:58:51 +03:00
fixed + + ;
if ( ! fixed ) {
log_error ( " No PV has been restored. " ) ;
return ECMD_FAILED ;
}
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) )
return_ECMD_FAILED ;
backup ( vg ) ;
log_print_unless_silent ( " Volume group \" %s \" successfully extended " , vg_name ) ;
return ECMD_PROCESSED ;
}
static int _vgextend_single ( struct cmd_context * cmd , const char * vg_name ,
struct volume_group * vg , struct processing_handle * handle )
{
struct vgextend_params * vp = ( struct vgextend_params * ) handle - > custom_handle ;
2016-02-19 00:38:23 +03:00
struct pvcreate_params * pp = & vp - > pp ;
2015-02-13 17:58:51 +03:00
uint32_t mda_copies ;
uint32_t mda_used ;
int ret = ECMD_FAILED ;
if ( arg_count ( cmd , metadataignore_ARG ) & &
( pp - > force = = PROMPT ) & & ! pp - > yes & &
( vg_mda_copies ( vg ) ! = VGMETADATACOPIES_UNMANAGED ) & &
( yes_no_prompt ( " Override preferred number of copies of VG %s metadata? [y/n]: " , vg_name ) = = ' n ' ) ) {
log_error ( " Volume group %s not changed " , vg_name ) ;
return ECMD_FAILED ;
}
2015-12-03 20:15:02 +03:00
if ( ! archive ( vg ) )
return_ECMD_FAILED ;
2016-02-15 23:28:09 +03:00
if ( ! vg_extend_each_pv ( vg , pp ) )
2015-02-13 17:58:51 +03:00
goto_out ;
if ( arg_count ( cmd , metadataignore_ARG ) ) {
mda_copies = vg_mda_copies ( vg ) ;
mda_used = vg_mda_used_count ( vg ) ;
if ( ( mda_copies ! = VGMETADATACOPIES_UNMANAGED ) & &
( mda_copies ! = mda_used ) ) {
log_warn ( " WARNING: Changing preferred number of copies of VG %s metadata from % " PRIu32 " to % " PRIu32 ,
vg_name , mda_copies , mda_used ) ;
vg_set_mda_copies ( vg , mda_used ) ;
}
}
2016-02-15 23:28:09 +03:00
log_verbose ( " Volume group \" %s \" will be extended by %d new physical volumes " , vg_name , pp - > pv_count ) ;
2015-02-13 17:58:51 +03:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) )
goto_out ;
backup ( vg ) ;
log_print_unless_silent ( " Volume group \" %s \" successfully extended " , vg_name ) ;
ret = ECMD_PROCESSED ;
out :
return ret ;
}
2002-02-11 23:50:53 +03:00
int vgextend ( struct cmd_context * cmd , int argc , char * * argv )
2001-10-16 02:04:27 +04:00
{
2016-02-15 23:28:09 +03:00
struct processing_handle * handle ;
2015-02-13 17:58:51 +03:00
struct vgextend_params vp ;
2016-02-19 00:38:23 +03:00
struct pvcreate_params * pp = & vp . pp ;
2015-02-13 17:58:51 +03:00
unsigned restoremissing = arg_is_set ( cmd , restoremissing_ARG ) ;
2016-02-15 23:28:09 +03:00
const char * vg_name ;
2015-02-13 17:58:51 +03:00
int ret ;
2001-10-16 02:04:27 +04:00
if ( ! argc ) {
log_error ( " Please enter volume group name and "
2002-02-12 00:00:35 +03:00
" physical volume(s) " ) ;
2001-10-16 02:04:27 +04:00
return EINVALID_CMD_LINE ;
}
2009-10-06 00:55:56 +04:00
if ( arg_count ( cmd , metadatacopies_ARG ) ) {
log_error ( " Invalid option --metadatacopies, "
" use --pvmetadatacopies instead. " ) ;
return EINVALID_CMD_LINE ;
}
2015-02-13 17:58:51 +03:00
2016-02-15 23:28:09 +03:00
vg_name = skip_dev_dir ( cmd , argv [ 0 ] , NULL ) ;
argc - - ;
argv + + ;
2015-02-13 17:58:51 +03:00
2016-02-19 00:38:23 +03:00
pvcreate_params_set_defaults ( pp ) ;
2015-02-13 17:58:51 +03:00
2016-02-19 00:38:23 +03:00
if ( ! pvcreate_params_from_args ( cmd , pp ) )
2016-02-15 23:28:09 +03:00
return EINVALID_CMD_LINE ;
2015-02-13 17:58:51 +03:00
2016-02-15 23:28:09 +03:00
pp - > pv_count = argc ;
pp - > pv_names = argv ;
/* Don't create a new PV on top of an existing PV like pvcreate does. */
pp - > preserve_existing = 1 ;
/* pvcreate within vgextend cannot be forced. */
pp - > force = 0 ;
/*
* Needed to change the set of orphan PVs .
* ( disable afterward to prevent process_each_pv from doing
* a shared global lock since it ' s already acquired it ex . )
*/
if ( ! lockd_gl ( cmd , " ex " , 0 ) )
return_ECMD_FAILED ;
cmd - > lockd_gl_disable = 1 ;
2016-05-31 13:24:05 +03:00
if ( ! ( handle = init_processing_handle ( cmd , NULL ) ) ) {
2016-02-15 23:28:09 +03:00
log_error ( " Failed to initialize processing handle. " ) ;
return ECMD_FAILED ;
}
if ( ! restoremissing ) {
if ( ! pvcreate_each_device ( cmd , handle , pp ) ) {
destroy_processing_handle ( cmd , handle ) ;
return_ECMD_FAILED ;
}
}
/*
* pvcreate_each_device returns with the VG_ORPHANS write lock held ,
* which was used to do pvcreate . Now to create the VG using those
* PVs , the VG lock will be taken ( with the orphan lock already held . )
*/
2009-10-06 00:03:37 +04:00
2012-07-27 02:06:06 +04:00
/*
* It is always ok to add new PVs to a VG - even if there are
* missing PVs . No LVs are affected by this operation , but
* repair processes - particularly for RAID segtypes - can
* be facilitated .
*/
cmd - > handles_missing_pvs = 1 ;
2010-10-13 14:34:31 +04:00
2016-02-15 23:28:09 +03:00
handle - > custom_handle = & vp ;
2015-03-05 23:00:44 +03:00
2016-04-29 00:18:20 +03:00
ret = process_each_vg ( cmd , 0 , NULL , vg_name , NULL ,
2016-05-03 12:46:28 +03:00
READ_FOR_UPDATE , 0 , handle ,
2015-02-13 17:58:51 +03:00
restoremissing ? & _vgextend_restoremissing : & _vgextend_single ) ;
2010-07-07 23:02:50 +04:00
2015-02-13 17:58:51 +03:00
destroy_processing_handle ( cmd , handle ) ;
2001-10-16 02:04:27 +04:00
2016-02-15 23:28:09 +03:00
if ( ! restoremissing )
unlock_vg ( cmd , VG_ORPHANS ) ;
2015-02-13 17:58:51 +03:00
return ret ;
2001-10-16 02:04:27 +04:00
}