2001-10-30 17:32:48 +03: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 lvchange_single ( char * lv_name ) ;
static int lvchange_permission ( struct logical_volume * lv ) ;
static int lvchange_availability ( struct logical_volume * lv ) ;
static int lvchange_contiguous ( struct logical_volume * lv ) ;
static int lvchange_readahead ( struct logical_volume * lv ) ;
int lvchange ( int argc , char * * argv )
{
int opt ;
int ret = 0 ;
int ret_max = 0 ;
if ( ! arg_count ( available_ARG ) & & ! arg_count ( contiguous_ARG )
& & ! arg_count ( permission_ARG ) & & ! arg_count ( readahead_ARG ) ) {
log_error ( " One or more of -a, -C, -p or -r required " ) ;
return EINVALID_CMD_LINE ;
}
if ( ! argc ) {
log_error ( " Please give logical volume path(s) " ) ;
return EINVALID_CMD_LINE ;
}
/* Walk through logical volumes */
for ( opt = 0 ; opt < argc ; opt + + ) {
ret = lvchange_single ( argv [ opt ] ) ;
if ( ret > ret_max )
ret_max = ret ;
}
return ret_max ;
}
static int lvchange_single ( char * lv_name )
{
char * vg_name ;
int doit = 0 ;
struct volume_group * vg ;
2001-10-31 15:47:01 +03:00
struct list * lvh ;
2001-10-30 17:32:48 +03:00
struct logical_volume * lv ;
/* FIXME Common code here - Add to a foreach? */
/* does VG exist? */
if ( ! ( vg_name = extract_vgname ( ios , lv_name ) ) ) {
return ECMD_FAILED ;
}
log_verbose ( " Finding volume group %s " , vg_name ) ;
if ( ! ( vg = ios - > vg_read ( ios , vg_name ) ) ) {
log_error ( " Volume group %s doesn't exist " , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! ( vg - > status & ACTIVE ) ) {
log_error ( " Volume group %s must be active before changing a "
" logical volume " , vg_name ) ;
return ECMD_FAILED ;
}
if ( ! ( lvh = find_lv_in_vg ( vg , lv_name ) ) ) {
log_error ( " Can't find logical volume %s in volume group %s " ,
lv_name , vg_name ) ;
return ECMD_FAILED ;
}
2001-10-31 15:47:01 +03:00
lv = & list_item ( lvh , struct lv_list ) - > lv ;
2001-10-30 17:32:48 +03:00
if ( lv - > status & SNAPSHOT_ORG ) {
log_error ( " Can't change logical volume %s under snapshot " ,
lv_name ) ;
return ECMD_FAILED ;
}
if ( lv - > status & SNAPSHOT ) {
log_error ( " Can't change snapshot logical volume %s " , lv_name ) ;
return ECMD_FAILED ;
}
/* access permission change */
if ( arg_count ( permission_ARG ) )
doit + = lvchange_permission ( lv ) ;
/* availability change */
if ( arg_count ( available_ARG ) )
doit + = lvchange_availability ( lv ) ;
/* allocation policy change */
if ( arg_count ( contiguous_ARG ) )
doit + = lvchange_contiguous ( lv ) ;
/* read ahead sector change */
if ( arg_count ( readahead_ARG ) )
doit + = lvchange_readahead ( lv ) ;
if ( ! doit ) {
return 0 ;
}
/* FIXME activate change */
log_print ( " Logical volume %s changed " , lv - > name ) ;
/* FIXME do_autobackup */
return 0 ;
}
static int lvchange_permission ( struct logical_volume * lv )
{
int lv_access ;
lv_access = arg_int_value ( permission_ARG , 0 ) ;
if ( ( lv_access & LVM_WRITE ) & & ( lv - > status & LVM_WRITE ) ) {
log_error ( " Logical volume %s is already writable " , lv - > name ) ;
return 0 ;
}
if ( ! ( lv_access & LVM_WRITE ) & & ! ( lv - > status & LVM_WRITE ) ) {
log_error ( " Logical volume %s is already read only " , lv - > name ) ;
return 0 ;
}
if ( lv_access & LVM_WRITE ) {
lv - > status | = LVM_WRITE ;
log_verbose ( " Setting logical volume %s read/write " , lv - > name ) ;
} else {
lv - > status & = ~ LVM_WRITE ;
log_verbose ( " Setting logical volume %s read-only " , lv - > name ) ;
}
2001-10-31 20:59:52 +03:00
log_very_verbose ( " Updating logical volume %s on disk(s) " , lv - > name ) ;
if ( ! ios - > vg_write ( ios , lv - > vg ) )
return 0 ;
log_very_verbose ( " Updating permissions for %s in kernel " , lv - > name ) ;
if ( ! lv_update_write_access ( lv ) )
return 0 ;
2001-10-30 17:32:48 +03:00
return 1 ;
}
static int lvchange_availability ( struct logical_volume * lv )
{
int lv_stat = 0 ;
if ( strcmp ( arg_str_value ( available_ARG , " n " ) , " n " ) )
lv_stat | = ACTIVE ;
if ( ( lv_stat & ACTIVE ) & & ( lv - > status & ACTIVE ) ) {
log_error ( " Logical volume %s is already active " , lv - > name ) ;
return 0 ;
}
if ( ! ( lv_stat & ACTIVE ) & & ! ( lv - > status & ACTIVE ) ) {
log_error ( " Logical volume %s is already not active " , lv - > name ) ;
return 0 ;
}
if ( lv_stat & ACTIVE ) {
lv - > status | = ACTIVE ;
log_verbose ( " Activating logical volume %s " , lv - > name ) ;
} else {
lv - > status & = ~ ACTIVE ;
log_verbose ( " Deactivating logical volume %s " , lv - > name ) ;
}
2001-10-31 20:59:52 +03:00
log_very_verbose ( " Updating logical volume %s on disk(s) " , lv - > name ) ;
if ( ! ios - > vg_write ( ios , lv - > vg ) )
return 0 ;
log_very_verbose ( " Updating %s in kernel " , lv - > name ) ;
if ( lv_stat & ACTIVE ) {
if ( ! lv_activate ( lv ) )
return 0 ;
} else
if ( ! lv_deactivate ( lv ) )
return 0 ;
2001-10-30 17:32:48 +03:00
return 1 ;
}
static int lvchange_contiguous ( struct logical_volume * lv )
{
int lv_allocation = 0 ;
if ( strcmp ( arg_str_value ( contiguous_ARG , " n " ) , " n " ) )
lv_allocation | = ALLOC_CONTIGUOUS ;
if ( ( lv_allocation & ALLOC_CONTIGUOUS ) & &
( lv - > status & ALLOC_CONTIGUOUS ) ) {
log_error ( " Allocation policy of logical volume %s is "
" already contiguous " , lv - > name ) ;
return 0 ;
}
if ( ! ( lv_allocation & ALLOC_CONTIGUOUS ) & &
! ( lv - > status & ALLOC_CONTIGUOUS ) ) {
log_error ( " Allocation policy of logical volume %s is already "
" not contiguous " , lv - > name ) ;
return 0 ;
}
/******** FIXME lv_check_contiguous?
if ( ( lv_allocation & ALLOC_CONTIGUOUS )
& & ( ret = lv_check_contiguous ( vg , lv_index + 1 ) ) = = FALSE ) {
log_error ( " No contiguous logical volume \" %s \" " , lv_name ) ;
return 0 ;
* * * * * * * * */
if ( lv_allocation & ALLOC_CONTIGUOUS ) {
lv - > status | = ALLOC_CONTIGUOUS ;
log_verbose ( " Setting contiguous allocation policy for %s " ,
lv - > name ) ;
} else {
lv - > status & = ~ ALLOC_CONTIGUOUS ;
log_verbose ( " Removing contiguous allocation policy for %s " ,
lv - > name ) ;
}
2001-10-31 20:59:52 +03:00
log_verbose ( " Updating logical volume %s on disk(s) " , lv - > name ) ;
if ( ! ios - > vg_write ( ios , lv - > vg ) )
return 0 ;
2001-10-30 17:32:48 +03:00
return 1 ;
}
static int lvchange_readahead ( struct logical_volume * lv )
{
int read_ahead = 0 ;
read_ahead = arg_int_value ( readahead_ARG , 0 ) ;
/******* FIXME Ranges?
if ( read_ahead < LVM_MIN_READ_AHEAD | | read_ahead > LVM_MAX_READ_AHEAD ) {
log_error ( " read ahead sector argument is invalid " ) ;
return 0 ;
}
* * * * * * * */
if ( lv - > read_ahead = = read_ahead ) {
log_error ( " Read ahead is already %u for %s " ,
read_ahead , lv - > name ) ;
return 0 ;
}
lv - > read_ahead = read_ahead ;
log_verbose ( " Setting read ahead to %u for %s " , read_ahead , lv - > name ) ;
2001-10-31 20:59:52 +03:00
log_verbose ( " Updating logical volume %s on disk(s) " , lv - > name ) ;
if ( ! ios - > vg_write ( ios , lv - > vg ) )
return 0 ;
2001-10-30 17:32:48 +03:00
return 1 ;
}