2001-10-16 20:25:28 +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-03-01 22:08:11 +03:00
static int _activate_lvs_in_vg ( struct cmd_context * cmd ,
struct volume_group * vg , int lock )
{
2003-10-16 00:02:46 +04:00
struct lv_list * lvl ;
2002-03-01 22:08:11 +03:00
struct logical_volume * lv ;
2003-05-06 16:14:36 +04:00
struct physical_volume * pv ;
2002-03-01 22:08:11 +03:00
int count = 0 ;
2003-10-16 00:02:46 +04:00
list_iterate_items ( lvl , & vg - > lvs ) {
lv = lvl - > lv ;
2002-03-01 22:08:11 +03:00
2003-05-06 16:14:36 +04:00
/* Only request activation of snapshot origin devices */
2002-03-18 16:09:27 +03:00
if ( lv_is_cow ( lv ) )
continue ;
2003-05-06 16:14:36 +04:00
/* Can't deactive a pvmove LV */
if ( ( lock = = LCK_LV_DEACTIVATE ) & & ( lv - > status & PVMOVE ) )
continue ;
2002-03-05 23:03:09 +03:00
if ( ! lock_vol ( cmd , lv - > lvid . s , lock | LCK_NONBLOCK ) )
2002-03-01 22:08:11 +03:00
continue ;
2003-05-06 16:14:36 +04:00
if ( ( lv - > status & PVMOVE ) & &
( pv = get_pvmove_pv_from_lv_mirr ( lv ) ) ) {
log_verbose ( " Spawning background process for %s %s " ,
lv - > name , dev_name ( pv - > dev ) ) ;
pvmove_poll ( cmd , dev_name ( pv - > dev ) , 1 ) ;
continue ;
}
2002-03-01 22:08:11 +03:00
count + + ;
}
return count ;
}
2003-10-22 02:06:07 +04:00
static int _vgchange_available ( struct cmd_context * cmd ,
struct volume_group * vg )
2001-10-16 20:25:28 +04:00
{
2002-03-01 22:08:11 +03:00
int lv_open , active ;
2002-02-12 00:00:35 +03:00
int available = ! strcmp ( arg_str_value ( cmd , available_ARG , " n " ) , " y " ) ;
2001-10-16 20:25:28 +04:00
/* FIXME: Force argument to deactivate them? */
2001-11-07 18:02:07 +03:00
if ( ! available & & ( lv_open = lvs_in_vg_opened ( vg ) ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Can't deactivate volume group \" %s \" with %d open "
2001-10-16 20:25:28 +04:00
" logical volume(s) " , vg - > name , lv_open ) ;
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
}
2002-03-01 22:08:11 +03:00
if ( available & & ( active = lvs_in_vg_activated ( vg ) ) )
2002-01-30 18:04:48 +03:00
log_verbose ( " %d logical volume(s) in volume group \" %s \" "
2002-03-01 22:08:11 +03:00
" already active " , active , vg - > name ) ;
2001-11-21 22:32:35 +03:00
2002-03-01 22:08:11 +03:00
if ( available & & _activate_lvs_in_vg ( cmd , vg , LCK_LV_ACTIVATE ) )
log_verbose ( " Activated logical volumes in "
" volume group \" %s \" " , vg - > name ) ;
2001-10-16 20:25:28 +04:00
2002-03-01 22:08:11 +03:00
if ( ! available & & _activate_lvs_in_vg ( cmd , vg , LCK_LV_DEACTIVATE ) )
log_verbose ( " Deactivated logical volumes in "
" volume group \" %s \" " , vg - > name ) ;
2001-10-16 20:25:28 +04:00
2002-01-30 18:04:48 +03:00
log_print ( " %d logical volume(s) in volume group \" %s \" now active " ,
2002-01-22 22:58:37 +03:00
lvs_in_vg_activated ( vg ) , vg - > name ) ;
2003-10-22 02:06:07 +04:00
return ECMD_PROCESSED ;
2001-10-16 20:25:28 +04:00
}
2003-10-22 02:06:07 +04:00
static int _vgchange_resizeable ( struct cmd_context * cmd ,
struct volume_group * vg )
2001-10-16 20:25:28 +04:00
{
2002-02-12 00:00:35 +03:00
int resizeable = ! strcmp ( arg_str_value ( cmd , resizeable_ARG , " n " ) , " y " ) ;
2001-10-16 20:25:28 +04:00
2002-01-10 18:09:51 +03:00
if ( resizeable & & ( vg - > status & RESIZEABLE_VG ) ) {
2002-02-12 00:00:35 +03:00
log_error ( " Volume group \" %s \" is already resizeable " ,
vg - > name ) ;
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
}
2002-01-10 18:09:51 +03:00
if ( ! resizeable & & ! ( vg - > status & RESIZEABLE_VG ) ) {
2002-01-30 18:04:48 +03:00
log_error ( " Volume group \" %s \" is already not resizeable " ,
2001-10-16 20:25:28 +04:00
vg - > name ) ;
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
}
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2002-01-09 16:17:14 +03:00
2002-01-10 18:09:51 +03:00
if ( resizeable )
vg - > status | = RESIZEABLE_VG ;
2001-10-16 20:25:28 +04:00
else
2002-01-10 18:09:51 +03:00
vg - > status & = ~ RESIZEABLE_VG ;
2001-10-16 20:25:28 +04:00
2003-07-05 02:34:56 +04:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) )
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2001-10-16 20:25:28 +04:00
2002-01-30 18:04:48 +03:00
log_print ( " Volume group \" %s \" successfully changed " , vg - > name ) ;
2001-10-16 20:25:28 +04:00
2003-10-22 02:06:07 +04:00
return ECMD_PROCESSED ;
2001-10-16 20:25:28 +04:00
}
2003-10-22 02:06:07 +04:00
static int _vgchange_logicalvolume ( struct cmd_context * cmd ,
struct volume_group * vg )
2001-10-16 20:25:28 +04:00
{
2002-12-20 02:25:55 +03:00
uint32_t max_lv = arg_uint_value ( cmd , logicalvolume_ARG , 0 ) ;
2001-10-16 20:25:28 +04:00
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 \" must be resizeable "
2001-10-16 20:25:28 +04:00
" to change MaxLogicalVolume " , vg - > name ) ;
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
}
2003-11-06 23:33:34 +03:00
if ( ! ( vg - > fid - > fmt - > features & FMT_UNLIMITED_VOLS ) ) {
if ( ! max_lv )
max_lv = 255 ;
else if ( max_lv > 255 ) {
log_error ( " MaxLogicalVolume limit is 255 " ) ;
return ECMD_FAILED ;
}
}
if ( max_lv & & max_lv < vg - > lv_count ) {
2001-10-16 20:25:28 +04:00
log_error ( " MaxLogicalVolume is less than the current number "
2002-01-30 18:04:48 +03:00
" %d of logical volume(s) for \" %s \" " , vg - > lv_count ,
2001-10-16 20:25:28 +04:00
vg - > name ) ;
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
}
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2002-01-09 16:17:14 +03:00
2001-10-16 20:25:28 +04:00
vg - > max_lv = max_lv ;
2003-07-05 02:34:56 +04:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) )
2003-10-22 02:06:07 +04:00
return ECMD_FAILED ;
2001-10-16 20:25:28 +04:00
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 ( " Volume group \" %s \" successfully changed " , vg - > name ) ;
2001-10-16 20:25:28 +04:00
2003-10-22 02:06:07 +04:00
return ECMD_PROCESSED ;
2001-10-16 20:25:28 +04:00
}
2002-11-18 17:04:08 +03:00
static int vgchange_single ( struct cmd_context * cmd , const char * vg_name ,
struct volume_group * vg , int consistent ,
void * handle )
{
2003-10-22 02:06:07 +04:00
int r = 0 ;
2002-11-18 17:04:08 +03:00
if ( ! vg ) {
log_error ( " Unable to find volume group \" %s \" " , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! consistent ) {
unlock_vg ( cmd , vg_name ) ;
log_error ( " Volume group \" %s \" inconsistent " , vg_name ) ;
if ( ! ( vg = recover_vg ( cmd , vg_name , LCK_VG_WRITE ) ) )
return ECMD_FAILED ;
}
if ( ! ( vg - > status & LVM_WRITE ) & & ! arg_count ( cmd , available_ARG ) ) {
log_error ( " Volume group \" %s \" is read-only " , vg - > name ) ;
return ECMD_FAILED ;
}
if ( vg - > status & EXPORTED_VG ) {
log_error ( " Volume group \" %s \" is exported " , vg_name ) ;
return ECMD_FAILED ;
}
if ( arg_count ( cmd , available_ARG ) )
2003-10-22 02:06:07 +04:00
r = _vgchange_available ( cmd , vg ) ;
2002-11-18 17:04:08 +03:00
2003-10-22 02:06:07 +04:00
else if ( arg_count ( cmd , resizeable_ARG ) )
r = _vgchange_resizeable ( cmd , vg ) ;
2002-11-18 17:04:08 +03:00
2003-10-22 02:06:07 +04:00
else if ( arg_count ( cmd , logicalvolume_ARG ) )
r = _vgchange_logicalvolume ( cmd , vg ) ;
2002-11-18 17:04:08 +03:00
2003-10-22 02:06:07 +04:00
return r ;
2002-11-18 17:04:08 +03:00
}
int vgchange ( struct cmd_context * cmd , int argc , char * * argv )
{
if ( !
( arg_count ( cmd , available_ARG ) + arg_count ( cmd , logicalvolume_ARG ) +
arg_count ( cmd , resizeable_ARG ) ) ) {
log_error ( " One of -a, -l or -x options required " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( cmd , available_ARG ) + arg_count ( cmd , logicalvolume_ARG ) +
arg_count ( cmd , resizeable_ARG ) > 1 ) {
log_error ( " Only one of -a, -l or -x options allowed " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( cmd , ignorelockingfailure_ARG ) & &
! arg_count ( cmd , available_ARG ) ) {
log_error ( " --ignorelockingfailure only available with -a " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( cmd , available_ARG ) = = 1
& & arg_count ( cmd , autobackup_ARG ) ) {
log_error ( " -A option not necessary with -a option " ) ;
return EINVALID_CMD_LINE ;
}
return process_each_vg ( cmd , argc , argv ,
( arg_count ( cmd , available_ARG ) ) ?
LCK_VG_READ : LCK_VG_WRITE , 0 , NULL ,
& vgchange_single ) ;
}