2001-10-03 21:03:25 +04:00
/*
2008-01-30 17:00:02 +03:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-21 00:55:30 +04:00
* Copyright ( C ) 2004 - 2007 Red Hat , Inc . All rights reserved .
2001-10-03 21:03:25 +04:00
*
2004-03-30 23:35:44 +04:00
* This file is part of LVM2 .
2001-10-03 21:03:25 +04:00
*
2004-03-30 23:35:44 +04:00
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-21 00:55:30 +04:00
* of the GNU Lesser General Public License v .2 .1 .
2001-10-03 21:03:25 +04:00
*
2007-08-21 00:55:30 +04:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 23:35:44 +04:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-10-03 21:03:25 +04:00
*/
# include "tools.h"
2002-11-18 17:04:08 +03:00
/* FIXME Locking. PVs in VG. */
2001-10-03 21:03:25 +04:00
2002-12-20 02:25:55 +03:00
static int _pvchange_single ( struct cmd_context * cmd , struct physical_volume * pv ,
2006-05-10 01:23:51 +04:00
void * handle __attribute ( ( unused ) ) )
2001-10-03 21:03:25 +04:00
{
2001-10-17 19:29:31 +04:00
struct volume_group * vg = NULL ;
2007-11-02 23:40:05 +03:00
const char * vg_name = NULL ;
2002-01-21 17:28:12 +03:00
struct pv_list * pvl ;
2002-11-18 17:04:08 +03:00
uint64_t sector ;
2005-05-24 21:38:26 +04:00
uint32_t orig_pe_alloc_count ;
2008-07-16 14:46:12 +04:00
/* FIXME Next three only required for format1. */
uint32_t orig_pe_count , orig_pe_size ;
uint64_t orig_pe_start ;
2001-10-03 21:03:25 +04:00
2007-10-12 18:29:32 +04:00
const char * pv_name = pv_dev_name ( pv ) ;
2004-03-08 20:19:15 +03:00
const char * tag = NULL ;
2005-05-24 21:38:26 +04:00
const char * orig_vg_name ;
2006-12-01 02:11:42 +03:00
char uuid [ 64 ] __attribute ( ( aligned ( 8 ) ) ) ;
2001-10-03 21:03:25 +04:00
2004-01-13 21:42:05 +03:00
int allocatable = 0 ;
2004-03-08 20:19:15 +03:00
int tagarg = 0 ;
if ( arg_count ( cmd , addtag_ARG ) )
tagarg = addtag_ARG ;
else if ( arg_count ( cmd , deltag_ARG ) )
tagarg = deltag_ARG ;
2004-01-13 21:42:05 +03:00
if ( arg_count ( cmd , allocatable_ARG ) )
allocatable = ! strcmp ( arg_str_value ( cmd , allocatable_ARG , " n " ) ,
" y " ) ;
2004-03-08 20:19:15 +03:00
else if ( tagarg & & ! ( tag = arg_str_value ( cmd , tagarg , NULL ) ) ) {
log_error ( " Failed to get tag " ) ;
return 0 ;
}
2001-10-03 21:03:25 +04:00
2002-01-29 20:23:33 +03:00
/* If in a VG, must change using volume group. */
2007-11-02 17:54:40 +03:00
if ( ! is_orphan ( pv ) ) {
2007-11-02 23:40:05 +03:00
vg_name = pv_vg_name ( pv ) ;
2002-01-29 20:23:33 +03:00
2007-11-15 05:20:03 +03:00
log_verbose ( " Finding volume group %s of physical volume %s " ,
vg_name , pv_name ) ;
if ( ! ( vg = vg_lock_and_read ( cmd , vg_name , NULL , LCK_VG_WRITE ,
CLUSTERED | EXPORTED_VG | LVM_WRITE ,
CORRECT_INCONSISTENT ) ) )
return_0 ;
2002-01-29 20:23:33 +03:00
2002-01-21 17:28:12 +03:00
if ( ! ( pvl = find_pv_in_vg ( vg , pv_name ) ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-02-12 00:00:35 +03:00
log_error
( " Unable to find \" %s \" in volume group \" %s \" " ,
pv_name , vg - > name ) ;
2001-10-17 19:29:31 +04:00
return 0 ;
}
2004-03-08 20:19:15 +03:00
if ( tagarg & & ! ( vg - > fid - > fmt - > features & FMT_TAGS ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-03-08 20:19:15 +03:00
log_error ( " Volume group containing %s does not "
" support tags " , pv_name ) ;
return 0 ;
}
2004-01-13 21:42:05 +03:00
if ( arg_count ( cmd , uuid_ARG ) & & lvs_in_vg_activated ( vg ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-01-13 21:42:05 +03:00
log_error ( " Volume group containing %s has active "
" logical volumes " , pv_name ) ;
return 0 ;
}
2002-01-21 19:05:23 +03:00
pv = pvl - > pv ;
2002-01-09 16:17:14 +03:00
if ( ! archive ( vg ) )
return 0 ;
2002-11-18 17:04:08 +03:00
} else {
2004-03-08 20:19:15 +03:00
if ( tagarg ) {
log_error ( " Can't change tag on Physical Volume %s not "
" in volume group " , pv_name ) ;
return 0 ;
}
2007-11-02 23:40:05 +03:00
vg_name = VG_ORPHANS ;
if ( ! lock_vol ( cmd , vg_name , LCK_VG_WRITE ) ) {
2002-11-18 17:04:08 +03:00
log_error ( " Can't get lock for orphans " ) ;
2004-01-13 21:42:05 +03:00
return 0 ;
2002-11-18 17:04:08 +03:00
}
2009-02-26 02:29:06 +03:00
if ( ! ( pv = pv_read ( cmd , pv_name , NULL , & sector , 1 , 0 ) ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-11-18 17:04:08 +03:00
log_error ( " Unable to read PV \" %s \" " , pv_name ) ;
return 0 ;
}
2001-10-17 19:29:31 +04:00
}
2001-10-03 21:03:25 +04:00
2004-01-13 21:42:05 +03:00
if ( arg_count ( cmd , allocatable_ARG ) ) {
2007-11-02 17:54:40 +03:00
if ( is_orphan ( pv ) & &
2004-01-13 21:42:05 +03:00
! ( pv - > fmt - > features & FMT_ORPHAN_ALLOCATABLE ) ) {
log_error ( " Allocatability not supported by orphan "
" %s format PV %s " , pv - > fmt - > name , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-01-13 21:42:05 +03:00
return 0 ;
}
2001-10-03 21:03:25 +04:00
2004-01-13 21:42:05 +03:00
/* change allocatability for a PV */
2007-06-16 02:16:55 +04:00
if ( allocatable & & ( pv_status ( pv ) & ALLOCATABLE_PV ) ) {
2004-01-13 21:42:05 +03:00
log_error ( " Physical volume \" %s \" is already "
" allocatable " , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-01-13 21:42:05 +03:00
return 1 ;
}
2001-10-03 21:03:25 +04:00
2007-06-16 02:16:55 +04:00
if ( ! allocatable & & ! ( pv_status ( pv ) & ALLOCATABLE_PV ) ) {
2004-01-13 21:42:05 +03:00
log_error ( " Physical volume \" %s \" is already "
" unallocatable " , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-01-13 21:42:05 +03:00
return 1 ;
}
if ( allocatable ) {
log_verbose ( " Setting physical volume \" %s \" "
" allocatable " , pv_name ) ;
pv - > status | = ALLOCATABLE_PV ;
} else {
log_verbose ( " Setting physical volume \" %s \" NOT "
" allocatable " , pv_name ) ;
pv - > status & = ~ ALLOCATABLE_PV ;
}
2004-03-08 20:19:15 +03:00
} else if ( tagarg ) {
/* tag or deltag */
if ( ( tagarg = = addtag_ARG ) ) {
if ( ! str_list_add ( cmd - > mem , & pv - > tags , tag ) ) {
log_error ( " Failed to add tag %s to physical "
" volume %s " , tag , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-03-08 20:19:15 +03:00
return 0 ;
}
} else {
if ( ! str_list_del ( & pv - > tags , tag ) ) {
log_error ( " Failed to remove tag %s from "
" physical volume " " %s " , tag , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2004-03-08 20:19:15 +03:00
return 0 ;
}
}
2001-10-17 19:29:31 +04:00
} else {
2004-01-13 21:42:05 +03:00
/* --uuid: Change PV ID randomly */
2005-01-20 21:11:53 +03:00
if ( ! id_create ( & pv - > id ) ) {
log_error ( " Failed to generate new random UUID for %s. " ,
pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2005-01-20 21:11:53 +03:00
return 0 ;
}
2005-05-24 21:38:26 +04:00
if ( ! id_write_format ( & pv - > id , uuid , sizeof ( uuid ) ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2008-01-30 16:19:47 +03:00
return_0 ;
2005-05-24 21:38:26 +04:00
}
log_verbose ( " Changing uuid of %s to %s. " , pv_name , uuid ) ;
2007-11-02 17:54:40 +03:00
if ( ! is_orphan ( pv ) ) {
2007-06-16 02:16:55 +04:00
orig_vg_name = pv_vg_name ( pv ) ;
orig_pe_alloc_count = pv_pe_alloc_count ( pv ) ;
2008-07-16 14:46:12 +04:00
/* FIXME format1 pv_write doesn't preserve these. */
orig_pe_size = pv_pe_size ( pv ) ;
orig_pe_start = pv_pe_start ( pv ) ;
orig_pe_count = pv_pe_count ( pv ) ;
2008-02-06 18:47:28 +03:00
pv - > vg_name = pv - > fmt - > orphan_vg_name ;
2005-05-24 21:38:26 +04:00
pv - > pe_alloc_count = 0 ;
if ( ! ( pv_write ( cmd , pv , NULL , INT64_C ( - 1 ) ) ) ) {
log_error ( " pv_write with new uuid failed "
" for %s. " , pv_name ) ;
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2005-05-24 21:38:26 +04:00
return 0 ;
}
pv - > vg_name = orig_vg_name ;
pv - > pe_alloc_count = orig_pe_alloc_count ;
2008-07-16 14:46:12 +04:00
pv - > pe_size = orig_pe_size ;
pv - > pe_start = orig_pe_start ;
pv - > pe_count = orig_pe_count ;
2005-05-24 21:38:26 +04:00
}
2001-10-03 21:03:25 +04:00
}
2002-01-30 18:04:48 +03:00
log_verbose ( " Updating physical volume \" %s \" " , pv_name ) ;
2007-11-02 17:54:40 +03:00
if ( ! is_orphan ( pv ) ) {
2003-07-05 02:34:56 +04:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) ) {
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-01-30 18:04:48 +03:00
log_error ( " Failed to store physical volume \" %s \" in "
" volume group \" %s \" " , pv_name , vg - > name ) ;
2001-10-17 19:29:31 +04:00
return 0 ;
2001-10-03 21:03:25 +04:00
}
2002-01-07 14:12:11 +03:00
backup ( vg ) ;
2007-11-02 23:40:05 +03:00
} else if ( ! ( pv_write ( cmd , pv , NULL , INT64_C ( - 1 ) ) ) ) {
unlock_vg ( cmd , vg_name ) ;
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
2007-11-02 23:40:05 +03:00
unlock_vg ( cmd , vg_name ) ;
2002-01-30 18:04:48 +03:00
log_print ( " Physical volume \" %s \" changed " , pv_name ) ;
2001-10-17 19:29:31 +04:00
return 1 ;
2001-10-03 21:03:25 +04:00
}
2002-11-18 17:04:08 +03:00
int pvchange ( struct cmd_context * cmd , int argc , char * * argv )
{
int opt = 0 ;
int done = 0 ;
int total = 0 ;
struct physical_volume * pv ;
char * pv_name ;
2003-10-16 00:02:46 +04:00
struct pv_list * pvl ;
2008-11-04 01:14:30 +03:00
struct dm_list * pvslist ;
struct dm_list mdas ;
2002-11-18 17:04:08 +03:00
2004-03-08 20:19:15 +03:00
if ( arg_count ( cmd , allocatable_ARG ) + arg_count ( cmd , addtag_ARG ) +
arg_count ( cmd , deltag_ARG ) + arg_count ( cmd , uuid_ARG ) ! = 1 ) {
log_error ( " Please give exactly one option of -x, -uuid, "
" --addtag or --deltag " ) ;
2002-11-18 17:04:08 +03:00
return EINVALID_CMD_LINE ;
}
if ( ! ( arg_count ( cmd , all_ARG ) ) & & ! argc ) {
log_error ( " Please give a physical volume path " ) ;
return EINVALID_CMD_LINE ;
}
if ( arg_count ( cmd , all_ARG ) & & argc ) {
log_error ( " Option a and PhysicalVolumePath are exclusive " ) ;
return EINVALID_CMD_LINE ;
}
if ( argc ) {
log_verbose ( " Using physical volume(s) on command line " ) ;
for ( ; opt < argc ; opt + + ) {
pv_name = argv [ opt ] ;
2008-11-04 01:14:30 +03:00
dm_list_init ( & mdas ) ;
2009-02-26 02:29:06 +03:00
if ( ! ( pv = pv_read ( cmd , pv_name , & mdas , NULL , 1 , 0 ) ) ) {
2004-06-19 23:27:00 +04:00
log_error ( " Failed to read physical volume %s " ,
pv_name ) ;
2002-11-18 17:04:08 +03:00
continue ;
}
2008-07-31 16:28:51 +04:00
/*
* If a PV has no MDAs it may appear to be an
* orphan until the metadata is read off
* another PV in the same VG . Detecting this
* means checking every VG by scanning every
* PV on the system .
*/
2008-11-04 01:14:30 +03:00
if ( is_orphan ( pv ) & & ! dm_list_size ( & mdas ) ) {
2008-07-31 16:28:51 +04:00
if ( ! scan_vgs_for_pvs ( cmd ) ) {
log_error ( " Rescan for PVs without "
" metadata areas failed. " ) ;
continue ;
}
if ( ! ( pv = pv_read ( cmd , pv_name ,
2009-02-26 02:29:06 +03:00
NULL , NULL , 1 , 0 ) ) ) {
2008-07-31 16:28:51 +04:00
log_error ( " Failed to read "
" physical volume %s " ,
pv_name ) ;
continue ;
}
}
2002-11-18 17:04:08 +03:00
total + + ;
2002-12-20 02:25:55 +03:00
done + = _pvchange_single ( cmd , pv , NULL ) ;
2002-11-18 17:04:08 +03:00
}
} else {
log_verbose ( " Scanning for physical volume names " ) ;
2002-12-20 02:25:55 +03:00
if ( ! ( pvslist = get_pvs ( cmd ) ) ) {
2002-11-18 17:04:08 +03:00
return ECMD_FAILED ;
}
2008-11-04 01:14:30 +03:00
dm_list_iterate_items ( pvl , pvslist ) {
2002-11-18 17:04:08 +03:00
total + + ;
2003-10-16 00:02:46 +04:00
done + = _pvchange_single ( cmd , pvl - > pv , NULL ) ;
2002-11-18 17:04:08 +03:00
}
}
log_print ( " %d physical volume%s changed / %d physical volume%s "
" not changed " ,
2004-01-13 21:42:05 +03:00
done , done = = 1 ? " " : " s " ,
total - done , ( total - done ) = = 1 ? " " : " s " ) ;
2002-11-18 17:04:08 +03:00
2003-10-22 02:06:07 +04:00
return ( total = = done ) ? ECMD_PROCESSED : ECMD_FAILED ;
2002-11-18 17:04:08 +03:00
}