mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-19 14:04:17 +03:00
lvconvert uses polldaemon now
This commit is contained in:
parent
b9c69aa63a
commit
ba0c495db7
@ -1,5 +1,6 @@
|
|||||||
Version 2.02.30 -
|
Version 2.02.30 -
|
||||||
===================================
|
===================================
|
||||||
|
Extend lvconvert to use polldaemon.
|
||||||
Add support for stacked mirrors.
|
Add support for stacked mirrors.
|
||||||
Major restructuring of pvmove and lvconvert layer manipulation code.
|
Major restructuring of pvmove and lvconvert layer manipulation code.
|
||||||
Replace tools/fsadm with scripts/fsadm.sh.
|
Replace tools/fsadm with scripts/fsadm.sh.
|
||||||
|
@ -1068,6 +1068,8 @@ static struct logical_volume *_set_up_mirror_log(struct cmd_context *cmd,
|
|||||||
int in_sync)
|
int in_sync)
|
||||||
{
|
{
|
||||||
struct logical_volume *log_lv;
|
struct logical_volume *log_lv;
|
||||||
|
const char *suffix;
|
||||||
|
struct lv_segment *seg;
|
||||||
|
|
||||||
init_mirror_in_sync(in_sync);
|
init_mirror_in_sync(in_sync);
|
||||||
|
|
||||||
@ -1076,9 +1078,15 @@ static struct logical_volume *_set_up_mirror_log(struct cmd_context *cmd,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(log_lv = _create_mirror_log(lv, ah, alloc, lv->name,
|
/* Check if the log is for temporary sync layer. */
|
||||||
strstr(lv->name, MIRROR_SYNC_LAYER)
|
seg = first_seg(lv);
|
||||||
? "_mlogtmp_%d" : "_mlog"))) {
|
if (seg_type(seg, 0) == AREA_LV &&
|
||||||
|
strstr(seg_lv(seg, 0)->name, MIRROR_SYNC_LAYER))
|
||||||
|
suffix = "_mlogtmp_%d";
|
||||||
|
else
|
||||||
|
suffix = "_mlog";
|
||||||
|
|
||||||
|
if (!(log_lv = _create_mirror_log(lv, ah, alloc, lv->name, suffix))) {
|
||||||
log_error("Failed to create mirror log.");
|
log_error("Failed to create mirror log.");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ lvconvert \- convert a logical volume from linear to mirror or snapshot
|
|||||||
.B lvconvert
|
.B lvconvert
|
||||||
\-m/\-\-mirrors Mirrors [\-\-mirrorlog {disk|core}] [\-\-corelog] [\-R/\-\-regionsize MirrorLogRegionSize]
|
\-m/\-\-mirrors Mirrors [\-\-mirrorlog {disk|core}] [\-\-corelog] [\-R/\-\-regionsize MirrorLogRegionSize]
|
||||||
[\-A/\-\-alloc AllocationPolicy]
|
[\-A/\-\-alloc AllocationPolicy]
|
||||||
|
[\-b/\-\-background] [\-i/\-\-interval Seconds]
|
||||||
[\-h/\-?/\-\-help]
|
[\-h/\-?/\-\-help]
|
||||||
[\-v/\-\-verbose]
|
[\-v/\-\-verbose]
|
||||||
[\-\-version]
|
[\-\-version]
|
||||||
@ -52,6 +53,12 @@ The optional argument "--corelog" is the same as specifying "--mirrorlog core".
|
|||||||
.I \-R, \-\-regionsize MirrorLogRegionSize
|
.I \-R, \-\-regionsize MirrorLogRegionSize
|
||||||
A mirror is divided into regions of this size (in MB), and the mirror log
|
A mirror is divided into regions of this size (in MB), and the mirror log
|
||||||
uses this granularity to track which regions are in sync.
|
uses this granularity to track which regions are in sync.
|
||||||
|
.TP
|
||||||
|
.I \-b, \-\-background
|
||||||
|
Run the daemon in the background.
|
||||||
|
.TP
|
||||||
|
.I \-i, \-\-interval Seconds
|
||||||
|
Report progress as a percentage at regular intervals.
|
||||||
.br
|
.br
|
||||||
.TP
|
.TP
|
||||||
.I \-s, \-\-snapshot
|
.I \-s, \-\-snapshot
|
||||||
|
@ -91,8 +91,10 @@ xx(lvconvert,
|
|||||||
"[-m|--mirrors Mirrors [{--mirrorlog {disk|core}|--corelog}]]\n"
|
"[-m|--mirrors Mirrors [{--mirrorlog {disk|core}|--corelog}]]\n"
|
||||||
"\t[-R|--regionsize MirrorLogRegionSize]\n"
|
"\t[-R|--regionsize MirrorLogRegionSize]\n"
|
||||||
"\t[--alloc AllocationPolicy]\n"
|
"\t[--alloc AllocationPolicy]\n"
|
||||||
|
"\t[-b|--background]\n"
|
||||||
"\t[-d|--debug]\n"
|
"\t[-d|--debug]\n"
|
||||||
"\t[-h|-?|--help]\n"
|
"\t[-h|-?|--help]\n"
|
||||||
|
"\t[-i|--interval seconds]\n"
|
||||||
"\t[-v|--verbose]\n"
|
"\t[-v|--verbose]\n"
|
||||||
"\t[--version]" "\n"
|
"\t[--version]" "\n"
|
||||||
"\tLogicalVolume[Path] [PhysicalVolume[Path]...]\n\n"
|
"\tLogicalVolume[Path] [PhysicalVolume[Path]...]\n\n"
|
||||||
@ -107,8 +109,8 @@ xx(lvconvert,
|
|||||||
"\t[--version]" "\n"
|
"\t[--version]" "\n"
|
||||||
"\tOriginalLogicalVolume[Path] SnapshotLogicalVolume[Path]\n",
|
"\tOriginalLogicalVolume[Path] SnapshotLogicalVolume[Path]\n",
|
||||||
|
|
||||||
alloc_ARG, chunksize_ARG, corelog_ARG, mirrorlog_ARG, mirrors_ARG,
|
alloc_ARG, background_ARG, chunksize_ARG, corelog_ARG, interval_ARG,
|
||||||
regionsize_ARG, snapshot_ARG, test_ARG, zero_ARG)
|
mirrorlog_ARG, mirrors_ARG, regionsize_ARG, snapshot_ARG, test_ARG, zero_ARG)
|
||||||
|
|
||||||
xx(lvcreate,
|
xx(lvcreate,
|
||||||
"Create a logical volume",
|
"Create a logical volume",
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "tools.h"
|
#include "tools.h"
|
||||||
|
#include "polldaemon.h"
|
||||||
#include "lv_alloc.h"
|
#include "lv_alloc.h"
|
||||||
|
|
||||||
struct lvconvert_params {
|
struct lvconvert_params {
|
||||||
@ -21,7 +22,9 @@ struct lvconvert_params {
|
|||||||
|
|
||||||
const char *origin;
|
const char *origin;
|
||||||
const char *lv_name;
|
const char *lv_name;
|
||||||
|
const char *lv_name_full;
|
||||||
const char *vg_name;
|
const char *vg_name;
|
||||||
|
int wait_daemon;
|
||||||
|
|
||||||
uint32_t chunk_size;
|
uint32_t chunk_size;
|
||||||
uint32_t region_size;
|
uint32_t region_size;
|
||||||
@ -70,11 +73,11 @@ static int _lvconvert_name_params(struct lvconvert_params *lp,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
lp->lv_name = (*pargv)[0];
|
lp->lv_name = lp->lv_name_full = (*pargv)[0];
|
||||||
(*pargv)++, (*pargc)--;
|
(*pargv)++, (*pargc)--;
|
||||||
|
|
||||||
if (strchr(lp->lv_name, '/') &&
|
if (strchr(lp->lv_name_full, '/') &&
|
||||||
(vg_name = extract_vgname(cmd, lp->lv_name)) &&
|
(vg_name = extract_vgname(cmd, lp->lv_name_full)) &&
|
||||||
lp->vg_name && strcmp(vg_name, lp->vg_name)) {
|
lp->vg_name && strcmp(vg_name, lp->vg_name)) {
|
||||||
log_error("Please use a single volume group name "
|
log_error("Please use a single volume group name "
|
||||||
"(\"%s\" or \"%s\")", vg_name, lp->vg_name);
|
"(\"%s\" or \"%s\")", vg_name, lp->vg_name);
|
||||||
@ -89,7 +92,7 @@ static int _lvconvert_name_params(struct lvconvert_params *lp,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((ptr = strrchr(lp->lv_name, '/')))
|
if ((ptr = strrchr(lp->lv_name_full, '/')))
|
||||||
lp->lv_name = ptr + 1;
|
lp->lv_name = ptr + 1;
|
||||||
|
|
||||||
if (!apply_lvname_restrictions(lp->lv_name))
|
if (!apply_lvname_restrictions(lp->lv_name))
|
||||||
@ -227,6 +230,90 @@ static int _read_params(struct lvconvert_params *lp, struct cmd_context *cmd,
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static struct volume_group *_get_lvconvert_vg(struct cmd_context *cmd,
|
||||||
|
const char *lv_name)
|
||||||
|
{
|
||||||
|
dev_close_all();
|
||||||
|
|
||||||
|
return vg_lock_and_read(cmd, extract_vgname(cmd, lv_name),
|
||||||
|
NULL, LCK_VG_WRITE,
|
||||||
|
CLUSTERED | EXPORTED_VG | LVM_WRITE,
|
||||||
|
CORRECT_INCONSISTENT | FAIL_INCONSISTENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct logical_volume *_get_lvconvert_lv(struct cmd_context *cmd __attribute((unused)),
|
||||||
|
struct volume_group *vg,
|
||||||
|
const char *name,
|
||||||
|
uint32_t lv_type __attribute((unused)))
|
||||||
|
{
|
||||||
|
return find_lv(vg, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _update_lvconvert_mirror(struct cmd_context *cmd __attribute((unused)),
|
||||||
|
struct volume_group *vg __attribute((unused)),
|
||||||
|
struct logical_volume *lv __attribute((unused)),
|
||||||
|
struct list *lvs_changed __attribute((unused)),
|
||||||
|
int first_time __attribute((unused)))
|
||||||
|
{
|
||||||
|
/* lvconvert mirror doesn't require periodical metadata update */
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _finish_lvconvert_mirror(struct cmd_context *cmd,
|
||||||
|
struct volume_group *vg,
|
||||||
|
struct logical_volume *lv,
|
||||||
|
struct list *lvs_changed __attribute((unused)))
|
||||||
|
{
|
||||||
|
if (!collapse_mirrored_lv(lv)) {
|
||||||
|
log_error("Failed to remove temporary sync layer.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
log_very_verbose("Updating logical volume \"%s\" on disk(s)", lv->name);
|
||||||
|
|
||||||
|
if (!vg_write(vg))
|
||||||
|
return_0;
|
||||||
|
|
||||||
|
backup(vg);
|
||||||
|
|
||||||
|
if (!suspend_lv(cmd, lv)) {
|
||||||
|
log_error("Failed to lock %s", lv->name);
|
||||||
|
vg_revert(vg);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!vg_commit(vg)) {
|
||||||
|
resume_lv(cmd, lv);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
log_very_verbose("Updating \"%s\" in kernel", lv->name);
|
||||||
|
|
||||||
|
if (!resume_lv(cmd, lv)) {
|
||||||
|
log_error("Problem reactivating %s", lv->name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
log_print("Logical volume %s converted.", lv->name);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct poll_functions _lvconvert_mirror_fns = {
|
||||||
|
.get_copy_vg = _get_lvconvert_vg,
|
||||||
|
.get_copy_lv = _get_lvconvert_lv,
|
||||||
|
.update_metadata = _update_lvconvert_mirror,
|
||||||
|
.finish_copy = _finish_lvconvert_mirror,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int _lvconvert_poll(struct cmd_context *cmd, const char *lv_name,
|
||||||
|
unsigned background)
|
||||||
|
{
|
||||||
|
return poll_daemon(cmd, lv_name, background, 0, &_lvconvert_mirror_fns,
|
||||||
|
"Converted");
|
||||||
|
}
|
||||||
|
|
||||||
static int _insert_lvconvert_layer(struct cmd_context *cmd,
|
static int _insert_lvconvert_layer(struct cmd_context *cmd,
|
||||||
struct logical_volume *lv)
|
struct logical_volume *lv)
|
||||||
{
|
{
|
||||||
@ -284,9 +371,8 @@ static int lvconvert_mirrors(struct cmd_context * cmd, struct logical_volume * l
|
|||||||
|
|
||||||
/* If called with no argument, try collapsing the resync layers */
|
/* If called with no argument, try collapsing the resync layers */
|
||||||
if (!arg_count(cmd, mirrors_ARG) && !arg_count(cmd, mirrorlog_ARG)) {
|
if (!arg_count(cmd, mirrors_ARG) && !arg_count(cmd, mirrorlog_ARG)) {
|
||||||
if (!collapse_mirrored_lv(lv))
|
lp->wait_daemon = 1;
|
||||||
return_0;
|
return 1;
|
||||||
goto commit_changes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -439,6 +525,7 @@ static int lvconvert_mirrors(struct cmd_context * cmd, struct logical_volume * l
|
|||||||
corelog ? 0U : 1U, lp->pvh, lp->alloc,
|
corelog ? 0U : 1U, lp->pvh, lp->alloc,
|
||||||
MIRROR_BY_LV))
|
MIRROR_BY_LV))
|
||||||
return_0;
|
return_0;
|
||||||
|
lp->wait_daemon = 1;
|
||||||
} else {
|
} else {
|
||||||
/* Reduce number of mirrors */
|
/* Reduce number of mirrors */
|
||||||
if (!lv_remove_mirrors(cmd, lv, existing_mirrors - lp->mirrors,
|
if (!lv_remove_mirrors(cmd, lv, existing_mirrors - lp->mirrors,
|
||||||
@ -472,6 +559,7 @@ commit_changes:
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!lp->wait_daemon)
|
||||||
log_print("Logical volume %s converted.", lv->name);
|
log_print("Logical volume %s converted.", lv->name);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
@ -625,5 +713,9 @@ int lvconvert(struct cmd_context * cmd, int argc, char **argv)
|
|||||||
|
|
||||||
error:
|
error:
|
||||||
unlock_vg(cmd, lp.vg_name);
|
unlock_vg(cmd, lp.vg_name);
|
||||||
|
|
||||||
|
if (lp.wait_daemon)
|
||||||
|
ret = _lvconvert_poll(cmd, lp.lv_name_full,
|
||||||
|
arg_count(cmd, background_ARG) ? 1U : 0);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -93,9 +93,11 @@ static int _check_mirror_status(struct cmd_context *cmd,
|
|||||||
|
|
||||||
overall_percent = copy_percent(lv_mirr);
|
overall_percent = copy_percent(lv_mirr);
|
||||||
if (parms->progress_display)
|
if (parms->progress_display)
|
||||||
log_print("%s: Moved: %.1f%%", name, overall_percent);
|
log_print("%s: %s: %.1f%%", name, parms->progress_title,
|
||||||
|
overall_percent);
|
||||||
else
|
else
|
||||||
log_verbose("%s: Moved: %.1f%%", name, overall_percent);
|
log_verbose("%s: %s: %.1f%%", name, parms->progress_title,
|
||||||
|
overall_percent);
|
||||||
|
|
||||||
if (segment_percent < 100.0) {
|
if (segment_percent < 100.0) {
|
||||||
/* The only case the caller *should* try again later */
|
/* The only case the caller *should* try again later */
|
||||||
@ -224,7 +226,8 @@ static void _poll_for_all_vgs(struct cmd_context *cmd,
|
|||||||
}
|
}
|
||||||
|
|
||||||
int poll_daemon(struct cmd_context *cmd, const char *name, unsigned background,
|
int poll_daemon(struct cmd_context *cmd, const char *name, unsigned background,
|
||||||
uint32_t lv_type, struct poll_functions *poll_fns)
|
uint32_t lv_type, struct poll_functions *poll_fns,
|
||||||
|
const char *progress_title)
|
||||||
{
|
{
|
||||||
struct daemon_parms parms;
|
struct daemon_parms parms;
|
||||||
|
|
||||||
@ -232,6 +235,7 @@ int poll_daemon(struct cmd_context *cmd, const char *name, unsigned background,
|
|||||||
parms.background = background;
|
parms.background = background;
|
||||||
parms.interval = arg_uint_value(cmd, interval_ARG, DEFAULT_INTERVAL);
|
parms.interval = arg_uint_value(cmd, interval_ARG, DEFAULT_INTERVAL);
|
||||||
parms.progress_display = 1;
|
parms.progress_display = 1;
|
||||||
|
parms.progress_title = progress_title;
|
||||||
parms.lv_type = lv_type;
|
parms.lv_type = lv_type;
|
||||||
parms.poll_fns = poll_fns;
|
parms.poll_fns = poll_fns;
|
||||||
|
|
||||||
|
@ -42,11 +42,13 @@ struct daemon_parms {
|
|||||||
unsigned background;
|
unsigned background;
|
||||||
unsigned outstanding_count;
|
unsigned outstanding_count;
|
||||||
unsigned progress_display;
|
unsigned progress_display;
|
||||||
|
const char *progress_title;
|
||||||
uint32_t lv_type;
|
uint32_t lv_type;
|
||||||
struct poll_functions *poll_fns;
|
struct poll_functions *poll_fns;
|
||||||
};
|
};
|
||||||
|
|
||||||
int poll_daemon(struct cmd_context *cmd, const char *name, unsigned background,
|
int poll_daemon(struct cmd_context *cmd, const char *name, unsigned background,
|
||||||
uint32_t lv_type, struct poll_functions *poll_fns);
|
uint32_t lv_type, struct poll_functions *poll_fns,
|
||||||
|
const char *progress_title);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -525,7 +525,8 @@ static struct poll_functions _pvmove_fns = {
|
|||||||
int pvmove_poll(struct cmd_context *cmd, const char *pv_name,
|
int pvmove_poll(struct cmd_context *cmd, const char *pv_name,
|
||||||
unsigned background)
|
unsigned background)
|
||||||
{
|
{
|
||||||
return poll_daemon(cmd, pv_name, background, PVMOVE, &_pvmove_fns);
|
return poll_daemon(cmd, pv_name, background, PVMOVE, &_pvmove_fns,
|
||||||
|
"Moved");
|
||||||
}
|
}
|
||||||
|
|
||||||
int pvmove(struct cmd_context *cmd, int argc, char **argv)
|
int pvmove(struct cmd_context *cmd, int argc, char **argv)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user