2001-10-30 14:32:48 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2011-11-18 19:22:49 +00:00
* Copyright ( C ) 2004 - 2011 Red Hat , Inc . All rights reserved .
2001-10-30 14:32:48 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
2001-10-30 14:32:48 +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-30 14:32:48 +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-30 14:32:48 +00:00
*/
# include "tools.h"
2002-02-11 21:00:35 +00:00
static int lvchange_permission ( struct cmd_context * cmd ,
struct logical_volume * lv )
2001-10-30 14:32:48 +00:00
{
2002-12-19 23:25:55 +00:00
uint32_t lv_access ;
2007-01-10 19:56:39 +00:00
struct lvinfo info ;
2009-04-21 14:31:57 +00:00
int r = 0 ;
2001-10-30 14:32:48 +00:00
2002-12-19 23:25:55 +00:00
lv_access = arg_uint_value ( cmd , permission_ARG , 0 ) ;
2001-10-30 14:32:48 +00:00
if ( ( lv_access & LVM_WRITE ) & & ( lv - > status & LVM_WRITE ) ) {
2002-02-11 21:00:35 +00:00
log_error ( " Logical volume \" %s \" is already writable " ,
lv - > name ) ;
2001-10-30 14:32:48 +00:00
return 0 ;
}
if ( ! ( lv_access & LVM_WRITE ) & & ! ( lv - > status & LVM_WRITE ) ) {
2002-02-11 21:00:35 +00:00
log_error ( " Logical volume \" %s \" is already read only " ,
lv - > name ) ;
2001-10-30 14:32:48 +00:00
return 0 ;
}
2008-04-10 17:09:32 +00:00
if ( ( lv - > status & MIRRORED ) & & ( vg_is_clustered ( lv - > vg ) ) & &
2010-08-17 16:25:32 +00:00
lv_info ( cmd , lv , 0 , & info , 0 , 0 ) & & info . exists ) {
2007-01-10 19:56:39 +00:00
log_error ( " Cannot change permissions of mirror \" %s \" "
" while active. " , lv - > name ) ;
return 0 ;
}
2011-12-01 00:04:21 +00:00
/* Not allowed to change permissions on RAID sub-LVs directly */
if ( ( lv - > status & RAID_META ) | | ( lv - > status & RAID_IMAGE ) ) {
log_error ( " Cannot change permissions of RAID %s \" %s \" " ,
( lv - > status & RAID_IMAGE ) ? " image " :
" metadata area " , lv - > name ) ;
return 0 ;
}
2001-10-30 14:32:48 +00:00
if ( lv_access & LVM_WRITE ) {
lv - > status | = LVM_WRITE ;
2002-02-11 21:00:35 +00:00
log_verbose ( " Setting logical volume \" %s \" read/write " ,
lv - > name ) ;
2001-10-30 14:32:48 +00:00
} else {
lv - > status & = ~ LVM_WRITE ;
2002-02-11 21:00:35 +00:00
log_verbose ( " Setting logical volume \" %s \" read-only " ,
lv - > name ) ;
2001-10-30 14:32:48 +00:00
}
2003-07-04 22:34:56 +00:00
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! vg_write ( lv - > vg ) )
return_0 ;
2003-07-04 22:34:56 +00:00
2005-08-15 12:00:04 +00:00
if ( ! suspend_lv ( cmd , lv ) ) {
2002-02-20 21:30:27 +00:00
log_error ( " Failed to lock %s " , lv - > name ) ;
2003-07-04 22:34:56 +00:00
vg_revert ( lv - > vg ) ;
2009-04-21 14:31:57 +00:00
goto out ;
2002-02-20 21:30:27 +00:00
}
2003-07-04 22:34:56 +00:00
if ( ! vg_commit ( lv - > vg ) ) {
2010-01-05 21:07:31 +00:00
if ( ! resume_lv ( cmd , lv ) )
stack ;
2009-04-21 14:31:57 +00:00
goto_out ;
2002-02-20 21:30:27 +00:00
}
2001-10-31 17:59:52 +00:00
2002-01-30 15:04:48 +00:00
log_very_verbose ( " Updating permissions for \" %s \" in kernel " , lv - > name ) ;
2005-08-15 12:00:04 +00:00
if ( ! resume_lv ( cmd , lv ) ) {
2002-04-24 18:20:51 +00:00
log_error ( " Problem reactivating %s " , lv - > name ) ;
2009-04-21 14:31:57 +00:00
goto out ;
2002-04-24 18:20:51 +00:00
}
2001-10-31 17:59:52 +00:00
2009-04-22 12:46:25 +00:00
r = 1 ;
2009-04-21 14:31:57 +00:00
out :
backup ( lv - > vg ) ;
return r ;
2001-10-30 14:32:48 +00:00
}
2007-01-19 22:21:45 +00:00
static int lvchange_monitoring ( struct cmd_context * cmd ,
struct logical_volume * lv )
2006-05-12 19:16:48 +00:00
{
struct lvinfo info ;
2012-03-23 09:58:04 +00:00
if ( ! lv_info ( cmd , lv , lv_is_thin_pool ( lv ) ? 1 : 0 ,
& info , 0 , 0 ) | | ! info . exists ) {
2006-05-12 19:16:48 +00:00
log_error ( " Logical volume, %s, is not active " , lv - > name ) ;
return 0 ;
}
2007-01-19 22:21:45 +00:00
/* do not monitor pvmove lv's */
2006-05-12 19:16:48 +00:00
if ( lv - > status & PVMOVE )
return 1 ;
2007-01-24 23:43:27 +00:00
if ( ( dmeventd_monitor_mode ( ) ! = DMEVENTD_MONITOR_IGNORE ) & &
2010-08-17 16:25:32 +00:00
! monitor_dev_for_events ( cmd , lv , 0 , dmeventd_monitor_mode ( ) ) )
2010-08-16 22:54:35 +00:00
return_0 ;
2006-05-12 19:16:48 +00:00
2007-01-19 22:21:45 +00:00
return 1 ;
2006-05-12 19:16:48 +00:00
}
2010-01-05 20:56:51 +00:00
static int lvchange_background_polling ( struct cmd_context * cmd ,
struct logical_volume * lv )
{
struct lvinfo info ;
2010-08-17 16:25:32 +00:00
if ( ! lv_info ( cmd , lv , 0 , & info , 0 , 0 ) | | ! info . exists ) {
2010-01-05 20:56:51 +00:00
log_error ( " Logical volume, %s, is not active " , lv - > name ) ;
return 0 ;
}
if ( background_polling ( ) )
lv_spawn_background_polling ( cmd , lv ) ;
return 1 ;
}
2002-02-11 21:00:35 +00:00
static int lvchange_availability ( struct cmd_context * cmd ,
struct logical_volume * lv )
2001-10-30 14:32:48 +00:00
{
2004-05-24 13:44:10 +00:00
int activate ;
2001-10-30 14:32:48 +00:00
2004-05-24 13:44:10 +00:00
activate = arg_uint_value ( cmd , available_ARG , 0 ) ;
2001-10-30 14:32:48 +00:00
2011-11-18 19:22:49 +00:00
if ( lv_is_cow ( lv ) & & ! lv_is_virtual_origin ( origin_from_cow ( lv ) ) )
lv = origin_from_cow ( lv ) ;
2004-06-16 17:13:41 +00:00
if ( activate = = CHANGE_ALN ) {
log_verbose ( " Deactivating logical volume \" %s \" locally " ,
lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! deactivate_lv_local ( cmd , lv ) )
return_0 ;
2004-06-16 17:13:41 +00:00
} else if ( activate = = CHANGE_AN ) {
log_verbose ( " Deactivating logical volume \" %s \" " , lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! deactivate_lv ( cmd , lv ) )
return_0 ;
2004-06-16 17:13:41 +00:00
} else {
2011-10-28 20:29:06 +00:00
if ( ( activate = = CHANGE_AE ) | |
lv_is_origin ( lv ) | |
lv_is_thin_type ( lv ) ) {
2004-06-16 17:13:41 +00:00
log_verbose ( " Activating logical volume \" %s \" "
" exclusively " , lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! activate_lv_excl ( cmd , lv ) )
return_0 ;
2004-06-16 17:13:41 +00:00
} else if ( activate = = CHANGE_ALY ) {
log_verbose ( " Activating logical volume \" %s \" locally " ,
lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! activate_lv_local ( cmd , lv ) )
return_0 ;
2004-06-16 17:13:41 +00:00
} else {
log_verbose ( " Activating logical volume \" %s \" " ,
lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! activate_lv ( cmd , lv ) )
return_0 ;
2004-05-05 17:56:20 +00:00
}
2004-06-16 17:13:41 +00:00
2010-01-05 20:56:51 +00:00
if ( background_polling ( ) )
lv_spawn_background_polling ( cmd , lv ) ;
2002-01-07 22:36:12 +00:00
}
2001-10-30 14:32:48 +00:00
return 1 ;
}
2004-03-26 21:24:03 +00:00
static int lvchange_refresh ( struct cmd_context * cmd , struct logical_volume * lv )
{
log_verbose ( " Refreshing logical volume \" %s \" (if active) " , lv - > name ) ;
2010-08-17 19:25:05 +00:00
2008-12-19 14:22:48 +00:00
return lv_refresh ( cmd , lv ) ;
2004-03-26 21:24:03 +00:00
}
2006-10-24 17:18:25 +00:00
static int lvchange_resync ( struct cmd_context * cmd ,
2006-10-23 23:03:55 +00:00
struct logical_volume * lv )
{
int active = 0 ;
2008-05-21 14:10:11 +00:00
int monitored ;
2006-10-23 23:03:55 +00:00
struct lvinfo info ;
struct logical_volume * log_lv ;
if ( ! ( lv - > status & MIRRORED ) ) {
log_error ( " Unable to resync %s because it is not mirrored. " ,
lv - > name ) ;
return 1 ;
}
if ( lv - > status & PVMOVE ) {
log_error ( " Unable to resync pvmove volume %s " , lv - > name ) ;
return 0 ;
}
if ( lv - > status & LOCKED ) {
log_error ( " Unable to resync locked volume %s " , lv - > name ) ;
return 0 ;
}
2010-08-17 16:25:32 +00:00
if ( lv_info ( cmd , lv , 0 , & info , 1 , 0 ) ) {
2006-10-23 23:03:55 +00:00
if ( info . open_count ) {
log_error ( " Can't resync open logical volume \" %s \" " ,
lv - > name ) ;
2009-05-20 09:55:33 +00:00
return 0 ;
2006-10-23 23:03:55 +00:00
}
2006-10-24 15:30:33 +00:00
if ( info . exists ) {
if ( ! arg_count ( cmd , yes_ARG ) & &
yes_no_prompt ( " Do you really want to deactivate "
2006-10-23 23:03:55 +00:00
" logical volume %s to resync it? [y/n]: " ,
lv - > name ) = = ' n ' ) {
2009-12-03 19:18:33 +00:00
log_error ( " Logical volume \" %s \" not resynced " ,
2006-10-23 23:03:55 +00:00
lv - > name ) ;
2009-05-20 09:55:33 +00:00
return 0 ;
2006-10-23 23:03:55 +00:00
}
2007-06-15 10:11:14 +00:00
if ( sigint_caught ( ) )
2009-05-20 09:55:33 +00:00
return 0 ;
2007-06-15 10:11:14 +00:00
2006-10-24 15:30:33 +00:00
active = 1 ;
}
2006-10-23 23:03:55 +00:00
}
2008-05-21 14:10:11 +00:00
/* Activate exclusively to ensure no nodes still have LV active */
monitored = dmeventd_monitor_mode ( ) ;
init_dmeventd_monitor ( 0 ) ;
2006-10-23 23:03:55 +00:00
if ( ! deactivate_lv ( cmd , lv ) ) {
log_error ( " Unable to deactivate %s for resync " , lv - > name ) ;
return 0 ;
}
2009-05-20 09:55:33 +00:00
if ( vg_is_clustered ( lv - > vg ) & & lv_is_active ( lv ) ) {
log_error ( " Can't get exclusive access to clustered volume %s " ,
lv - > name ) ;
return 0 ;
}
2008-05-21 14:10:11 +00:00
init_dmeventd_monitor ( monitored ) ;
2006-10-23 23:03:55 +00:00
log_lv = first_seg ( lv ) - > log_lv ;
log_very_verbose ( " Starting resync of %s%s%s mirror \" %s \" " ,
( active ) ? " active " : " " ,
2008-04-10 17:09:32 +00:00
vg_is_clustered ( lv - > vg ) ? " clustered " : " " ,
2006-10-23 23:03:55 +00:00
( log_lv ) ? " disk-logged " : " core-logged " ,
lv - > name ) ;
/*
* If this mirror has a core log ( i . e . ! log_lv ) ,
* then simply deactivating / activating will cause
* it to reset the sync status . We only need to
* worry about persistent logs .
*/
2011-03-29 12:51:57 +00:00
if ( ! log_lv & & ! ( lv - > status & LV_NOTSYNCED ) ) {
2006-10-23 23:03:55 +00:00
if ( active & & ! activate_lv ( cmd , lv ) ) {
log_error ( " Failed to reactivate %s to resynchronize "
" mirror " , lv - > name ) ;
return 0 ;
}
return 1 ;
}
2011-03-29 12:51:57 +00:00
lv - > status & = ~ LV_NOTSYNCED ;
2006-10-23 23:03:55 +00:00
if ( log_lv ) {
/* Separate mirror log so we can clear it */
2008-01-26 00:25:04 +00:00
detach_mirror_log ( first_seg ( lv ) ) ;
2006-10-23 23:03:55 +00:00
if ( ! vg_write ( lv - > vg ) ) {
log_error ( " Failed to write intermediate VG metadata. " ) ;
2008-01-26 00:25:04 +00:00
if ( ! attach_mirror_log ( first_seg ( lv ) , log_lv ) )
stack ;
if ( active & & ! activate_lv ( cmd , lv ) )
stack ;
2006-10-23 23:03:55 +00:00
return 0 ;
}
if ( ! vg_commit ( lv - > vg ) ) {
log_error ( " Failed to commit intermediate VG metadata. " ) ;
2008-01-26 00:25:04 +00:00
if ( ! attach_mirror_log ( first_seg ( lv ) , log_lv ) )
stack ;
if ( active & & ! activate_lv ( cmd , lv ) )
stack ;
2006-10-23 23:03:55 +00:00
return 0 ;
}
2009-04-21 14:31:57 +00:00
backup ( lv - > vg ) ;
2006-10-23 23:03:55 +00:00
if ( ! activate_lv ( cmd , log_lv ) ) {
log_error ( " Unable to activate %s for mirror log resync " ,
log_lv - > name ) ;
return 0 ;
}
log_very_verbose ( " Clearing log device %s " , log_lv - > name ) ;
2006-11-03 21:07:15 +00:00
if ( ! set_lv ( cmd , log_lv , log_lv - > size , 0 ) ) {
2006-10-23 23:03:55 +00:00
log_error ( " Unable to reset sync status for %s " , lv - > name ) ;
if ( ! deactivate_lv ( cmd , log_lv ) )
log_error ( " Failed to deactivate log LV after "
" wiping failed " ) ;
return 0 ;
}
if ( ! deactivate_lv ( cmd , log_lv ) ) {
log_error ( " Unable to deactivate log LV %s after wiping "
" for resync " , log_lv - > name ) ;
return 0 ;
}
/* Put mirror log back in place */
2008-01-26 00:25:04 +00:00
if ( ! attach_mirror_log ( first_seg ( lv ) , log_lv ) )
stack ;
2006-10-23 23:03:55 +00:00
}
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
if ( ! vg_write ( lv - > vg ) | | ! vg_commit ( lv - > vg ) ) {
log_error ( " Failed to update metadata on disk. " ) ;
return 0 ;
}
if ( active & & ! activate_lv ( cmd , lv ) ) {
log_error ( " Failed to reactivate %s after resync " , lv - > name ) ;
return 0 ;
}
return 1 ;
}
2004-05-18 22:12:53 +00:00
static int lvchange_alloc ( struct cmd_context * cmd , struct logical_volume * lv )
2001-10-30 14:32:48 +00:00
{
2002-07-11 14:21:49 +00:00
int want_contiguous = 0 ;
2004-05-18 22:12:53 +00:00
alloc_policy_t alloc ;
2001-10-30 14:32:48 +00:00
2004-05-18 22:12:53 +00:00
want_contiguous = strcmp ( arg_str_value ( cmd , contiguous_ARG , " n " ) , " n " ) ;
alloc = want_contiguous ? ALLOC_CONTIGUOUS : ALLOC_INHERIT ;
2012-02-28 14:24:57 +00:00
alloc = ( alloc_policy_t ) arg_uint_value ( cmd , alloc_ARG , alloc ) ;
2001-10-30 14:32:48 +00:00
2004-05-18 22:12:53 +00:00
if ( alloc = = lv - > alloc ) {
2002-01-30 15:04:48 +00:00
log_error ( " Allocation policy of logical volume \" %s \" is "
2004-05-18 22:12:53 +00:00
" already %s " , lv - > name , get_alloc_string ( alloc ) ) ;
2001-10-30 14:32:48 +00:00
return 0 ;
}
2004-05-18 22:12:53 +00:00
lv - > alloc = alloc ;
2001-10-30 14:32:48 +00:00
2004-05-18 22:12:53 +00:00
/* FIXME If contiguous, check existing extents already are */
2001-10-30 14:32:48 +00:00
2004-05-18 22:12:53 +00:00
log_verbose ( " Setting contiguous allocation policy for \" %s \" to %s " ,
lv - > name , get_alloc_string ( alloc ) ) ;
2001-10-30 14:32:48 +00:00
2003-07-04 22:34:56 +00:00
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
2004-05-18 22:12:53 +00:00
2009-04-21 14:31:57 +00:00
/* No need to suspend LV for this change */
if ( ! vg_write ( lv - > vg ) | | ! vg_commit ( lv - > vg ) )
2008-01-30 13:19:47 +00:00
return_0 ;
2003-07-04 22:34:56 +00:00
backup ( lv - > vg ) ;
2001-10-30 14:32:48 +00:00
return 1 ;
}
2002-02-11 21:00:35 +00:00
static int lvchange_readahead ( struct cmd_context * cmd ,
struct logical_volume * lv )
2001-10-30 14:32:48 +00:00
{
2006-05-09 21:23:51 +00:00
unsigned read_ahead = 0 ;
2007-12-05 19:24:32 +00:00
unsigned pagesize = ( unsigned ) lvm_getpagesize ( ) > > SECTOR_SHIFT ;
2009-04-21 14:31:57 +00:00
int r = 0 ;
2001-10-30 14:32:48 +00:00
2002-12-19 23:25:55 +00:00
read_ahead = arg_uint_value ( cmd , readahead_ARG , 0 ) ;
2001-10-30 14:32:48 +00:00
2007-11-09 16:51:54 +00:00
if ( read_ahead ! = DM_READ_AHEAD_AUTO & &
( lv - > vg - > fid - > fmt - > features & FMT_RESTRICTED_READAHEAD ) & &
( read_ahead < 2 | | read_ahead > 120 ) ) {
log_error ( " Metadata only supports readahead values between 2 and 120. " ) ;
2001-10-30 14:32:48 +00:00
return 0 ;
}
2007-12-05 19:24:32 +00:00
if ( read_ahead ! = DM_READ_AHEAD_AUTO & &
read_ahead ! = DM_READ_AHEAD_NONE & & read_ahead % pagesize ) {
2009-06-06 22:00:20 +00:00
if ( read_ahead < pagesize )
read_ahead = pagesize ;
else
read_ahead = ( read_ahead / pagesize ) * pagesize ;
log_warn ( " WARNING: Overriding readahead to %u sectors, a multiple "
" of %uK page size. " , read_ahead , pagesize > > 1 ) ;
2007-12-05 19:24:32 +00:00
}
2001-10-30 14:32:48 +00:00
if ( lv - > read_ahead = = read_ahead ) {
2008-06-18 11:32:14 +00:00
if ( read_ahead = = DM_READ_AHEAD_AUTO )
log_error ( " Read ahead is already auto for \" %s \" " , lv - > name ) ;
else
log_error ( " Read ahead is already %u for \" %s \" " ,
read_ahead , lv - > name ) ;
2001-10-30 14:32:48 +00:00
return 0 ;
}
lv - > read_ahead = read_ahead ;
2002-02-20 21:30:27 +00:00
2002-02-11 21:00:35 +00:00
log_verbose ( " Setting read ahead to %u for \" %s \" " , read_ahead ,
lv - > name ) ;
2001-10-30 14:32:48 +00:00
2003-07-04 22:34:56 +00:00
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
2008-01-30 13:19:47 +00:00
if ( ! vg_write ( lv - > vg ) )
return_0 ;
2003-07-04 22:34:56 +00:00
2005-08-15 12:00:04 +00:00
if ( ! suspend_lv ( cmd , lv ) ) {
2002-02-20 21:30:27 +00:00
log_error ( " Failed to lock %s " , lv - > name ) ;
2003-07-04 22:34:56 +00:00
vg_revert ( lv - > vg ) ;
2009-04-21 14:31:57 +00:00
goto out ;
2002-02-20 21:30:27 +00:00
}
2001-10-31 17:59:52 +00:00
2003-07-04 22:34:56 +00:00
if ( ! vg_commit ( lv - > vg ) ) {
2010-01-05 21:07:31 +00:00
if ( ! resume_lv ( cmd , lv ) )
stack ;
2009-04-21 14:31:57 +00:00
goto_out ;
2002-02-20 21:30:27 +00:00
}
2001-10-31 17:59:52 +00:00
2003-07-04 22:34:56 +00:00
log_very_verbose ( " Updating permissions for \" %s \" in kernel " , lv - > name ) ;
2005-08-15 12:00:04 +00:00
if ( ! resume_lv ( cmd , lv ) ) {
2002-04-24 18:20:51 +00:00
log_error ( " Problem reactivating %s " , lv - > name ) ;
2009-04-21 14:31:57 +00:00
goto out ;
2002-04-24 18:20:51 +00:00
}
2002-02-20 21:30:27 +00:00
2009-04-21 14:31:57 +00:00
r = 1 ;
out :
backup ( lv - > vg ) ;
return r ;
2001-10-30 14:32:48 +00:00
}
2002-02-01 17:54:39 +00:00
2002-02-11 21:00:35 +00:00
static int lvchange_persistent ( struct cmd_context * cmd ,
struct logical_volume * lv )
2002-02-01 17:54:39 +00:00
{
2003-07-11 17:10:19 +00:00
struct lvinfo info ;
2004-05-05 12:03:07 +00:00
int active = 0 ;
2002-02-25 12:56:16 +00:00
2002-02-11 21:00:35 +00:00
if ( ! strcmp ( arg_str_value ( cmd , persistent_ARG , " n " ) , " n " ) ) {
2002-02-01 17:54:39 +00:00
if ( ! ( lv - > status & FIXED_MINOR ) ) {
log_error ( " Minor number is already not persistent "
" for \" %s \" " , lv - > name ) ;
return 0 ;
}
lv - > status & = ~ FIXED_MINOR ;
lv - > minor = - 1 ;
2003-04-02 19:14:43 +00:00
lv - > major = - 1 ;
log_verbose ( " Disabling persistent device number for \" %s \" " ,
lv - > name ) ;
2002-02-01 17:54:39 +00:00
} else {
2003-04-02 19:14:43 +00:00
if ( ! arg_count ( cmd , minor_ARG ) & & lv - > minor < 0 ) {
2002-02-01 17:54:39 +00:00
log_error ( " Minor number must be specified with -My " ) ;
return 0 ;
}
2012-03-06 02:30:49 +00:00
if ( arg_count ( cmd , major_ARG ) > 1 ) {
log_error ( " Option -j/--major may not be repeated. " ) ;
return 0 ;
}
if ( arg_count ( cmd , minor_ARG ) > 1 ) {
log_error ( " Option --minor may not be repeated. " ) ;
return 0 ;
}
2003-04-02 19:14:43 +00:00
if ( ! arg_count ( cmd , major_ARG ) & & lv - > major < 0 ) {
log_error ( " Major number must be specified with -My " ) ;
return 0 ;
}
2010-08-17 16:25:32 +00:00
if ( lv_info ( cmd , lv , 0 , & info , 0 , 0 ) & & info . exists )
2004-05-05 12:03:07 +00:00
active = 1 ;
2006-10-13 21:33:31 +00:00
if ( active & & ! arg_count ( cmd , force_ARG ) & &
yes_no_prompt ( " Logical volume %s will be "
" deactivated temporarily. "
" Continue? [y/n]: " , lv - > name ) = = ' n ' ) {
2009-12-03 19:18:33 +00:00
log_error ( " %s device number not changed. " ,
2006-10-13 21:33:31 +00:00
lv - > name ) ;
return 0 ;
2003-07-11 17:10:19 +00:00
}
2007-06-15 10:11:14 +00:00
if ( sigint_caught ( ) )
return 0 ;
2004-05-05 18:27:56 +00:00
log_verbose ( " Ensuring %s is inactive. " , lv - > name ) ;
2005-08-15 12:00:04 +00:00
if ( ! deactivate_lv ( cmd , lv ) ) {
2002-03-11 19:02:28 +00:00
log_error ( " %s: deactivation failed " , lv - > name ) ;
return 0 ;
}
2002-02-01 17:54:39 +00:00
lv - > status | = FIXED_MINOR ;
2003-04-02 19:14:43 +00:00
lv - > minor = arg_int_value ( cmd , minor_ARG , lv - > minor ) ;
lv - > major = arg_int_value ( cmd , major_ARG , lv - > major ) ;
log_verbose ( " Setting persistent device number to (%d, %d) "
" for \" %s \" " , lv - > major , lv - > minor , lv - > name ) ;
2006-07-10 19:39:14 +00:00
2002-02-01 17:54:39 +00:00
}
2003-07-04 22:34:56 +00:00
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
2009-04-21 14:31:57 +00:00
if ( ! vg_write ( lv - > vg ) | | ! vg_commit ( lv - > vg ) )
2008-01-30 13:19:47 +00:00
return_0 ;
2003-07-04 22:34:56 +00:00
backup ( lv - > vg ) ;
2006-07-10 19:39:14 +00:00
if ( active ) {
log_verbose ( " Re-activating logical volume \" %s \" " , lv - > name ) ;
if ( ! activate_lv ( cmd , lv ) ) {
log_error ( " %s: reactivation failed " , lv - > name ) ;
return 0 ;
}
2002-04-24 18:20:51 +00:00
}
2002-02-20 21:30:27 +00:00
2002-02-01 17:54:39 +00:00
return 1 ;
}
2002-11-18 14:04:08 +00:00
2011-01-24 13:38:31 +00:00
static int lvchange_tag ( struct cmd_context * cmd , struct logical_volume * lv , int arg )
2004-03-08 17:19:15 +00:00
{
2011-01-24 13:38:31 +00:00
if ( ! change_tag ( cmd , NULL , lv , NULL , arg ) )
return_0 ;
2004-03-08 17:19:15 +00:00
log_very_verbose ( " Updating logical volume \" %s \" on disk(s) " , lv - > name ) ;
/* No need to suspend LV for this change */
2009-04-21 14:31:57 +00:00
if ( ! vg_write ( lv - > vg ) | | ! vg_commit ( lv - > vg ) )
2008-01-30 13:19:47 +00:00
return_0 ;
2004-03-08 17:19:15 +00:00
2009-04-21 14:31:57 +00:00
backup ( lv - > vg ) ;
2004-03-08 17:19:15 +00:00
return 1 ;
}
2005-03-21 22:55:12 +00:00
2002-11-18 14:04:08 +00:00
static int lvchange_single ( struct cmd_context * cmd , struct logical_volume * lv ,
2010-07-09 15:34:40 +00:00
void * handle __attribute__ ( ( unused ) ) )
2002-11-18 14:04:08 +00:00
{
2008-07-31 13:03:01 +00:00
int doit = 0 , docmds = 0 ;
2012-02-15 15:18:43 +00:00
int archived = 0 ;
2009-05-27 18:19:21 +00:00
struct logical_volume * origin ;
2011-11-18 19:22:49 +00:00
char snaps_msg [ 128 ] ;
2002-11-18 14:04:08 +00:00
if ( ! ( lv - > vg - > status & LVM_WRITE ) & &
( arg_count ( cmd , contiguous_ARG ) | | arg_count ( cmd , permission_ARG ) | |
2004-05-18 22:12:53 +00:00
arg_count ( cmd , readahead_ARG ) | | arg_count ( cmd , persistent_ARG ) | |
arg_count ( cmd , alloc_ARG ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Only -a permitted with read-only volume "
" group \" %s \" " , lv - > vg - > name ) ;
return EINVALID_CMD_LINE ;
}
if ( lv_is_origin ( lv ) & &
( arg_count ( cmd , contiguous_ARG ) | | arg_count ( cmd , permission_ARG ) | |
2004-05-18 22:12:53 +00:00
arg_count ( cmd , readahead_ARG ) | | arg_count ( cmd , persistent_ARG ) | |
arg_count ( cmd , alloc_ARG ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Can't change logical volume \" %s \" under snapshot " ,
lv - > name ) ;
return ECMD_FAILED ;
}
2011-11-18 19:22:49 +00:00
if ( lv_is_cow ( lv ) & & ! lv_is_virtual_origin ( origin = origin_from_cow ( lv ) ) & &
2009-04-25 01:17:59 +00:00
arg_count ( cmd , available_ARG ) ) {
2011-11-18 19:22:49 +00:00
if ( origin - > origin_count < 2 )
snaps_msg [ 0 ] = ' \0 ' ;
else if ( dm_snprintf ( snaps_msg , sizeof ( snaps_msg ) ,
" and %u other snapshot(s) " ,
origin - > origin_count - 1 ) < 0 ) {
log_error ( " Failed to prepare message. " ) ;
return ECMD_FAILED ;
}
if ( ! arg_count ( cmd , yes_ARG ) & &
( yes_no_prompt ( " Change of snapshot %s will also change its "
" origin %s%s. Proceed? [y/n]: " , lv - > name ,
origin - > name , snaps_msg ) = = ' n ' ) ) {
log_error ( " Logical volume %s not changed. " , lv - > name ) ;
return ECMD_FAILED ;
}
2002-11-18 14:04:08 +00:00
}
2003-05-06 12:14:36 +00:00
if ( lv - > status & PVMOVE ) {
log_error ( " Unable to change pvmove LV %s " , lv - > name ) ;
if ( arg_count ( cmd , available_ARG ) )
log_error ( " Use 'pvmove --abort' to abandon a pvmove " ) ;
return ECMD_FAILED ;
}
2005-06-01 16:51:55 +00:00
if ( lv - > status & MIRROR_LOG ) {
log_error ( " Unable to change mirror log LV %s directly " , lv - > name ) ;
return ECMD_FAILED ;
}
2005-06-03 14:49:51 +00:00
if ( lv - > status & MIRROR_IMAGE ) {
log_error ( " Unable to change mirror image LV %s directly " ,
lv - > name ) ;
return ECMD_FAILED ;
}
2009-05-27 18:19:21 +00:00
/* If LV is sparse, activate origin instead */
if ( arg_count ( cmd , available_ARG ) & & lv_is_cow ( lv ) & &
lv_is_virtual_origin ( origin = origin_from_cow ( lv ) ) )
lv = origin ;
if ( ! ( lv_is_visible ( lv ) ) & & ! lv_is_virtual_origin ( lv ) ) {
2005-06-03 14:49:51 +00:00
log_error ( " Unable to change internal LV %s directly " ,
lv - > name ) ;
return ECMD_FAILED ;
}
2010-01-06 19:08:58 +00:00
/*
* FIXME : DEFAULT_BACKGROUND_POLLING should be " unspecified " .
* If - - poll is explicitly provided use it ; otherwise polling
* should only be started if the LV is not already active . So :
* 1 ) change the activation code to say if the LV was actually activated
* 2 ) make polling of an LV tightly coupled with LV activation
2010-05-06 11:15:55 +00:00
*
* Do not initiate any polling if - - sysinit option is used .
2010-01-06 19:08:58 +00:00
*/
2010-05-06 11:15:55 +00:00
init_background_polling ( arg_count ( cmd , sysinit_ARG ) ? 0 :
arg_int_value ( cmd , poll_ARG ,
DEFAULT_BACKGROUND_POLLING ) ) ;
2010-01-05 20:56:51 +00:00
2002-11-18 14:04:08 +00:00
/* access permission change */
if ( arg_count ( cmd , permission_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archive ( lv - > vg ) ) {
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2002-11-18 14:04:08 +00:00
archived = 1 ;
doit + = lvchange_permission ( cmd , lv ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2002-11-18 14:04:08 +00:00
}
/* allocation policy change */
2004-05-18 22:12:53 +00:00
if ( arg_count ( cmd , contiguous_ARG ) | | arg_count ( cmd , alloc_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archived & & ! archive ( lv - > vg ) ) {
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2002-11-18 14:04:08 +00:00
archived = 1 ;
2004-05-18 22:12:53 +00:00
doit + = lvchange_alloc ( cmd , lv ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2002-11-18 14:04:08 +00:00
}
/* read ahead sector change */
if ( arg_count ( cmd , readahead_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archived & & ! archive ( lv - > vg ) ) {
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2002-11-18 14:04:08 +00:00
archived = 1 ;
doit + = lvchange_readahead ( cmd , lv ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2002-11-18 14:04:08 +00:00
}
2009-11-04 12:39:56 +00:00
/* persistent device number change */
2002-11-18 14:04:08 +00:00
if ( arg_count ( cmd , persistent_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archived & & ! archive ( lv - > vg ) ) {
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2002-11-18 14:04:08 +00:00
archived = 1 ;
doit + = lvchange_persistent ( cmd , lv ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2009-09-14 22:47:49 +00:00
if ( sigint_caught ( ) ) {
stack ;
2007-06-15 10:11:14 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2002-11-18 14:04:08 +00:00
}
2004-03-08 17:19:15 +00:00
/* add tag */
if ( arg_count ( cmd , addtag_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archived & & ! archive ( lv - > vg ) ) {
stack ;
2004-03-08 17:19:15 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2004-03-08 17:19:15 +00:00
archived = 1 ;
doit + = lvchange_tag ( cmd , lv , addtag_ARG ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2004-03-08 17:19:15 +00:00
}
/* del tag */
if ( arg_count ( cmd , deltag_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! archived & & ! archive ( lv - > vg ) ) {
stack ;
2004-03-08 17:19:15 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2004-03-08 17:19:15 +00:00
archived = 1 ;
doit + = lvchange_tag ( cmd , lv , deltag_ARG ) ;
2008-07-31 13:03:01 +00:00
docmds + + ;
2004-03-08 17:19:15 +00:00
}
2002-11-18 14:04:08 +00:00
if ( doit )
log_print ( " Logical volume \" %s \" changed " , lv - > name ) ;
2006-10-24 17:18:25 +00:00
if ( arg_count ( cmd , resync_ARG ) )
2009-09-14 22:47:49 +00:00
if ( ! lvchange_resync ( cmd , lv ) ) {
stack ;
2006-10-23 23:03:55 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2006-10-23 23:03:55 +00:00
2002-11-18 14:04:08 +00:00
/* availability change */
2006-05-12 19:16:48 +00:00
if ( arg_count ( cmd , available_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! lvchange_availability ( cmd , lv ) ) {
stack ;
2002-11-18 14:04:08 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2006-05-12 19:16:48 +00:00
}
2002-11-18 14:04:08 +00:00
2004-03-26 21:24:03 +00:00
if ( arg_count ( cmd , refresh_ARG ) )
2009-09-14 22:47:49 +00:00
if ( ! lvchange_refresh ( cmd , lv ) ) {
stack ;
2004-03-26 21:24:03 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2004-03-26 21:24:03 +00:00
2006-05-12 19:16:48 +00:00
if ( ! arg_count ( cmd , available_ARG ) & &
! arg_count ( cmd , refresh_ARG ) & &
arg_count ( cmd , monitor_ARG ) ) {
2009-09-14 22:47:49 +00:00
if ( ! lvchange_monitoring ( cmd , lv ) ) {
stack ;
2006-05-12 19:16:48 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2006-05-12 19:16:48 +00:00
}
2010-01-05 20:56:51 +00:00
if ( ! arg_count ( cmd , available_ARG ) & &
! arg_count ( cmd , refresh_ARG ) & &
arg_count ( cmd , poll_ARG ) ) {
if ( ! lvchange_background_polling ( cmd , lv ) ) {
stack ;
return ECMD_FAILED ;
}
}
2009-09-14 22:47:49 +00:00
if ( doit ! = docmds ) {
stack ;
2008-07-31 13:03:01 +00:00
return ECMD_FAILED ;
2009-09-14 22:47:49 +00:00
}
2008-07-31 13:03:01 +00:00
2003-10-21 22:06:07 +00:00
return ECMD_PROCESSED ;
2002-11-18 14:04:08 +00:00
}
int lvchange ( struct cmd_context * cmd , int argc , char * * argv )
{
2010-10-26 01:37:59 +00:00
int update = /* options other than -a, --refresh, --monitor or --poll */
2010-06-28 19:10:16 +00:00
arg_count ( cmd , contiguous_ARG ) | | arg_count ( cmd , permission_ARG ) | |
arg_count ( cmd , readahead_ARG ) | | arg_count ( cmd , persistent_ARG ) | |
arg_count ( cmd , addtag_ARG ) | | arg_count ( cmd , deltag_ARG ) | |
arg_count ( cmd , resync_ARG ) | | arg_count ( cmd , alloc_ARG ) ;
if ( ! update & &
! arg_count ( cmd , available_ARG ) & & ! arg_count ( cmd , refresh_ARG ) & &
2011-09-05 12:54:29 +00:00
! arg_count ( cmd , monitor_ARG ) & & ! arg_count ( cmd , poll_ARG ) ) {
log_error ( " Need 1 or more of -a, -C, -M, -p, -r, "
2010-01-05 20:56:51 +00:00
" --resync, --refresh, --alloc, --addtag, --deltag, "
" --monitor or --poll " ) ;
2002-11-18 14:04:08 +00:00
return EINVALID_CMD_LINE ;
}
2010-10-26 01:37:59 +00:00
if ( arg_count ( cmd , available_ARG ) & & arg_count ( cmd , refresh_ARG ) ) {
log_error ( " Only one of -a and --refresh permitted. " ) ;
return EINVALID_CMD_LINE ;
}
2009-07-15 05:47:55 +00:00
2010-05-06 11:15:55 +00:00
if ( ( arg_count ( cmd , ignorelockingfailure_ARG ) | |
2010-06-28 19:10:16 +00:00
arg_count ( cmd , sysinit_ARG ) ) & & update ) {
2010-05-06 11:15:55 +00:00
log_error ( " Only -a permitted with --ignorelockingfailure and --sysinit " ) ;
2002-11-18 14:04:08 +00:00
return EINVALID_CMD_LINE ;
}
2010-06-28 19:10:16 +00:00
if ( ! update )
2009-07-15 05:47:55 +00:00
cmd - > handles_missing_pvs = 1 ;
2002-11-18 14:04:08 +00:00
if ( ! argc ) {
log_error ( " Please give logical volume path(s) " ) ;
return EINVALID_CMD_LINE ;
}
2003-04-02 19:14:43 +00:00
if ( ( arg_count ( cmd , minor_ARG ) | | arg_count ( cmd , major_ARG ) ) & &
! arg_count ( cmd , persistent_ARG ) ) {
log_error ( " --major and --minor require -My " ) ;
return EINVALID_CMD_LINE ;
}
2002-11-18 14:04:08 +00:00
if ( arg_count ( cmd , minor_ARG ) & & argc ! = 1 ) {
log_error ( " Only give one logical volume when specifying minor " ) ;
return EINVALID_CMD_LINE ;
}
2004-05-18 22:12:53 +00:00
if ( arg_count ( cmd , contiguous_ARG ) & & arg_count ( cmd , alloc_ARG ) ) {
log_error ( " Only one of --alloc and --contiguous permitted " ) ;
return EINVALID_CMD_LINE ;
}
2010-05-06 11:15:55 +00:00
if ( arg_count ( cmd , poll_ARG ) & & arg_count ( cmd , sysinit_ARG ) ) {
log_error ( " Only one of --poll and --sysinit permitted " ) ;
return EINVALID_CMD_LINE ;
}
2009-07-15 05:48:36 +00:00
return process_each_lv ( cmd , argc , argv ,
2010-06-28 19:10:16 +00:00
update ? READ_FOR_UPDATE : 0 , NULL ,
2002-11-18 14:04:08 +00:00
& lvchange_single ) ;
}