mirror of
				git://sourceware.org/git/lvm2.git
				synced 2025-10-30 20:23:49 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			315 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			315 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved. 
 | |
|  * Copyright (C) 2004 Red Hat, Inc. All rights reserved.
 | |
|  *
 | |
|  * This file is part of LVM2.
 | |
|  *
 | |
|  * This copyrighted material is made available to anyone wishing to use,
 | |
|  * modify, copy, or redistribute it subject to the terms and conditions
 | |
|  * of the GNU General Public License v.2.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software Foundation,
 | |
|  * Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 | |
|  */
 | |
| 
 | |
| #include "tools.h"
 | |
| 
 | |
| static int _move_pv(struct volume_group *vg_from, struct volume_group *vg_to,
 | |
| 		    char *pv_name)
 | |
| {
 | |
| 	struct pv_list *pvl;
 | |
| 	struct physical_volume *pv;
 | |
| 
 | |
| 	if (!(pvl = find_pv_in_vg(vg_from, pv_name))) {
 | |
| 		log_error("Physical volume %s not in volume group %s",
 | |
| 			  pv_name, vg_from->name);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	list_del(&pvl->list);
 | |
| 	list_add(&vg_to->pvs, &pvl->list);
 | |
| 
 | |
| 	vg_from->pv_count--;
 | |
| 	vg_to->pv_count++;
 | |
| 
 | |
| 	pv = list_item(pvl, struct pv_list)->pv;
 | |
| 
 | |
| 	vg_from->extent_count -= pv->pe_count;
 | |
| 	vg_to->extent_count += pv->pe_count;
 | |
| 
 | |
| 	vg_from->free_count -= pv->pe_count - pv->pe_alloc_count;
 | |
| 	vg_to->free_count += pv->pe_count - pv->pe_alloc_count;
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| /* FIXME Why not (lv->vg == vg) ? */
 | |
| static int _lv_is_in_vg(struct volume_group *vg, struct logical_volume *lv)
 | |
| {
 | |
| 	struct lv_list *lvl;
 | |
| 
 | |
| 	list_iterate_items(lvl, &vg->lvs)
 | |
| 		if (lv == lvl->lv)
 | |
| 			 return 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int _move_lvs(struct volume_group *vg_from, struct volume_group *vg_to)
 | |
| {
 | |
| 	struct list *lvh, *lvht;
 | |
| 	struct logical_volume *lv;
 | |
| 	struct lv_segment *seg;
 | |
| 	struct physical_volume *pv;
 | |
| 	struct volume_group *vg_with;
 | |
| 	unsigned int s;
 | |
| 
 | |
| 	list_iterate_safe(lvh, lvht, &vg_from->lvs) {
 | |
| 		lv = list_item(lvh, struct lv_list)->lv;
 | |
| 
 | |
| 		if ((lv->status & SNAPSHOT))
 | |
| 			continue;
 | |
| 
 | |
| 		/* Ensure all the PVs used by this LV remain in the same */
 | |
| 		/* VG as each other */
 | |
| 		vg_with = NULL;
 | |
| 		list_iterate_items(seg, &lv->segments) {
 | |
| 			for (s = 0; s < seg->area_count; s++) {
 | |
| 				/* FIXME Check AREA_LV too */
 | |
| 				if (seg_type(seg, s) != AREA_PV)
 | |
| 					continue;
 | |
| 
 | |
| 				pv = seg_pv(seg, s);
 | |
| 				if (vg_with) {
 | |
| 					if (!pv_is_in_vg(vg_with, pv)) {
 | |
| 						log_error("Logical Volume %s "
 | |
| 							  "split between "
 | |
| 							  "Volume Groups",
 | |
| 							  lv->name);
 | |
| 						return 0;
 | |
| 					}
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if (pv_is_in_vg(vg_from, pv)) {
 | |
| 					vg_with = vg_from;
 | |
| 					continue;
 | |
| 				}
 | |
| 				if (pv_is_in_vg(vg_to, pv)) {
 | |
| 					vg_with = vg_to;
 | |
| 					continue;
 | |
| 				}
 | |
| 				log_error("Physical Volume %s not found",
 | |
| 					  dev_name(pv->dev));
 | |
| 				return 0;
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 			
 | |
| 		if (vg_with == vg_from)
 | |
| 			continue;
 | |
| 
 | |
| 		/* Move this LV */
 | |
| 		list_del(lvh);
 | |
| 		list_add(&vg_to->lvs, lvh);
 | |
| 
 | |
| 		vg_from->lv_count--;
 | |
| 		vg_to->lv_count++;
 | |
| 	}
 | |
| 
 | |
| 	/* FIXME Ensure no LVs contain segs pointing at LVs in the other VG */
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int _move_snapshots(struct volume_group *vg_from,
 | |
| 			   struct volume_group *vg_to)
 | |
| {
 | |
| 	struct list *lvh, *lvht;
 | |
| 	struct logical_volume *lv;
 | |
| 	struct lv_segment *seg;
 | |
| 	int cow_from = 0;
 | |
| 	int origin_from = 0;
 | |
| 
 | |
| 	list_iterate_safe(lvh, lvht, &vg_from->lvs) {
 | |
| 		lv = list_item(lvh, struct lv_list)->lv;
 | |
| 
 | |
| 		if (!(lv->status & SNAPSHOT))
 | |
| 			continue;
 | |
| 
 | |
| 		list_iterate_items(seg, &lv->segments) {
 | |
| 			cow_from = _lv_is_in_vg(vg_from, seg->cow);
 | |
| 			origin_from = _lv_is_in_vg(vg_from, seg->origin);
 | |
| 		}
 | |
| 		if (cow_from && origin_from)
 | |
| 			continue;
 | |
| 		if ((!cow_from && origin_from) || (cow_from && !origin_from)) {
 | |
| 			log_error("Snapshot %s split", seg->cow->name);
 | |
| 			return 0;
 | |
| 		}	
 | |
| 
 | |
| 		/* Move this snapshot */
 | |
| 		list_del(lvh);
 | |
| 		list_add(&vg_to->lvs, lvh);
 | |
| 
 | |
| 		vg_from->snapshot_count--;
 | |
| 		vg_to->snapshot_count++;
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| int vgsplit(struct cmd_context *cmd, int argc, char **argv)
 | |
| {
 | |
| 	char *vg_name_from, *vg_name_to;
 | |
| 	struct volume_group *vg_to, *vg_from;
 | |
| 	int opt;
 | |
| 	int active;
 | |
| 	int consistent = 1;
 | |
| 
 | |
| 	if (argc < 3) {
 | |
| 		log_error("Existing VG, new VG and physical volumes required.");
 | |
| 		return EINVALID_CMD_LINE;
 | |
| 	}
 | |
| 
 | |
| 	vg_name_from = argv[0];
 | |
| 	vg_name_to = argv[1];
 | |
| 	argc -= 2;
 | |
| 	argv += 2;
 | |
| 
 | |
| 	if (!strcmp(vg_name_to, vg_name_from)) {
 | |
| 		log_error("Duplicate volume group name \"%s\"", vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	log_verbose("Checking for volume group \"%s\"", vg_name_from);
 | |
| 	if (!lock_vol(cmd, vg_name_from, LCK_VG_WRITE)) {
 | |
| 		log_error("Can't get lock for %s", vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	if (!(vg_from = vg_read(cmd, vg_name_from, &consistent)) || !consistent) {
 | |
| 		log_error("Volume group \"%s\" doesn't exist", vg_name_from);
 | |
| 		unlock_vg(cmd, vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	if (vg_from->status & EXPORTED_VG) {
 | |
| 		log_error("Volume group \"%s\" is exported", vg_from->name);
 | |
| 		unlock_vg(cmd, vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	if (!(vg_from->status & LVM_WRITE)) {
 | |
| 		log_error("Volume group \"%s\" is read-only", vg_from->name);
 | |
| 		unlock_vg(cmd, vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	log_verbose("Checking for volume group \"%s\"", vg_name_to);
 | |
| 	if (!lock_vol(cmd, vg_name_to, LCK_VG_WRITE | LCK_NONBLOCK)) {
 | |
| 		log_error("Can't get lock for %s", vg_name_to);
 | |
| 		unlock_vg(cmd, vg_name_from);
 | |
| 		return ECMD_FAILED;
 | |
| 	}
 | |
| 
 | |
| 	consistent = 0;
 | |
| 	if ((vg_to = vg_read(cmd, vg_name_to, &consistent))) {
 | |
| 		/* FIXME Remove this restriction */
 | |
| 		log_error("Volume group \"%s\" already exists", vg_name_to);
 | |
| 		goto error;
 | |
| 	}
 | |
| 
 | |
| 	if (!validate_vg_name(cmd, vg_name_to)) {
 | |
| 		log_error("New volume group name \"%s\" is invalid",
 | |
| 			   vg_name_to);
 | |
| 		goto error;
 | |
| 	}
 | |
| 
 | |
| 	if ((active = lvs_in_vg_activated(vg_from))) {
 | |
| 		/* FIXME Remove this restriction */
 | |
| 		log_error("Logical volumes in \"%s\" must be inactive",
 | |
| 			  vg_name_from);
 | |
| 		goto error;
 | |
| 	}
 | |
| 
 | |
| 	/* Create new VG structure */
 | |
| 	if (!(vg_to = vg_create(cmd, vg_name_to, vg_from->extent_size,
 | |
| 				vg_from->max_pv, vg_from->max_lv,
 | |
| 				vg_from->alloc, 0, NULL)))
 | |
| 		goto error;
 | |
| 
 | |
| 	/* Archive vg_from before changing it */
 | |
| 	if (!archive(vg_from))
 | |
| 		goto error;
 | |
| 
 | |
| 	/* Move PVs across to new structure */
 | |
| 	for (opt = 0; opt < argc; opt++) {
 | |
| 		if (!_move_pv(vg_from, vg_to, argv[opt]))
 | |
| 			goto error;
 | |
| 	}
 | |
| 
 | |
| 	/* Move required LVs across, checking consistency */
 | |
| 	if (!(_move_lvs(vg_from, vg_to)))
 | |
| 		goto error;
 | |
| 
 | |
| 	/* Move required snapshots across */
 | |
| 	if (!(_move_snapshots(vg_from, vg_to)))
 | |
| 		goto error;
 | |
| 
 | |
| 	/* FIXME Split mdas properly somehow too! */
 | |
| 	/* Currently we cheat by sharing the format instance and relying on 
 | |
| 	 * vg_write to ignore mdas outside the VG!  Done this way, with text 
 | |
| 	 * format, vg_from disappears for a short time. */
 | |
| 	vg_to->fid = vg_from->fid;
 | |
| 
 | |
| 	/* store it on disks */
 | |
| 	log_verbose("Writing out updated volume groups");
 | |
| 
 | |
| 	/* Write out new VG as EXPORTED */
 | |
| 	vg_to->status |= EXPORTED_VG;
 | |
| 
 | |
| 	if (!archive(vg_to))
 | |
| 		goto error;
 | |
| 
 | |
| 	if (!vg_write(vg_to) || !vg_commit(vg_to))
 | |
| 		goto error;
 | |
| 
 | |
| 	backup(vg_to);
 | |
| 
 | |
| 	/* Write out updated old VG */
 | |
| 	if (!vg_write(vg_from) || !vg_commit(vg_from))
 | |
| 		goto error;
 | |
| 
 | |
| 	backup(vg_from);
 | |
| 
 | |
| 	/* Remove EXPORTED flag from new VG */
 | |
| 	consistent = 1;
 | |
| 	if (!(vg_to = vg_read(cmd, vg_name_to, &consistent)) || !consistent) {
 | |
| 		log_error("Volume group \"%s\" became inconsistent: please "
 | |
| 			  "fix manually", vg_name_to);
 | |
| 		goto error;
 | |
| 	}
 | |
| 
 | |
| 	vg_to->status &= ~EXPORTED_VG;
 | |
| 
 | |
| 	if (!vg_write(vg_to) || !vg_commit(vg_to))
 | |
| 		goto error;
 | |
| 
 | |
| 	backup(vg_to);
 | |
| 
 | |
| 	unlock_vg(cmd, vg_name_from);
 | |
| 	unlock_vg(cmd, vg_name_to);
 | |
| 
 | |
| 	log_print("Volume group \"%s\" successfully split from \"%s\"",
 | |
| 		  vg_to->name, vg_from->name);
 | |
| 	return ECMD_PROCESSED;
 | |
| 
 | |
|       error:
 | |
| 	unlock_vg(cmd, vg_name_from);
 | |
| 	unlock_vg(cmd, vg_name_to);
 | |
| 	return ECMD_FAILED;
 | |
| }
 |