2001-10-11 21:35:55 +00:00
/*
2004-03-30 19:35:44 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 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
* of the GNU General Public License v .2 .
2001-10-11 21:35:55 +00:00
*
* You should have received a copy of the GNU General Public License
2004-03-30 19:35:44 +00:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-10-11 21:35:55 +00:00
*/
# include "tools.h"
2003-01-17 21:04:26 +00:00
static int _remove_pv ( struct volume_group * vg , struct pv_list * pvl )
{
char uuid [ 64 ] ;
if ( vg - > pv_count = = 1 ) {
log_error ( " Volume Groups must always contain at least one PV " ) ;
return 0 ;
}
if ( ! id_write_format ( & pvl - > pv - > id , uuid , sizeof ( uuid ) ) ) {
stack ;
return 0 ;
}
log_verbose ( " Removing PV with UUID %s from VG %s " , uuid , vg - > name ) ;
if ( pvl - > pv - > pe_alloc_count ) {
log_error ( " LVs still present on PV with UUID %s: Can't remove "
" from VG %s " , uuid , vg - > name ) ;
return 0 ;
}
vg - > free_count - = pvl - > pv - > pe_count ;
vg - > extent_count - = pvl - > pv - > pe_count ;
vg - > pv_count - - ;
list_del ( & pvl - > list ) ;
return 1 ;
}
static int _remove_lv ( struct cmd_context * cmd , struct logical_volume * lv ,
int * list_unsafe )
{
2005-04-07 12:39:44 +00:00
struct lv_segment * snap_seg ;
struct list * snh , * snht ;
struct logical_volume * cow ;
2003-01-17 21:04:26 +00:00
log_verbose ( " %s/%s has missing extents: removing (including "
" dependencies) " , lv - > vg - > name , lv - > name ) ;
/* Deactivate if necessary */
if ( ! lv_is_cow ( lv ) ) {
log_verbose ( " Deactivating (if active) logical volume %s " ,
lv - > name ) ;
2004-05-05 12:03:07 +00:00
if ( ! deactivate_lv ( cmd , lv - > lvid . s ) ) {
2003-01-17 21:04:26 +00:00
log_error ( " Failed to deactivate LV %s " , lv - > name ) ;
return 0 ;
}
2005-04-07 12:39:44 +00:00
} else if ( ( snap_seg = find_cow ( lv ) ) ) {
2003-01-17 21:04:26 +00:00
log_verbose ( " Deactivating (if active) logical volume %s "
2005-04-07 12:39:44 +00:00
" (origin of %s) " , snap_seg - > origin - > name , lv - > name ) ;
2003-01-17 21:04:26 +00:00
2005-04-07 12:39:44 +00:00
if ( ! deactivate_lv ( cmd , snap_seg - > origin - > lvid . s ) ) {
2003-01-17 21:04:26 +00:00
log_error ( " Failed to deactivate LV %s " ,
2005-04-07 12:39:44 +00:00
snap_seg - > origin - > name ) ;
2003-01-17 21:04:26 +00:00
return 0 ;
}
/* Use the origin LV */
2005-04-07 12:39:44 +00:00
lv = snap_seg - > origin ;
2003-01-17 21:04:26 +00:00
}
/* Remove snapshot dependencies */
2005-04-07 12:39:44 +00:00
list_iterate_safe ( snh , snht , & lv - > snapshot_segs ) {
snap_seg = list_struct_base ( snh , struct lv_segment ,
origin_list ) ;
cow = snap_seg - > cow ;
2003-01-17 21:04:26 +00:00
* list_unsafe = 1 ; /* May remove caller's lvht! */
2005-04-07 12:39:44 +00:00
if ( ! vg_remove_snapshot ( lv - > vg , cow ) ) {
2003-01-17 21:04:26 +00:00
stack ;
return 0 ;
}
2005-04-07 12:39:44 +00:00
log_verbose ( " Removing LV %s from VG %s " , cow - > name ,
2003-01-17 21:04:26 +00:00
lv - > vg - > name ) ;
2005-04-07 12:39:44 +00:00
if ( ! lv_remove ( lv - > vg , cow ) ) {
2003-01-17 21:04:26 +00:00
stack ;
return 0 ;
}
}
/* Remove the LV itself */
log_verbose ( " Removing LV %s from VG %s " , lv - > name , lv - > vg - > name ) ;
if ( ! lv_remove ( lv - > vg , lv ) ) {
stack ;
return 0 ;
}
return 1 ;
}
static int _make_vg_consistent ( struct cmd_context * cmd , struct volume_group * vg )
{
struct list * pvh , * pvht ;
struct list * lvh , * lvht ;
struct pv_list * pvl ;
struct logical_volume * lv ;
struct physical_volume * pv ;
struct lv_segment * seg ;
unsigned int s ;
int list_unsafe ;
/* Deactivate & remove necessary LVs */
restart_loop :
list_unsafe = 0 ; /* Set if we delete a different list-member */
list_iterate_safe ( lvh , lvht , & vg - > lvs ) {
lv = list_item ( lvh , struct lv_list ) - > lv ;
/* Are any segments of this LV on missing PVs? */
2003-10-15 20:02:46 +00:00
list_iterate_items ( seg , & lv - > segments ) {
2003-04-24 22:23:24 +00:00
for ( s = 0 ; s < seg - > area_count ; s + + ) {
if ( seg - > area [ s ] . type ! = AREA_PV )
continue ;
/* FIXME Also check for segs on deleted LVs */
pv = seg - > area [ s ] . u . pv . pv ;
2003-01-17 21:04:26 +00:00
if ( ! pv | | ! pv - > dev ) {
if ( ! _remove_lv ( cmd , lv , & list_unsafe ) ) {
stack ;
return 0 ;
}
if ( list_unsafe )
goto restart_loop ;
}
}
}
}
/* Remove missing PVs */
list_iterate_safe ( pvh , pvht , & vg - > pvs ) {
pvl = list_item ( pvh , struct pv_list ) ;
if ( pvl - > pv - > dev )
continue ;
if ( ! _remove_pv ( vg , pvl ) ) {
stack ;
return 0 ;
}
}
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 ,
struct physical_volume * pv , void * handle )
2002-11-18 14:04:08 +00:00
{
struct pv_list * pvl ;
const char * name = dev_name ( pv - > dev ) ;
if ( pv - > pe_alloc_count ) {
log_error ( " Physical volume \" %s \" still in use " , name ) ;
return ECMD_FAILED ;
}
if ( vg - > pv_count = = 1 ) {
log_error ( " Can't remove final physical volume \" %s \" from "
" volume group \" %s \" " , name , vg - > name ) ;
return ECMD_FAILED ;
}
pvl = find_pv_in_vg ( vg , name ) ;
if ( ! archive ( vg ) )
return ECMD_FAILED ;
2001-10-11 21:35:55 +00:00
2002-11-18 14:04:08 +00:00
log_verbose ( " Removing \" %s \" from volume group \" %s \" " , name , vg - > name ) ;
if ( pvl )
list_del ( & pvl - > list ) ;
2002-12-19 23:25:55 +00:00
pv - > vg_name = ORPHAN ;
2002-11-18 14:04:08 +00:00
vg - > pv_count - - ;
vg - > free_count - = pv - > pe_count - pv - > pe_alloc_count ;
vg - > extent_count - = pv - > pe_count ;
2003-07-04 22:34:56 +00:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Removal of physical volume \" %s \" from "
" \" %s \" failed " , name , vg - > name ) ;
return ECMD_FAILED ;
}
2003-03-24 18:08:53 +00:00
if ( ! pv_write ( cmd , pv , NULL , INT64_C ( - 1 ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Failed to clear metadata from physical "
" volume \" %s \" "
" after removal from \" %s \" " , name , vg - > name ) ;
return ECMD_FAILED ;
}
backup ( vg ) ;
log_print ( " Removed \" %s \" from volume group \" %s \" " , name , vg - > name ) ;
2003-10-21 22:06:07 +00:00
return ECMD_PROCESSED ;
2002-11-18 14:04:08 +00:00
}
2003-01-17 21:04:26 +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 ;
char * vg_name ;
2003-01-17 21:04:26 +00:00
int ret = 1 ;
2002-11-18 14:04:08 +00:00
int consistent = 1 ;
2001-10-11 21:35:55 +00:00
2003-01-17 21:04:26 +00:00
if ( ! argc & ! arg_count ( cmd , removemissing_ARG ) ) {
2001-10-11 21:35:55 +00:00
log_error ( " Please give volume group name and "
" physical volume paths " ) ;
return EINVALID_CMD_LINE ;
}
2003-01-17 21:04:26 +00:00
if ( ! argc & arg_count ( cmd , removemissing_ARG ) ) {
log_error ( " Please give volume group name " ) ;
return EINVALID_CMD_LINE ;
}
if ( argc = = 1 & & ! arg_count ( cmd , all_ARG )
& & ! arg_count ( cmd , removemissing_ARG ) ) {
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 ;
}
2003-01-17 21:04:26 +00:00
if ( argc > 1 & & arg_count ( cmd , removemissing_ARG ) ) {
log_error ( " Please only specify the volume group " ) ;
return EINVALID_CMD_LINE ;
}
2001-10-11 21:35:55 +00:00
vg_name = argv [ 0 ] ;
argv + + ;
argc - - ;
2002-01-30 15:04:48 +00:00
log_verbose ( " Finding volume group \" %s \" " , vg_name ) ;
2002-02-27 12:26:41 +00:00
if ( ! lock_vol ( cmd , vg_name , LCK_VG_WRITE ) ) {
2002-02-11 15:42:34 +00:00
log_error ( " Can't get lock for %s " , vg_name ) ;
return ECMD_FAILED ;
}
2003-01-17 21:04:26 +00:00
if ( ( ! ( vg = vg_read ( cmd , vg_name , & consistent ) ) | | ! consistent ) & &
! arg_count ( cmd , removemissing_ARG ) ) {
2002-01-30 15:04:48 +00:00
log_error ( " Volume group \" %s \" doesn't exist " , vg_name ) ;
2002-03-11 15:08:39 +00:00
unlock_vg ( cmd , vg_name ) ;
2001-10-11 21:35:55 +00:00
return ECMD_FAILED ;
}
2003-01-17 21:04:26 +00:00
if ( arg_count ( cmd , removemissing_ARG ) ) {
if ( vg & & consistent ) {
log_error ( " Volume group \" %s \" is already consistent " ,
vg_name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
2002-01-29 17:23:33 +00:00
2003-01-17 21:04:26 +00:00
init_partial ( 1 ) ;
consistent = 0 ;
if ( ! ( vg = vg_read ( cmd , vg_name , & consistent ) ) ) {
log_error ( " Volume group \" %s \" not found " , vg_name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! archive ( vg ) ) {
init_partial ( 0 ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
2002-01-29 17:23:33 +00:00
2003-01-17 21:04:26 +00:00
if ( ! _make_vg_consistent ( cmd , vg ) ) {
init_partial ( 0 ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
init_partial ( 0 ) ;
vg - > status & = ~ PARTIAL_VG ;
vg - > status | = LVM_WRITE ;
2001-10-11 21:35:55 +00:00
2003-07-04 22:34:56 +00:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) ) {
2003-01-17 21:04:26 +00:00
log_error ( " Failed to write out a consistent VG for %s " ,
vg_name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
backup ( vg ) ;
log_print ( " Wrote out consistent volume group %s " , vg_name ) ;
} else {
if ( vg - > status & EXPORTED_VG ) {
log_error ( " Volume group \" %s \" is exported " , vg - > name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! ( vg - > status & LVM_WRITE ) ) {
log_error ( " Volume group \" %s \" is read-only " , vg_name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! ( vg - > status & RESIZEABLE_VG ) ) {
log_error ( " Volume group \" %s \" is not reducible " ,
vg_name ) ;
unlock_vg ( cmd , vg_name ) ;
return ECMD_FAILED ;
}
/* FIXME: Pass private struct through to all these functions */
/* and update in batch here? */
ret = process_each_pv ( cmd , argc , argv , vg , NULL ,
_vgreduce_single ) ;
}
2002-02-11 15:42:34 +00:00
2002-03-11 15:08:39 +00:00
unlock_vg ( cmd , vg_name ) ;
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
2001-10-11 21:35:55 +00:00
log_verbose
( " volume group \" %s \" will be reduced by %d physical volume%s " ,
vg_name , np , np > 1 ? " s " : " " ) ;
2002-01-30 15:04:48 +00:00
log_verbose ( " reducing volume group \" %s \" by physical volume \" %s \" " ,
vg_name , pv_names [ p ] ) ;
2001-10-11 21:35:55 +00: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 ) ;
* * * * * * * */
}