mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-10 05:18:36 +03:00
cbfbc6766e
When vg_lock_and_read() calls were added, they were done so incorrectly for the destination VG (vg_to). This resulted in the VG lock not obtained when a new VG was the destination (vg_lock_and_read() would fail in the vg_read() clause, which would then release the lock before returning NULL), and could result in corrupted destination VG. The fix was to put back the original lock_vol() and vg_read() calls for 'vg_to'. The failure of vg_read() indicates "vg does not exist", and we key off that to determine whether we are dealing with a new or existing VG as the destination. The first two error messages were also the result of the incorrect vg_lock_and_read() calls: Volume group "new" not found cluster request failed: Invalid argument New volume group "new" successfully split from "vg" Fixes https://bugzilla.redhat.com/show_bug.cgi?id=438249
409 lines
9.8 KiB
C
409 lines
9.8 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2007 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 Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser 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 void _move_pv(struct volume_group *vg_from, struct volume_group *vg_to,
|
|
struct pv_list *pvl)
|
|
{
|
|
struct physical_volume *pv;
|
|
|
|
list_move(&pvl->list, &vg_to->pvs);
|
|
|
|
vg_from->pv_count--;
|
|
vg_to->pv_count++;
|
|
|
|
pv = list_item(pvl, struct pv_list)->pv;
|
|
|
|
vg_from->extent_count -= pv_pe_count(pv);
|
|
vg_to->extent_count += pv_pe_count(pv);
|
|
|
|
vg_from->free_count -= pv_pe_count(pv) - pv_pe_alloc_count(pv);
|
|
vg_to->free_count += pv_pe_count(pv) - pv_pe_alloc_count(pv);
|
|
}
|
|
|
|
/* 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 s;
|
|
|
|
list_iterate_safe(lvh, lvht, &vg_from->lvs) {
|
|
lv = list_item(lvh, struct lv_list)->lv;
|
|
|
|
if ((lv->status & SNAPSHOT))
|
|
continue;
|
|
|
|
if ((lv->status & MIRRORED))
|
|
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("Can't split Logical "
|
|
"Volume %s between "
|
|
"two 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",
|
|
pv_dev_name(pv));
|
|
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("Can't split snapshot %s between"
|
|
" two Volume Groups", 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;
|
|
}
|
|
|
|
static int _move_mirrors(struct volume_group *vg_from,
|
|
struct volume_group *vg_to)
|
|
{
|
|
struct list *lvh, *lvht;
|
|
struct logical_volume *lv;
|
|
struct lv_segment *seg;
|
|
unsigned s, seg_in, log_in;
|
|
|
|
list_iterate_safe(lvh, lvht, &vg_from->lvs) {
|
|
lv = list_item(lvh, struct lv_list)->lv;
|
|
|
|
if (!(lv->status & MIRRORED))
|
|
continue;
|
|
|
|
seg = first_seg(lv);
|
|
|
|
seg_in = 0;
|
|
for (s = 0; s < seg->area_count; s++)
|
|
if (_lv_is_in_vg(vg_to, seg_lv(seg, s)))
|
|
seg_in++;
|
|
|
|
log_in = (!seg->log_lv || _lv_is_in_vg(vg_to, seg->log_lv));
|
|
|
|
if ((seg_in && seg_in < seg->area_count) ||
|
|
(seg_in && seg->log_lv && !log_in) ||
|
|
(!seg_in && seg->log_lv && log_in)) {
|
|
log_error("Can't split mirror %s between "
|
|
"two Volume Groups", lv->name);
|
|
return 0;
|
|
}
|
|
|
|
if (seg_in == seg->area_count && log_in) {
|
|
list_del(lvh);
|
|
list_add(&vg_to->lvs, lvh);
|
|
|
|
vg_from->lv_count--;
|
|
vg_to->lv_count++;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Has the user given an option related to a new vg as the split destination?
|
|
*/
|
|
static int new_vg_option_specified(struct cmd_context *cmd)
|
|
{
|
|
return(arg_count(cmd, clustered_ARG) ||
|
|
arg_count(cmd, alloc_ARG) ||
|
|
arg_count(cmd, maxphysicalvolumes_ARG) ||
|
|
arg_count(cmd, maxlogicalvolumes_ARG));
|
|
}
|
|
|
|
int vgsplit(struct cmd_context *cmd, int argc, char **argv)
|
|
{
|
|
struct vgcreate_params vp_new;
|
|
struct vgcreate_params vp_def;
|
|
char *vg_name_from, *vg_name_to;
|
|
struct volume_group *vg_to, *vg_from;
|
|
int opt;
|
|
int active;
|
|
int existing_vg;
|
|
int old_suppress;
|
|
struct pv_list *pvl;
|
|
int consistent;
|
|
|
|
if (argc < 3) {
|
|
log_error("Existing VG, new VG and physical volumes required.");
|
|
return EINVALID_CMD_LINE;
|
|
}
|
|
|
|
vg_name_from = skip_dev_dir(cmd, argv[0], NULL);
|
|
vg_name_to = skip_dev_dir(cmd, argv[1], NULL);
|
|
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 (!(vg_from = vg_lock_and_read(cmd, vg_name_from, NULL, LCK_VG_WRITE,
|
|
CLUSTERED | EXPORTED_VG |
|
|
RESIZEABLE_VG | LVM_WRITE,
|
|
CORRECT_INCONSISTENT | FAIL_INCONSISTENT)))
|
|
return ECMD_FAILED;
|
|
|
|
if ((active = lvs_in_vg_activated(vg_from))) {
|
|
/* FIXME Remove this restriction */
|
|
log_error("Logical volumes in \"%s\" must be inactive",
|
|
vg_name_from);
|
|
unlock_vg(cmd, vg_name_from);
|
|
return ECMD_FAILED;
|
|
}
|
|
|
|
log_verbose("Checking for new 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, NULL, &consistent))) {
|
|
existing_vg = 1;
|
|
if (new_vg_option_specified(cmd)) {
|
|
log_error("Volume group \"%s\" exists, but new VG "
|
|
"option specified", vg_name_to);
|
|
goto error;
|
|
}
|
|
if (!vgs_are_compatible(cmd, vg_from,vg_to))
|
|
goto error;
|
|
} else {
|
|
log_suppress(old_suppress);
|
|
existing_vg = 0;
|
|
|
|
/* Set metadata format of original VG */
|
|
/* FIXME: need some common logic */
|
|
cmd->fmt = vg_from->fid->fmt;
|
|
|
|
vp_def.vg_name = NULL;
|
|
vp_def.extent_size = vg_from->extent_size;
|
|
vp_def.max_pv = vg_from->max_pv;
|
|
vp_def.max_lv = vg_from->max_lv;
|
|
vp_def.alloc = vg_from->alloc;
|
|
vp_def.clustered = 0;
|
|
|
|
if (fill_vg_create_params(cmd, vg_name_to, &vp_new, &vp_def)) {
|
|
unlock_vg(cmd, vg_name_from);
|
|
return EINVALID_CMD_LINE;
|
|
}
|
|
|
|
if (validate_vg_create_params(cmd, &vp_new)) {
|
|
unlock_vg(cmd, vg_name_from);
|
|
return EINVALID_CMD_LINE;
|
|
}
|
|
|
|
if (!(vg_to = vg_create(cmd, vg_name_to, vp_new.extent_size,
|
|
vp_new.max_pv, vp_new.max_lv,
|
|
vp_new.alloc, 0, NULL)))
|
|
goto error;
|
|
|
|
if (vg_from->status & CLUSTERED)
|
|
vg_to->status |= CLUSTERED;
|
|
}
|
|
|
|
/* 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 (!(pvl = find_pv_in_vg(vg_from, argv[opt]))) {
|
|
log_error("Physical volume %s not in volume group %s",
|
|
argv[opt], vg_from->name);
|
|
goto error;
|
|
}
|
|
|
|
_move_pv(vg_from, vg_to, pvl);
|
|
}
|
|
|
|
/* 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;
|
|
|
|
/* Move required mirrors across */
|
|
if (!(_move_mirrors(vg_from, vg_to)))
|
|
goto error;
|
|
|
|
/* Split metadata areas and check if both vgs have at least one area */
|
|
if (!(vg_split_mdas(cmd, vg_from, vg_to)) && vg_from->pv_count) {
|
|
log_error("Cannot split: Nowhere to store metadata for new Volume Group");
|
|
goto error;
|
|
}
|
|
|
|
/* Set proper name for all PVs in new VG */
|
|
if (!vg_rename(cmd, vg_to, vg_name_to))
|
|
goto error;
|
|
|
|
/* store it on disks */
|
|
log_verbose("Writing out updated volume groups");
|
|
|
|
/*
|
|
* First, write out the new VG as EXPORTED. We do this first in case
|
|
* there is a crash - we will still have the new VG information, in an
|
|
* exported state. Recovery after this point would be removal of the
|
|
* new VG and redoing the vgsplit.
|
|
* FIXME: recover automatically or instruct the user?
|
|
*/
|
|
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);
|
|
|
|
/*
|
|
* Next, write out the updated old VG. If we crash after this point,
|
|
* recovery is a vgimport on the new VG.
|
|
* FIXME: recover automatically or instruct the user the user?
|
|
*/
|
|
if (vg_from->pv_count) {
|
|
if (!vg_write(vg_from) || !vg_commit(vg_from))
|
|
goto error;
|
|
|
|
backup(vg_from);
|
|
}
|
|
|
|
/*
|
|
* Finally, remove the EXPORTED flag from the new VG and write it out.
|
|
*/
|
|
consistent = 1;
|
|
if (!test_mode() &&
|
|
(!(vg_to = vg_read(cmd, vg_name_to, NULL, &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("%s volume group \"%s\" successfully split from \"%s\"",
|
|
existing_vg ? "Existing" : "New",
|
|
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;
|
|
}
|