mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-20 18:09:23 +03:00
vgreduce, vgremove, vgrename & vgscan
This commit is contained in:
parent
c4bf9638e3
commit
6e91eeef13
@ -91,6 +91,10 @@ static struct volume_group *_vg_read(struct io_space *is, const char *vg_name)
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* Strip prefix if present */
|
||||
if (!strncmp(vg_name, is->prefix, strlen(is->prefix)))
|
||||
vg_name += strlen(is->prefix);
|
||||
|
||||
if (!read_pvs_in_vg(vg_name, is->filter, mem, &pvs)) {
|
||||
stack;
|
||||
return NULL;
|
||||
@ -295,7 +299,8 @@ static struct list_head *_get_vgs(struct io_space *is)
|
||||
list_for_each(tmp, pvs) {
|
||||
struct pv_list *pvl = list_entry(tmp, struct pv_list, list);
|
||||
|
||||
if (_find_vg_name(names, pvl->pv.vg_name))
|
||||
if (!(*pvl->pv.vg_name) ||
|
||||
_find_vg_name(names, pvl->pv.vg_name))
|
||||
continue;
|
||||
|
||||
if (!(nl = pool_alloc(is->mem, sizeof(*nl)))) {
|
||||
@ -358,8 +363,9 @@ static int _pv_write(struct io_space *is, struct physical_volume *pv)
|
||||
|
||||
INIT_LIST_HEAD(&pvs);
|
||||
|
||||
if (pv->vg_name) {
|
||||
log_info("pv_write should only be called for an orphan pv");
|
||||
if (*pv->vg_name) {
|
||||
log_error("Assertion failed: can't _pv_write non-orphan PV "
|
||||
"(in VG %s)", pv->vg_name);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -25,8 +25,10 @@ SOURCES=\
|
||||
lvmchange.c\
|
||||
toollib.c\
|
||||
vgck.c\
|
||||
vgreduce.c\
|
||||
vgrename.c\
|
||||
vgremove.c
|
||||
vgremove.c\
|
||||
vgscan.c
|
||||
|
||||
TARGETS=\
|
||||
lvm
|
||||
|
@ -19,7 +19,7 @@
|
||||
*/
|
||||
|
||||
xx(available_ARG, 'a', "available", yes_no_arg)
|
||||
xx(all_ARG, 'a', "all", yes_no_arg)
|
||||
xx(all_ARG, 'a', "all", NULL)
|
||||
xx(autobackup_ARG, 'A', "autobackup", yes_no_arg)
|
||||
xx(activevolumegroups_ARG, 'A', "activevolumegroups", NULL)
|
||||
xx(blockdevice_ARG, 'b', "blockdevice", NULL)
|
||||
|
@ -45,7 +45,5 @@ int vgextend(int argc, char **argv) {return 1;}
|
||||
int vgimport(int argc, char **argv) {return 1;}
|
||||
int vgmerge(int argc, char **argv) {return 1;}
|
||||
int vgmknodes(int argc, char **argv) {return 1;}
|
||||
int vgreduce(int argc, char **argv) {return 1;}
|
||||
int vgscan(int argc, char **argv) {return 1;}
|
||||
int vgsplit(int argc, char **argv) {return 1;}
|
||||
|
||||
|
@ -120,13 +120,60 @@ int process_each_vg(int argc, char **argv,
|
||||
return ret_max;
|
||||
}
|
||||
|
||||
int process_each_pv(int argc, char **argv, struct volume_group *vg,
|
||||
int (*process_single) (struct volume_group * vg,
|
||||
struct physical_volume * pv))
|
||||
{
|
||||
int opt = 0;
|
||||
int ret_max = 0;
|
||||
int ret = 0;
|
||||
|
||||
struct list_head *pvh;
|
||||
|
||||
if (argc) {
|
||||
log_verbose("Using physical volume(s) on command line");
|
||||
for (; opt < argc; opt++) {
|
||||
if (!(pvh = find_pv_in_vg(vg, argv[opt]))) {
|
||||
log_error("Physical Volume %s not found in "
|
||||
"Volume Group %s", argv[opt],
|
||||
vg->name);
|
||||
continue;
|
||||
}
|
||||
ret = process_single(vg, &list_entry
|
||||
(pvh, struct pv_list, list)->pv);
|
||||
if (ret > ret_max)
|
||||
ret_max = ret;
|
||||
}
|
||||
} else {
|
||||
log_verbose("Using all physical volume(s) in volume group");
|
||||
list_for_each(pvh, &vg->pvs) {
|
||||
ret = process_single(vg, &list_entry
|
||||
(pvh, struct pv_list, list)->pv);
|
||||
if (ret > ret_max)
|
||||
ret_max = ret;
|
||||
}
|
||||
}
|
||||
|
||||
return ret_max;
|
||||
}
|
||||
|
||||
int is_valid_chars(char *n)
|
||||
{
|
||||
register char c;
|
||||
while ((c = *n++))
|
||||
if (!isalnum(c) && c != '.' && c != '_' && c != '-' &&
|
||||
c != '+')
|
||||
if (!isalnum(c) && c != '.' && c != '_' && c != '-' && c != '+')
|
||||
return 0;
|
||||
return 1;
|
||||
}
|
||||
|
||||
struct list_head *find_pv_in_vg(struct volume_group *vg, const char *pv_name)
|
||||
{
|
||||
struct list_head *pvh;
|
||||
list_for_each(pvh, &vg->pvs) {
|
||||
if (!strcmp(list_entry(pvh, struct pv_list, list)->pv.dev->name,
|
||||
pv_name)) return pvh;
|
||||
}
|
||||
|
||||
return NULL;
|
||||
|
||||
}
|
||||
|
@ -28,6 +28,12 @@ int do_autobackup(struct volume_group *vg);
|
||||
int process_each_vg(int argc, char **argv,
|
||||
int (*process_single) (const char *vg_name));
|
||||
|
||||
int process_each_pv(int argc, char **argv, struct volume_group *vg,
|
||||
int (*process_single) (struct volume_group * vg,
|
||||
struct physical_volume * pv));
|
||||
|
||||
int is_valid_chars(char *n);
|
||||
|
||||
struct list_head *find_pv_in_vg(struct volume_group *vg, const char *pv_name);
|
||||
|
||||
#endif
|
||||
|
132
tools/vgreduce.c
Normal file
132
tools/vgreduce.c
Normal file
@ -0,0 +1,132 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Sistina Software
|
||||
*
|
||||
* LVM is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* LVM is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with LVM; see the file COPYING. If not, write to
|
||||
* the Free Software Foundation, 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*
|
||||
*/
|
||||
|
||||
#include "tools.h"
|
||||
|
||||
static int vgreduce_single(struct volume_group *vg, struct physical_volume *pv);
|
||||
|
||||
int vgreduce(int argc, char **argv)
|
||||
{
|
||||
struct volume_group *vg;
|
||||
char *vg_name;
|
||||
|
||||
if (!argc) {
|
||||
log_error("Please give volume group name and "
|
||||
"physical volume paths");
|
||||
return EINVALID_CMD_LINE;
|
||||
}
|
||||
|
||||
if (argc == 1 && !arg_count(all_ARG)) {
|
||||
log_error("Please enter physical volume paths or option -a");
|
||||
return EINVALID_CMD_LINE;
|
||||
}
|
||||
|
||||
if (argc > 1 && arg_count(all_ARG)) {
|
||||
log_error("Option -a and physical volume paths mutually "
|
||||
"exclusive");
|
||||
return EINVALID_CMD_LINE;
|
||||
}
|
||||
|
||||
vg_name = argv[0];
|
||||
argv++;
|
||||
argc--;
|
||||
|
||||
log_verbose("Finding volume group %s", vg_name);
|
||||
if (!(vg = ios->vg_read(ios, vg_name))) {
|
||||
log_error("Volume group %s doesn't exist", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
if (!(vg->status & ACTIVE)) {
|
||||
log_error("Volume group %s is not active", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
if (!(vg->status & EXTENDABLE_VG)) {
|
||||
log_error("Volume group %s is not reducable", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
/* FIXME: Pass private structure through to all these functions */
|
||||
/* and update in batch here? */
|
||||
return process_each_pv(argc, argv, vg, vgreduce_single);
|
||||
|
||||
/******* FIXME
|
||||
log_error ("no empty physical volumes found in volume group \"%s\"", vg_name);
|
||||
|
||||
log_verbose
|
||||
("volume group \"%s\" will be reduced by %d physical volume%s",
|
||||
vg_name, np, np > 1 ? "s" : "");
|
||||
log_verbose ("reducing volume group \"%s\" by physical volume \"%s\"", vg_name, pv_names[p]);
|
||||
|
||||
log_print
|
||||
("volume group \"%s\" %ssuccessfully reduced by physical volume%s:",
|
||||
vg_name, error > 0 ? "NOT " : "", p > 1 ? "s" : "");
|
||||
log_print("%s", pv_this[p]->pv_name);
|
||||
********/
|
||||
|
||||
}
|
||||
|
||||
/* Or take pv_name instead? */
|
||||
static int vgreduce_single(struct volume_group *vg, struct physical_volume *pv)
|
||||
{
|
||||
struct list_head *pvh;
|
||||
|
||||
if (pv->pe_allocated) {
|
||||
log_error("Physical volume %s still in use", pv->dev->name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
/********* FIXME: Is this unnecessary after checking pe_allocated?
|
||||
if (pv->lv_cur > 0) {
|
||||
log_error ("can't reduce volume group \"%s\" by used physical volume \"%s\"", vg_name, error_pv_name);
|
||||
}
|
||||
*********/
|
||||
|
||||
if (vg->pv_count == 1) {
|
||||
log_error("Can't remove final physical volume %s from "
|
||||
"volume group %s", pv->dev->name, vg->name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
pvh = find_pv_in_vg(vg, pv->dev->name);
|
||||
|
||||
log_verbose("Removing %s from volume group %s", pv->dev->name,
|
||||
vg->name);
|
||||
list_del(pvh);
|
||||
*pv->vg_name = '\0';
|
||||
vg->pv_count--;
|
||||
|
||||
if (!(ios->vg_write(ios, vg))) {
|
||||
log_error("Removal of physical volume %s from %s failed",
|
||||
pv->dev->name, vg->name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
if (!ios->pv_write(ios, pv)) {
|
||||
log_error("Failed to clear metadata from physical volume %s "
|
||||
"after removal from %s", pv->dev->name, vg->name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
log_print("Removed %s from volume group %s", pv->dev->name, vg->name);
|
||||
|
||||
return 0;
|
||||
}
|
@ -32,13 +32,13 @@ static int vgremove_single(const char *vg_name)
|
||||
struct volume_group *vg;
|
||||
struct physical_volume *pv;
|
||||
struct list_head *pvh;
|
||||
int ret = 0;
|
||||
|
||||
log_verbose("Checking for volume group %s", vg_name);
|
||||
if (!(vg = ios->vg_read(ios, vg_name))) {
|
||||
log_error("Volume group %s doesn't exist", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
log_verbose("Checking for volume group %s", vg_name);
|
||||
if (!(vg = ios->vg_read(ios, vg_name))) {
|
||||
log_error("Volume group %s doesn't exist", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
if (vg->status & ACTIVE) {
|
||||
log_error("Volume group %s is still active", vg_name);
|
||||
@ -47,7 +47,7 @@ static int vgremove_single(const char *vg_name)
|
||||
|
||||
if (vg->lv_count) {
|
||||
log_error("Volume group %s still contains %d logical volume(s)",
|
||||
vg_name, vg->lv_count);
|
||||
vg_name, vg->lv_count);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
@ -58,19 +58,23 @@ static int vgremove_single(const char *vg_name)
|
||||
}
|
||||
*************/
|
||||
|
||||
|
||||
/* init physical volumes */
|
||||
list_for_each(pvh, &vg->pvs) {
|
||||
pv = &list_entry(pvh, struct pv_list, list)->pv;
|
||||
log_verbose("Removing physical volume %s from volume group %s",
|
||||
pv->dev->name, vg_name);
|
||||
pv->vg_name = '\0';
|
||||
if (!(ios->pv_write(ios, pv)))
|
||||
*pv->vg_name = '\0';
|
||||
if (!(ios->pv_write(ios, pv))) {
|
||||
log_error("Failed to remove physical volume %s from "
|
||||
"volume group %s", pv->dev->name, vg_name);
|
||||
"volume group %s", pv->dev->name, vg_name);
|
||||
ret = ECMD_FAILED;
|
||||
}
|
||||
}
|
||||
|
||||
log_print("Volume group %s successfully removed", vg_name);
|
||||
if (!ret)
|
||||
log_print("Volume group %s successfully removed", vg_name);
|
||||
else
|
||||
log_error("Volume group %s not properly removed", vg_name);
|
||||
|
||||
return 0;
|
||||
return ret;
|
||||
}
|
||||
|
@ -92,8 +92,8 @@ int vgrename(int argc, char **argv)
|
||||
|
||||
/* FIXME Should vg_write fix these implicitly? It has to check them. */
|
||||
list_for_each(pvh, &vg_old->pvs) {
|
||||
strcpy(list_entry(pvh, struct pv_list, list)->pv.vg_name,
|
||||
vg_name_new);
|
||||
strcpy(list_entry(pvh, struct pv_list, list)->pv.vg_name,
|
||||
vg_name_new);
|
||||
}
|
||||
|
||||
/********** FIXME: Check within vg_write now
|
||||
@ -147,4 +147,3 @@ char *lv_change_vgname(char *vg_name, char *lv_name)
|
||||
strncpy(lv_name_buf, lv_name, NAME_LEN - 1); return lv_name_buf;}
|
||||
|
||||
**********************/
|
||||
|
||||
|
66
tools/vgscan.c
Normal file
66
tools/vgscan.c
Normal file
@ -0,0 +1,66 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Sistina Software
|
||||
*
|
||||
* LVM is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* LVM is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with LVM; see the file COPYING. If not, write to
|
||||
* the Free Software Foundation, 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*
|
||||
*/
|
||||
|
||||
#include "tools.h"
|
||||
|
||||
static int vgscan_single(const char *vg_name);
|
||||
|
||||
int vgscan(int argc, char **argv)
|
||||
{
|
||||
if (argc) {
|
||||
log_error("Too many parameters on command line");
|
||||
return EINVALID_CMD_LINE;
|
||||
}
|
||||
|
||||
log_print("Reading all physical volumes (this may take a while...)");
|
||||
|
||||
return process_each_vg(argc, argv, &vgscan_single);
|
||||
}
|
||||
|
||||
static int vgscan_single(const char *vg_name)
|
||||
{
|
||||
struct volume_group *vg;
|
||||
|
||||
log_verbose("Checking for volume group %s", vg_name);
|
||||
if (!(vg = ios->vg_read(ios, vg_name))) {
|
||||
log_error("Volume group %s not found", vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
log_print("Found %sactive volume group %s",
|
||||
(vg->status & ACTIVE) ? "" : "in", vg_name);
|
||||
|
||||
/* FIXME: Now try to activate any inactive logical volumes */
|
||||
/* activate_lvs(vg) ? */
|
||||
|
||||
if (!(vg->status & ACTIVE)) {
|
||||
vg->status |= ACTIVE;
|
||||
if (!(ios->vg_write(ios, vg))) {
|
||||
log_error("Failed to activate volume group %s",
|
||||
vg_name);
|
||||
return ECMD_FAILED;
|
||||
}
|
||||
|
||||
/* FIXME: Create /dev/vg entries? */
|
||||
log_print("Volume Group %s activated", vg_name);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user