mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-21 13:34:40 +03:00
565 lines
12 KiB
C
565 lines
12 KiB
C
/*
|
|
* Copyright (C) 2008-2013 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 "lib/misc/lib.h"
|
|
#include "lib/commands/toolcontext.h"
|
|
#include "lib/metadata/metadata.h"
|
|
#include "lib/format_text/archiver.h"
|
|
#include "lib/locking/locking.h"
|
|
#include "lib/cache/lvmcache.h"
|
|
#include "lib/cache/lvmetad.h"
|
|
#include "lvm_misc.h"
|
|
#include "liblvm/lvm2app.h"
|
|
#include "lib/display/display.h"
|
|
#include "lib/cache/lvmetad.h"
|
|
|
|
int lvm_vg_add_tag(vg_t vg, const char *tag)
|
|
{
|
|
int rc = -1;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (!vg_read_error(vg) && vg_check_write_mode(vg) &&
|
|
vg_change_tag(vg, tag, 1))
|
|
rc = 0;
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
|
|
int lvm_vg_remove_tag(vg_t vg, const char *tag)
|
|
{
|
|
int rc = -1;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (!vg_read_error(vg) && vg_check_write_mode(vg) &&
|
|
vg_change_tag(vg, tag, 0))
|
|
rc = 0;
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
|
|
vg_t lvm_vg_create(lvm_t libh, const char *vg_name)
|
|
{
|
|
struct volume_group *vg = NULL;
|
|
struct saved_env e = store_user_env((struct cmd_context *)libh);
|
|
|
|
vg = vg_lock_and_create((struct cmd_context *)libh, vg_name);
|
|
/* FIXME: error handling is still TBD */
|
|
if (vg_read_error(vg)) {
|
|
release_vg(vg);
|
|
vg = NULL;
|
|
} else {
|
|
vg->open_mode = 'w';
|
|
}
|
|
|
|
restore_user_env(&e);
|
|
return (vg_t) vg;
|
|
}
|
|
|
|
static int _lvm_vg_extend(vg_t vg, const char *device)
|
|
{
|
|
struct pvcreate_params pp;
|
|
|
|
if (vg_read_error(vg))
|
|
return -1;
|
|
|
|
if (!vg_check_write_mode(vg))
|
|
return -1;
|
|
|
|
if (!lock_vol(vg->cmd, VG_ORPHANS, LCK_VG_WRITE, NULL)) {
|
|
log_error("Can't get lock for orphan PVs");
|
|
return -1;
|
|
}
|
|
|
|
pvcreate_params_set_defaults(&pp);
|
|
if (!vg_extend(vg, 1, &device, &pp)) {
|
|
unlock_vg(vg->cmd, NULL, VG_ORPHANS);
|
|
return -1;
|
|
}
|
|
/*
|
|
* FIXME: Either commit to disk, or keep holding VG_ORPHANS and
|
|
* release in lvm_vg_close().
|
|
*/
|
|
unlock_vg(vg->cmd, NULL, VG_ORPHANS);
|
|
return 0;
|
|
}
|
|
|
|
int lvm_vg_extend(vg_t vg, const char *device)
|
|
{
|
|
int rc = 0;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = _lvm_vg_extend(vg, device);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_vg_reduce(vg_t vg, const char *device)
|
|
{
|
|
int rc = -1;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (!vg_read_error(vg) && vg_check_write_mode(vg) && vg_reduce(vg, device))
|
|
rc = 0;
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_vg_set_extent_size(vg_t vg, uint32_t new_size)
|
|
{
|
|
int rc = -1;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (!vg_read_error(vg) && vg_check_write_mode(vg) &&
|
|
vg_set_extent_size(vg, new_size / SECTOR_SIZE))
|
|
rc = 0;
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
static int _lvm_vg_write(vg_t vg)
|
|
{
|
|
struct pv_list *pvl;
|
|
|
|
if (vg_read_error(vg))
|
|
return -1;
|
|
if (!vg_check_write_mode(vg))
|
|
return -1;
|
|
|
|
if (dm_list_empty(&vg->pvs)) {
|
|
if (!vg_remove(vg))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
if (! dm_list_empty(&vg->removed_pvs)) {
|
|
if (!lock_vol(vg->cmd, VG_ORPHANS, LCK_VG_WRITE, NULL)) {
|
|
log_error("Can't get lock for orphan PVs");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (!archive(vg))
|
|
return -1;
|
|
|
|
/* Store VG on disk(s) */
|
|
if (!vg_write(vg) || !vg_commit(vg))
|
|
return -1;
|
|
|
|
if (! dm_list_empty(&vg->removed_pvs)) {
|
|
dm_list_iterate_items(pvl, &vg->removed_pvs) {
|
|
pv_write_orphan(vg->cmd, pvl->pv);
|
|
pv_set_fid(pvl->pv, NULL);
|
|
/* FIXME: do pvremove / label_remove()? */
|
|
}
|
|
dm_list_init(&vg->removed_pvs);
|
|
unlock_vg(vg->cmd, NULL, VG_ORPHANS);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int lvm_vg_write(vg_t vg)
|
|
{
|
|
int rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = _lvm_vg_write(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_vg_close(vg_t vg)
|
|
{
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
if (vg_read_error(vg) == FAILED_LOCKING)
|
|
release_vg(vg);
|
|
else if (!lvmcache_vgname_is_locked(vg->name))
|
|
release_vg(vg);
|
|
else
|
|
unlock_and_release_vg(vg->cmd, vg, vg->name);
|
|
restore_user_env(&e);
|
|
return 0;
|
|
}
|
|
|
|
int lvm_vg_remove(vg_t vg)
|
|
{
|
|
int rc = -1;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (!vg_read_error(vg) && vg_check_write_mode(vg) && vg_remove_check(vg)) {
|
|
vg_remove_pvs(vg);
|
|
rc = 0;
|
|
}
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
static vg_t _lvm_vg_open(lvm_t libh, const char *vgname, const char *mode,
|
|
uint32_t flags)
|
|
{
|
|
uint32_t internal_flags = 0;
|
|
struct volume_group *vg;
|
|
|
|
if (!strncmp(mode, "w", 1))
|
|
internal_flags |= READ_FOR_UPDATE;
|
|
else if (strncmp(mode, "r", 1)) {
|
|
log_errno(EINVAL, "Invalid VG open mode");
|
|
return NULL;
|
|
}
|
|
|
|
lvmcache_label_scan((struct cmd_context *)libh);
|
|
|
|
vg = vg_read((struct cmd_context *)libh, vgname, NULL, internal_flags, 0);
|
|
if (vg_read_error(vg)) {
|
|
/* FIXME: use log_errno either here in inside vg_read */
|
|
release_vg(vg);
|
|
return NULL;
|
|
}
|
|
/* FIXME: combine this with locking ? */
|
|
vg->open_mode = mode[0];
|
|
|
|
return (vg_t) vg;
|
|
}
|
|
|
|
vg_t lvm_vg_open(lvm_t libh, const char *vgname, const char *mode,
|
|
uint32_t flags)
|
|
{
|
|
vg_t rc;
|
|
struct saved_env e = store_user_env((struct cmd_context*)libh);
|
|
rc = _lvm_vg_open(libh, vgname, mode, flags);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
static struct dm_list *_lvm_vg_list_pvs(vg_t vg)
|
|
{
|
|
struct dm_list *list;
|
|
pv_list_t *pvs;
|
|
struct pv_list *pvl;
|
|
|
|
if (dm_list_empty(&vg->pvs))
|
|
return NULL;
|
|
|
|
if (!(list = dm_pool_zalloc(vg->vgmem, sizeof(*list)))) {
|
|
log_errno(ENOMEM, "Memory allocation fail for dm_list.");
|
|
return NULL;
|
|
}
|
|
dm_list_init(list);
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (!(pvs = dm_pool_zalloc(vg->vgmem, sizeof(*pvs)))) {
|
|
log_errno(ENOMEM,
|
|
"Memory allocation fail for lvm_pv_list.");
|
|
return NULL;
|
|
}
|
|
pvs->pv = pvl->pv;
|
|
dm_list_add(list, &pvs->list);
|
|
}
|
|
return list;
|
|
}
|
|
|
|
struct dm_list *lvm_vg_list_pvs(vg_t vg)
|
|
{
|
|
struct dm_list *rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = _lvm_vg_list_pvs(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
static struct dm_list *_lvm_vg_list_lvs(vg_t vg)
|
|
{
|
|
struct dm_list *list;
|
|
lv_list_t *lvs;
|
|
struct lv_list *lvl;
|
|
|
|
if (dm_list_empty(&vg->lvs))
|
|
return NULL;
|
|
|
|
if (!(list = dm_pool_zalloc(vg->vgmem, sizeof(*list)))) {
|
|
log_errno(ENOMEM, "Memory allocation fail for dm_list.");
|
|
return NULL;
|
|
}
|
|
dm_list_init(list);
|
|
|
|
dm_list_iterate_items(lvl, &vg->lvs) {
|
|
if (!(lvs = dm_pool_zalloc(vg->vgmem, sizeof(*lvs)))) {
|
|
log_errno(ENOMEM,
|
|
"Memory allocation fail for lvm_lv_list.");
|
|
return NULL;
|
|
}
|
|
lvs->lv = lvl->lv;
|
|
dm_list_add(list, &lvs->list);
|
|
}
|
|
return list;
|
|
}
|
|
|
|
struct dm_list *lvm_vg_list_lvs(vg_t vg)
|
|
{
|
|
struct dm_list *rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = _lvm_vg_list_lvs(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
struct dm_list *lvm_vg_get_tags(const vg_t vg)
|
|
{
|
|
struct dm_list *rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = tag_list_copy(vg->vgmem, &vg->tags);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_seqno(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_seqno(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_is_clustered(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_is_clustered(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_is_exported(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_is_exported(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_is_partial(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = (vg_missing_pv_count(vg) != 0);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
/* FIXME: invalid handle? return INTMAX? */
|
|
uint64_t lvm_vg_get_size(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = SECTOR_SIZE * vg_size(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_free_size(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = SECTOR_SIZE * vg_free(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_extent_size(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = SECTOR_SIZE * vg_extent_size(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_extent_count(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_extent_count(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_free_extent_count(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_free_count(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_pv_count(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_pv_count(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_max_pv(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_max_pv(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
uint64_t lvm_vg_get_max_lv(const vg_t vg)
|
|
{
|
|
uint64_t rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_max_lv(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
const char *lvm_vg_get_uuid(const vg_t vg)
|
|
{
|
|
const char *rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = vg_uuid_dup(vg);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
const char *lvm_vg_get_name(const vg_t vg)
|
|
{
|
|
const char *rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = dm_pool_strndup(vg->vgmem, vg->name, NAME_LEN+1);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
|
|
struct lvm_property_value lvm_vg_get_property(const vg_t vg, const char *name)
|
|
{
|
|
struct lvm_property_value rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
rc = get_property(NULL, vg, NULL, NULL, NULL, NULL, NULL, name);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_vg_set_property(const vg_t vg, const char *name,
|
|
struct lvm_property_value *value)
|
|
{
|
|
/* At this point it is unknown if all property set paths make the
|
|
* appropriate copy of the string. We will allocate a copy on the vg so
|
|
* that worst case we have two copies which will get freed when the vg gets
|
|
* released.
|
|
*/
|
|
int rc;
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
if (value->is_valid && value->is_string && value->value.string) {
|
|
value->value.string = dm_pool_strndup(vg->vgmem, value->value.string,
|
|
strlen(value->value.string) + 1);
|
|
}
|
|
|
|
rc = set_property(NULL, vg, NULL, NULL, NULL, name, value);
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
struct dm_list *lvm_list_vg_names(lvm_t libh)
|
|
{
|
|
struct dm_list *rc = NULL;
|
|
struct saved_env e = store_user_env((struct cmd_context *)libh);
|
|
|
|
if (lvmetad_vg_list_to_lvmcache((struct cmd_context *)libh)) {
|
|
rc = get_vgnames((struct cmd_context *)libh, 0);
|
|
}
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
struct dm_list *lvm_list_vg_uuids(lvm_t libh)
|
|
{
|
|
struct dm_list *rc = NULL;
|
|
struct saved_env e = store_user_env((struct cmd_context *)libh);
|
|
|
|
if (lvmetad_vg_list_to_lvmcache((struct cmd_context *)libh)) {
|
|
rc = get_vgids((struct cmd_context *)libh, 0);
|
|
}
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* FIXME: Elaborate on when to use, side-effects, .cache file, etc
|
|
*/
|
|
int lvm_scan(lvm_t libh)
|
|
{
|
|
int rc = 0;
|
|
struct saved_env e = store_user_env((struct cmd_context *)libh);
|
|
|
|
if (!lvmcache_label_scan((struct cmd_context *)libh))
|
|
rc = -1;
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_lv_name_validate(const vg_t vg, const char *name)
|
|
{
|
|
int rc = -1;
|
|
name_error_t name_error;
|
|
int historical;
|
|
|
|
struct saved_env e = store_user_env(vg->cmd);
|
|
|
|
name_error = validate_name_detailed(name);
|
|
|
|
if (NAME_VALID == name_error) {
|
|
if (apply_lvname_restrictions(name)) {
|
|
if (!lv_name_is_used_in_vg(vg, name, &historical)) {
|
|
rc = 0;
|
|
} else {
|
|
log_errno(EINVAL, "%sLV name exists in VG",
|
|
historical ? "historical " : "");
|
|
}
|
|
}
|
|
} else {
|
|
display_name_error(name_error);
|
|
}
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|
|
|
|
int lvm_vg_name_validate(lvm_t libh, const char *name)
|
|
{
|
|
int rc = -1;
|
|
struct cmd_context *cmd = (struct cmd_context *)libh;
|
|
struct saved_env e = store_user_env(cmd);
|
|
|
|
if (validate_new_vg_name(cmd, name))
|
|
rc = 0;
|
|
|
|
restore_user_env(&e);
|
|
return rc;
|
|
}
|