xref: /dflybsd-src/contrib/lvm2/dist/tools/vgconvert.c (revision 86d7f5d305c6adaa56ff4582ece9859d73106103)
186d7f5d3SJohn Marino /*	$NetBSD: vgconvert.c,v 1.1.1.2 2009/12/02 00:25:56 haad Exp $	*/
286d7f5d3SJohn Marino 
386d7f5d3SJohn Marino /*
486d7f5d3SJohn Marino  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
586d7f5d3SJohn Marino  * Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
686d7f5d3SJohn Marino  *
786d7f5d3SJohn Marino  * This file is part of LVM2.
886d7f5d3SJohn Marino  *
986d7f5d3SJohn Marino  * This copyrighted material is made available to anyone wishing to use,
1086d7f5d3SJohn Marino  * modify, copy, or redistribute it subject to the terms and conditions
1186d7f5d3SJohn Marino  * of the GNU Lesser General Public License v.2.1.
1286d7f5d3SJohn Marino  *
1386d7f5d3SJohn Marino  * You should have received a copy of the GNU Lesser General Public License
1486d7f5d3SJohn Marino  * along with this program; if not, write to the Free Software Foundation,
1586d7f5d3SJohn Marino  * Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
1686d7f5d3SJohn Marino  */
1786d7f5d3SJohn Marino 
1886d7f5d3SJohn Marino #include "tools.h"
1986d7f5d3SJohn Marino 
vgconvert_single(struct cmd_context * cmd,const char * vg_name,struct volume_group * vg,void * handle __attribute ((unused)))2086d7f5d3SJohn Marino static int vgconvert_single(struct cmd_context *cmd, const char *vg_name,
2186d7f5d3SJohn Marino 			    struct volume_group *vg,
2286d7f5d3SJohn Marino 			    void *handle __attribute((unused)))
2386d7f5d3SJohn Marino {
2486d7f5d3SJohn Marino 	struct physical_volume *pv, *existing_pv;
2586d7f5d3SJohn Marino 	struct logical_volume *lv;
2686d7f5d3SJohn Marino 	struct lv_list *lvl;
2786d7f5d3SJohn Marino 	uint64_t size = 0;
2886d7f5d3SJohn Marino 	struct dm_list mdas;
2986d7f5d3SJohn Marino 	int pvmetadatacopies = 0;
3086d7f5d3SJohn Marino 	uint64_t pvmetadatasize = 0;
3186d7f5d3SJohn Marino 	uint64_t pe_end = 0, pe_start = 0;
3286d7f5d3SJohn Marino 	struct pv_list *pvl;
3386d7f5d3SJohn Marino 	int change_made = 0;
3486d7f5d3SJohn Marino 	struct lvinfo info;
3586d7f5d3SJohn Marino 	int active = 0;
3686d7f5d3SJohn Marino 
3786d7f5d3SJohn Marino 	if (!vg_check_status(vg, LVM_WRITE | EXPORTED_VG)) {
3886d7f5d3SJohn Marino 		stack;
3986d7f5d3SJohn Marino 		return ECMD_FAILED;
4086d7f5d3SJohn Marino 	}
4186d7f5d3SJohn Marino 
4286d7f5d3SJohn Marino 	if (vg->fid->fmt == cmd->fmt) {
4386d7f5d3SJohn Marino 		log_error("Volume group \"%s\" already uses format %s",
4486d7f5d3SJohn Marino 			  vg_name, cmd->fmt->name);
4586d7f5d3SJohn Marino 		return ECMD_FAILED;
4686d7f5d3SJohn Marino 	}
4786d7f5d3SJohn Marino 
4886d7f5d3SJohn Marino 	if (cmd->fmt->features & FMT_MDAS) {
4986d7f5d3SJohn Marino 		if (arg_sign_value(cmd, metadatasize_ARG, 0) == SIGN_MINUS) {
5086d7f5d3SJohn Marino 			log_error("Metadata size may not be negative");
5186d7f5d3SJohn Marino 			return EINVALID_CMD_LINE;
5286d7f5d3SJohn Marino 		}
5386d7f5d3SJohn Marino 
5486d7f5d3SJohn Marino 		pvmetadatasize = arg_uint64_value(cmd, metadatasize_ARG,
5586d7f5d3SJohn Marino 						  UINT64_C(0));
5686d7f5d3SJohn Marino 		if (!pvmetadatasize)
5786d7f5d3SJohn Marino 			pvmetadatasize =
5886d7f5d3SJohn Marino 			    find_config_tree_int(cmd,
5986d7f5d3SJohn Marino 					    "metadata/pvmetadatasize",
6086d7f5d3SJohn Marino 					    DEFAULT_PVMETADATASIZE);
6186d7f5d3SJohn Marino 
6286d7f5d3SJohn Marino 		pvmetadatacopies = arg_int_value(cmd, pvmetadatacopies_ARG, -1);
6386d7f5d3SJohn Marino 		if (pvmetadatacopies < 0)
6486d7f5d3SJohn Marino 			pvmetadatacopies =
6586d7f5d3SJohn Marino 			    find_config_tree_int(cmd,
6686d7f5d3SJohn Marino 					    "metadata/pvmetadatacopies",
6786d7f5d3SJohn Marino 					     DEFAULT_PVMETADATACOPIES);
6886d7f5d3SJohn Marino 	}
6986d7f5d3SJohn Marino 
7086d7f5d3SJohn Marino 	if (!archive(vg)) {
7186d7f5d3SJohn Marino 		log_error("Archive of \"%s\" metadata failed.", vg_name);
7286d7f5d3SJohn Marino 		return ECMD_FAILED;
7386d7f5d3SJohn Marino 	}
7486d7f5d3SJohn Marino 
7586d7f5d3SJohn Marino 	/* Set PV/LV limit if converting from unlimited metadata format */
7686d7f5d3SJohn Marino 	if (vg->fid->fmt->features & FMT_UNLIMITED_VOLS &&
7786d7f5d3SJohn Marino 	    !(cmd->fmt->features & FMT_UNLIMITED_VOLS)) {
7886d7f5d3SJohn Marino 		if (!vg->max_lv)
7986d7f5d3SJohn Marino 			vg->max_lv = 255;
8086d7f5d3SJohn Marino 		if (!vg->max_pv)
8186d7f5d3SJohn Marino 			vg->max_pv = 255;
8286d7f5d3SJohn Marino 	}
8386d7f5d3SJohn Marino 
8486d7f5d3SJohn Marino 	/* If converting to restricted lvid, check if lvid is compatible */
8586d7f5d3SJohn Marino 	if (!(vg->fid->fmt->features & FMT_RESTRICTED_LVIDS) &&
8686d7f5d3SJohn Marino 	    cmd->fmt->features & FMT_RESTRICTED_LVIDS)
8786d7f5d3SJohn Marino 		dm_list_iterate_items(lvl, &vg->lvs)
8886d7f5d3SJohn Marino 			if (!lvid_in_restricted_range(&lvl->lv->lvid)) {
8986d7f5d3SJohn Marino 				log_error("Logical volume %s lvid format is"
9086d7f5d3SJohn Marino 					  " incompatible with requested"
9186d7f5d3SJohn Marino 					  " metadata format.", lvl->lv->name);
9286d7f5d3SJohn Marino 				return ECMD_FAILED;
9386d7f5d3SJohn Marino 			}
9486d7f5d3SJohn Marino 
9586d7f5d3SJohn Marino 	/* Attempt to change any LVIDs that are too big */
9686d7f5d3SJohn Marino 	if (cmd->fmt->features & FMT_RESTRICTED_LVIDS) {
9786d7f5d3SJohn Marino 		dm_list_iterate_items(lvl, &vg->lvs) {
9886d7f5d3SJohn Marino 			lv = lvl->lv;
9986d7f5d3SJohn Marino 			if (lv->status & SNAPSHOT)
10086d7f5d3SJohn Marino 				continue;
10186d7f5d3SJohn Marino 			if (lvnum_from_lvid(&lv->lvid) < MAX_RESTRICTED_LVS)
10286d7f5d3SJohn Marino 				continue;
10386d7f5d3SJohn Marino 			if (lv_info(cmd, lv, &info, 0, 0) && info.exists) {
10486d7f5d3SJohn Marino 				log_error("Logical volume %s must be "
10586d7f5d3SJohn Marino 					  "deactivated before conversion.",
10686d7f5d3SJohn Marino 					   lv->name);
10786d7f5d3SJohn Marino 				active++;
10886d7f5d3SJohn Marino 				continue;
10986d7f5d3SJohn Marino 			}
11086d7f5d3SJohn Marino 			lvid_from_lvnum(&lv->lvid, &lv->vg->id, find_free_lvnum(lv));
11186d7f5d3SJohn Marino 
11286d7f5d3SJohn Marino 		}
11386d7f5d3SJohn Marino 	}
11486d7f5d3SJohn Marino 
11586d7f5d3SJohn Marino 	if (active) {
11686d7f5d3SJohn Marino 		stack;
11786d7f5d3SJohn Marino 		return ECMD_FAILED;
11886d7f5d3SJohn Marino 	}
11986d7f5d3SJohn Marino 
12086d7f5d3SJohn Marino 	dm_list_iterate_items(pvl, &vg->pvs) {
12186d7f5d3SJohn Marino 		existing_pv = pvl->pv;
12286d7f5d3SJohn Marino 
12386d7f5d3SJohn Marino 		pe_start = pv_pe_start(existing_pv);
12486d7f5d3SJohn Marino 		pe_end = pv_pe_count(existing_pv) * pv_pe_size(existing_pv)
12586d7f5d3SJohn Marino 		    + pe_start - 1;
12686d7f5d3SJohn Marino 
12786d7f5d3SJohn Marino 		dm_list_init(&mdas);
12886d7f5d3SJohn Marino 		if (!(pv = pv_create(cmd, pv_dev(existing_pv),
12986d7f5d3SJohn Marino 				     &existing_pv->id, size, 0, 0,
13086d7f5d3SJohn Marino 				     pe_start, pv_pe_count(existing_pv),
13186d7f5d3SJohn Marino 				     pv_pe_size(existing_pv), pvmetadatacopies,
13286d7f5d3SJohn Marino 				     pvmetadatasize, &mdas))) {
13386d7f5d3SJohn Marino 			log_error("Failed to setup physical volume \"%s\"",
13486d7f5d3SJohn Marino 				  pv_dev_name(existing_pv));
13586d7f5d3SJohn Marino 			if (change_made)
13686d7f5d3SJohn Marino 				log_error("Use pvcreate and vgcfgrestore to "
13786d7f5d3SJohn Marino 					  "repair from archived metadata.");
13886d7f5d3SJohn Marino 			return ECMD_FAILED;
13986d7f5d3SJohn Marino 		}
14086d7f5d3SJohn Marino 
14186d7f5d3SJohn Marino 		/* Need to revert manually if it fails after this point */
14286d7f5d3SJohn Marino 		change_made = 1;
14386d7f5d3SJohn Marino 
14486d7f5d3SJohn Marino 		log_verbose("Set up physical volume for \"%s\" with %" PRIu64
14586d7f5d3SJohn Marino 			    " available sectors", pv_dev_name(pv), pv_size(pv));
14686d7f5d3SJohn Marino 
14786d7f5d3SJohn Marino 		/* Wipe existing label first */
14886d7f5d3SJohn Marino 		if (!label_remove(pv_dev(pv))) {
14986d7f5d3SJohn Marino 			log_error("Failed to wipe existing label on %s",
15086d7f5d3SJohn Marino 				  pv_dev_name(pv));
15186d7f5d3SJohn Marino 			log_error("Use pvcreate and vgcfgrestore to repair "
15286d7f5d3SJohn Marino 				  "from archived metadata.");
15386d7f5d3SJohn Marino 			return ECMD_FAILED;
15486d7f5d3SJohn Marino 		}
15586d7f5d3SJohn Marino 
15686d7f5d3SJohn Marino 		log_very_verbose("Writing physical volume data to disk \"%s\"",
15786d7f5d3SJohn Marino 				 pv_dev_name(pv));
15886d7f5d3SJohn Marino 		if (!(pv_write(cmd, pv, &mdas,
15986d7f5d3SJohn Marino 			       arg_int64_value(cmd, labelsector_ARG,
16086d7f5d3SJohn Marino 					       DEFAULT_LABELSECTOR)))) {
16186d7f5d3SJohn Marino 			log_error("Failed to write physical volume \"%s\"",
16286d7f5d3SJohn Marino 				  pv_dev_name(pv));
16386d7f5d3SJohn Marino 			log_error("Use pvcreate and vgcfgrestore to repair "
16486d7f5d3SJohn Marino 				  "from archived metadata.");
16586d7f5d3SJohn Marino 			return ECMD_FAILED;
16686d7f5d3SJohn Marino 		}
16786d7f5d3SJohn Marino 		log_verbose("Physical volume \"%s\" successfully created",
16886d7f5d3SJohn Marino 			    pv_dev_name(pv));
16986d7f5d3SJohn Marino 
17086d7f5d3SJohn Marino 	}
17186d7f5d3SJohn Marino 
17286d7f5d3SJohn Marino 	log_verbose("Deleting existing metadata for VG %s", vg_name);
17386d7f5d3SJohn Marino 	if (!vg_remove_mdas(vg)) {
17486d7f5d3SJohn Marino 		log_error("Removal of existing metadata for %s failed.",
17586d7f5d3SJohn Marino 			  vg_name);
17686d7f5d3SJohn Marino 		log_error("Use pvcreate and vgcfgrestore to repair "
17786d7f5d3SJohn Marino 			  "from archived metadata.");
17886d7f5d3SJohn Marino 		return ECMD_FAILED;
17986d7f5d3SJohn Marino 	}
18086d7f5d3SJohn Marino 
18186d7f5d3SJohn Marino 	/* FIXME Cache the label format change so we don't have to skip this */
18286d7f5d3SJohn Marino 	if (test_mode()) {
18386d7f5d3SJohn Marino 		log_verbose("Test mode: Skipping metadata writing for VG %s in"
18486d7f5d3SJohn Marino 			    " format %s", vg_name, cmd->fmt->name);
18586d7f5d3SJohn Marino 		return ECMD_PROCESSED;
18686d7f5d3SJohn Marino 	}
18786d7f5d3SJohn Marino 
18886d7f5d3SJohn Marino 	log_verbose("Writing metadata for VG %s using format %s", vg_name,
18986d7f5d3SJohn Marino 		    cmd->fmt->name);
19086d7f5d3SJohn Marino 	if (!backup_restore_vg(cmd, vg)) {
19186d7f5d3SJohn Marino 		log_error("Conversion failed for volume group %s.", vg_name);
19286d7f5d3SJohn Marino 		log_error("Use pvcreate and vgcfgrestore to repair from "
19386d7f5d3SJohn Marino 			  "archived metadata.");
19486d7f5d3SJohn Marino 		return ECMD_FAILED;
19586d7f5d3SJohn Marino 	}
19686d7f5d3SJohn Marino 	log_print("Volume group %s successfully converted", vg_name);
19786d7f5d3SJohn Marino 
19886d7f5d3SJohn Marino 	backup(vg);
19986d7f5d3SJohn Marino 
20086d7f5d3SJohn Marino 	return ECMD_PROCESSED;
20186d7f5d3SJohn Marino }
20286d7f5d3SJohn Marino 
vgconvert(struct cmd_context * cmd,int argc,char ** argv)20386d7f5d3SJohn Marino int vgconvert(struct cmd_context *cmd, int argc, char **argv)
20486d7f5d3SJohn Marino {
20586d7f5d3SJohn Marino 	if (!argc) {
20686d7f5d3SJohn Marino 		log_error("Please enter volume group(s)");
20786d7f5d3SJohn Marino 		return EINVALID_CMD_LINE;
20886d7f5d3SJohn Marino 	}
20986d7f5d3SJohn Marino 
21086d7f5d3SJohn Marino 	if (arg_int_value(cmd, labelsector_ARG, 0) >= LABEL_SCAN_SECTORS) {
21186d7f5d3SJohn Marino 		log_error("labelsector must be less than %lu",
21286d7f5d3SJohn Marino 			  LABEL_SCAN_SECTORS);
21386d7f5d3SJohn Marino 		return EINVALID_CMD_LINE;
21486d7f5d3SJohn Marino 	}
21586d7f5d3SJohn Marino 
21686d7f5d3SJohn Marino 	if (arg_count(cmd, metadatacopies_ARG)) {
21786d7f5d3SJohn Marino 		log_error("Invalid option --metadatacopies, "
21886d7f5d3SJohn Marino 			  "use --pvmetadatacopies instead.");
21986d7f5d3SJohn Marino 		return EINVALID_CMD_LINE;
22086d7f5d3SJohn Marino 	}
22186d7f5d3SJohn Marino 	if (!(cmd->fmt->features & FMT_MDAS) &&
22286d7f5d3SJohn Marino 	    (arg_count(cmd, pvmetadatacopies_ARG) ||
22386d7f5d3SJohn Marino 	     arg_count(cmd, metadatasize_ARG))) {
22486d7f5d3SJohn Marino 		log_error("Metadata parameters only apply to text format");
22586d7f5d3SJohn Marino 		return EINVALID_CMD_LINE;
22686d7f5d3SJohn Marino 	}
22786d7f5d3SJohn Marino 
22886d7f5d3SJohn Marino 	if (arg_count(cmd, pvmetadatacopies_ARG) &&
22986d7f5d3SJohn Marino 	    arg_int_value(cmd, pvmetadatacopies_ARG, -1) > 2) {
23086d7f5d3SJohn Marino 		log_error("Metadatacopies may only be 0, 1 or 2");
23186d7f5d3SJohn Marino 		return EINVALID_CMD_LINE;
23286d7f5d3SJohn Marino 	}
23386d7f5d3SJohn Marino 
23486d7f5d3SJohn Marino 	return process_each_vg(cmd, argc, argv, READ_FOR_UPDATE, NULL,
23586d7f5d3SJohn Marino 			       &vgconvert_single);
23686d7f5d3SJohn Marino }
237