2001-10-03 21:03:25 +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"
2001-10-17 19:29:31 +04:00
int pvchange_single ( struct physical_volume * pv ) ;
2001-10-03 21:03:25 +04:00
int pvchange ( int argc , char * * argv )
{
int opt = 0 ;
2001-10-17 19:29:31 +04:00
int done = 0 ;
int total = 0 ;
2001-10-03 21:03:25 +04:00
2001-10-17 19:29:31 +04:00
struct physical_volume * pv ;
2001-10-03 21:03:25 +04:00
char * pv_name ;
2001-10-31 15:47:01 +03:00
struct list * pvh , * pvs ;
2001-10-03 21:03:25 +04:00
2002-01-10 18:09:51 +03:00
if ( arg_count ( allocatable_ARG ) = = 0 ) {
2001-10-03 21:03:25 +04:00
log_error ( " Please give the x option " ) ;
2001-10-05 02:53:37 +04:00
return EINVALID_CMD_LINE ;
2001-10-03 21:03:25 +04:00
}
if ( ! ( arg_count ( all_ARG ) ) & & ! argc ) {
log_error ( " Please give a physical volume path " ) ;
2001-10-05 02:53:37 +04:00
return EINVALID_CMD_LINE ;
2001-10-03 21:03:25 +04:00
}
if ( arg_count ( all_ARG ) & & argc ) {
log_error ( " Option a and PhysicalVolumePath are exclusive " ) ;
2001-10-05 02:53:37 +04:00
return EINVALID_CMD_LINE ;
2001-10-03 21:03:25 +04:00
}
2001-10-17 19:29:31 +04:00
if ( argc ) {
log_verbose ( " Using physical volume(s) on command line " ) ;
2001-10-03 21:03:25 +04:00
for ( ; opt < argc ; opt + + ) {
pv_name = argv [ opt ] ;
2001-11-12 18:10:01 +03:00
if ( ! ( pv = fid - > ops - > pv_read ( fid , pv_name ) ) ) {
2001-10-03 21:03:25 +04:00
log_error ( " Failed to read physical volume %s " ,
pv_name ) ;
continue ;
}
2001-10-17 19:29:31 +04:00
total + + ;
done + = pvchange_single ( pv ) ;
}
} else {
log_verbose ( " Scanning for physical volume names " ) ;
2001-11-12 18:10:01 +03:00
if ( ! ( pvs = fid - > ops - > get_pvs ( fid ) ) ) {
2001-10-17 19:29:31 +04:00
return ECMD_FAILED ;
}
2001-10-31 15:47:01 +03:00
list_iterate ( pvh , pvs ) {
2001-10-17 19:29:31 +04:00
total + + ;
2001-10-31 15:47:01 +03:00
done + = pvchange_single (
& list_item ( pvh , struct pv_list ) - > pv ) ;
2001-10-03 21:03:25 +04:00
}
}
2001-10-17 19:29:31 +04:00
log_print ( " %d physical volume(s) changed / %d physical volume(s) "
" not changed " , done , total - done ) ;
2001-10-03 21:03:25 +04:00
return 0 ;
}
2001-10-17 19:29:31 +04:00
int pvchange_single ( struct physical_volume * pv )
2001-10-03 21:03:25 +04:00
{
2001-10-17 19:29:31 +04:00
struct volume_group * vg = NULL ;
2001-10-31 15:47:01 +03:00
struct list * pvh ;
2001-10-03 21:03:25 +04:00
2001-10-25 18:04:18 +04:00
const char * pv_name = dev_name ( pv - > dev ) ;
2001-10-03 21:03:25 +04:00
2002-01-10 18:09:51 +03:00
int allocatable = ! strcmp ( arg_str_value ( allocatable_ARG , " n " ) , " y " ) ;
2001-10-03 21:03:25 +04:00
2001-10-17 19:29:31 +04:00
/* If in a VG, must change using volume group. Pointless. */
/* FIXME: Provide a direct pv_write_pv that *only* touches PV structs*/
if ( * pv - > vg_name ) {
log_verbose ( " Finding volume group of physical volume %s " ,
pv_name ) ;
2001-11-12 18:10:01 +03:00
if ( ! ( vg = fid - > ops - > vg_read ( fid , pv - > vg_name ) ) ) {
2001-10-17 19:29:31 +04:00
log_error ( " Unable to find volume group of %s " , pv_name ) ;
return 0 ;
}
if ( ! ( pvh = find_pv_in_vg ( vg , pv_name ) ) ) {
log_error ( " Unable to find %s in volume group %s " ,
pv_name , vg - > name ) ;
return 0 ;
}
2001-10-31 15:47:01 +03:00
pv = & list_item ( pvh , struct pv_list ) - > pv ;
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
return 0 ;
2001-10-17 19:29:31 +04:00
}
2001-10-03 21:03:25 +04:00
2001-10-17 19:29:31 +04:00
/* change allocatability for a PV */
2002-01-10 18:09:51 +03:00
if ( allocatable & & ( pv - > status & ALLOCATABLE_PV ) ) {
2001-10-17 19:29:31 +04:00
log_error ( " Physical volume %s is already allocatable " , pv_name ) ;
return 0 ;
2001-10-03 21:03:25 +04:00
}
2002-01-10 18:09:51 +03:00
if ( ! allocatable & & ! ( pv - > status & ALLOCATABLE_PV ) ) {
2001-10-17 19:29:31 +04:00
log_error ( " Physical volume %s is already unallocatable " ,
pv_name ) ;
return 0 ;
}
2001-10-03 21:03:25 +04:00
2002-01-10 18:09:51 +03:00
if ( allocatable ) {
2001-10-17 19:29:31 +04:00
log_verbose ( " Setting physical volume %s allocatable " , pv_name ) ;
2002-01-10 18:09:51 +03:00
pv - > status | = ALLOCATABLE_PV ;
2001-10-17 19:29:31 +04:00
} else {
log_verbose ( " Setting physical volume %s NOT allocatable " ,
pv_name ) ;
2002-01-10 18:09:51 +03:00
pv - > status & = ~ ALLOCATABLE_PV ;
2001-10-03 21:03:25 +04:00
}
2001-10-25 18:04:18 +04:00
log_verbose ( " Updating physical volume %s " , pv_name ) ;
2001-10-17 19:29:31 +04:00
if ( * pv - > vg_name ) {
2001-11-12 18:10:01 +03:00
if ( ! ( fid - > ops - > vg_write ( fid , vg ) ) ) {
2001-10-17 19:29:31 +04:00
log_error ( " Failed to store physical volume %s in "
" volume group %s " , pv_name , vg - > name ) ;
return 0 ;
2001-10-03 21:03:25 +04:00
}
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2001-10-17 19:29:31 +04:00
} else {
2001-11-12 18:10:01 +03:00
if ( ! ( fid - > ops - > pv_write ( fid , pv ) ) ) {
2001-10-17 19:29:31 +04:00
log_error ( " Failed to store physical volume %s " ,
pv_name ) ;
return 0 ;
2001-10-03 21:03:25 +04:00
}
}
2001-10-17 19:29:31 +04:00
log_print ( " Physical volume %s changed " , pv_name ) ;
return 1 ;
2001-10-03 21:03:25 +04:00
}