2001-10-03 17:03:25 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-20 20:55:30 +00:00
* Copyright ( C ) 2004 - 2007 Red Hat , Inc . All rights reserved .
2001-10-03 17:03:25 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
2001-10-03 17:03:25 +00:00
*
2004-03-30 19:35:44 +00: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-20 20:55:30 +00:00
* of the GNU Lesser General Public License v .2 .1 .
2001-10-03 17:03:25 +00:00
*
2007-08-20 20:55:30 +00:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 19:35:44 +00: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 17:03:25 +00:00
*/
# include "tools.h"
2002-11-18 14:04:08 +00:00
/* FIXME Locking. PVs in VG. */
2001-10-03 17:03:25 +00:00
2002-12-19 23:25:55 +00:00
static int _pvchange_single ( struct cmd_context * cmd , struct physical_volume * pv ,
2006-05-09 21:23:51 +00:00
void * handle __attribute ( ( unused ) ) )
2001-10-03 17:03:25 +00:00
{
2001-10-17 15:29:31 +00:00
struct volume_group * vg = NULL ;
2007-11-02 20:40:05 +00:00
const char * vg_name = NULL ;
2002-01-21 14:28:12 +00:00
struct pv_list * pvl ;
2002-11-18 14:04:08 +00:00
uint64_t sector ;
2005-05-24 17:38:26 +00:00
uint32_t orig_pe_alloc_count ;
2008-07-16 10:46:12 +00:00
/* FIXME Next three only required for format1. */
uint32_t orig_pe_count , orig_pe_size ;
uint64_t orig_pe_start ;
2001-10-03 17:03:25 +00:00
2007-10-12 14:29:32 +00:00
const char * pv_name = pv_dev_name ( pv ) ;
2004-03-08 17:19:15 +00:00
const char * tag = NULL ;
2005-05-24 17:38:26 +00:00
const char * orig_vg_name ;
2006-11-30 23:11:42 +00:00
char uuid [ 64 ] __attribute ( ( aligned ( 8 ) ) ) ;
2001-10-03 17:03:25 +00:00
2004-01-13 18:42:05 +00:00
int allocatable = 0 ;
2004-03-08 17:19:15 +00:00
int tagarg = 0 ;
2009-04-10 10:01:38 +00:00
int r = 0 ;
2004-03-08 17:19:15 +00:00
if ( arg_count ( cmd , addtag_ARG ) )
tagarg = addtag_ARG ;
else if ( arg_count ( cmd , deltag_ARG ) )
tagarg = deltag_ARG ;
2004-01-13 18:42:05 +00:00
if ( arg_count ( cmd , allocatable_ARG ) )
allocatable = ! strcmp ( arg_str_value ( cmd , allocatable_ARG , " n " ) ,
" y " ) ;
2004-03-08 17:19:15 +00:00
else if ( tagarg & & ! ( tag = arg_str_value ( cmd , tagarg , NULL ) ) ) {
log_error ( " Failed to get tag " ) ;
return 0 ;
}
2001-10-03 17:03:25 +00:00
2002-01-29 17:23:33 +00:00
/* If in a VG, must change using volume group. */
2007-11-02 14:54:40 +00:00
if ( ! is_orphan ( pv ) ) {
2007-11-02 20:40:05 +00:00
vg_name = pv_vg_name ( pv ) ;
2002-01-29 17:23:33 +00:00
2007-11-15 02:20:03 +00:00
log_verbose ( " Finding volume group %s of physical volume %s " ,
vg_name , pv_name ) ;
2009-07-01 16:59:37 +00:00
vg = vg_read_for_update ( cmd , vg_name , NULL , 0 ) ;
2009-07-07 01:18:35 +00:00
if ( vg_read_error ( vg ) ) {
vg_release ( vg ) ;
2007-11-15 02:20:03 +00:00
return_0 ;
2009-07-07 01:18:35 +00:00
}
2002-01-29 17:23:33 +00:00
2002-01-21 14:28:12 +00:00
if ( ! ( pvl = find_pv_in_vg ( vg , pv_name ) ) ) {
2009-07-08 18:15:51 +00:00
log_error ( " Unable to find \" %s \" in volume group \" %s \" " ,
pv_name , vg - > name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2001-10-17 15:29:31 +00:00
}
2004-03-08 17:19:15 +00:00
if ( tagarg & & ! ( vg - > fid - > fmt - > features & FMT_TAGS ) ) {
log_error ( " Volume group containing %s does not "
" support tags " , pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2004-03-08 17:19:15 +00:00
}
2004-01-13 18:42:05 +00:00
if ( arg_count ( cmd , uuid_ARG ) & & lvs_in_vg_activated ( vg ) ) {
log_error ( " Volume group containing %s has active "
" logical volumes " , pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2004-01-13 18:42:05 +00:00
}
2002-01-21 16:05:23 +00:00
pv = pvl - > pv ;
2002-01-09 13:17:14 +00:00
if ( ! archive ( vg ) )
2009-04-10 10:01:38 +00:00
goto out ;
2002-11-18 14:04:08 +00:00
} else {
2004-03-08 17:19:15 +00:00
if ( tagarg ) {
log_error ( " Can't change tag on Physical Volume %s not "
" in volume group " , pv_name ) ;
return 0 ;
}
2007-11-02 20:40:05 +00:00
vg_name = VG_ORPHANS ;
if ( ! lock_vol ( cmd , vg_name , LCK_VG_WRITE ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Can't get lock for orphans " ) ;
2004-01-13 18:42:05 +00:00
return 0 ;
2002-11-18 14:04:08 +00:00
}
2009-02-25 23:29:06 +00:00
if ( ! ( pv = pv_read ( cmd , pv_name , NULL , & sector , 1 , 0 ) ) ) {
2007-11-02 20:40:05 +00:00
unlock_vg ( cmd , vg_name ) ;
2002-11-18 14:04:08 +00:00
log_error ( " Unable to read PV \" %s \" " , pv_name ) ;
return 0 ;
}
2001-10-17 15:29:31 +00:00
}
2001-10-03 17:03:25 +00:00
2004-01-13 18:42:05 +00:00
if ( arg_count ( cmd , allocatable_ARG ) ) {
2007-11-02 14:54:40 +00:00
if ( is_orphan ( pv ) & &
2004-01-13 18:42:05 +00:00
! ( pv - > fmt - > features & FMT_ORPHAN_ALLOCATABLE ) ) {
log_error ( " Allocatability not supported by orphan "
" %s format PV %s " , pv - > fmt - > name , pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2004-01-13 18:42:05 +00:00
}
2001-10-03 17:03:25 +00:00
2004-01-13 18:42:05 +00:00
/* change allocatability for a PV */
2007-06-15 22:16:55 +00:00
if ( allocatable & & ( pv_status ( pv ) & ALLOCATABLE_PV ) ) {
2004-01-13 18:42:05 +00:00
log_error ( " Physical volume \" %s \" is already "
" allocatable " , pv_name ) ;
2009-04-10 10:01:38 +00:00
r = 1 ;
goto out ;
2004-01-13 18:42:05 +00:00
}
2001-10-03 17:03:25 +00:00
2007-06-15 22:16:55 +00:00
if ( ! allocatable & & ! ( pv_status ( pv ) & ALLOCATABLE_PV ) ) {
2004-01-13 18:42:05 +00:00
log_error ( " Physical volume \" %s \" is already "
" unallocatable " , pv_name ) ;
2009-04-10 10:01:38 +00:00
r = 1 ;
goto out ;
2004-01-13 18:42:05 +00:00
}
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 17:19:15 +00: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 ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2004-03-08 17:19:15 +00:00
}
} else {
if ( ! str_list_del ( & pv - > tags , tag ) ) {
log_error ( " Failed to remove tag %s from "
" physical volume " " %s " , tag , pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2004-03-08 17:19:15 +00:00
}
}
2001-10-17 15:29:31 +00:00
} else {
2004-01-13 18:42:05 +00:00
/* --uuid: Change PV ID randomly */
2005-01-20 18:11:53 +00:00
if ( ! id_create ( & pv - > id ) ) {
log_error ( " Failed to generate new random UUID for %s. " ,
pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2005-05-24 17:38:26 +00:00
}
2009-04-10 10:01:38 +00:00
if ( ! id_write_format ( & pv - > id , uuid , sizeof ( uuid ) ) )
goto_out ;
2005-05-24 17:38:26 +00:00
log_verbose ( " Changing uuid of %s to %s. " , pv_name , uuid ) ;
2007-11-02 14:54:40 +00:00
if ( ! is_orphan ( pv ) ) {
2007-06-15 22:16:55 +00:00
orig_vg_name = pv_vg_name ( pv ) ;
orig_pe_alloc_count = pv_pe_alloc_count ( pv ) ;
2008-07-16 10:46:12 +00: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 15:47:28 +00:00
pv - > vg_name = pv - > fmt - > orphan_vg_name ;
2005-05-24 17:38:26 +00: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 ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2005-05-24 17:38:26 +00:00
}
pv - > vg_name = orig_vg_name ;
pv - > pe_alloc_count = orig_pe_alloc_count ;
2008-07-16 10:46:12 +00:00
pv - > pe_size = orig_pe_size ;
pv - > pe_start = orig_pe_start ;
pv - > pe_count = orig_pe_count ;
2005-05-24 17:38:26 +00:00
}
2001-10-03 17:03:25 +00:00
}
2002-01-30 15:04:48 +00:00
log_verbose ( " Updating physical volume \" %s \" " , pv_name ) ;
2007-11-02 14:54:40 +00:00
if ( ! is_orphan ( pv ) ) {
2003-07-04 22:34:56 +00:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) ) {
2002-01-30 15:04:48 +00:00
log_error ( " Failed to store physical volume \" %s \" in "
" volume group \" %s \" " , pv_name , vg - > name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2001-10-03 17:03:25 +00:00
}
2002-01-07 11:12:11 +00:00
backup ( vg ) ;
2007-11-02 20:40:05 +00:00
} else if ( ! ( pv_write ( cmd , pv , NULL , INT64_C ( - 1 ) ) ) ) {
log_error ( " Failed to store physical volume \" %s \" " ,
pv_name ) ;
2009-04-10 10:01:38 +00:00
goto out ;
2001-10-03 17:03:25 +00:00
}
2001-10-17 15:29:31 +00:00
2002-01-30 15:04:48 +00:00
log_print ( " Physical volume \" %s \" changed " , pv_name ) ;
2009-04-10 10:01:38 +00:00
r = 1 ;
out :
2009-05-21 03:04:52 +00:00
unlock_and_release_vg ( cmd , vg , vg_name ) ;
2009-04-10 10:01:38 +00:00
return r ;
2001-10-17 15:29:31 +00:00
2001-10-03 17:03:25 +00:00
}
2002-11-18 14:04:08 +00: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-15 20:02:46 +00:00
struct pv_list * pvl ;
2008-11-03 22:14:30 +00:00
struct dm_list * pvslist ;
struct dm_list mdas ;
2002-11-18 14:04:08 +00:00
2004-03-08 17:19:15 +00: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 14:04:08 +00: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-03 22:14:30 +00:00
dm_list_init ( & mdas ) ;
2009-02-25 23:29:06 +00:00
if ( ! ( pv = pv_read ( cmd , pv_name , & mdas , NULL , 1 , 0 ) ) ) {
2004-06-19 19:27:00 +00:00
log_error ( " Failed to read physical volume %s " ,
pv_name ) ;
2002-11-18 14:04:08 +00:00
continue ;
}
2008-07-31 12:28:51 +00: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-03 22:14:30 +00:00
if ( is_orphan ( pv ) & & ! dm_list_size ( & mdas ) ) {
2008-07-31 12:28:51 +00: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-25 23:29:06 +00:00
NULL , NULL , 1 , 0 ) ) ) {
2008-07-31 12:28:51 +00:00
log_error ( " Failed to read "
" physical volume %s " ,
pv_name ) ;
continue ;
}
}
2002-11-18 14:04:08 +00:00
total + + ;
2002-12-19 23:25:55 +00:00
done + = _pvchange_single ( cmd , pv , NULL ) ;
2002-11-18 14:04:08 +00:00
}
} else {
log_verbose ( " Scanning for physical volume names " ) ;
2002-12-19 23:25:55 +00:00
if ( ! ( pvslist = get_pvs ( cmd ) ) ) {
2009-09-14 22:47:49 +00:00
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
}
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( pvl , pvslist ) {
2002-11-18 14:04:08 +00:00
total + + ;
2003-10-15 20:02:46 +00:00
done + = _pvchange_single ( cmd , pvl - > pv , NULL ) ;
2002-11-18 14:04:08 +00:00
}
}
log_print ( " %d physical volume%s changed / %d physical volume%s "
" not changed " ,
2004-01-13 18:42:05 +00:00
done , done = = 1 ? " " : " s " ,
total - done , ( total - done ) = = 1 ? " " : " s " ) ;
2002-11-18 14:04:08 +00:00
2003-10-21 22:06:07 +00:00
return ( total = = done ) ? ECMD_PROCESSED : ECMD_FAILED ;
2002-11-18 14:04:08 +00:00
}