2001-10-12 01:35:55 +04:00
/*
2008-01-30 17:00:02 +03:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2009-02-09 12:45:49 +03:00
* Copyright ( C ) 2004 - 2009 Red Hat , Inc . All rights reserved .
2001-10-12 01:35:55 +04:00
*
2004-03-30 23:35:44 +04:00
* This file is part of LVM2 .
2001-10-12 01:35:55 +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-12 01:35:55 +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-12 01:35:55 +04:00
*/
# include "tools.h"
2008-09-19 10:42:00 +04:00
static int _remove_pv ( struct volume_group * vg , struct pv_list * pvl , int silent )
2003-01-18 00:04:26 +03:00
{
2010-07-09 19:34:40 +04:00
char uuid [ 64 ] __attribute__ ( ( aligned ( 8 ) ) ) ;
2003-01-18 00:04:26 +03:00
if ( vg - > pv_count = = 1 ) {
log_error ( " Volume Groups must always contain at least one PV " ) ;
return 0 ;
}
2008-01-30 16:19:47 +03:00
if ( ! id_write_format ( & pvl - > pv - > id , uuid , sizeof ( uuid ) ) )
return_0 ;
2003-01-18 00:04:26 +03:00
log_verbose ( " Removing PV with UUID %s from VG %s " , uuid , vg - > name ) ;
if ( pvl - > pv - > pe_alloc_count ) {
2008-09-19 10:42:00 +04:00
if ( ! silent )
log_error ( " LVs still present on PV with UUID %s: "
" Can't remove from VG %s " , uuid , vg - > name ) ;
2003-01-18 00:04:26 +03:00
return 0 ;
}
vg - > free_count - = pvl - > pv - > pe_count ;
vg - > extent_count - = pvl - > pv - > pe_count ;
2010-04-13 21:26:03 +04:00
del_pvl_from_vgs ( vg , pvl ) ;
2011-03-11 17:56:56 +03:00
free_pv_fid ( pvl - > pv ) ;
2003-01-18 00:04:26 +03:00
return 1 ;
}
2008-09-19 10:42:00 +04:00
static int _consolidate_vg ( struct cmd_context * cmd , struct volume_group * vg )
{
struct pv_list * pvl ;
struct lv_list * lvl ;
int r = 1 ;
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( lvl , & vg - > lvs )
2008-09-19 10:42:00 +04:00
if ( lvl - > lv - > status & PARTIAL_LV ) {
log_warn ( " WARNING: Partial LV %s needs to be repaired "
" or removed. " , lvl - > lv - > name ) ;
r = 0 ;
}
if ( ! r ) {
cmd - > handles_missing_pvs = 1 ;
2012-02-15 16:30:46 +04:00
log_error ( " There are still partial LVs in VG %s. " , vg - > name ) ;
log_error ( " To remove them unconditionally use: vgreduce --removemissing --force. " ) ;
2008-09-19 10:42:00 +04:00
log_warn ( " Proceeding to remove empty missing PVs. " ) ;
}
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( pvl , & vg - > pvs ) {
2010-03-16 17:37:38 +03:00
if ( pvl - > pv - > dev & & ! is_missing_pv ( pvl - > pv ) )
2008-09-19 10:42:00 +04:00
continue ;
if ( r & & ! _remove_pv ( vg , pvl , 0 ) )
return_0 ;
}
return r ;
}
2003-01-18 00:04:26 +03:00
static int _make_vg_consistent ( struct cmd_context * cmd , struct volume_group * vg )
{
2011-05-07 19:52:16 +04:00
struct lv_list * lvl ;
2003-01-18 00:04:26 +03:00
struct logical_volume * lv ;
2005-12-21 21:51:50 +03:00
2011-05-07 19:52:16 +04:00
cmd - > partial_activation = 1 ;
2005-12-21 21:51:50 +03:00
2011-05-07 19:52:16 +04:00
restart :
vg_mark_partial_lvs ( vg , 1 ) ;
2006-05-11 23:01:11 +04:00
2011-05-07 19:52:16 +04:00
dm_list_iterate_items ( lvl , & vg - > lvs ) {
lv = lvl - > lv ;
2006-05-11 23:01:11 +04:00
2011-05-07 19:52:16 +04:00
/* Are any segments of this LV on missing PVs? */
if ( lv - > status & PARTIAL_LV ) {
2013-02-21 00:52:46 +04:00
if ( seg_is_raid ( first_seg ( lv ) ) ) {
if ( ! lv_raid_remove_missing ( lv ) )
return_0 ;
goto restart ;
}
2014-09-16 03:13:46 +04:00
if ( lv_is_mirror ( lv ) ) {
2011-05-07 19:52:16 +04:00
if ( ! mirror_remove_missing ( cmd , lv , 1 ) )
2008-01-30 16:19:47 +03:00
return_0 ;
2011-05-07 19:52:16 +04:00
goto restart ;
2006-01-04 21:09:52 +03:00
}
2005-12-21 21:51:50 +03:00
2014-09-16 00:33:53 +04:00
if ( arg_count ( cmd , mirrorsonly_ARG ) & & ! lv_is_mirrored ( lv ) ) {
2011-05-07 19:52:16 +04:00
log_error ( " Non-mirror-image LV %s found: can't remove. " , lv - > name ) ;
continue ;
2005-12-21 21:51:50 +03:00
}
2011-05-07 19:52:16 +04:00
if ( ! lv_is_visible ( lv ) )
continue ;
log_warn ( " Removing partial LV %s. " , lv - > name ) ;
2012-02-27 14:06:58 +04:00
if ( ! lv_remove_with_dependencies ( cmd , lv , DONT_PROMPT , 0 ) )
2011-05-07 19:52:16 +04:00
return_0 ;
goto restart ;
2005-12-21 21:51:50 +03:00
}
}
2011-05-07 19:52:16 +04:00
_consolidate_vg ( cmd , vg ) ;
2003-01-18 00:04:26 +03:00
return 1 ;
}
2002-11-18 17:04:08 +03:00
/* Or take pv_name instead? */
2003-01-18 00:04:26 +03:00
static int _vgreduce_single ( struct cmd_context * cmd , struct volume_group * vg ,
2006-05-10 01:23:51 +04:00
struct physical_volume * pv ,
2014-11-27 17:02:13 +03:00
struct processing_handle * handle __attribute__ ( ( unused ) ) )
2002-11-18 17:04:08 +03:00
{
2014-10-07 03:53:56 +04:00
int r ;
2012-06-21 14:43:31 +04:00
2014-10-07 03:53:56 +04:00
if ( ! vg_check_status ( vg , EXPORTED_VG | LVM_WRITE | RESIZEABLE_VG ) )
return ECMD_FAILED ;
r = vgreduce_single ( cmd , vg , pv , 1 ) ;
2013-09-04 02:31:45 +04:00
if ( ! r )
2002-11-18 17:04:08 +03:00
return ECMD_FAILED ;
2013-09-04 02:31:45 +04:00
return ECMD_PROCESSED ;
}
2002-11-18 17:04:08 +03:00
2002-02-11 23:50:53 +03:00
int vgreduce ( struct cmd_context * cmd , int argc , char * * argv )
2001-10-12 01:35:55 +04:00
{
struct volume_group * vg ;
2011-02-18 17:47:28 +03:00
const char * vg_name ;
2015-07-08 15:53:23 +03:00
uint32_t lockd_state = 0 ;
2009-04-10 14:01:38 +04:00
int ret = ECMD_FAILED ;
2008-09-19 10:42:00 +04:00
int fixed = 1 ;
int repairing = arg_count ( cmd , removemissing_ARG ) ;
2009-06-15 18:47:39 +04:00
int saved_ignore_suspended_devices = ignore_suspended_devices ( ) ;
2012-03-30 18:59:35 +04:00
int locked = 0 ;
2001-10-12 01:35:55 +04:00
2008-09-19 10:42:00 +04:00
if ( ! argc & & ! repairing ) {
2001-10-12 01:35:55 +04:00
log_error ( " Please give volume group name and "
" physical volume paths " ) ;
return EINVALID_CMD_LINE ;
}
2008-09-19 10:42:00 +04:00
2012-02-08 15:41:18 +04:00
if ( ! argc ) { /* repairing */
2003-01-18 00:04:26 +03:00
log_error ( " Please give volume group name " ) ;
return EINVALID_CMD_LINE ;
}
2008-09-19 10:42:00 +04:00
if ( arg_count ( cmd , mirrorsonly_ARG ) & & ! repairing ) {
2005-12-22 00:21:45 +03:00
log_error ( " --mirrorsonly requires --removemissing " ) ;
return EINVALID_CMD_LINE ;
}
2008-09-19 10:42:00 +04:00
if ( argc = = 1 & & ! arg_count ( cmd , all_ARG ) & & ! repairing ) {
2001-10-12 01:35:55 +04:00
log_error ( " Please enter physical volume paths or option -a " ) ;
return EINVALID_CMD_LINE ;
}
2002-02-12 00:00:35 +03:00
if ( argc > 1 & & arg_count ( cmd , all_ARG ) ) {
2001-10-12 01:35:55 +04:00
log_error ( " Option -a and physical volume paths mutually "
" exclusive " ) ;
return EINVALID_CMD_LINE ;
}
2008-09-19 10:42:00 +04:00
if ( argc > 1 & & repairing ) {
2003-01-18 00:04:26 +03:00
log_error ( " Please only specify the volume group " ) ;
return EINVALID_CMD_LINE ;
}
2007-03-09 23:47:41 +03:00
vg_name = skip_dev_dir ( cmd , argv [ 0 ] , NULL ) ;
2001-10-12 01:35:55 +04:00
argv + + ;
argc - - ;
2014-10-07 03:53:56 +04:00
if ( ! repairing )
/* FIXME: Pass private struct through to all these functions */
/* and update in batch afterwards? */
return process_each_pv ( cmd , argc , argv , vg_name ,
READ_FOR_UPDATE , NULL ,
_vgreduce_single ) ;
2002-01-30 18:04:48 +03:00
log_verbose ( " Finding volume group \" %s \" " , vg_name ) ;
2002-02-11 18:42:34 +03:00
2014-10-07 03:53:56 +04:00
init_ignore_suspended_devices ( 1 ) ;
cmd - > handles_missing_pvs = 1 ;
2009-06-15 18:47:39 +04:00
2015-03-05 23:00:44 +03:00
/* Needed to change the set of orphan PVs. */
if ( ! lockd_gl ( cmd , " ex " , 0 ) )
return_ECMD_FAILED ;
if ( ! lockd_vg ( cmd , vg_name , " ex " , 0 , & lockd_state ) )
return_ECMD_FAILED ;
vg = vg_read_for_update ( cmd , vg_name , NULL , READ_ALLOW_EXPORTED , lockd_state ) ;
2009-07-01 21:01:46 +04:00
if ( vg_read_error ( vg ) = = FAILED_ALLOCATION | |
vg_read_error ( vg ) = = FAILED_NOTFOUND )
2009-09-15 02:47:49 +04:00
goto_out ;
2001-10-12 01:35:55 +04:00
2009-07-01 21:01:46 +04:00
/* FIXME We want to allow read-only VGs to be changed here? */
2014-10-07 03:53:56 +04:00
if ( vg_read_error ( vg ) & &
( vg_read_error ( vg ) ! = FAILED_READ_ONLY ) & &
! arg_count ( cmd , removemissing_ARG ) )
2009-09-15 02:47:49 +04:00
goto_out ;
2006-09-02 05:18:17 +04:00
2012-03-30 18:59:35 +04:00
locked = ! vg_read_error ( vg ) ;
2014-10-07 03:53:56 +04:00
if ( ! vg_read_error ( vg ) & & ! vg_missing_pv_count ( vg ) ) {
log_error ( " Volume group \" %s \" is already consistent " , vg_name ) ;
ret = ECMD_PROCESSED ;
goto out ;
}
2002-01-29 20:23:33 +03:00
2014-10-07 03:53:56 +04:00
release_vg ( vg ) ;
log_verbose ( " Trying to open VG %s for recovery... " , vg_name ) ;
2009-07-01 21:01:46 +04:00
2014-10-07 03:53:56 +04:00
vg = vg_read_for_update ( cmd , vg_name , NULL ,
2015-03-05 23:00:44 +03:00
READ_ALLOW_INCONSISTENT | READ_ALLOW_EXPORTED , lockd_state ) ;
2009-07-01 21:01:46 +04:00
2014-10-07 03:53:56 +04:00
locked | = ! vg_read_error ( vg ) ;
2009-07-01 21:01:46 +04:00
2014-10-07 03:53:56 +04:00
if ( vg_read_error ( vg ) & &
( vg_read_error ( vg ) ! = FAILED_READ_ONLY ) & &
( vg_read_error ( vg ) ! = FAILED_INCONSISTENT ) )
goto_out ;
2002-01-29 20:23:33 +03:00
2014-10-07 03:53:56 +04:00
if ( ! archive ( vg ) )
goto_out ;
2003-01-18 00:04:26 +03:00
2014-10-07 03:53:56 +04:00
if ( arg_count ( cmd , force_ARG ) ) {
if ( ! _make_vg_consistent ( cmd , vg ) )
goto_out ;
} else
fixed = _consolidate_vg ( cmd , vg ) ;
2003-01-18 00:04:26 +03:00
2014-10-07 03:53:56 +04:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) ) {
log_error ( " Failed to write out a consistent VG for %s " , vg_name ) ;
goto out ;
}
2003-01-18 00:04:26 +03:00
2014-10-07 03:53:56 +04:00
backup ( vg ) ;
2003-01-18 00:04:26 +03:00
2014-10-07 03:53:56 +04:00
if ( fixed ) {
log_print_unless_silent ( " Wrote out consistent volume group %s " , vg_name ) ;
ret = ECMD_PROCESSED ;
} else
ret = ECMD_FAILED ;
2003-01-18 00:04:26 +03:00
2009-04-10 14:01:38 +04:00
out :
2009-06-15 18:47:39 +04:00
init_ignore_suspended_devices ( saved_ignore_suspended_devices ) ;
2012-03-30 18:59:35 +04:00
if ( locked )
unlock_vg ( cmd , vg_name ) ;
release_vg ( vg ) ;
2002-02-11 18:42:34 +03:00
return ret ;
2001-10-12 01:35:55 +04:00
/******* FIXME
log_error ( " no empty physical volumes found in volume group \" %s \" " , vg_name ) ;
2002-01-21 19:05:23 +03:00
2014-10-07 03:53:56 +04:00
log_verbose ( " volume group \" %s \" will be reduced by %d physical volume%s " ,
vg_name , np , np > 1 ? " s " : " " ) ;
log_verbose ( " reducing volume group \" %s \" by physical volume \" %s \" " ,
vg_name , pv_names [ p ] ) ;
2001-10-12 01:35:55 +04:00
2014-10-07 03:53:56 +04:00
log_print ( " volume group \" %s \" %ssuccessfully reduced by physical volume%s: " ,
vg_name , error > 0 ? " NOT " : " " , p > 1 ? " s " : " " ) ;
log_print ( " %s " , pv_this [ p ] - > pv_name ) ;
2001-10-12 01:35:55 +04:00
* * * * * * * */
}