mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
65623b63a2
So far pvmove_update_metadata (originaly _update_metadata) was used for both initial and subsequent metadata updates during polling. With a new polldaemon (lvmpolld) all operations that require polling have to be split in two parts: The initiating one and the polling one. The later step will be used from lvm command spawned by lvmpolld to monitor and advance the mirror on next segment if required. 1) The initiation part is _update_metadata in pvmove.c which performs only the first update, setting up the pvmove itself in metadata. 2) pvmove_update_metadata in pvmove_poll.c now handles all other subsequent metadata updates except the last one. Due to the split we could remove some code. Also some functions were moved back to pvmove.c as they were suited for initialisation of pvmove only.
222 lines
5.6 KiB
C
222 lines
5.6 KiB
C
/*
|
|
* Copyright (C) 2015 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 "pvmove_poll.h"
|
|
#include "tools.h"
|
|
|
|
struct volume_group *get_vg(struct cmd_context *cmd, const char *vgname)
|
|
{
|
|
dev_close_all();
|
|
|
|
return vg_read_for_update(cmd, vgname, NULL, 0);
|
|
}
|
|
|
|
static int _is_pvmove_image_removable(struct logical_volume *mimage_lv,
|
|
void *baton)
|
|
{
|
|
uint32_t mimage_to_remove = *((uint32_t *)baton);
|
|
struct lv_segment *mirror_seg;
|
|
|
|
if (!(mirror_seg = get_only_segment_using_this_lv(mimage_lv))) {
|
|
log_error(INTERNAL_ERROR "%s is not a proper mirror image",
|
|
mimage_lv->name);
|
|
return 0;
|
|
}
|
|
|
|
if (seg_type(mirror_seg, 0) != AREA_LV) {
|
|
log_error(INTERNAL_ERROR "%s is not a pvmove mirror of LV-type",
|
|
mirror_seg->lv->name);
|
|
return 0;
|
|
}
|
|
|
|
if (mimage_to_remove > mirror_seg->area_count) {
|
|
log_error(INTERNAL_ERROR "Mirror image %" PRIu32 " not found in segment",
|
|
mimage_to_remove);
|
|
return 0;
|
|
}
|
|
|
|
if (seg_lv(mirror_seg, mimage_to_remove) == mimage_lv)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _detach_pvmove_mirror(struct cmd_context *cmd,
|
|
struct logical_volume *lv_mirr)
|
|
{
|
|
uint32_t mimage_to_remove = 0;
|
|
struct dm_list lvs_completed;
|
|
struct lv_list *lvl;
|
|
|
|
/* Update metadata to remove mirror segments and break dependencies */
|
|
dm_list_init(&lvs_completed);
|
|
|
|
if (arg_is_set(cmd, abort_ARG) &&
|
|
(seg_type(first_seg(lv_mirr), 0) == AREA_LV))
|
|
mimage_to_remove = 1; /* remove the second mirror leg */
|
|
|
|
if (!lv_remove_mirrors(cmd, lv_mirr, 1, 0, _is_pvmove_image_removable, &mimage_to_remove, PVMOVE) ||
|
|
!remove_layers_for_segments_all(cmd, lv_mirr, PVMOVE,
|
|
&lvs_completed)) {
|
|
return 0;
|
|
}
|
|
|
|
dm_list_iterate_items(lvl, &lvs_completed)
|
|
/* FIXME Assumes only one pvmove at a time! */
|
|
lvl->lv->status &= ~LOCKED;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Called to advance the mirror to successive sections of it.
|
|
* (Not called first time or after the last section completes.)
|
|
*/
|
|
int pvmove_update_metadata(struct cmd_context *cmd, struct volume_group *vg,
|
|
struct logical_volume *lv_mirr,
|
|
struct dm_list *lvs_changed __attribute__((unused)),
|
|
unsigned flags __attribute__((unused)))
|
|
{
|
|
log_verbose("Updating volume group metadata.");
|
|
if (!vg_write(vg)) {
|
|
log_error("ABORTING: Volume group metadata update failed.");
|
|
return 0;
|
|
}
|
|
|
|
if (!suspend_lv(cmd, lv_mirr)) {
|
|
if (vg)
|
|
vg_revert(vg);
|
|
log_error("ABORTING: Temporary pvmove mirror reload failed.");
|
|
if (!revert_lv(cmd, lv_mirr))
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
/* Commit on-disk metadata */
|
|
if (!vg_commit(vg)) {
|
|
log_error("ABORTING: Volume group metadata update failed.");
|
|
if (!resume_lv(cmd, lv_mirr))
|
|
log_error("Unable to reactivate logical volume \"%s\".",
|
|
lv_mirr->name);
|
|
if (!revert_lv(cmd, lv_mirr))
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
if (!resume_lv(cmd, lv_mirr)) {
|
|
log_error("Unable to reactivate logical volume \"%s\".",
|
|
lv_mirr->name);
|
|
return 0;
|
|
}
|
|
|
|
backup(vg);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int pvmove_finish(struct cmd_context *cmd, struct volume_group *vg,
|
|
struct logical_volume *lv_mirr, struct dm_list *lvs_changed)
|
|
{
|
|
int r = 1;
|
|
|
|
if (!dm_list_empty(lvs_changed) &&
|
|
(!_detach_pvmove_mirror(cmd, lv_mirr) ||
|
|
!replace_lv_with_error_segment(lv_mirr))) {
|
|
log_error("ABORTING: Removal of temporary mirror failed");
|
|
return 0;
|
|
}
|
|
|
|
/* Store metadata without dependencies on mirror segments */
|
|
if (!vg_write(vg)) {
|
|
log_error("ABORTING: Failed to write new data locations "
|
|
"to disk.");
|
|
return 0;
|
|
}
|
|
|
|
/* Suspend LVs changed (implicitly suspends lv_mirr) */
|
|
if (!suspend_lvs(cmd, lvs_changed, vg)) {
|
|
log_error("ABORTING: Locking LVs to remove temporary mirror failed");
|
|
if (!revert_lv(cmd, lv_mirr))
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
/* Store metadata without dependencies on mirror segments */
|
|
if (!vg_commit(vg)) {
|
|
log_error("ABORTING: Failed to write new data locations "
|
|
"to disk.");
|
|
if (!revert_lv(cmd, lv_mirr))
|
|
stack;
|
|
if (!revert_lvs(cmd, lvs_changed))
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
/* Release mirror LV. (No pending I/O because it's been suspended.) */
|
|
if (!resume_lv(cmd, lv_mirr)) {
|
|
log_error("Unable to reactivate logical volume \"%s\"",
|
|
lv_mirr->name);
|
|
r = 0;
|
|
}
|
|
|
|
/* Unsuspend LVs */
|
|
if (!resume_lvs(cmd, lvs_changed))
|
|
stack;
|
|
|
|
/* Deactivate mirror LV */
|
|
if (!deactivate_lv(cmd, lv_mirr)) {
|
|
log_error("ABORTING: Unable to deactivate temporary logical "
|
|
"volume \"%s\"", lv_mirr->name);
|
|
r = 0;
|
|
}
|
|
|
|
log_verbose("Removing temporary pvmove LV");
|
|
if (!lv_remove(lv_mirr)) {
|
|
log_error("ABORTING: Removal of temporary pvmove LV failed");
|
|
return 0;
|
|
}
|
|
|
|
/* Store it on disks */
|
|
log_verbose("Writing out final volume group after pvmove");
|
|
if (!vg_write(vg) || !vg_commit(vg)) {
|
|
log_error("ABORTING: Failed to write new data locations "
|
|
"to disk.");
|
|
return 0;
|
|
}
|
|
|
|
/* FIXME backup positioning */
|
|
backup(vg);
|
|
|
|
return r;
|
|
}
|
|
|
|
struct volume_group *pvmove_get_copy_vg(struct cmd_context *cmd, const char *name,
|
|
const char *uuid __attribute__((unused)))
|
|
{
|
|
struct physical_volume *pv;
|
|
struct volume_group *vg;
|
|
|
|
/* Reread all metadata in case it got changed */
|
|
if (!(pv = find_pv_by_name(cmd, name, 0, 0))) {
|
|
log_error("ABORTING: Can't reread PV %s", name);
|
|
/* What more could we do here? */
|
|
return NULL;
|
|
}
|
|
|
|
vg = get_vg(cmd, pv_vg_name(pv));
|
|
free_pv_fid(pv);
|
|
|
|
return vg;
|
|
}
|