1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-20 18:09:23 +03:00
lvm2/lib/metadata/merge.c

245 lines
5.9 KiB
C
Raw Normal View History

/*
2004-03-30 19:35:44 +00:00
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
* Copyright (C) 2004 Red Hat, Inc. All rights reserved.
*
2004-03-30 19:35:44 +00:00
* 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
*/
2002-11-18 14:04:08 +00:00
#include "lib.h"
2002-01-10 18:12:26 +00:00
#include "metadata.h"
2003-09-15 18:22:50 +00:00
#include "toolcontext.h"
#include "lv_alloc.h"
#include "pv_alloc.h"
2004-03-08 17:19:15 +00:00
#include "str_list.h"
2004-09-16 18:40:56 +00:00
#include "segtype.h"
/*
* Attempt to merge two adjacent segments.
2004-05-04 21:25:57 +00:00
* Currently only supports striped segments on AREA_PV.
* Returns success if successful, in which case 'first'
* gets adjusted to contain both areas.
*/
static int _merge(struct lv_segment *first, struct lv_segment *second)
{
2004-05-04 21:25:57 +00:00
if (!first || !second || first->segtype != second->segtype ||
!first->segtype->ops->merge_segments) return 0;
2004-05-04 21:25:57 +00:00
return first->segtype->ops->merge_segments(first, second);
}
int lv_merge_segments(struct logical_volume *lv)
{
struct list *segh, *t;
2002-11-18 14:04:08 +00:00
struct lv_segment *current, *prev = NULL;
if (lv->status & LOCKED || lv->status & PVMOVE)
return 1;
list_iterate_safe(segh, t, &lv->segments) {
2002-11-18 14:04:08 +00:00
current = list_item(segh, struct lv_segment);
if (_merge(prev, current))
list_del(&current->list);
else
prev = current;
}
return 1;
}
/*
* Verify that an LV's segments are consecutive, complete and don't overlap.
*/
2005-06-01 16:51:55 +00:00
int check_lv_segments(struct logical_volume *lv)
{
struct lv_segment *seg;
uint32_t le = 0;
unsigned seg_count = 0;
2005-06-01 16:51:55 +00:00
int r = 1;
uint32_t area_multiplier, s;
list_iterate_items(seg, &lv->segments) {
seg_count++;
if (seg->le != le) {
log_error("LV %s invalid: segment %u should begin at "
"LE %" PRIu32 " (found %" PRIu32 ").",
lv->name, seg_count, le, seg->le);
2005-06-01 16:51:55 +00:00
r = 0;
}
area_multiplier = segtype_is_striped(seg->segtype) ?
seg->area_count : 1;
if (seg->area_len * area_multiplier != seg->len) {
log_error("LV %s: segment %u has inconsistent "
"area_len %u",
lv->name, seg_count, seg->area_len);
r = 0;
}
for (s = 0; s < seg->area_count; s++) {
if (seg_type(seg, s) == AREA_PV) {
if (!seg_pvseg(seg, s) ||
seg_pvseg(seg, s)->lvseg != seg ||
seg_pvseg(seg, s)->lv_area != s) {
log_error("LV %s: segment %u has "
"inconsistent PV area %u",
lv->name, seg_count, s);
r = 0;
}
} else {
if (!seg_lv(seg, s) ||
seg_lv(seg, s)->vg != lv->vg ||
seg_lv(seg, s) == lv) {
log_error("LV %s: segment %u has "
"inconsistent LV area %u",
lv->name, seg_count, s);
r = 0;
}
if (seg_le(seg, s) != le) {
log_error("LV %s: segment %u has "
"inconsistent LV area %u "
"size",
lv->name, seg_count, s);
r = 0;
}
}
}
le += seg->len;
}
2005-06-01 16:51:55 +00:00
if (le != lv->le_count) {
log_error("LV %s: inconsistent LE count %u != %u",
lv->name, le, lv->le_count);
r = 0;
}
return r;
}
2003-09-15 18:22:50 +00:00
/*
* Split the supplied segment at the supplied logical extent
* NB Use LE numbering that works across stripes PV1: 0,2,4 PV2: 1,3,5 etc.
2003-09-15 18:22:50 +00:00
*/
static int _lv_split_segment(struct logical_volume *lv, struct lv_segment *seg,
uint32_t le)
{
struct lv_segment *split_seg;
uint32_t s;
uint32_t offset = le - seg->le;
uint32_t area_offset;
2003-09-15 18:22:50 +00:00
if (!seg_can_split(seg)) {
2004-05-04 21:25:57 +00:00
log_error("Unable to split the %s segment at LE %" PRIu32
" in LV %s", seg->segtype->name, le, lv->name);
2003-09-15 18:22:50 +00:00
return 0;
}
/* Clone the existing segment */
2005-04-22 15:44:00 +00:00
if (!(split_seg = alloc_lv_segment(lv->vg->cmd->mem, seg->segtype,
seg->lv, seg->le, seg->len,
seg->status, seg->stripe_size,
2005-06-01 16:51:55 +00:00
seg->log_lv,
2005-04-22 15:44:00 +00:00
seg->area_count, seg->area_len,
2005-06-01 16:51:55 +00:00
seg->chunk_size, seg->region_size,
2005-04-22 15:44:00 +00:00
seg->extents_copied))) {
log_error("Couldn't allocate cloned LV segment.");
2003-09-15 18:22:50 +00:00
return 0;
}
2004-03-08 17:19:15 +00:00
if (!str_list_dup(lv->vg->cmd->mem, &split_seg->tags, &seg->tags)) {
log_error("LV segment tags duplication failed");
return 0;
}
2003-09-15 18:22:50 +00:00
/* In case of a striped segment, the offset has to be / stripes */
area_offset = offset;
if (seg_is_striped(seg))
area_offset /= seg->area_count;
2003-09-15 18:22:50 +00:00
split_seg->area_len -= area_offset;
seg->area_len = area_offset;
split_seg->len -= offset;
seg->len = offset;
split_seg->le = seg->le + seg->len;
2003-09-15 18:22:50 +00:00
/* Adjust the PV mapping */
for (s = 0; s < seg->area_count; s++) {
2005-06-01 16:51:55 +00:00
seg_type(split_seg, s) = seg_type(seg, s);
2003-09-15 18:22:50 +00:00
/* Split area at the offset */
2005-06-01 16:51:55 +00:00
switch (seg_type(seg, s)) {
2003-09-15 18:22:50 +00:00
case AREA_LV:
2005-06-01 16:51:55 +00:00
seg_lv(split_seg, s) = seg_lv(seg, s);
seg_le(split_seg, s) =
seg_le(seg, s) + seg->area_len;
log_debug("Split %s:%u[%u] at %u: %s LE %u", lv->name,
2005-06-01 16:51:55 +00:00
seg->le, s, le, seg_lv(seg, s)->name,
seg_le(split_seg, s));
2003-09-15 18:22:50 +00:00
break;
case AREA_PV:
2005-06-01 16:51:55 +00:00
if (!assign_peg_to_lvseg(seg_pv(seg, s),
seg_pe(seg, s) +
seg->area_len,
2005-06-01 16:51:55 +00:00
seg_pvseg(seg, s)->len -
seg->area_len,
split_seg, s)) {
stack;
return 0;
}
log_debug("Split %s:%u[%u] at %u: %s PE %u", lv->name,
seg->le, s, le,
2005-06-01 16:51:55 +00:00
dev_name(seg_dev(seg, s)),
seg_pe(split_seg, s));
2003-09-15 18:22:50 +00:00
break;
default:
log_error("Unrecognised segment type %u",
2005-06-01 16:51:55 +00:00
seg_type(seg, s));
2003-09-15 18:22:50 +00:00
return 0;
}
}
/* Add split off segment to the list _after_ the original one */
list_add_h(&seg->list, &split_seg->list);
return 1;
}
/*
* Ensure there's a segment boundary at the given logical extent
*/
int lv_split_segment(struct logical_volume *lv, uint32_t le)
{
struct lv_segment *seg;
if (!(seg = find_seg_by_le(lv, le))) {
log_error("Segment with extent %" PRIu32 " in LV %s not found",
le, lv->name);
return 0;
}
/* This is a segment start already */
if (le == seg->le)
return 1;
if (!_lv_split_segment(lv, seg, le)) {
stack;
return 0;
}
return 1;
}