2001-10-16 16:25:28 +00: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"
static int vgchange_single ( const char * vg_name ) ;
void vgchange_available ( struct volume_group * vg ) ;
void vgchange_allocation ( struct volume_group * vg ) ;
void vgchange_logicalvolume ( struct volume_group * vg ) ;
int vgchange ( int argc , char * * argv )
{
if ( ! ( arg_count ( available_ARG ) + arg_count ( logicalvolume_ARG ) +
arg_count ( allocation_ARG ) ) ) {
log_error ( " One of -a, -l or -x options required " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( available_ARG ) + arg_count ( logicalvolume_ARG ) +
arg_count ( allocation_ARG ) > 1 ) {
log_error ( " Only one of -a, -l or -x options allowed " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( available_ARG ) = = 1 & & arg_count ( autobackup_ARG ) ) {
log_error ( " -A option not necessary with -a option " ) ;
return EINVALID_CMD_LINE ;
}
return process_each_vg ( argc , argv , & vgchange_single ) ;
}
static int vgchange_single ( const char * vg_name )
{
struct volume_group * vg ;
2001-11-12 15:10:01 +00:00
if ( ! ( vg = fid - > ops - > vg_read ( fid , vg_name ) ) ) {
2001-10-16 16:25:28 +00:00
log_error ( " Unable to find volume group %s " , 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 ( available_ARG ) ) {
vgchange_available ( vg ) ;
}
if ( arg_count ( allocation_ARG ) ) {
vgchange_allocation ( vg ) ;
}
if ( arg_count ( logicalvolume_ARG ) ) {
vgchange_logicalvolume ( vg ) ;
}
return 0 ;
}
void vgchange_available ( struct volume_group * vg )
{
2001-11-21 19:32:35 +00:00
int lv_open , lv_active ;
struct list * lvh ;
2001-10-16 16:25:28 +00:00
int available = ! strcmp ( arg_str_value ( available_ARG , " n " ) , " y " ) ;
2001-11-21 19:32:35 +00:00
/* Ignore existing disk status */
if ( available & & ( vg - > status & ACTIVE ) )
log_verbose ( " Volume group %s is already active on disk " ,
vg - > name ) ;
2001-10-16 16:25:28 +00:00
2001-11-21 19:32:35 +00:00
if ( ! available & & ! ( vg - > status & ACTIVE ) )
log_verbose ( " Volume group %s is already inactive on disk " ,
vg - > name ) ;
2001-10-16 16:25:28 +00:00
/* FIXME: Force argument to deactivate them? */
2001-11-07 15:02:07 +00:00
if ( ! available & & ( lv_open = lvs_in_vg_opened ( vg ) ) ) {
log_error ( " Can't deactivate volume group '%s' with %d open "
2001-10-16 16:25:28 +00:00
" logical volume(s) " , vg - > name , lv_open ) ;
return ;
}
2001-11-21 19:32:35 +00:00
if ( available & & ( lv_active = lvs_in_vg_activated ( vg ) ) )
log_verbose ( " %d logical volume(s) in volume group %s "
" already active " , lv_active , vg - > name ) ;
2001-10-16 18:07:54 +00:00
if ( available ) {
2001-10-16 16:25:28 +00:00
vg - > status | = ACTIVE ;
2001-11-21 19:32:35 +00:00
list_iterate ( lvh , & vg - > lvs )
list_item ( lvh , struct lv_list ) - > lv . status
2001-10-16 18:07:54 +00:00
| = ACTIVE ;
} else {
2001-10-16 16:25:28 +00:00
vg - > status & = ~ ACTIVE ;
2001-11-21 19:32:35 +00:00
list_iterate ( lvh , & vg - > lvs )
list_item ( lvh , struct lv_list ) - > lv . status
2001-10-16 18:07:54 +00:00
& = ~ ACTIVE ;
}
2001-10-16 16:25:28 +00:00
2001-11-12 15:10:01 +00:00
if ( ! fid - > ops - > vg_write ( fid , vg ) )
2001-10-16 16:25:28 +00:00
return ;
2001-12-31 21:27:39 +00:00
autobackup ( vg ) ;
2001-10-16 16:25:28 +00:00
if ( available & & ( lv_open = activate_lvs_in_vg ( vg ) ) )
log_verbose ( " Activated %d logical volumes in "
2001-11-21 19:32:35 +00:00
" volume group %s " , lv_open , vg - > name ) ;
2001-10-16 16:25:28 +00:00
if ( ! available & & ( lv_open = deactivate_lvs_in_vg ( vg ) ) )
log_verbose ( " Deactivated %d logical volumes in "
2001-11-21 19:32:35 +00:00
" volume group %s " , lv_open , vg - > name ) ;
2001-10-16 16:25:28 +00:00
2001-11-21 19:32:35 +00:00
log_print ( " Volume group %s updated " , vg - > name ) ;
2001-10-16 16:25:28 +00:00
return ;
}
void vgchange_allocation ( struct volume_group * vg )
{
int extendable = ! strcmp ( arg_str_value ( allocation_ARG , " n " ) , " y " ) ;
if ( extendable & & ( vg - > status & EXTENDABLE_VG ) ) {
log_error ( " Volume group %s is already extendable " , vg - > name ) ;
return ;
}
if ( ! extendable & & ! ( vg - > status & EXTENDABLE_VG ) ) {
log_error ( " Volume group %s is already not extendable " ,
vg - > name ) ;
return ;
}
if ( extendable )
vg - > status | = EXTENDABLE_VG ;
else
vg - > status & = ~ EXTENDABLE_VG ;
2001-11-12 15:10:01 +00:00
if ( ! fid - > ops - > vg_write ( fid , vg ) )
2001-10-16 16:25:28 +00:00
return ;
2001-12-31 21:27:39 +00:00
autobackup ( vg ) ;
2001-10-16 16:25:28 +00:00
2001-12-31 21:27:39 +00:00
log_print ( " Volume group %s successfully changed " , vg - > name ) ;
2001-10-16 16:25:28 +00:00
return ;
}
void vgchange_logicalvolume ( struct volume_group * vg )
{
int max_lv = arg_int_value ( logicalvolume_ARG , 0 ) ;
if ( vg - > status & ACTIVE ) {
log_error ( " MaxLogicalVolume can't be changed in "
" active volume group '%s' " , vg - > name ) ;
return ;
}
if ( ! ( vg - > status & EXTENDABLE_VG ) ) {
log_error ( " Volume group '%s' must be extendable "
" to change MaxLogicalVolume " , vg - > name ) ;
return ;
}
if ( max_lv < vg - > lv_count ) {
log_error ( " MaxLogicalVolume is less than the current number "
" %d of logical volume(s) for '%s' " , vg - > lv_count ,
vg - > name ) ;
return ;
}
/************** FIXME To be handled within vg_write
for ( p = 0 , pp = * vg - > pv ; p < vg - > pv_max ; p + + , pp + + ) {
if ( pp ! = NULL ) {
pp - > lv_on_disk . size =
round_up ( ( max_lv + 1 ) * sizeof ( lv_disk_t ) , LVM_VGDA_ALIGN ) ;
pe_on_disk_base_sav = pp - > pe_on_disk . base ;
pp - > pe_on_disk . base = pp - > lv_on_disk . base + pp - > lv_on_disk . size ;
pp - > pe_on_disk . size - = pp - > pe_on_disk . base - pe_on_disk_base_sav ;
if ( LVM_VGDA_SIZE ( pp ) / SECTOR_SIZE >
( pp - > pv_size & ~ LVM_PE_ALIGN ) - pp - > pe_total * pp - > pe_size ) {
log_error ( " extended VGDA would overlap first physical extent " ) ;
return LVM_E_PE_OVERLAP ;
}
}
}
* * * * * * * * * * * * * * * */
vg - > max_lv = max_lv ;
2001-11-12 15:10:01 +00:00
if ( ! fid - > ops - > vg_write ( fid , vg ) )
2001-10-16 16:25:28 +00:00
return ;
2001-12-31 21:27:39 +00:00
autobackup ( vg ) ;
2001-10-16 16:25:28 +00:00
log_print ( " Volume group %s successfully changed " , vg - > name ) ;
return ;
}