1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-06 17:18:29 +03:00

raid_manip: report index of tracking LV back; activate added striped image pairs

This commit is contained in:
Heinz Mauelshagen 2016-02-04 13:31:37 +01:00
parent ebda5f8a65
commit 10f5c83851
2 changed files with 63 additions and 55 deletions

View File

@ -295,8 +295,7 @@ static int is_same_level(const struct segment_type *t1, const struct segment_typ
return _cmp_level(t1, t2); return _cmp_level(t1, t2);
} }
static int _lv_is_raid_with_tracking(const struct logical_volume *lv, static int _lv_is_raid_with_tracking(const struct logical_volume *lv, uint32_t *ss)
struct logical_volume **tracking)
{ {
uint32_t s; uint32_t s;
const struct lv_segment *seg; const struct lv_segment *seg;
@ -307,7 +306,9 @@ static int _lv_is_raid_with_tracking(const struct logical_volume *lv,
for (s = 0; s < seg->area_count; s++) for (s = 0; s < seg->area_count; s++)
if (lv_is_visible(seg_lv(seg, s))) { if (lv_is_visible(seg_lv(seg, s))) {
if (!(seg_lv(seg, s)->status & LVM_WRITE)) { if (!(seg_lv(seg, s)->status & LVM_WRITE)) {
*tracking = seg_lv(seg, s); if (ss)
*ss = s;
return 1; return 1;
} }
@ -320,9 +321,7 @@ static int _lv_is_raid_with_tracking(const struct logical_volume *lv,
/* API function to check for @lv to be a tracking one */ /* API function to check for @lv to be a tracking one */
int lv_is_raid_with_tracking(const struct logical_volume *lv) int lv_is_raid_with_tracking(const struct logical_volume *lv)
{ {
struct logical_volume *tracking; return _lv_is_raid_with_tracking(lv, NULL);
return _lv_is_raid_with_tracking(lv, &tracking);
} }
/* HM Helper: return true in case this is a raid1 top-level LV inserted to do synchronization of 2 given sub LVs */ /* HM Helper: return true in case this is a raid1 top-level LV inserted to do synchronization of 2 given sub LVs */
@ -3230,7 +3229,7 @@ int lv_raid_split(struct logical_volume *lv, int yes,
const char *split_name, uint32_t new_image_count, const char *split_name, uint32_t new_image_count,
struct dm_list *splittable_pvs) struct dm_list *splittable_pvs)
{ {
uint32_t split_count; uint32_t split_count, s;
struct lv_list *lvl; struct lv_list *lvl;
struct dm_list meta_lvs, data_lvs; struct dm_list meta_lvs, data_lvs;
struct cmd_context *cmd; struct cmd_context *cmd;
@ -3274,8 +3273,8 @@ int lv_raid_split(struct logical_volume *lv, int yes,
* complete the split of the tracking sub-LV * complete the split of the tracking sub-LV
*/ */
log_debug_metadata("Check if LV %s is tracking changes", display_lvname(lv)); log_debug_metadata("Check if LV %s is tracking changes", display_lvname(lv));
if (_lv_is_raid_with_tracking(lv, &tracking)) { if (_lv_is_raid_with_tracking(lv, &s)) {
if (!lv_is_on_pvs(tracking, splittable_pvs)) { if (!lv_is_on_pvs((tracking = seg_lv(seg, s)), splittable_pvs)) {
log_error("Unable to split additional image from %s " log_error("Unable to split additional image from %s "
"while tracking changes for %s", "while tracking changes for %s",
lv->name, tracking->name); lv->name, tracking->name);
@ -3407,10 +3406,12 @@ int lv_raid_split_and_track(struct logical_volume *lv,
int s; int s;
struct logical_volume *split_lv; struct logical_volume *split_lv;
struct lv_segment *seg; struct lv_segment *seg;
struct volume_group *vg;
RETURN_IF_LV_SEG_ZERO(lv, (seg = first_seg(lv))); RETURN_IF_LV_SEG_ZERO(lv, (seg = first_seg(lv)));
RETURN_IF_NONZERO(!seg_is_mirrored(seg) && !seg_is_raid01(seg), RETURN_IF_NONZERO(!seg_is_mirrored(seg) && !seg_is_raid01(seg),
"mirrored/raid10 segment to split off"); "mirrored/raid10 segment to split off");
vg = lv->vg;
if (!_raid_in_sync(lv)) { if (!_raid_in_sync(lv)) {
log_error("Unable to split image from %s while not in-sync", log_error("Unable to split image from %s while not in-sync",
@ -3498,19 +3499,18 @@ int lv_raid_split_and_track(struct logical_volume *lv,
return 0; return 0;
} }
if (!lv_update_and_reload(lv)) if (!vg_write(vg) || !vg_commit(vg) || !backup(vg))
return_0; return_0;
log_print_unless_silent("%s split from %s for read-only purposes.",
split_lv->name, lv->name);
/* Suspend+resume the tracking LV to create its devnode */ /* Suspend+resume the tracking LV to create its devnode */
if (!suspend_lv(lv->vg->cmd, split_lv) || !resume_lv(lv->vg->cmd, split_lv)) { if (!suspend_lv(vg->cmd, split_lv) || !resume_lv(vg->cmd, split_lv)) {
log_error("Failed to suspend+resume %s after committing changes", log_error("Failed to suspend+resume %s after committing changes",
display_lvname(split_lv)); display_lvname(split_lv));
return 0; return 0;
} }
log_print_unless_silent("%s split from %s for read-only purposes.",
split_lv->name, lv->name);
log_print_unless_silent("Use 'lvconvert --merge %s' to merge back into %s", log_print_unless_silent("Use 'lvconvert --merge %s' to merge back into %s",
display_lvname(split_lv), display_lvname(lv)); display_lvname(split_lv), display_lvname(lv));
return 1; return 1;
@ -3589,13 +3589,12 @@ int lv_raid_merge(struct logical_volume *image_lv)
"mirrored/raid10 to merge into, rejecting request"); "mirrored/raid10 to merge into, rejecting request");
RETURN_IF_ZERO(seg->meta_areas, "metadata LV areas"); RETURN_IF_ZERO(seg->meta_areas, "metadata LV areas");
if (!_lv_is_raid_with_tracking(lv, &tracking)) { if (!_lv_is_raid_with_tracking(lv, &s)) {
log_error("%s is not a tracking LV.", log_error("%s is not a tracking LV.", display_lvname(lv));
display_lvname(lv));
return 0; return 0;
} }
if (tracking != image_lv) { if ((tracking = seg_lv(seg, s)) != image_lv) {
log_error("%s is not the tracking LV of %s but %s is.", log_error("%s is not the tracking LV of %s but %s is.",
display_lvname(image_lv), display_lvname(lv), display_lvname(tracking)); display_lvname(image_lv), display_lvname(lv), display_lvname(tracking));
return 0; return 0;
@ -3614,13 +3613,7 @@ int lv_raid_merge(struct logical_volume *image_lv)
display_lvname(image_lv)); display_lvname(image_lv));
} }
for (s = 0; s < seg->area_count; s++) if (!(meta_lv = seg_metalv(seg, s))) {
if (seg_lv(seg, s) == image_lv) {
meta_lv = seg_metalv(seg, s);
break;
}
if (!meta_lv) {
log_error("Failed to find metadata LV for %s in %s.", log_error("Failed to find metadata LV for %s in %s.",
display_lvname(image_lv), display_lvname(lv)); display_lvname(image_lv), display_lvname(lv));
return 0; return 0;
@ -3641,7 +3634,7 @@ int lv_raid_merge(struct logical_volume *image_lv)
log_print_unless_silent("LV %s successfully merged back into %s", log_print_unless_silent("LV %s successfully merged back into %s",
display_lvname(image_lv), display_lvname(lv)); display_lvname(image_lv), display_lvname(lv));
return _lv_cond_repair(lv); return 1;
} }
/* /*
@ -4351,7 +4344,7 @@ PFLA("lv->size=%s seg->len=%u seg->area_len=%u seg->area_count=%u old_image_coun
* Reshape: add disks to existing raid lv * Reshape: add disks to existing raid lv
* *
*/ */
static int _raid_reshape_add_disks(struct logical_volume *lv, static int _raid_reshape_add_images(struct logical_volume *lv,
const struct segment_type *new_segtype, const struct segment_type *new_segtype,
int yes, int force, int yes, int force,
uint32_t old_image_count, uint32_t new_image_count, uint32_t old_image_count, uint32_t new_image_count,
@ -4359,15 +4352,19 @@ static int _raid_reshape_add_disks(struct logical_volume *lv,
struct dm_list *allocate_pvs) struct dm_list *allocate_pvs)
{ {
uint32_t grown_le_count, current_le_count, s; uint32_t grown_le_count, current_le_count, s;
struct volume_group *vg;
struct logical_volume *slv;
struct lv_segment *seg; struct lv_segment *seg;
struct lvinfo info = { 0 }; struct lvinfo info = { 0 };
RETURN_IF_LV_SEG_SEGTYPE_ZERO(lv, (seg = first_seg(lv)), new_segtype); RETURN_IF_LV_SEG_SEGTYPE_ZERO(lv, (seg = first_seg(lv)), new_segtype);
vg = lv->vg;
if (!lv_info(lv->vg->cmd, lv, 0, &info, 1, 0) && driver_version(NULL, 0)) { if (!lv_info(vg->cmd, lv, 0, &info, 1, 0) && driver_version(NULL, 0)) {
log_error("lv_info failed: aborting"); log_error("lv_info failed: aborting");
return 0; return 0;
} }
if (seg->segtype != new_segtype) if (seg->segtype != new_segtype)
log_print_unless_silent("Ignoring layout change on device adding reshape"); log_print_unless_silent("Ignoring layout change on device adding reshape");
PFL(); PFL();
@ -4404,7 +4401,7 @@ PFL();
if (!_lv_alloc_reshape_space(lv, alloc_begin, NULL, allocate_pvs)) if (!_lv_alloc_reshape_space(lv, alloc_begin, NULL, allocate_pvs))
return 0; return 0;
PFLA("lv->size=%s", display_size(lv->vg->cmd, lv->size)); PFLA("lv->size=%s", display_size(vg->cmd, lv->size));
PFLA("lv->le_count=%u", lv->le_count); PFLA("lv->le_count=%u", lv->le_count);
PFLA("seg->len=%u", first_seg(lv)->len); PFLA("seg->len=%u", first_seg(lv)->len);
/* /*
@ -4415,10 +4412,21 @@ PFLA("seg->len=%u", first_seg(lv)->len);
*/ */
log_debug_metadata("Setting delta disk flag on new data LVs of %s", log_debug_metadata("Setting delta disk flag on new data LVs of %s",
display_lvname(lv)); display_lvname(lv));
if (old_image_count < seg->area_count) {
if (!vg_write(vg) || !vg_commit(vg) || !backup(vg)) {
log_error("metadata commit/backup failed");
return 0;
}
for (s = old_image_count; s < seg->area_count; s++) { for (s = old_image_count; s < seg->area_count; s++) {
PFLA("seg_lv(seg, %u)=%s", s, seg_lv(seg, s)->name); slv = seg_lv(seg, s);
seg_lv(seg, s)->status &= ~LV_REBUILD; PFLA("seg_lv(seg, %u)=%s", s, slv);
seg_lv(seg, s)->status |= LV_RESHAPE_DELTA_DISKS_PLUS; slv->status &= ~LV_REBUILD;
slv->status |= LV_RESHAPE_DELTA_DISKS_PLUS;
if (!activate_lv_excl_local(vg->cmd, slv) ||
!activate_lv_excl_local(vg->cmd, seg_metalv(seg, s)))
return_0;
}
} }
return 1; return 1;
@ -4430,7 +4438,7 @@ PFLA("seg_lv(seg, %u)=%s", s, seg_lv(seg, s)->name);
* Reshape: remove disks from existing raid lv * Reshape: remove disks from existing raid lv
* *
*/ */
static int _raid_reshape_remove_disks(struct logical_volume *lv, static int _raid_reshape_remove_images(struct logical_volume *lv,
const struct segment_type *new_segtype, const struct segment_type *new_segtype,
int yes, int force, int yes, int force,
uint32_t old_image_count, uint32_t new_image_count, uint32_t old_image_count, uint32_t new_image_count,
@ -4928,14 +4936,14 @@ PFLA("devs_in_sync=%u old_image_count=%u new_image_count=%u", devs_in_sync,old_i
/* Handle disk addition reshaping */ /* Handle disk addition reshaping */
if (old_image_count < new_image_count) { if (old_image_count < new_image_count) {
PFL(); PFL();
if (!_raid_reshape_add_disks(lv, new_segtype, yes, force, if (!_raid_reshape_add_images(lv, new_segtype, yes, force,
old_image_count, new_image_count, old_image_count, new_image_count,
new_stripes, new_stripe_size, allocate_pvs)) new_stripes, new_stripe_size, allocate_pvs))
return 0; return 0;
/* Handle disk removal reshaping */ /* Handle disk removal reshaping */
} else if (old_image_count > new_image_count) { } else if (old_image_count > new_image_count) {
if (!_raid_reshape_remove_disks(lv, new_segtype, yes, force, if (!_raid_reshape_remove_images(lv, new_segtype, yes, force,
old_image_count, new_image_count, old_image_count, new_image_count,
new_stripes, new_stripe_size, new_stripes, new_stripe_size,
allocate_pvs, &removal_lvs)) allocate_pvs, &removal_lvs))
@ -9120,7 +9128,7 @@ static int _raid_convert_define_parms(const struct lv_segment *seg,
!segtype_is_raid1(*segtype) && !segtype_is_raid1(*segtype) &&
!segtype_is_raid01(*segtype)) { !segtype_is_raid01(*segtype)) {
if (!segtype_is_any_raid6(*segtype) && *data_copies > *stripes) { if (!segtype_is_any_raid6(*segtype) && *data_copies > *stripes) {
log_error("Number of data copies %u is larger than number of stripes %u!", log_error("Number of data copies %u is larger than number of stripes %u",
*data_copies, *stripes); *data_copies, *stripes);
return 0; return 0;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (C) 2005-2014 Red Hat, Inc. All rights reserved. * Copyright (C) 2005-2016 Red Hat, Inc. All rights reserved.
* *
* This file is part of the device-mapper userspace tools. * This file is part of the device-mapper userspace tools.
* *