Home | History | Annotate | Line # | Download | only in tools
vgchange.c revision 1.1.1.2
      1      1.1  haad /*	$NetBSD: vgchange.c,v 1.1.1.2 2009/02/18 11:17:49 haad 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.2  haad static int _vgchange_refresh(struct cmd_context *cmd, struct volume_group *vg)
    516  1.1.1.2  haad {
    517  1.1.1.2  haad 	log_verbose("Refreshing volume group \"%s\"", vg->name);
    518  1.1.1.2  haad 
    519  1.1.1.2  haad 	if (!vg_refresh_visible(cmd, vg))
    520  1.1.1.2  haad 		return ECMD_FAILED;
    521  1.1.1.2  haad 
    522  1.1.1.2  haad 	return ECMD_PROCESSED;
    523  1.1.1.2  haad }
    524  1.1.1.2  haad 
    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.2  haad 					    (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.2  haad 	else if (arg_count(cmd, refresh_ARG))
    592  1.1.1.2  haad 		r = _vgchange_refresh(cmd, vg);
    593  1.1.1.2  haad 
    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.2  haad 	     arg_count(cmd, monitor_ARG) + arg_count(cmd, refresh_ARG))) {
    607  1.1.1.2  haad 		log_error("One of -a, -c, -l, -p, -s, -x, --refresh, "
    608  1.1.1.2  haad 				"--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