Home | History | Annotate | Line # | Download | only in tools
vgchange.c revision 1.1.1.1.2.1
      1  1.1.1.1.2.1   jym /*	$NetBSD: vgchange.c,v 1.1.1.1.2.1 2009/05/13 18:52:47 jym Exp $	*/
      2          1.1  haad 
      3          1.1  haad /*
      4          1.1  haad  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
      5          1.1  haad  * Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
      6          1.1  haad  *
      7          1.1  haad  * This file is part of LVM2.
      8          1.1  haad  *
      9          1.1  haad  * This copyrighted material is made available to anyone wishing to use,
     10          1.1  haad  * modify, copy, or redistribute it subject to the terms and conditions
     11          1.1  haad  * of the GNU Lesser General Public License v.2.1.
     12          1.1  haad  *
     13          1.1  haad  * You should have received a copy of the GNU Lesser General Public License
     14          1.1  haad  * along with this program; if not, write to the Free Software Foundation,
     15          1.1  haad  * Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
     16          1.1  haad  */
     17          1.1  haad 
     18          1.1  haad #include "tools.h"
     19          1.1  haad 
     20          1.1  haad static int _monitor_lvs_in_vg(struct cmd_context *cmd,
     21          1.1  haad 			       struct volume_group *vg, int reg)
     22          1.1  haad {
     23          1.1  haad 	struct lv_list *lvl;
     24          1.1  haad 	struct logical_volume *lv;
     25          1.1  haad 	struct lvinfo info;
     26          1.1  haad 	int lv_active;
     27          1.1  haad 	int count = 0;
     28          1.1  haad 
     29          1.1  haad 	dm_list_iterate_items(lvl, &vg->lvs) {
     30          1.1  haad 		lv = lvl->lv;
     31          1.1  haad 
     32          1.1  haad 		if (!lv_info(cmd, lv, &info, 0, 0))
     33          1.1  haad 			lv_active = 0;
     34          1.1  haad 		else
     35          1.1  haad 			lv_active = info.exists;
     36          1.1  haad 
     37          1.1  haad 		/*
     38          1.1  haad 		 * FIXME: Need to consider all cases... PVMOVE, etc
     39          1.1  haad 		 */
     40          1.1  haad 		if ((lv->status & PVMOVE) || !lv_active)
     41          1.1  haad 			continue;
     42          1.1  haad 
     43          1.1  haad 		if (!monitor_dev_for_events(cmd, lv, reg)) {
     44          1.1  haad 			continue;
     45          1.1  haad 		} else
     46          1.1  haad 			count++;
     47          1.1  haad 	}
     48          1.1  haad 
     49          1.1  haad 	/*
     50          1.1  haad 	 * returns the number of _new_ monitored devices
     51          1.1  haad 	 */
     52          1.1  haad 
     53          1.1  haad 	return count;
     54          1.1  haad }
     55          1.1  haad 
     56          1.1  haad static int _activate_lvs_in_vg(struct cmd_context *cmd,
     57          1.1  haad 			       struct volume_group *vg, int activate)
     58          1.1  haad {
     59          1.1  haad 	struct lv_list *lvl;
     60          1.1  haad 	struct logical_volume *lv;
     61          1.1  haad 	const char *pvname;
     62          1.1  haad 	int count = 0;
     63          1.1  haad 
     64          1.1  haad 	dm_list_iterate_items(lvl, &vg->lvs) {
     65          1.1  haad 		lv = lvl->lv;
     66          1.1  haad 
     67          1.1  haad 		/* Only request activation of snapshot origin devices */
     68          1.1  haad 		if ((lv->status & SNAPSHOT) || lv_is_cow(lv))
     69          1.1  haad 			continue;
     70          1.1  haad 
     71          1.1  haad 		/* Only request activation of mirror LV */
     72          1.1  haad 		if ((lv->status & MIRROR_IMAGE) || (lv->status & MIRROR_LOG))
     73          1.1  haad 			continue;
     74          1.1  haad 
     75          1.1  haad 		/* Can't deactivate a pvmove LV */
     76          1.1  haad 		/* FIXME There needs to be a controlled way of doing this */
     77          1.1  haad 		if (((activate == CHANGE_AN) || (activate == CHANGE_ALN)) &&
     78          1.1  haad 		    ((lv->status & PVMOVE) ))
     79          1.1  haad 			continue;
     80          1.1  haad 
     81          1.1  haad 		if (activate == CHANGE_AN) {
     82          1.1  haad 			if (!deactivate_lv(cmd, lv))
     83          1.1  haad 				continue;
     84          1.1  haad 		} else if (activate == CHANGE_ALN) {
     85          1.1  haad 			if (!deactivate_lv_local(cmd, lv))
     86          1.1  haad 				continue;
     87          1.1  haad 		} else if (lv_is_origin(lv) || (activate == CHANGE_AE)) {
     88          1.1  haad 			if (!activate_lv_excl(cmd, lv))
     89          1.1  haad 				continue;
     90          1.1  haad 		} else if (activate == CHANGE_ALY) {
     91          1.1  haad 			if (!activate_lv_local(cmd, lv))
     92          1.1  haad 				continue;
     93          1.1  haad 		} else if (!activate_lv(cmd, lv))
     94          1.1  haad 			continue;
     95          1.1  haad 
     96          1.1  haad 		if ((lv->status & PVMOVE) &&
     97          1.1  haad 		    (pvname = get_pvmove_pvname_from_lv_mirr(lv))) {
     98          1.1  haad 			log_verbose("Spawning background process for %s %s",
     99          1.1  haad 				    lv->name, pvname);
    100          1.1  haad 			pvmove_poll(cmd, pvname, 1);
    101          1.1  haad 			continue;
    102          1.1  haad 		}
    103          1.1  haad 
    104          1.1  haad 		count++;
    105          1.1  haad 	}
    106          1.1  haad 
    107          1.1  haad 	return count;
    108          1.1  haad }
    109          1.1  haad 
    110          1.1  haad static int _vgchange_monitoring(struct cmd_context *cmd, struct volume_group *vg)
    111          1.1  haad {
    112          1.1  haad 	int active, monitored;
    113          1.1  haad 
    114          1.1  haad 	if ((active = lvs_in_vg_activated(vg)) &&
    115          1.1  haad 	    dmeventd_monitor_mode() != DMEVENTD_MONITOR_IGNORE) {
    116          1.1  haad 		monitored = _monitor_lvs_in_vg(cmd, vg, dmeventd_monitor_mode());
    117          1.1  haad 		log_print("%d logical volume(s) in volume group "
    118          1.1  haad 			    "\"%s\" %smonitored",
    119          1.1  haad 			    monitored, vg->name, (dmeventd_monitor_mode()) ? "" : "un");
    120          1.1  haad 	}
    121          1.1  haad 
    122          1.1  haad 	return ECMD_PROCESSED;
    123          1.1  haad }
    124          1.1  haad 
    125          1.1  haad static int _vgchange_available(struct cmd_context *cmd, struct volume_group *vg)
    126          1.1  haad {
    127          1.1  haad 	int lv_open, active, monitored;
    128          1.1  haad 	int available;
    129          1.1  haad 	int activate = 1;
    130          1.1  haad 
    131          1.1  haad 	available = arg_uint_value(cmd, available_ARG, 0);
    132          1.1  haad 
    133          1.1  haad 	if ((available == CHANGE_AN) || (available == CHANGE_ALN))
    134          1.1  haad 		activate = 0;
    135          1.1  haad 
    136          1.1  haad 	/* FIXME: Force argument to deactivate them? */
    137          1.1  haad 	if (!activate && (lv_open = lvs_in_vg_opened(vg))) {
    138          1.1  haad 		log_error("Can't deactivate volume group \"%s\" with %d open "
    139          1.1  haad 			  "logical volume(s)", vg->name, lv_open);
    140          1.1  haad 		return ECMD_FAILED;
    141          1.1  haad 	}
    142          1.1  haad 
    143          1.1  haad 	if (activate && lockingfailed() && (vg_is_clustered(vg))) {
    144          1.1  haad 		log_error("Locking inactive: ignoring clustered "
    145          1.1  haad 			  "volume group %s", vg->name);
    146          1.1  haad 		return ECMD_FAILED;
    147          1.1  haad 	}
    148          1.1  haad 
    149          1.1  haad 	/* FIXME Move into library where clvmd can use it */
    150          1.1  haad 	if (activate && !lockingfailed())
    151          1.1  haad 		check_current_backup(vg);
    152          1.1  haad 
    153          1.1  haad 	if (activate && (active = lvs_in_vg_activated(vg))) {
    154          1.1  haad 		log_verbose("%d logical volume(s) in volume group \"%s\" "
    155          1.1  haad 			    "already active", active, vg->name);
    156          1.1  haad 		if (dmeventd_monitor_mode() != DMEVENTD_MONITOR_IGNORE) {
    157          1.1  haad 			monitored = _monitor_lvs_in_vg(cmd, vg, dmeventd_monitor_mode());
    158          1.1  haad 			log_verbose("%d existing logical volume(s) in volume "
    159          1.1  haad 				    "group \"%s\" %smonitored",
    160          1.1  haad 				    monitored, vg->name,
    161          1.1  haad 				    dmeventd_monitor_mode() ? "" : "un");
    162          1.1  haad 		}
    163          1.1  haad 	}
    164          1.1  haad 
    165          1.1  haad 	if (activate && _activate_lvs_in_vg(cmd, vg, available))
    166          1.1  haad 		log_verbose("Activated logical volumes in "
    167          1.1  haad 			    "volume group \"%s\"", vg->name);
    168          1.1  haad 
    169          1.1  haad 	if (!activate && _activate_lvs_in_vg(cmd, vg, available))
    170          1.1  haad 		log_verbose("Deactivated logical volumes in "
    171          1.1  haad 			    "volume group \"%s\"", vg->name);
    172          1.1  haad 
    173          1.1  haad 	log_print("%d logical volume(s) in volume group \"%s\" now active",
    174          1.1  haad 		  lvs_in_vg_activated(vg), vg->name);
    175          1.1  haad 	return ECMD_PROCESSED;
    176          1.1  haad }
    177          1.1  haad 
    178          1.1  haad static int _vgchange_alloc(struct cmd_context *cmd, struct volume_group *vg)
    179          1.1  haad {
    180          1.1  haad 	alloc_policy_t alloc;
    181          1.1  haad 
    182          1.1  haad 	alloc = arg_uint_value(cmd, alloc_ARG, ALLOC_NORMAL);
    183          1.1  haad 
    184          1.1  haad 	if (alloc == ALLOC_INHERIT) {
    185          1.1  haad 		log_error("Volume Group allocation policy cannot inherit "
    186          1.1  haad 			  "from anything");
    187          1.1  haad 		return EINVALID_CMD_LINE;
    188          1.1  haad 	}
    189          1.1  haad 
    190          1.1  haad 	if (alloc == vg->alloc) {
    191          1.1  haad 		log_error("Volume group allocation policy is already %s",
    192          1.1  haad 			  get_alloc_string(vg->alloc));
    193          1.1  haad 		return ECMD_FAILED;
    194          1.1  haad 	}
    195          1.1  haad 
    196          1.1  haad 	if (!archive(vg))
    197          1.1  haad 		return ECMD_FAILED;
    198          1.1  haad 
    199          1.1  haad 	vg->alloc = alloc;
    200          1.1  haad 
    201          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    202          1.1  haad 		return ECMD_FAILED;
    203          1.1  haad 
    204          1.1  haad 	backup(vg);
    205          1.1  haad 
    206          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    207          1.1  haad 
    208          1.1  haad 	return ECMD_PROCESSED;
    209          1.1  haad }
    210          1.1  haad 
    211          1.1  haad static int _vgchange_resizeable(struct cmd_context *cmd,
    212          1.1  haad 				struct volume_group *vg)
    213          1.1  haad {
    214          1.1  haad 	int resizeable = !strcmp(arg_str_value(cmd, resizeable_ARG, "n"), "y");
    215          1.1  haad 
    216          1.1  haad 	if (resizeable && (vg_status(vg) & RESIZEABLE_VG)) {
    217          1.1  haad 		log_error("Volume group \"%s\" is already resizeable",
    218          1.1  haad 			  vg->name);
    219          1.1  haad 		return ECMD_FAILED;
    220          1.1  haad 	}
    221          1.1  haad 
    222          1.1  haad 	if (!resizeable && !(vg_status(vg) & RESIZEABLE_VG)) {
    223          1.1  haad 		log_error("Volume group \"%s\" is already not resizeable",
    224          1.1  haad 			  vg->name);
    225          1.1  haad 		return ECMD_FAILED;
    226          1.1  haad 	}
    227          1.1  haad 
    228          1.1  haad 	if (!archive(vg))
    229          1.1  haad 		return ECMD_FAILED;
    230          1.1  haad 
    231          1.1  haad 	if (resizeable)
    232          1.1  haad 		vg->status |= RESIZEABLE_VG;
    233          1.1  haad 	else
    234          1.1  haad 		vg->status &= ~RESIZEABLE_VG;
    235          1.1  haad 
    236          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    237          1.1  haad 		return ECMD_FAILED;
    238          1.1  haad 
    239          1.1  haad 	backup(vg);
    240          1.1  haad 
    241          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    242          1.1  haad 
    243          1.1  haad 	return ECMD_PROCESSED;
    244          1.1  haad }
    245          1.1  haad 
    246          1.1  haad static int _vgchange_clustered(struct cmd_context *cmd,
    247          1.1  haad 			       struct volume_group *vg)
    248          1.1  haad {
    249          1.1  haad 	int clustered = !strcmp(arg_str_value(cmd, clustered_ARG, "n"), "y");
    250          1.1  haad 	struct lv_list *lvl;
    251          1.1  haad 
    252          1.1  haad 	if (clustered && (vg_is_clustered(vg))) {
    253          1.1  haad 		log_error("Volume group \"%s\" is already clustered",
    254          1.1  haad 			  vg->name);
    255          1.1  haad 		return ECMD_FAILED;
    256          1.1  haad 	}
    257          1.1  haad 
    258          1.1  haad 	if (!clustered && !(vg_is_clustered(vg))) {
    259          1.1  haad 		log_error("Volume group \"%s\" is already not clustered",
    260          1.1  haad 			  vg->name);
    261          1.1  haad 		return ECMD_FAILED;
    262          1.1  haad 	}
    263          1.1  haad 
    264          1.1  haad 	if (clustered) {
    265          1.1  haad 		dm_list_iterate_items(lvl, &vg->lvs) {
    266          1.1  haad 			if (lv_is_origin(lvl->lv) || lv_is_cow(lvl->lv)) {
    267          1.1  haad 				log_error("Volume group %s contains snapshots "
    268          1.1  haad 					  "that are not yet supported.",
    269          1.1  haad 					  vg->name);
    270          1.1  haad 				return ECMD_FAILED;
    271          1.1  haad 			}
    272          1.1  haad 		}
    273          1.1  haad 	}
    274          1.1  haad 
    275          1.1  haad 	if (!archive(vg))
    276          1.1  haad 		return ECMD_FAILED;
    277          1.1  haad 
    278          1.1  haad 	if (clustered)
    279          1.1  haad 		vg->status |= CLUSTERED;
    280          1.1  haad 	else
    281          1.1  haad 		vg->status &= ~CLUSTERED;
    282          1.1  haad 
    283          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    284          1.1  haad 		return ECMD_FAILED;
    285          1.1  haad 
    286          1.1  haad 	backup(vg);
    287          1.1  haad 
    288          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    289          1.1  haad 
    290          1.1  haad 	return ECMD_PROCESSED;
    291          1.1  haad }
    292          1.1  haad 
    293          1.1  haad static int _vgchange_logicalvolume(struct cmd_context *cmd,
    294          1.1  haad 				   struct volume_group *vg)
    295          1.1  haad {
    296          1.1  haad 	uint32_t max_lv = arg_uint_value(cmd, logicalvolume_ARG, 0);
    297          1.1  haad 
    298          1.1  haad 	if (!(vg_status(vg) & RESIZEABLE_VG)) {
    299          1.1  haad 		log_error("Volume group \"%s\" must be resizeable "
    300          1.1  haad 			  "to change MaxLogicalVolume", vg->name);
    301          1.1  haad 		return ECMD_FAILED;
    302          1.1  haad 	}
    303          1.1  haad 
    304          1.1  haad 	if (!(vg->fid->fmt->features & FMT_UNLIMITED_VOLS)) {
    305          1.1  haad 		if (!max_lv)
    306          1.1  haad 			max_lv = 255;
    307          1.1  haad 		else if (max_lv > 255) {
    308          1.1  haad 			log_error("MaxLogicalVolume limit is 255");
    309          1.1  haad 			return ECMD_FAILED;
    310          1.1  haad 		}
    311          1.1  haad 	}
    312          1.1  haad 
    313          1.1  haad 	if (max_lv && max_lv < vg->lv_count) {
    314          1.1  haad 		log_error("MaxLogicalVolume is less than the current number "
    315          1.1  haad 			  "%d of LVs for \"%s\"", vg->lv_count,
    316          1.1  haad 			  vg->name);
    317          1.1  haad 		return ECMD_FAILED;
    318          1.1  haad 	}
    319          1.1  haad 
    320          1.1  haad 	if (!archive(vg))
    321          1.1  haad 		return ECMD_FAILED;
    322          1.1  haad 
    323          1.1  haad 	vg->max_lv = max_lv;
    324          1.1  haad 
    325          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    326          1.1  haad 		return ECMD_FAILED;
    327          1.1  haad 
    328          1.1  haad 	backup(vg);
    329          1.1  haad 
    330          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    331          1.1  haad 
    332          1.1  haad 	return ECMD_PROCESSED;
    333          1.1  haad }
    334          1.1  haad 
    335          1.1  haad static int _vgchange_physicalvolumes(struct cmd_context *cmd,
    336          1.1  haad 				     struct volume_group *vg)
    337          1.1  haad {
    338          1.1  haad 	uint32_t max_pv = arg_uint_value(cmd, maxphysicalvolumes_ARG, 0);
    339          1.1  haad 
    340          1.1  haad 	if (!(vg_status(vg) & RESIZEABLE_VG)) {
    341          1.1  haad 		log_error("Volume group \"%s\" must be resizeable "
    342          1.1  haad 			  "to change MaxPhysicalVolumes", vg->name);
    343          1.1  haad 		return ECMD_FAILED;
    344          1.1  haad 	}
    345          1.1  haad 
    346          1.1  haad 	if (arg_sign_value(cmd, maxphysicalvolumes_ARG, 0) == SIGN_MINUS) {
    347          1.1  haad 		log_error("MaxPhysicalVolumes may not be negative");
    348          1.1  haad 		return EINVALID_CMD_LINE;
    349          1.1  haad 	}
    350          1.1  haad 
    351          1.1  haad 	if (!(vg->fid->fmt->features & FMT_UNLIMITED_VOLS)) {
    352          1.1  haad 		if (!max_pv)
    353          1.1  haad 			max_pv = 255;
    354          1.1  haad 		else if (max_pv > 255) {
    355          1.1  haad 			log_error("MaxPhysicalVolume limit is 255");
    356          1.1  haad 			return ECMD_FAILED;
    357          1.1  haad 		}
    358          1.1  haad 	}
    359          1.1  haad 
    360          1.1  haad 	if (max_pv && max_pv < vg->pv_count) {
    361          1.1  haad 		log_error("MaxPhysicalVolumes is less than the current number "
    362          1.1  haad 			  "%d of PVs for \"%s\"", vg->pv_count,
    363          1.1  haad 			  vg->name);
    364          1.1  haad 		return ECMD_FAILED;
    365          1.1  haad 	}
    366          1.1  haad 
    367          1.1  haad 	if (!archive(vg))
    368          1.1  haad 		return ECMD_FAILED;
    369          1.1  haad 
    370          1.1  haad 	vg->max_pv = max_pv;
    371          1.1  haad 
    372          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    373          1.1  haad 		return ECMD_FAILED;
    374          1.1  haad 
    375          1.1  haad 	backup(vg);
    376          1.1  haad 
    377          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    378          1.1  haad 
    379          1.1  haad 	return ECMD_PROCESSED;
    380          1.1  haad }
    381          1.1  haad 
    382          1.1  haad static int _vgchange_pesize(struct cmd_context *cmd, struct volume_group *vg)
    383          1.1  haad {
    384          1.1  haad 	uint32_t extent_size;
    385          1.1  haad 
    386          1.1  haad 	if (!(vg_status(vg) & RESIZEABLE_VG)) {
    387          1.1  haad 		log_error("Volume group \"%s\" must be resizeable "
    388          1.1  haad 			  "to change PE size", vg->name);
    389          1.1  haad 		return ECMD_FAILED;
    390          1.1  haad 	}
    391          1.1  haad 
    392          1.1  haad 	if (arg_sign_value(cmd, physicalextentsize_ARG, 0) == SIGN_MINUS) {
    393          1.1  haad 		log_error("Physical extent size may not be negative");
    394          1.1  haad 		return EINVALID_CMD_LINE;
    395          1.1  haad 	}
    396          1.1  haad 
    397          1.1  haad 	extent_size = arg_uint_value(cmd, physicalextentsize_ARG, 0);
    398          1.1  haad 	if (!extent_size) {
    399          1.1  haad 		log_error("Physical extent size may not be zero");
    400          1.1  haad 		return EINVALID_CMD_LINE;
    401          1.1  haad 	}
    402          1.1  haad 
    403          1.1  haad 	if (extent_size == vg->extent_size) {
    404          1.1  haad 		log_error("Physical extent size of VG %s is already %s",
    405          1.1  haad 			  vg->name, display_size(cmd, (uint64_t) extent_size));
    406          1.1  haad 		return ECMD_PROCESSED;
    407          1.1  haad 	}
    408          1.1  haad 
    409          1.1  haad 	if (extent_size & (extent_size - 1)) {
    410          1.1  haad 		log_error("Physical extent size must be a power of 2.");
    411          1.1  haad 		return EINVALID_CMD_LINE;
    412          1.1  haad 	}
    413          1.1  haad 
    414          1.1  haad 	if (extent_size > vg->extent_size) {
    415          1.1  haad 		if ((uint64_t) vg->extent_size * vg->extent_count % extent_size) {
    416          1.1  haad 			/* FIXME Adjust used PV sizes instead */
    417          1.1  haad 			log_error("New extent size is not a perfect fit");
    418          1.1  haad 			return EINVALID_CMD_LINE;
    419          1.1  haad 		}
    420          1.1  haad 	}
    421          1.1  haad 
    422          1.1  haad 	if (!archive(vg))
    423          1.1  haad 		return ECMD_FAILED;
    424          1.1  haad 
    425          1.1  haad 	if (!vg_change_pesize(cmd, vg, extent_size)) {
    426          1.1  haad 		stack;
    427          1.1  haad 		return ECMD_FAILED;
    428          1.1  haad 	}
    429          1.1  haad 
    430          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    431          1.1  haad 		return ECMD_FAILED;
    432          1.1  haad 
    433          1.1  haad 	backup(vg);
    434          1.1  haad 
    435          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    436          1.1  haad 
    437          1.1  haad 	return ECMD_PROCESSED;
    438          1.1  haad }
    439          1.1  haad 
    440          1.1  haad static int _vgchange_tag(struct cmd_context *cmd, struct volume_group *vg,
    441          1.1  haad 			 int arg)
    442          1.1  haad {
    443          1.1  haad 	const char *tag;
    444          1.1  haad 
    445          1.1  haad 	if (!(tag = arg_str_value(cmd, arg, NULL))) {
    446          1.1  haad 		log_error("Failed to get tag");
    447          1.1  haad 		return ECMD_FAILED;
    448          1.1  haad 	}
    449          1.1  haad 
    450          1.1  haad 	if (!(vg->fid->fmt->features & FMT_TAGS)) {
    451          1.1  haad 		log_error("Volume group %s does not support tags", vg->name);
    452          1.1  haad 		return ECMD_FAILED;
    453          1.1  haad 	}
    454          1.1  haad 
    455          1.1  haad 	if (!archive(vg))
    456          1.1  haad 		return ECMD_FAILED;
    457          1.1  haad 
    458          1.1  haad 	if ((arg == addtag_ARG)) {
    459          1.1  haad 		if (!str_list_add(cmd->mem, &vg->tags, tag)) {
    460          1.1  haad 			log_error("Failed to add tag %s to volume group %s",
    461          1.1  haad 				  tag, vg->name);
    462          1.1  haad 			return ECMD_FAILED;
    463          1.1  haad 		}
    464          1.1  haad 	} else {
    465          1.1  haad 		if (!str_list_del(&vg->tags, tag)) {
    466          1.1  haad 			log_error("Failed to remove tag %s from volume group "
    467          1.1  haad 				  "%s", tag, vg->name);
    468          1.1  haad 			return ECMD_FAILED;
    469          1.1  haad 		}
    470          1.1  haad 	}
    471          1.1  haad 
    472          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    473          1.1  haad 		return ECMD_FAILED;
    474          1.1  haad 
    475          1.1  haad 	backup(vg);
    476          1.1  haad 
    477          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    478          1.1  haad 
    479          1.1  haad 	return ECMD_PROCESSED;
    480          1.1  haad }
    481          1.1  haad 
    482          1.1  haad static int _vgchange_uuid(struct cmd_context *cmd __attribute((unused)),
    483          1.1  haad 			  struct volume_group *vg)
    484          1.1  haad {
    485          1.1  haad 	struct lv_list *lvl;
    486          1.1  haad 
    487          1.1  haad 	if (lvs_in_vg_activated(vg)) {
    488          1.1  haad 		log_error("Volume group has active logical volumes");
    489          1.1  haad 		return ECMD_FAILED;
    490          1.1  haad 	}
    491          1.1  haad 
    492          1.1  haad 	if (!archive(vg))
    493          1.1  haad 		return ECMD_FAILED;
    494          1.1  haad 
    495          1.1  haad 	if (!id_create(&vg->id)) {
    496          1.1  haad 		log_error("Failed to generate new random UUID for VG %s.",
    497          1.1  haad 			  vg->name);
    498          1.1  haad 		return ECMD_FAILED;
    499          1.1  haad 	}
    500          1.1  haad 
    501          1.1  haad 	dm_list_iterate_items(lvl, &vg->lvs) {
    502          1.1  haad 		memcpy(&lvl->lv->lvid, &vg->id, sizeof(vg->id));
    503          1.1  haad 	}
    504          1.1  haad 
    505          1.1  haad 	if (!vg_write(vg) || !vg_commit(vg))
    506          1.1  haad 		return ECMD_FAILED;
    507          1.1  haad 
    508          1.1  haad 	backup(vg);
    509          1.1  haad 
    510          1.1  haad 	log_print("Volume group \"%s\" successfully changed", vg->name);
    511          1.1  haad 
    512          1.1  haad 	return ECMD_PROCESSED;
    513          1.1  haad }
    514          1.1  haad 
    515  1.1.1.1.2.1   jym static int _vgchange_refresh(struct cmd_context *cmd, struct volume_group *vg)
    516  1.1.1.1.2.1   jym {
    517  1.1.1.1.2.1   jym 	log_verbose("Refreshing volume group \"%s\"", vg->name);
    518  1.1.1.1.2.1   jym 
    519  1.1.1.1.2.1   jym 	if (!vg_refresh_visible(cmd, vg))
    520  1.1.1.1.2.1   jym 		return ECMD_FAILED;
    521  1.1.1.1.2.1   jym 
    522  1.1.1.1.2.1   jym 	return ECMD_PROCESSED;
    523  1.1.1.1.2.1   jym }
    524  1.1.1.1.2.1   jym 
    525          1.1  haad static int vgchange_single(struct cmd_context *cmd, const char *vg_name,
    526          1.1  haad 			   struct volume_group *vg, int consistent,
    527          1.1  haad 			   void *handle __attribute((unused)))
    528          1.1  haad {
    529          1.1  haad 	int r = ECMD_FAILED;
    530          1.1  haad 
    531          1.1  haad 	if (!vg) {
    532          1.1  haad 		log_error("Unable to find volume group \"%s\"", vg_name);
    533          1.1  haad 		return ECMD_FAILED;
    534          1.1  haad 	}
    535          1.1  haad 
    536          1.1  haad 	if (!consistent) {
    537          1.1  haad 		unlock_vg(cmd, vg_name);
    538          1.1  haad 		dev_close_all();
    539          1.1  haad 		log_error("Volume group \"%s\" inconsistent", vg_name);
    540          1.1  haad 		if (!(vg = recover_vg(cmd, vg_name, LCK_VG_WRITE)))
    541          1.1  haad 			return ECMD_FAILED;
    542          1.1  haad 	}
    543          1.1  haad 
    544          1.1  haad 	if (!(vg_status(vg) & LVM_WRITE) && !arg_count(cmd, available_ARG)) {
    545          1.1  haad 		log_error("Volume group \"%s\" is read-only", vg->name);
    546          1.1  haad 		return ECMD_FAILED;
    547          1.1  haad 	}
    548          1.1  haad 
    549          1.1  haad 	if (vg_status(vg) & EXPORTED_VG) {
    550          1.1  haad 		log_error("Volume group \"%s\" is exported", vg_name);
    551          1.1  haad 		return ECMD_FAILED;
    552          1.1  haad 	}
    553          1.1  haad 
    554          1.1  haad 	init_dmeventd_monitor(arg_int_value(cmd, monitor_ARG,
    555  1.1.1.1.2.1   jym 					    (is_static() || arg_count(cmd, ignoremonitoring_ARG)) ?
    556          1.1  haad 					    DMEVENTD_MONITOR_IGNORE : DEFAULT_DMEVENTD_MONITOR));
    557          1.1  haad 
    558          1.1  haad 	if (arg_count(cmd, available_ARG))
    559          1.1  haad 		r = _vgchange_available(cmd, vg);
    560          1.1  haad 
    561          1.1  haad 	else if (arg_count(cmd, monitor_ARG))
    562          1.1  haad 		r = _vgchange_monitoring(cmd, vg);
    563          1.1  haad 
    564          1.1  haad 	else if (arg_count(cmd, resizeable_ARG))
    565          1.1  haad 		r = _vgchange_resizeable(cmd, vg);
    566          1.1  haad 
    567          1.1  haad 	else if (arg_count(cmd, logicalvolume_ARG))
    568          1.1  haad 		r = _vgchange_logicalvolume(cmd, vg);
    569          1.1  haad 
    570          1.1  haad 	else if (arg_count(cmd, maxphysicalvolumes_ARG))
    571          1.1  haad 		r = _vgchange_physicalvolumes(cmd, vg);
    572          1.1  haad 
    573          1.1  haad 	else if (arg_count(cmd, addtag_ARG))
    574          1.1  haad 		r = _vgchange_tag(cmd, vg, addtag_ARG);
    575          1.1  haad 
    576          1.1  haad 	else if (arg_count(cmd, deltag_ARG))
    577          1.1  haad 		r = _vgchange_tag(cmd, vg, deltag_ARG);
    578          1.1  haad 
    579          1.1  haad 	else if (arg_count(cmd, physicalextentsize_ARG))
    580          1.1  haad 		r = _vgchange_pesize(cmd, vg);
    581          1.1  haad 
    582          1.1  haad 	else if (arg_count(cmd, uuid_ARG))
    583          1.1  haad 		r = _vgchange_uuid(cmd, vg);
    584          1.1  haad 
    585          1.1  haad 	else if (arg_count(cmd, alloc_ARG))
    586          1.1  haad 		r = _vgchange_alloc(cmd, vg);
    587          1.1  haad 
    588          1.1  haad 	else if (arg_count(cmd, clustered_ARG))
    589          1.1  haad 		r = _vgchange_clustered(cmd, vg);
    590          1.1  haad 
    591  1.1.1.1.2.1   jym 	else if (arg_count(cmd, refresh_ARG))
    592  1.1.1.1.2.1   jym 		r = _vgchange_refresh(cmd, vg);
    593  1.1.1.1.2.1   jym 
    594          1.1  haad 	return r;
    595          1.1  haad }
    596          1.1  haad 
    597          1.1  haad int vgchange(struct cmd_context *cmd, int argc, char **argv)
    598          1.1  haad {
    599          1.1  haad 	if (!
    600          1.1  haad 	    (arg_count(cmd, available_ARG) + arg_count(cmd, logicalvolume_ARG) +
    601          1.1  haad 	     arg_count(cmd, maxphysicalvolumes_ARG) +
    602          1.1  haad 	     arg_count(cmd, resizeable_ARG) + arg_count(cmd, deltag_ARG) +
    603          1.1  haad 	     arg_count(cmd, addtag_ARG) + arg_count(cmd, uuid_ARG) +
    604          1.1  haad 	     arg_count(cmd, physicalextentsize_ARG) +
    605          1.1  haad 	     arg_count(cmd, clustered_ARG) + arg_count(cmd, alloc_ARG) +
    606  1.1.1.1.2.1   jym 	     arg_count(cmd, monitor_ARG) + arg_count(cmd, refresh_ARG))) {
    607  1.1.1.1.2.1   jym 		log_error("One of -a, -c, -l, -p, -s, -x, --refresh, "
    608  1.1.1.1.2.1   jym 				"--uuid, --alloc, --addtag or --deltag required");
    609          1.1  haad 		return EINVALID_CMD_LINE;
    610          1.1  haad 	}
    611          1.1  haad 
    612          1.1  haad 	/* FIXME Cope with several changes at once! */
    613          1.1  haad 	if (arg_count(cmd, available_ARG) + arg_count(cmd, logicalvolume_ARG) +
    614          1.1  haad 	    arg_count(cmd, maxphysicalvolumes_ARG) +
    615          1.1  haad 	    arg_count(cmd, resizeable_ARG) + arg_count(cmd, deltag_ARG) +
    616          1.1  haad 	    arg_count(cmd, addtag_ARG) + arg_count(cmd, alloc_ARG) +
    617          1.1  haad 	    arg_count(cmd, uuid_ARG) + arg_count(cmd, clustered_ARG) +
    618          1.1  haad 	    arg_count(cmd, physicalextentsize_ARG) > 1) {
    619          1.1  haad 		log_error("Only one of -a, -c, -l, -p, -s, -x, --uuid, "
    620          1.1  haad 			  "--alloc, --addtag or --deltag allowed");
    621          1.1  haad 		return EINVALID_CMD_LINE;
    622          1.1  haad 	}
    623          1.1  haad 
    624          1.1  haad 	if (arg_count(cmd, ignorelockingfailure_ARG) &&
    625          1.1  haad 	    !arg_count(cmd, available_ARG)) {
    626          1.1  haad 		log_error("--ignorelockingfailure only available with -a");
    627          1.1  haad 		return EINVALID_CMD_LINE;
    628          1.1  haad 	}
    629          1.1  haad 
    630          1.1  haad 	if (arg_count(cmd, available_ARG) == 1
    631          1.1  haad 	    && arg_count(cmd, autobackup_ARG)) {
    632          1.1  haad 		log_error("-A option not necessary with -a option");
    633          1.1  haad 		return EINVALID_CMD_LINE;
    634          1.1  haad 	}
    635          1.1  haad 
    636          1.1  haad 	return process_each_vg(cmd, argc, argv,
    637          1.1  haad 			       (arg_count(cmd, available_ARG)) ?
    638          1.1  haad 			       LCK_VG_READ : LCK_VG_WRITE, 0, NULL,
    639          1.1  haad 			       &vgchange_single);
    640          1.1  haad }
    641