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