2001-10-12 01:35:55 +04:00
/*
* Copyright ( C ) 2001 Sistina Software
*
* LVM is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 , or ( at your option )
* any later version .
*
* LVM is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with LVM ; see the file COPYING . If not , write to
* the Free Software Foundation , 59 Temple Place - Suite 330 ,
* Boston , MA 02111 - 1307 , USA .
*
*/
# include "tools.h"
2002-02-12 00:00:35 +03:00
static int vgreduce_single ( struct cmd_context * cmd , struct volume_group * vg ,
struct physical_volume * pv ) ;
2001-10-12 01:35:55 +04: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 ;
char * vg_name ;
2002-02-11 18:42:34 +03:00
int ret ;
2001-10-12 01:35:55 +04:00
if ( ! argc ) {
log_error ( " Please give volume group name and "
" physical volume paths " ) ;
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 ( " 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 ;
}
vg_name = argv [ 0 ] ;
argv + + ;
argc - - ;
2002-01-30 18:04:48 +03:00
log_verbose ( " Finding volume group \" %s \" " , vg_name ) ;
2002-02-27 15:26:41 +03:00
if ( ! lock_vol ( cmd , vg_name , LCK_VG_WRITE ) ) {
2002-02-11 18:42:34 +03:00
log_error ( " Can't get lock for %s " , vg_name ) ;
return ECMD_FAILED ;
}
2002-04-24 22:20:51 +04:00
if ( ! ( vg = vg_read ( cmd , vg_name ) ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" doesn't exist " , vg_name ) ;
2002-03-11 18:08:39 +03:00
unlock_vg ( cmd , vg_name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
2002-02-12 00:00:35 +03:00
if ( vg - > status & EXPORTED_VG ) {
log_error ( " Volume group \" %s \" is exported " , vg - > name ) ;
2002-03-11 18:08:39 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-02-12 00:00:35 +03:00
return ECMD_FAILED ;
}
2002-01-29 20:23:33 +03:00
if ( ! ( vg - > status & LVM_WRITE ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" is read-only " , vg_name ) ;
2002-03-11 18:08:39 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-01-29 20:23:33 +03:00
return ECMD_FAILED ;
}
2002-01-10 18:09:51 +03:00
if ( ! ( vg - > status & RESIZEABLE_VG ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" is not reducable " , vg_name ) ;
2002-03-11 18:08:39 +03:00
unlock_vg ( cmd , vg_name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
/* FIXME: Pass private structure through to all these functions */
/* and update in batch here? */
2002-02-11 23:50:53 +03:00
ret = process_each_pv ( cmd , argc , argv , vg , vgreduce_single ) ;
2002-02-11 18:42:34 +03:00
2002-03-11 18:08:39 +03:00
unlock_vg ( cmd , vg_name ) ;
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
2001-10-12 01:35:55 +04:00
log_verbose
( " volume group \" %s \" will be reduced by %d physical volume%s " ,
vg_name , np , np > 1 ? " s " : " " ) ;
2002-01-30 18:04:48 +03:00
log_verbose ( " reducing volume group \" %s \" by physical volume \" %s \" " ,
vg_name , pv_names [ p ] ) ;
2001-10-12 01:35:55 +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 ) ;
* * * * * * * */
}
/* Or take pv_name instead? */
2002-02-12 00:00:35 +03:00
static int vgreduce_single ( struct cmd_context * cmd , struct volume_group * vg ,
struct physical_volume * pv )
2001-10-12 01:35:55 +04:00
{
2002-01-21 17:28:12 +03:00
struct pv_list * pvl ;
2001-10-25 18:04:18 +04:00
const char * name = dev_name ( pv - > dev ) ;
2001-10-12 01:35:55 +04:00
2002-04-24 22:20:51 +04:00
if ( pv - > pe_alloc_count ) {
2002-01-30 18:04:48 +03:00
log_error ( " Physical volume \" %s \" still in use " , name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
2002-04-24 22:20:51 +04:00
/********* FIXME: Is this unnecessary after checking pe_alloc_count?
2001-10-12 01:35:55 +04:00
if ( pv - > lv_cur > 0 ) {
log_error ( " can't reduce volume group \" %s \" by used physical volume \" %s \" " , vg_name , error_pv_name ) ;
}
* * * * * * * * */
if ( vg - > pv_count = = 1 ) {
2002-01-30 18:04:48 +03:00
log_error ( " Can't remove final physical volume \" %s \" from "
" volume group \" %s \" " , name , vg - > name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
2002-01-21 17:28:12 +03:00
pvl = find_pv_in_vg ( vg , name ) ;
2001-10-12 01:35:55 +04:00
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
return ECMD_FAILED ;
2002-01-30 18:04:48 +03:00
log_verbose ( " Removing \" %s \" from volume group \" %s \" " , name , vg - > name ) ;
2002-01-21 19:05:23 +03:00
2002-01-21 17:28:12 +03:00
if ( pvl )
list_del ( & pvl - > list ) ;
2001-10-12 01:35:55 +04:00
* pv - > vg_name = ' \0 ' ;
vg - > pv_count - - ;
2002-04-24 22:20:51 +04:00
vg - > free_count - = pv - > pe_count - pv - > pe_alloc_count ;
2001-11-13 20:53:06 +03:00
vg - > extent_count - = pv - > pe_count ;
2001-10-12 01:35:55 +04:00
2002-04-24 22:20:51 +04:00
if ( ! vg_write ( vg ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Removal of physical volume \" %s \" from "
2002-02-12 00:00:35 +03:00
" \" %s \" failed " , name , vg - > name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
2002-04-24 22:20:51 +04:00
if ( ! pv_write ( cmd , pv ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Failed to clear metadata from physical "
" volume \" %s \" "
" after removal from \" %s \" " , name , vg - > name ) ;
2001-10-12 01:35:55 +04:00
return ECMD_FAILED ;
}
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2002-01-01 00:27:39 +03:00
2002-01-30 18:04:48 +03:00
log_print ( " Removed \" %s \" from volume group \" %s \" " , name , vg - > name ) ;
2001-10-12 01:35:55 +04:00
return 0 ;
}