mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-30 17:18:21 +03:00
259 lines
7.0 KiB
C
259 lines
7.0 KiB
C
/*
|
|
* Copyright (C) 2003-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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "base/memory/zalloc.h"
|
|
#include "lib/misc/lib.h"
|
|
#include "lib/commands/toolcontext.h"
|
|
#include "lib/metadata/segtype.h"
|
|
#include "lib/display/display.h"
|
|
#include "lib/format_text/text_export.h"
|
|
#include "lib/format_text/text_import.h"
|
|
#include "lib/config/config.h"
|
|
#include "lib/datastruct/str_list.h"
|
|
#include "lib/activate/targets.h"
|
|
#include "lib/misc/lvm-string.h"
|
|
#include "lib/activate/activate.h"
|
|
#include "lib/metadata/pv_alloc.h"
|
|
#include "lib/metadata/metadata.h"
|
|
|
|
static const char *_striped_name(const struct lv_segment *seg)
|
|
{
|
|
return (seg->area_count == 1) ? SEG_TYPE_NAME_LINEAR : seg->segtype->name;
|
|
}
|
|
|
|
static void _striped_display(const struct lv_segment *seg)
|
|
{
|
|
uint32_t s;
|
|
|
|
if (seg->area_count == 1)
|
|
display_stripe(seg, 0, " ");
|
|
else {
|
|
log_print(" Stripes\t\t%u", seg->area_count);
|
|
|
|
if (seg->lv->vg->cmd->si_unit_consistency)
|
|
log_print(" Stripe size\t\t%s",
|
|
display_size(seg->lv->vg->cmd,
|
|
(uint64_t) seg->stripe_size));
|
|
else
|
|
log_print(" Stripe size\t\t%u KB",
|
|
seg->stripe_size / 2);
|
|
|
|
for (s = 0; s < seg->area_count; s++) {
|
|
log_print(" Stripe %d:", s);
|
|
display_stripe(seg, s, " ");
|
|
}
|
|
}
|
|
log_print(" ");
|
|
}
|
|
|
|
static int _striped_text_import_area_count(const struct dm_config_node *sn, uint32_t *area_count)
|
|
{
|
|
if (!dm_config_get_uint32(sn, "stripe_count", area_count)) {
|
|
log_error("Couldn't read 'stripe_count' for "
|
|
"segment '%s'.", dm_config_parent_name(sn));
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _striped_text_import(struct lv_segment *seg, const struct dm_config_node *sn,
|
|
struct dm_hash_table *pv_hash)
|
|
{
|
|
const struct dm_config_value *cv;
|
|
|
|
if ((seg->area_count != 1) &&
|
|
!dm_config_get_uint32(sn, "stripe_size", &seg->stripe_size)) {
|
|
log_error("Couldn't read stripe_size for segment %s "
|
|
"of logical volume %s.", dm_config_parent_name(sn), seg->lv->name);
|
|
return 0;
|
|
}
|
|
|
|
if (!dm_config_get_list(sn, "stripes", &cv)) {
|
|
log_error("Couldn't find stripes array for segment %s "
|
|
"of logical volume %s.", dm_config_parent_name(sn), seg->lv->name);
|
|
return 0;
|
|
}
|
|
|
|
seg->area_len /= seg->area_count;
|
|
|
|
return text_import_areas(seg, sn, cv, pv_hash, 0);
|
|
}
|
|
|
|
static int _striped_text_export(const struct lv_segment *seg, struct formatter *f)
|
|
{
|
|
|
|
outfc(f, (seg->area_count == 1) ? "# linear" : NULL,
|
|
"stripe_count = %u", seg->area_count);
|
|
|
|
if (seg->area_count > 1)
|
|
outsize(f, (uint64_t) seg->stripe_size,
|
|
"stripe_size = %u", seg->stripe_size);
|
|
|
|
return out_areas(f, seg, "stripe");
|
|
}
|
|
|
|
/*
|
|
* Test whether two segments could be merged by the current merging code
|
|
*/
|
|
static int _striped_segments_compatible(struct lv_segment *first,
|
|
struct lv_segment *second)
|
|
{
|
|
uint32_t width;
|
|
unsigned s;
|
|
|
|
if ((first->area_count != second->area_count) ||
|
|
(first->stripe_size != second->stripe_size))
|
|
return 0;
|
|
|
|
for (s = 0; s < first->area_count; s++) {
|
|
|
|
/* FIXME Relax this to first area type != second area type */
|
|
/* plus the additional AREA_LV checks needed */
|
|
if ((seg_type(first, s) != AREA_PV) ||
|
|
(seg_type(second, s) != AREA_PV))
|
|
return 0;
|
|
|
|
width = first->area_len;
|
|
|
|
if ((seg_pv(first, s) !=
|
|
seg_pv(second, s)) ||
|
|
(seg_pe(first, s) + width !=
|
|
seg_pe(second, s)))
|
|
return 0;
|
|
}
|
|
|
|
if (!str_list_lists_equal(&first->tags, &second->tags))
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _striped_merge_segments(struct lv_segment *seg1, struct lv_segment *seg2)
|
|
{
|
|
uint32_t s;
|
|
|
|
if (!_striped_segments_compatible(seg1, seg2))
|
|
return 0;
|
|
|
|
seg1->len += seg2->len;
|
|
seg1->area_len += seg2->area_len;
|
|
|
|
for (s = 0; s < seg1->area_count; s++)
|
|
if (seg_type(seg1, s) == AREA_PV)
|
|
merge_pv_segments(seg_pvseg(seg1, s),
|
|
seg_pvseg(seg2, s));
|
|
|
|
return 1;
|
|
}
|
|
|
|
#ifdef DEVMAPPER_SUPPORT
|
|
static int _striped_target_status_compatible(const char *type)
|
|
{
|
|
return (strcmp(type, TARGET_NAME_LINEAR) == 0);
|
|
}
|
|
|
|
static int _striped_add_target_line(struct dev_manager *dm,
|
|
struct dm_pool *mem __attribute__((unused)),
|
|
struct cmd_context *cmd __attribute__((unused)),
|
|
void **target_state __attribute__((unused)),
|
|
struct lv_segment *seg,
|
|
const struct lv_activate_opts *laopts __attribute__((unused)),
|
|
struct dm_tree_node *node, uint64_t len,
|
|
uint32_t *pvmove_mirror_count __attribute__((unused)))
|
|
{
|
|
if (!seg->area_count) {
|
|
log_error(INTERNAL_ERROR "striped add_target_line called "
|
|
"with no areas for %s.", seg->lv->name);
|
|
return 0;
|
|
}
|
|
if (seg->area_count == 1) {
|
|
if (!add_linear_area_to_dtree(node, len, seg->lv->vg->extent_size,
|
|
cmd->use_linear_target,
|
|
seg->lv->vg->name, seg->lv->name))
|
|
return_0;
|
|
} else if (!dm_tree_node_add_striped_target(node, len,
|
|
seg->stripe_size))
|
|
return_0;
|
|
|
|
return add_areas_line(dm, seg, node, 0u, seg->area_count);
|
|
}
|
|
|
|
static int _striped_target_present(struct cmd_context *cmd,
|
|
const struct lv_segment *seg __attribute__((unused)),
|
|
unsigned *attributes __attribute__((unused)))
|
|
{
|
|
static int _striped_checked = 0;
|
|
static int _striped_present = 0;
|
|
|
|
if (!activation())
|
|
return 0;
|
|
|
|
if (!_striped_checked) {
|
|
_striped_checked = 1;
|
|
_striped_present = target_present(cmd, TARGET_NAME_LINEAR, 0) &&
|
|
target_present(cmd, TARGET_NAME_STRIPED, 0);
|
|
}
|
|
|
|
return _striped_present;
|
|
}
|
|
#endif
|
|
|
|
static void _striped_destroy(struct segment_type *segtype)
|
|
{
|
|
free(segtype);
|
|
}
|
|
|
|
static const struct segtype_handler _striped_ops = {
|
|
.name = _striped_name,
|
|
.display = _striped_display,
|
|
.text_import_area_count = _striped_text_import_area_count,
|
|
.text_import = _striped_text_import,
|
|
.text_export = _striped_text_export,
|
|
.merge_segments = _striped_merge_segments,
|
|
#ifdef DEVMAPPER_SUPPORT
|
|
.target_status_compatible = _striped_target_status_compatible,
|
|
.add_target_line = _striped_add_target_line,
|
|
.target_present = _striped_target_present,
|
|
#endif
|
|
.destroy = _striped_destroy,
|
|
};
|
|
|
|
static struct segment_type *_init_segtype(struct cmd_context *cmd, const char *name, uint64_t target)
|
|
{
|
|
struct segment_type *segtype = zalloc(sizeof(*segtype));
|
|
|
|
if (!segtype)
|
|
return_NULL;
|
|
|
|
segtype->ops = &_striped_ops;
|
|
segtype->name = name;
|
|
segtype->flags = target | SEG_CAN_SPLIT | SEG_AREAS_STRIPED;
|
|
|
|
log_very_verbose("Initialised segtype: %s", segtype->name);
|
|
return segtype;
|
|
}
|
|
|
|
struct segment_type *init_striped_segtype(struct cmd_context *cmd)
|
|
{
|
|
return _init_segtype(cmd, SEG_TYPE_NAME_STRIPED, SEG_STRIPED_TARGET);
|
|
}
|
|
|
|
|
|
struct segment_type *init_linear_segtype(struct cmd_context *cmd)
|
|
{
|
|
return _init_segtype(cmd, SEG_TYPE_NAME_LINEAR, SEG_LINEAR_TARGET);
|
|
}
|