mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
a6ca9ac0bc
Add lvm2app functions to manage LV tags. For lvm_lv_get_tags(), we return a list of tags, similar to other functions that return lists. An empty list is returned if there are no tags. NULL is returned if there is a problem obtaining the list of tags. Signed-off-by: Dave Wysochanski <dwysocha@redhat.com>
231 lines
5.2 KiB
C
231 lines
5.2 KiB
C
/*
|
|
* Copyright (C) 2008,2009 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "lib.h"
|
|
#include "lvm2app.h"
|
|
#include "metadata-exported.h"
|
|
#include "lvm-string.h"
|
|
#include "defaults.h"
|
|
#include "segtype.h"
|
|
#include "locking.h"
|
|
#include "activate.h"
|
|
#include "lvm_misc.h"
|
|
|
|
#include <string.h>
|
|
|
|
static int _lv_check_handle(const lv_t lv, const int vg_writeable)
|
|
{
|
|
if (!lv || !lv->vg || vg_read_error(lv->vg))
|
|
return -1;
|
|
if (vg_writeable && !vg_check_write_mode(lv->vg))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
/* FIXME: have lib/report/report.c _disp function call lv_size()? */
|
|
uint64_t lvm_lv_get_size(const lv_t lv)
|
|
{
|
|
return SECTOR_SIZE * lv_size(lv);
|
|
}
|
|
|
|
char *lvm_lv_get_uuid(const lv_t lv)
|
|
{
|
|
char uuid[64] __attribute((aligned(8)));
|
|
|
|
if (!id_write_format(&lv->lvid.id[1], uuid, sizeof(uuid))) {
|
|
log_error("Internal error converting uuid");
|
|
return NULL;
|
|
}
|
|
return strndup((const char *)uuid, 64);
|
|
}
|
|
|
|
char *lvm_lv_get_name(const lv_t lv)
|
|
{
|
|
char *name;
|
|
|
|
name = dm_malloc(NAME_LEN + 1);
|
|
strncpy(name, (const char *)lv->name, NAME_LEN);
|
|
name[NAME_LEN] = '\0';
|
|
return name;
|
|
}
|
|
|
|
uint64_t lvm_lv_is_active(const lv_t lv)
|
|
{
|
|
struct lvinfo info;
|
|
if (lv_info(lv->vg->cmd, lv, &info, 1, 0) &&
|
|
info.exists && info.live_table)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
uint64_t lvm_lv_is_suspended(const lv_t lv)
|
|
{
|
|
struct lvinfo info;
|
|
if (lv_info(lv->vg->cmd, lv, &info, 1, 0) &&
|
|
info.exists && info.suspended)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
int lvm_lv_add_tag(lv_t lv, const char *tag)
|
|
{
|
|
if (_lv_check_handle(lv, 1))
|
|
return -1;
|
|
if (!lv_change_tag(lv, tag, 1))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int lvm_lv_remove_tag(lv_t lv, const char *tag)
|
|
{
|
|
if (_lv_check_handle(lv, 1))
|
|
return -1;
|
|
if (!lv_change_tag(lv, tag, 0))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
struct dm_list *lvm_lv_get_tags(const lv_t lv)
|
|
{
|
|
return tag_list_copy(lv->vg->vgmem, &lv->tags);
|
|
}
|
|
|
|
/* Set defaults for non-segment specific LV parameters */
|
|
static void _lv_set_default_params(struct lvcreate_params *lp,
|
|
vg_t vg, const char *lvname,
|
|
uint64_t extents)
|
|
{
|
|
lp->zero = 1;
|
|
lp->major = -1;
|
|
lp->minor = -1;
|
|
lp->vg_name = vg->name;
|
|
lp->lv_name = lvname; /* FIXME: check this for safety */
|
|
lp->pvh = &vg->pvs;
|
|
|
|
lp->extents = extents;
|
|
lp->permission = LVM_READ | LVM_WRITE;
|
|
lp->read_ahead = DM_READ_AHEAD_NONE;
|
|
lp->alloc = ALLOC_INHERIT;
|
|
lp->tag = NULL;
|
|
}
|
|
|
|
/* Set default for linear segment specific LV parameters */
|
|
static void _lv_set_default_linear_params(struct cmd_context *cmd,
|
|
struct lvcreate_params *lp)
|
|
{
|
|
lp->segtype = get_segtype_from_string(cmd, "striped");
|
|
lp->stripes = 1;
|
|
lp->stripe_size = DEFAULT_STRIPESIZE * 2;
|
|
}
|
|
|
|
/*
|
|
* FIXME: This function should probably not commit to disk but require calling
|
|
* lvm_vg_write. However, this appears to be non-trivial change until
|
|
* lv_create_single is refactored by segtype.
|
|
*/
|
|
lv_t lvm_vg_create_lv_linear(vg_t vg, const char *name, uint64_t size)
|
|
{
|
|
struct lvcreate_params lp;
|
|
uint64_t extents;
|
|
struct lv_list *lvl;
|
|
|
|
if (vg_read_error(vg))
|
|
return NULL;
|
|
if (!vg_check_write_mode(vg))
|
|
return NULL;
|
|
memset(&lp, 0, sizeof(lp));
|
|
extents = extents_from_size(vg->cmd, size / SECTOR_SIZE,
|
|
vg->extent_size);
|
|
_lv_set_default_params(&lp, vg, name, extents);
|
|
_lv_set_default_linear_params(vg->cmd, &lp);
|
|
if (!lv_create_single(vg, &lp))
|
|
return NULL;
|
|
lvl = find_lv_in_vg(vg, name);
|
|
if (!lvl)
|
|
return NULL;
|
|
return (lv_t) lvl->lv;
|
|
}
|
|
|
|
/*
|
|
* FIXME: This function should probably not commit to disk but require calling
|
|
* lvm_vg_write.
|
|
*/
|
|
int lvm_vg_remove_lv(lv_t lv)
|
|
{
|
|
if (!lv || !lv->vg || vg_read_error(lv->vg))
|
|
return -1;
|
|
if (!vg_check_write_mode(lv->vg))
|
|
return -1;
|
|
if (!lv_remove_single(lv->vg->cmd, lv, DONT_PROMPT))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
int lvm_lv_activate(lv_t lv)
|
|
{
|
|
if (!lv || !lv->vg || vg_read_error(lv->vg) || !lv->vg->cmd)
|
|
return -1;
|
|
|
|
/* FIXME: handle pvmove stuff later */
|
|
if (lv->status & LOCKED) {
|
|
log_error("Unable to activate locked LV");
|
|
return -1;
|
|
}
|
|
|
|
/* FIXME: handle lvconvert stuff later */
|
|
if (lv->status & CONVERTING) {
|
|
log_error("Unable to activate LV with in-progress lvconvert");
|
|
return -1;
|
|
}
|
|
|
|
if (lv_is_origin(lv)) {
|
|
log_verbose("Activating logical volume \"%s\" "
|
|
"exclusively", lv->name);
|
|
if (!activate_lv_excl(lv->vg->cmd, lv)) {
|
|
log_error("Activate exclusive failed.");
|
|
return -1;
|
|
}
|
|
} else {
|
|
log_verbose("Activating logical volume \"%s\"",
|
|
lv->name);
|
|
if (!activate_lv(lv->vg->cmd, lv)) {
|
|
log_error("Activate failed.");
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int lvm_lv_deactivate(lv_t lv)
|
|
{
|
|
if (!lv || !lv->vg || vg_read_error(lv->vg) || !lv->vg->cmd)
|
|
return -1;
|
|
|
|
log_verbose("Deactivating logical volume \"%s\"", lv->name);
|
|
if (!deactivate_lv(lv->vg->cmd, lv)) {
|
|
log_error("Deactivate failed.");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int lvm_lv_resize(const lv_t lv, uint64_t new_size)
|
|
{
|
|
/* FIXME: add lv resize code here */
|
|
log_error("NOT IMPLEMENTED YET");
|
|
return -1;
|
|
}
|