mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-04 09:18:36 +03:00
7c6526aae2
Start separating the validation from the action in the basic lvresize code moved to the library. Remove incorrect use of command line error codes from lvresize library functions. Move errors.h to tools directory to reinforce this, exporting public versions of the error codes in lvm2cmd.h for dmeventd plugins to use.
243 lines
5.1 KiB
C
243 lines
5.1 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <stddef.h>
|
|
#include "lib.h"
|
|
#include "metadata.h"
|
|
#include "lvm-string.h"
|
|
#include "lvm_misc.h"
|
|
#include "lvm2app.h"
|
|
#include "locking.h"
|
|
#include "toolcontext.h"
|
|
|
|
const char *lvm_pv_get_uuid(const pv_t pv)
|
|
{
|
|
return pv_uuid_dup(pv);
|
|
}
|
|
|
|
const char *lvm_pv_get_name(const pv_t pv)
|
|
{
|
|
return dm_pool_strndup(pv->vg->vgmem, pv_dev_name(pv), NAME_LEN);
|
|
}
|
|
|
|
uint64_t lvm_pv_get_mda_count(const pv_t pv)
|
|
{
|
|
return (uint64_t) pv_mda_count(pv);
|
|
}
|
|
|
|
uint64_t lvm_pv_get_dev_size(const pv_t pv)
|
|
{
|
|
return SECTOR_SIZE * pv_dev_size(pv);
|
|
}
|
|
|
|
uint64_t lvm_pv_get_size(const pv_t pv)
|
|
{
|
|
return SECTOR_SIZE * pv_size_field(pv);
|
|
}
|
|
|
|
uint64_t lvm_pv_get_free(const pv_t pv)
|
|
{
|
|
return SECTOR_SIZE * pv_free(pv);
|
|
}
|
|
|
|
struct lvm_property_value lvm_pv_get_property(const pv_t pv, const char *name)
|
|
{
|
|
return get_property(pv, NULL, NULL, NULL, NULL, NULL, name);
|
|
}
|
|
|
|
struct lvm_property_value lvm_pvseg_get_property(const pvseg_t pvseg,
|
|
const char *name)
|
|
{
|
|
return get_property(NULL, NULL, NULL, NULL, pvseg, NULL, name);
|
|
}
|
|
|
|
struct lvm_list_wrapper
|
|
{
|
|
unsigned long magic;
|
|
struct dm_list pvslist;
|
|
struct dm_list vgslist;
|
|
};
|
|
|
|
int lvm_pv_remove(lvm_t libh, const char *pv_name)
|
|
{
|
|
struct cmd_context *cmd = (struct cmd_context *)libh;
|
|
|
|
if (!pvremove_single(cmd, pv_name, NULL, 0, 0))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct dm_list *lvm_list_pvs(lvm_t libh)
|
|
{
|
|
struct lvm_list_wrapper *rc = NULL;
|
|
struct cmd_context *cmd = (struct cmd_context *)libh;
|
|
|
|
if (!(rc = dm_pool_zalloc(cmd->mem, sizeof(*rc)))) {
|
|
log_errno(ENOMEM, "Memory allocation fail for pv list.");
|
|
return NULL;
|
|
}
|
|
|
|
if (!lock_vol(cmd, VG_GLOBAL, LCK_VG_WRITE, NULL)) {
|
|
log_errno(ENOLCK, "Unable to obtain global lock.");
|
|
} else {
|
|
dm_list_init(&rc->pvslist);
|
|
dm_list_init(&rc->vgslist);
|
|
if( !get_pvs_perserve_vg(cmd, &rc->pvslist, &rc->vgslist) ) {
|
|
dm_pool_free(cmd->mem, rc);
|
|
return NULL;
|
|
}
|
|
rc->magic = 0xF005BA11;
|
|
}
|
|
|
|
return &rc->pvslist;
|
|
}
|
|
|
|
int lvm_list_pvs_free(struct dm_list *pvlist)
|
|
{
|
|
struct lvm_list_wrapper *to_delete;
|
|
struct vg_list *vgl;
|
|
struct pv_list *pvl;
|
|
struct cmd_context *cmd = NULL;
|
|
|
|
if (pvlist) {
|
|
to_delete = dm_list_struct_base(pvlist, struct lvm_list_wrapper, pvslist);
|
|
if (to_delete->magic != 0xF005BA11) {
|
|
log_errno(EINVAL, "Not a correct pvlist structure");
|
|
return -1;
|
|
}
|
|
|
|
dm_list_iterate_items(vgl, &to_delete->vgslist) {
|
|
cmd = vgl->vg->cmd;
|
|
release_vg(vgl->vg);
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &to_delete->pvslist)
|
|
free_pv_fid(pvl->pv);
|
|
|
|
unlock_vg(cmd, VG_GLOBAL);
|
|
|
|
to_delete->magic = 0xA5A5A5A5;
|
|
dm_pool_free(cmd->mem, to_delete);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct dm_list *lvm_pv_list_pvsegs(pv_t pv)
|
|
{
|
|
struct dm_list *list;
|
|
pvseg_list_t *pvseg;
|
|
struct pv_segment *pvl;
|
|
|
|
if (dm_list_empty(&pv->segments))
|
|
return NULL;
|
|
|
|
if (!(list = dm_pool_zalloc(pv->vg->vgmem, sizeof(*list)))) {
|
|
log_errno(ENOMEM, "Memory allocation fail for dm_list.");
|
|
return NULL;
|
|
}
|
|
|
|
dm_list_init(list);
|
|
|
|
dm_list_iterate_items(pvl, &pv->segments) {
|
|
if (!(pvseg = dm_pool_zalloc(pv->vg->vgmem, sizeof(*pvseg)))) {
|
|
log_errno(ENOMEM,
|
|
"Memory allocation fail for lvm_pvseg_list.");
|
|
return NULL;
|
|
}
|
|
pvseg->pvseg = pvl;
|
|
dm_list_add(list, &pvseg->list);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
pv_t lvm_pv_from_name(vg_t vg, const char *name)
|
|
{
|
|
struct pv_list *pvl;
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs)
|
|
if (!strcmp(name, pv_dev_name(pvl->pv)))
|
|
return pvl->pv;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
pv_t lvm_pv_from_uuid(vg_t vg, const char *uuid)
|
|
{
|
|
struct pv_list *pvl;
|
|
struct id id;
|
|
|
|
if (strlen(uuid) < ID_LEN) {
|
|
log_errno (EINVAL, "Invalid UUID string length");
|
|
return NULL;
|
|
}
|
|
|
|
if (!id_read_format(&id, uuid)) {
|
|
log_errno(EINVAL, "Invalid UUID format.");
|
|
return NULL;
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs)
|
|
if (id_equal(&id, &pvl->pv->id))
|
|
return pvl->pv;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int lvm_pv_resize(const pv_t pv, uint64_t new_size)
|
|
{
|
|
uint64_t size = new_size >> SECTOR_SHIFT;
|
|
|
|
if (new_size % SECTOR_SIZE) {
|
|
log_errno(EINVAL, "Size not a multiple of 512");
|
|
return -1;
|
|
}
|
|
|
|
if (!vg_check_write_mode(pv->vg))
|
|
return -1;
|
|
|
|
if (!pv_resize_single(pv->vg->cmd, pv->vg, pv, size)) {
|
|
log_error("PV re-size failed!");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int lvm_pv_create(lvm_t libh, const char *pv_name, uint64_t size)
|
|
{
|
|
struct pvcreate_params pp;
|
|
struct cmd_context *cmd = (struct cmd_context *)libh;
|
|
uint64_t size_sectors = size;
|
|
|
|
pvcreate_params_set_defaults(&pp);
|
|
|
|
if (size_sectors != 0) {
|
|
if (size_sectors % SECTOR_SIZE) {
|
|
log_errno(EINVAL, "Size not a multiple of 512");
|
|
return -1;
|
|
}
|
|
size_sectors = size_sectors >> SECTOR_SHIFT;
|
|
}
|
|
|
|
pp.size = size_sectors;
|
|
|
|
if (!pvcreate_locked(cmd, pv_name, &pp))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|