xref: /netbsd-src/external/gpl2/lvm2/dist/tools/pvresize.c (revision 7c604eea85b4f330dc75ffe65e947f4d73758aa0)
1*7c604eeaShaad /*	$NetBSD: pvresize.c,v 1.1.1.2 2009/12/02 00:25:54 haad Exp $	*/
256a34939Shaad 
356a34939Shaad /*
456a34939Shaad  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
556a34939Shaad  * Copyright (C) 2004-2005 Red Hat, Inc. All rights reserved.
656a34939Shaad  * Copyright (C) 2005 Zak Kipling. All rights reserved.
756a34939Shaad  *
856a34939Shaad  * This file is part of LVM2.
956a34939Shaad  *
1056a34939Shaad  * This copyrighted material is made available to anyone wishing to use,
1156a34939Shaad  * modify, copy, or redistribute it subject to the terms and conditions
1256a34939Shaad  * of the GNU Lesser General Public License v.2.1.
1356a34939Shaad  *
1456a34939Shaad  * You should have received a copy of the GNU Lesser General Public License
1556a34939Shaad  * along with this program; if not, write to the Free Software Foundation,
1656a34939Shaad  * Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
1756a34939Shaad  */
1856a34939Shaad 
1956a34939Shaad #include "tools.h"
2056a34939Shaad 
2156a34939Shaad struct pvresize_params {
2256a34939Shaad 	uint64_t new_size;
2356a34939Shaad 
2456a34939Shaad 	unsigned done;
2556a34939Shaad 	unsigned total;
2656a34939Shaad };
2756a34939Shaad 
_pv_resize_single(struct cmd_context * cmd,struct volume_group * vg,struct physical_volume * pv,const uint64_t new_size)2856a34939Shaad static int _pv_resize_single(struct cmd_context *cmd,
2956a34939Shaad 			     struct volume_group *vg,
3056a34939Shaad 			     struct physical_volume *pv,
3156a34939Shaad 			     const uint64_t new_size)
3256a34939Shaad {
3356a34939Shaad 	struct pv_list *pvl;
3456a34939Shaad 	uint64_t size = 0;
3556a34939Shaad 	uint32_t new_pe_count = 0;
36*7c604eeaShaad 	int r = 0;
3756a34939Shaad 	struct dm_list mdas;
3856a34939Shaad 	const char *pv_name = pv_dev_name(pv);
3956a34939Shaad 	const char *vg_name;
4056a34939Shaad 	struct lvmcache_info *info;
4156a34939Shaad 	int mda_count = 0;
42*7c604eeaShaad 	struct volume_group *old_vg = vg;
4356a34939Shaad 
4456a34939Shaad 	dm_list_init(&mdas);
4556a34939Shaad 
4656a34939Shaad 	if (is_orphan_vg(pv_vg_name(pv))) {
4756a34939Shaad 		vg_name = VG_ORPHANS;
4856a34939Shaad 		if (!lock_vol(cmd, vg_name, LCK_VG_WRITE)) {
4956a34939Shaad 			log_error("Can't get lock for orphans");
5056a34939Shaad 			return 0;
5156a34939Shaad 		}
5256a34939Shaad 
53*7c604eeaShaad 		if (!(pv = pv_read(cmd, pv_name, &mdas, NULL, 1, 0))) {
5456a34939Shaad 			unlock_vg(cmd, vg_name);
5556a34939Shaad 			log_error("Unable to read PV \"%s\"", pv_name);
5656a34939Shaad 			return 0;
5756a34939Shaad 		}
5856a34939Shaad 
5956a34939Shaad 		mda_count = dm_list_size(&mdas);
6056a34939Shaad 	} else {
6156a34939Shaad 		vg_name = pv_vg_name(pv);
6256a34939Shaad 
63*7c604eeaShaad 		vg = vg_read_for_update(cmd, vg_name, NULL, 0);
6456a34939Shaad 
65*7c604eeaShaad 		if (vg_read_error(vg))
66*7c604eeaShaad 			goto bad;
6756a34939Shaad 
6856a34939Shaad 		if (!(pvl = find_pv_in_vg(vg, pv_name))) {
6956a34939Shaad 			log_error("Unable to find \"%s\" in volume group \"%s\"",
7056a34939Shaad 				  pv_name, vg->name);
71*7c604eeaShaad 			goto bad;
7256a34939Shaad 		}
7356a34939Shaad 
7456a34939Shaad 		pv = pvl->pv;
7556a34939Shaad 
7656a34939Shaad 		if (!(info = info_from_pvid(pv->dev->pvid, 0))) {
7756a34939Shaad 			log_error("Can't get info for PV %s in volume group %s",
7856a34939Shaad 				  pv_name, vg->name);
79*7c604eeaShaad 			goto bad;
8056a34939Shaad 		}
8156a34939Shaad 
8256a34939Shaad 		mda_count = dm_list_size(&info->mdas);
8356a34939Shaad 
8456a34939Shaad 		if (!archive(vg))
85*7c604eeaShaad 			goto bad;
8656a34939Shaad 	}
8756a34939Shaad 
8856a34939Shaad 	/* FIXME Create function to test compatibility properly */
8956a34939Shaad 	if (mda_count > 1) {
9056a34939Shaad 		log_error("%s: too many metadata areas for pvresize", pv_name);
91*7c604eeaShaad 		goto bad;
9256a34939Shaad 	}
9356a34939Shaad 
9456a34939Shaad 	if (!(pv->fmt->features & FMT_RESIZE_PV)) {
9556a34939Shaad 		log_error("Physical volume %s format does not support resizing.",
9656a34939Shaad 			  pv_name);
97*7c604eeaShaad 		goto bad;
9856a34939Shaad 	}
9956a34939Shaad 
10056a34939Shaad 	/* Get new size */
10156a34939Shaad 	if (!dev_get_size(pv_dev(pv), &size)) {
10256a34939Shaad 		log_error("%s: Couldn't get size.", pv_name);
103*7c604eeaShaad 		goto bad;
10456a34939Shaad 	}
10556a34939Shaad 
10656a34939Shaad 	if (new_size) {
10756a34939Shaad 		if (new_size > size)
10856a34939Shaad 			log_warn("WARNING: %s: Overriding real size. "
10956a34939Shaad 				  "You could lose data.", pv_name);
11056a34939Shaad 		log_verbose("%s: Pretending size is %" PRIu64 " not %" PRIu64
11156a34939Shaad 			    " sectors.", pv_name, new_size, pv_size(pv));
11256a34939Shaad 		size = new_size;
11356a34939Shaad 	}
11456a34939Shaad 
11556a34939Shaad 	if (size < PV_MIN_SIZE) {
11656a34939Shaad 		log_error("%s: Size must exceed minimum of %ld sectors.",
11756a34939Shaad 			  pv_name, PV_MIN_SIZE);
118*7c604eeaShaad 		goto bad;
11956a34939Shaad 	}
12056a34939Shaad 
12156a34939Shaad 	if (size < pv_pe_start(pv)) {
12256a34939Shaad 		log_error("%s: Size must exceed physical extent start of "
12356a34939Shaad 			  "%" PRIu64 " sectors.", pv_name, pv_pe_start(pv));
124*7c604eeaShaad 		goto bad;
12556a34939Shaad 	}
12656a34939Shaad 
12756a34939Shaad 	pv->size = size;
12856a34939Shaad 
12956a34939Shaad 	if (vg) {
13056a34939Shaad 		pv->size -= pv_pe_start(pv);
13156a34939Shaad 		new_pe_count = pv_size(pv) / vg->extent_size;
13256a34939Shaad 
13356a34939Shaad  		if (!new_pe_count) {
13456a34939Shaad 			log_error("%s: Size must leave space for at "
13556a34939Shaad 				  "least one physical extent of "
13656a34939Shaad 				  "%" PRIu32 " sectors.", pv_name,
13756a34939Shaad 				  pv_pe_size(pv));
138*7c604eeaShaad 			goto bad;
13956a34939Shaad 		}
14056a34939Shaad 
141*7c604eeaShaad 		if (!pv_resize(pv, vg, new_pe_count))
142*7c604eeaShaad 			goto_bad;
14356a34939Shaad 	}
14456a34939Shaad 
14556a34939Shaad 	log_verbose("Resizing volume \"%s\" to %" PRIu64 " sectors.",
14656a34939Shaad 		    pv_name, pv_size(pv));
14756a34939Shaad 
14856a34939Shaad 	log_verbose("Updating physical volume \"%s\"", pv_name);
14956a34939Shaad 	if (!is_orphan_vg(pv_vg_name(pv))) {
15056a34939Shaad 		if (!vg_write(vg) || !vg_commit(vg)) {
15156a34939Shaad 			log_error("Failed to store physical volume \"%s\" in "
15256a34939Shaad 				  "volume group \"%s\"", pv_name, vg->name);
153*7c604eeaShaad 			goto bad;
15456a34939Shaad 		}
15556a34939Shaad 		backup(vg);
156*7c604eeaShaad 	} else if (!(pv_write(cmd, pv, NULL, INT64_C(-1)))) {
15756a34939Shaad 		log_error("Failed to store physical volume \"%s\"",
15856a34939Shaad 			  pv_name);
159*7c604eeaShaad 		goto bad;;
16056a34939Shaad 	}
16156a34939Shaad 
16256a34939Shaad 	log_print("Physical volume \"%s\" changed", pv_name);
163*7c604eeaShaad 	r = 1;
164*7c604eeaShaad 
165*7c604eeaShaad bad:
166*7c604eeaShaad 	unlock_vg(cmd, vg_name);
167*7c604eeaShaad 	if (!old_vg)
168*7c604eeaShaad 		vg_release(vg);
169*7c604eeaShaad 	return r;
17056a34939Shaad }
17156a34939Shaad 
_pvresize_single(struct cmd_context * cmd,struct volume_group * vg,struct physical_volume * pv,void * handle)17256a34939Shaad static int _pvresize_single(struct cmd_context *cmd,
17356a34939Shaad 			    struct volume_group *vg,
17456a34939Shaad 			    struct physical_volume *pv,
17556a34939Shaad 			    void *handle)
17656a34939Shaad {
17756a34939Shaad 	struct pvresize_params *params = (struct pvresize_params *) handle;
17856a34939Shaad 
17956a34939Shaad 	params->total++;
18056a34939Shaad 
181*7c604eeaShaad 	if (!_pv_resize_single(cmd, vg, pv, params->new_size)) {
182*7c604eeaShaad 		stack;
18356a34939Shaad 		return ECMD_FAILED;
184*7c604eeaShaad 	}
18556a34939Shaad 
18656a34939Shaad 	params->done++;
18756a34939Shaad 
18856a34939Shaad 	return ECMD_PROCESSED;
18956a34939Shaad }
19056a34939Shaad 
pvresize(struct cmd_context * cmd,int argc,char ** argv)19156a34939Shaad int pvresize(struct cmd_context *cmd, int argc, char **argv)
19256a34939Shaad {
19356a34939Shaad 	struct pvresize_params params;
19456a34939Shaad 	int ret;
19556a34939Shaad 
19656a34939Shaad 	if (!argc) {
19756a34939Shaad 		log_error("Please supply physical volume(s)");
19856a34939Shaad 		return EINVALID_CMD_LINE;
19956a34939Shaad 	}
20056a34939Shaad 
20156a34939Shaad 	if (arg_sign_value(cmd, physicalvolumesize_ARG, 0) == SIGN_MINUS) {
20256a34939Shaad 		log_error("Physical volume size may not be negative");
20356a34939Shaad 		return 0;
20456a34939Shaad 	}
20556a34939Shaad 
20656a34939Shaad 	params.new_size = arg_uint64_value(cmd, physicalvolumesize_ARG,
20756a34939Shaad 					   UINT64_C(0));
20856a34939Shaad 
20956a34939Shaad 	params.done = 0;
21056a34939Shaad 	params.total = 0;
21156a34939Shaad 
212*7c604eeaShaad 	ret = process_each_pv(cmd, argc, argv, NULL, READ_FOR_UPDATE, 0, &params,
21356a34939Shaad 			      _pvresize_single);
21456a34939Shaad 
21556a34939Shaad 	log_print("%d physical volume(s) resized / %d physical volume(s) "
21656a34939Shaad 		  "not resized", params.done, params.total - params.done);
21756a34939Shaad 
21856a34939Shaad 	return ret;
21956a34939Shaad }
220