1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-05 13:18:20 +03:00
lvm2/lib/format1/format1.c

576 lines
13 KiB
C
Raw Normal View History

2001-10-04 21:48:55 +04:00
/*
2004-03-30 23:35:44 +04:00
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
* Copyright (C) 2004 Red Hat, Inc. All rights reserved.
2001-10-04 21:48:55 +04:00
*
2004-03-30 23:35:44 +04:00
* 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 General Public License v.2.
*
* You should have received a copy of the GNU 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
2001-10-04 21:48:55 +04:00
*/
2002-11-18 17:01:16 +03:00
#include "lib.h"
2001-10-04 21:48:55 +04:00
#include "disk-rep.h"
#include "limits.h"
2001-10-16 00:29:15 +04:00
#include "display.h"
#include "toolcontext.h"
#include "lvmcache.h"
2002-11-18 17:01:16 +03:00
#include "lvm1-label.h"
#include "format1.h"
2004-09-16 22:40:56 +04:00
#include "segtype.h"
2001-10-04 21:48:55 +04:00
2003-08-20 19:48:27 +04:00
#define FMT_LVM1_NAME "lvm1"
2001-10-16 20:25:28 +04:00
/* VG consistency checks */
static int _check_vgs(struct list *pvs, int *partial)
2001-10-04 21:48:55 +04:00
{
struct list *pvh, *t;
struct disk_list *dl = NULL;
2001-10-16 20:25:28 +04:00
struct disk_list *first = NULL;
uint32_t pv_count = 0;
uint32_t exported = 0;
int first_time = 1;
*partial = 0;
2001-10-04 21:48:55 +04:00
/*
* If there are exported and unexported PVs, ignore exported ones.
* This means an active VG won't be affected if disks are inserted
* bearing an exported VG with the same name.
*/
2005-06-01 20:51:55 +04:00
list_iterate_items(dl, pvs) {
if (first_time) {
exported = dl->pvd.pv_status & VG_EXPORTED;
first_time = 0;
continue;
}
if (exported != (dl->pvd.pv_status & VG_EXPORTED)) {
/* Remove exported PVs */
list_iterate_safe(pvh, t, pvs) {
dl = list_item(pvh, struct disk_list);
if (dl->pvd.pv_status & VG_EXPORTED)
list_del(pvh);
}
break;
}
}
/* Remove any PVs with VG structs that differ from the first */
list_iterate_safe(pvh, t, pvs) {
dl = list_item(pvh, struct disk_list);
2001-10-09 18:26:45 +04:00
if (!first)
2001-10-16 20:25:28 +04:00
first = dl;
else if (memcmp(&first->vgd, &dl->vgd, sizeof(first->vgd))) {
2002-01-28 00:30:47 +03:00
log_error("VG data differs between PVs %s and %s",
dev_name(first->dev), dev_name(dl->dev));
log_debug("VG data on %s: %s %s %" PRIu32 " %" PRIu32
" %" PRIu32 " %" PRIu32 " %" PRIu32 " %"
PRIu32 " %" PRIu32 " %" PRIu32 " %" PRIu32
" %" PRIu32 " %" PRIu32 " %" PRIu32 " %"
PRIu32 " %" PRIu32 " %" PRIu32,
dev_name(first->dev), first->vgd.vg_uuid,
first->vgd.vg_name_dummy,
first->vgd.vg_number, first->vgd.vg_access,
first->vgd.vg_status, first->vgd.lv_max,
first->vgd.lv_cur, first->vgd.lv_open,
first->vgd.pv_max, first->vgd.pv_cur,
first->vgd.pv_act, first->vgd.dummy,
first->vgd.vgda, first->vgd.pe_size,
first->vgd.pe_total, first->vgd.pe_allocated,
first->vgd.pvg_total);
log_debug("VG data on %s: %s %s %" PRIu32 " %" PRIu32
" %" PRIu32 " %" PRIu32 " %" PRIu32 " %"
PRIu32 " %" PRIu32 " %" PRIu32 " %" PRIu32
" %" PRIu32 " %" PRIu32 " %" PRIu32 " %"
PRIu32 " %" PRIu32 " %" PRIu32,
dev_name(dl->dev), dl->vgd.vg_uuid,
dl->vgd.vg_name_dummy, dl->vgd.vg_number,
dl->vgd.vg_access, dl->vgd.vg_status,
dl->vgd.lv_max, dl->vgd.lv_cur,
dl->vgd.lv_open, dl->vgd.pv_max,
dl->vgd.pv_cur, dl->vgd.pv_act, dl->vgd.dummy,
dl->vgd.vgda, dl->vgd.pe_size,
dl->vgd.pe_total, dl->vgd.pe_allocated,
dl->vgd.pvg_total);
list_del(pvh);
if (partial_mode()) {
*partial = 1;
continue;
}
2001-10-04 21:48:55 +04:00
return 0;
}
2001-10-16 20:25:28 +04:00
pv_count++;
}
/* On entry to fn, list known to be non-empty */
2003-08-20 19:48:27 +04:00
if (pv_count != first->vgd.pv_cur) {
log_error("%d PV(s) found for VG %s: expected %d",
2003-08-20 19:48:27 +04:00
pv_count, first->pvd.vg_name, first->vgd.pv_cur);
if (!partial_mode())
return 0;
*partial = 1;
2001-10-04 21:48:55 +04:00
}
return 1;
}
static struct volume_group *_build_vg(struct format_instance *fid,
struct list *pvs)
2001-10-04 21:48:55 +04:00
{
struct dm_pool *mem = fid->fmt->cmd->mem;
struct volume_group *vg = dm_pool_alloc(mem, sizeof(*vg));
2001-10-09 21:09:46 +04:00
struct disk_list *dl;
int partial;
2001-10-09 21:09:46 +04:00
2001-10-15 22:39:40 +04:00
if (!vg)
goto_bad;
2001-10-04 21:48:55 +04:00
2001-10-15 22:39:40 +04:00
if (list_empty(pvs))
goto_bad;
2001-10-09 21:09:46 +04:00
2001-10-04 21:48:55 +04:00
memset(vg, 0, sizeof(*vg));
vg->cmd = fid->fmt->cmd;
vg->fid = fid;
vg->seqno = 0;
2001-10-31 15:47:01 +03:00
list_init(&vg->pvs);
list_init(&vg->lvs);
2004-03-08 20:19:15 +03:00
list_init(&vg->tags);
2001-10-08 20:08:16 +04:00
if (!_check_vgs(pvs, &partial))
goto_bad;
2001-10-09 18:26:45 +04:00
dl = list_item(pvs->n, struct disk_list);
if (!import_vg(mem, vg, dl, partial))
goto_bad;
2001-10-04 21:48:55 +04:00
2002-11-18 17:01:16 +03:00
if (!import_pvs(fid->fmt, mem, vg, pvs, &vg->pvs, &vg->pv_count))
goto_bad;
2001-10-04 21:48:55 +04:00
2001-10-09 18:26:45 +04:00
if (!import_lvs(mem, vg, pvs))
goto_bad;
2001-10-04 21:48:55 +04:00
2004-05-05 01:25:57 +04:00
if (!import_extents(fid->fmt->cmd, vg, pvs))
goto_bad;
2001-10-08 13:45:16 +04:00
if (!import_snapshots(mem, vg, pvs))
goto_bad;
2001-10-04 21:48:55 +04:00
return vg;
bad:
2001-10-04 21:48:55 +04:00
stack;
dm_pool_free(mem, vg);
2001-10-04 21:48:55 +04:00
return NULL;
}
static struct volume_group *_format1_vg_read(struct format_instance *fid,
2002-11-18 17:01:16 +03:00
const char *vg_name,
struct metadata_area *mda)
2001-10-04 21:48:55 +04:00
{
struct dm_pool *mem = dm_pool_create("lvm1 vg_read", 1024 * 10);
2001-10-31 15:47:01 +03:00
struct list pvs;
2001-11-12 18:10:01 +03:00
struct volume_group *vg = NULL;
2001-10-31 15:47:01 +03:00
list_init(&pvs);
2001-10-04 21:48:55 +04:00
if (!mem) {
stack;
return NULL;
}
/* Strip dev_dir if present */
vg_name = strip_dir(vg_name, fid->fmt->cmd->dev_dir);
2001-10-12 01:35:55 +04:00
if (!read_pvs_in_vg
(fid->fmt, vg_name, fid->fmt->cmd->filter, mem, &pvs))
goto_bad;
2001-10-04 21:48:55 +04:00
if (!(vg = _build_vg(fid, &pvs)))
goto_bad;
2001-11-12 18:10:01 +03:00
bad:
dm_pool_destroy(mem);
2001-10-04 21:48:55 +04:00
return vg;
}
2004-03-08 20:25:59 +03:00
static struct disk_list *_flatten_pv(struct format_instance *fid,
struct dm_pool *mem, struct volume_group *vg,
2001-10-09 18:26:45 +04:00
struct physical_volume *pv,
2001-11-14 16:52:38 +03:00
const char *dev_dir)
2001-10-05 20:36:53 +04:00
{
struct disk_list *dl = dm_pool_alloc(mem, sizeof(*dl));
2001-10-08 16:11:33 +04:00
2001-10-09 14:47:52 +04:00
if (!dl) {
stack;
return NULL;
}
dl->mem = mem;
dl->dev = pv->dev;
2001-10-31 15:47:01 +03:00
list_init(&dl->uuids);
list_init(&dl->lvds);
2001-10-09 14:47:52 +04:00
2004-03-08 20:25:59 +03:00
if (!export_pv(fid->fmt->cmd, mem, vg, &dl->pvd, pv) ||
!export_vg(&dl->vgd, vg) ||
2001-10-09 18:26:45 +04:00
!export_uuids(dl, vg) ||
2002-11-18 17:01:16 +03:00
!export_lvs(dl, vg, pv, dev_dir) || !calculate_layout(dl)) {
2001-10-09 14:47:52 +04:00
stack;
dm_pool_free(mem, dl);
2001-10-09 14:47:52 +04:00
return NULL;
}
return dl;
2001-10-05 20:36:53 +04:00
}
static int _flatten_vg(struct format_instance *fid, struct dm_pool *mem,
struct volume_group *vg,
2001-11-14 16:52:38 +03:00
struct list *pvds, const char *dev_dir,
struct dev_filter *filter)
2001-10-05 20:36:53 +04:00
{
2001-10-09 14:47:52 +04:00
struct pv_list *pvl;
2001-10-05 20:36:53 +04:00
struct disk_list *data;
2005-06-01 20:51:55 +04:00
list_iterate_items(pvl, &vg->pvs) {
2004-03-08 20:25:59 +03:00
if (!(data = _flatten_pv(fid, mem, vg, pvl->pv, dev_dir))) {
2001-10-05 20:36:53 +04:00
stack;
return 0;
}
2001-11-14 16:52:38 +03:00
list_add(pvds, &data->list);
2001-10-05 20:36:53 +04:00
}
2001-11-14 16:52:38 +03:00
export_numbers(pvds, vg);
export_pv_act(pvds);
2001-10-11 17:05:55 +04:00
if (!export_vg_number(fid, pvds, vg->name, filter)) {
stack;
return 0;
}
2001-10-05 20:36:53 +04:00
return 1;
}
static int _format1_vg_write(struct format_instance *fid, struct volume_group *vg,
2002-11-18 17:01:16 +03:00
struct metadata_area *mda)
2001-10-05 20:36:53 +04:00
{
struct dm_pool *mem = dm_pool_create("lvm1 vg_write", 1024 * 10);
2001-11-14 16:52:38 +03:00
struct list pvds;
2001-10-05 20:36:53 +04:00
int r = 0;
if (!mem) {
stack;
return 0;
}
2001-11-14 16:52:38 +03:00
list_init(&pvds);
2001-10-10 17:09:40 +04:00
r = (_flatten_vg(fid, mem, vg, &pvds, fid->fmt->cmd->dev_dir,
fid->fmt->cmd->filter) &&
write_disks(fid->fmt, &pvds));
2002-11-18 17:01:16 +03:00
lvmcache_update_vg(vg);
dm_pool_destroy(mem);
2001-10-05 20:36:53 +04:00
return r;
}
2001-10-04 21:48:55 +04:00
static int _format1_pv_read(const struct format_type *fmt, const char *pv_name,
struct physical_volume *pv, struct list *mdas)
2001-10-09 12:58:52 +04:00
{
struct dm_pool *mem = dm_pool_create("lvm1 pv_read", 1024);
2001-10-09 12:58:52 +04:00
struct disk_list *dl;
struct device *dev;
int r = 0;
2001-10-09 12:58:52 +04:00
2002-11-18 17:01:16 +03:00
log_very_verbose("Reading physical volume data %s from disk", pv_name);
2001-10-17 19:29:31 +04:00
2001-10-09 12:58:52 +04:00
if (!mem) {
stack;
return 0;
2001-10-09 12:58:52 +04:00
}
2002-11-18 17:01:16 +03:00
if (!(dev = dev_cache_get(pv_name, fmt->cmd->filter))) {
stack;
goto out;
}
if (!(dl = read_disk(fmt, dev, mem, NULL))) {
2001-10-09 12:58:52 +04:00
stack;
goto out;
2001-10-09 12:58:52 +04:00
}
if (!import_pv(fmt, fmt->cmd->mem, dl->dev, NULL, pv, &dl->pvd, &dl->vgd)) {
2001-10-09 12:58:52 +04:00
stack;
goto out;
2001-10-09 12:58:52 +04:00
}
2002-11-18 17:01:16 +03:00
pv->fmt = fmt;
r = 1;
2001-10-09 12:58:52 +04:00
out:
dm_pool_destroy(mem);
return r;
2001-10-09 12:58:52 +04:00
}
static int _format1_pv_setup(const struct format_type *fmt,
2002-11-18 17:01:16 +03:00
uint64_t pe_start, uint32_t extent_count,
uint32_t extent_size,
int pvmetadatacopies,
uint64_t pvmetadatasize, struct list *mdas,
struct physical_volume *pv, struct volume_group *vg)
{
if (pv->size > MAX_PV_SIZE)
pv->size--;
2002-02-15 04:26:16 +03:00
if (pv->size > MAX_PV_SIZE) {
2002-11-18 17:01:16 +03:00
log_error("Physical volumes cannot be bigger than %s",
display_size(fmt->cmd, (uint64_t) MAX_PV_SIZE));
2002-02-15 04:26:16 +03:00
return 0;
}
2002-11-18 17:01:16 +03:00
/* Nothing more to do if extent size isn't provided */
if (!extent_size)
return 1;
2002-02-15 17:33:59 +03:00
/*
* This works out pe_start and pe_count.
*/
if (!calculate_extent_count(pv, extent_size, extent_count, pe_start)) {
stack;
return 0;
}
2002-11-18 17:01:16 +03:00
/* Retain existing extent locations exactly */
if (((pe_start || extent_count) && (pe_start != pv->pe_start)) ||
(extent_count && (extent_count != pv->pe_count))) {
log_error("Metadata would overwrite physical extents");
return 0;
}
return 1;
}
static int _format1_lv_setup(struct format_instance *fid, struct logical_volume *lv)
{
uint64_t max_size = UINT_MAX;
if (!*lv->lvid.s)
lvid_from_lvnum(&lv->lvid, &lv->vg->id, find_free_lvnum(lv));
if (lv->le_count > MAX_LE_TOTAL) {
2002-01-28 00:30:47 +03:00
log_error("logical volumes cannot contain more than "
"%d extents.", MAX_LE_TOTAL);
return 0;
}
if (lv->size > max_size) {
log_error("logical volumes cannot be larger than %s",
display_size(fid->fmt->cmd, max_size));
return 0;
}
return 1;
}
static int _format1_pv_write(const struct format_type *fmt, struct physical_volume *pv,
2002-11-18 17:01:16 +03:00
struct list *mdas, int64_t sector)
2001-10-09 21:44:58 +04:00
{
struct dm_pool *mem;
2001-10-10 14:05:29 +04:00
struct disk_list *dl;
2001-10-31 15:47:01 +03:00
struct list pvs;
2002-11-18 17:01:16 +03:00
struct label *label;
struct lvmcache_info *info;
2001-10-10 14:05:29 +04:00
if (!(info = lvmcache_add(fmt->labeller, (char *) &pv->id, pv->dev,
pv->vg_name, NULL, 0))) {
2002-11-18 17:01:16 +03:00
stack;
2001-10-10 14:05:29 +04:00
return 0;
}
2002-11-18 17:01:16 +03:00
label = info->label;
info->device_size = pv->size << SECTOR_SHIFT;
info->fmt = fmt;
list_init(&info->mdas);
list_init(&pvs);
2001-10-10 14:05:29 +04:00
2001-10-18 20:55:19 +04:00
/* Ensure any residual PE structure is gone */
pv->pe_size = pv->pe_count = 0;
2006-08-17 23:15:27 +04:00
pv->pe_start = LVM1_PE_ALIGN;
2001-10-18 20:55:19 +04:00
if (!(mem = dm_pool_create("lvm1 pv_write", 1024))) {
2001-10-10 14:05:29 +04:00
stack;
return 0;
}
if (!(dl = dm_pool_alloc(mem, sizeof(*dl))))
goto_bad;
2001-10-10 14:05:29 +04:00
dl->mem = mem;
dl->dev = pv->dev;
if (!export_pv(fmt->cmd, mem, NULL, &dl->pvd, pv))
goto_bad;
2001-10-10 14:05:29 +04:00
/* must be set to be able to zero gap after PV structure in
dev_write in order to make other disk tools happy */
dl->pvd.pv_on_disk.base = METADATA_BASE;
dl->pvd.pv_on_disk.size = PV_SIZE;
2006-08-17 23:15:27 +04:00
dl->pvd.pe_on_disk.base = LVM1_PE_ALIGN << SECTOR_SHIFT;
2001-10-31 15:47:01 +03:00
list_add(&pvs, &dl->list);
if (!write_disks(fmt, &pvs))
goto_bad;
2001-10-10 14:05:29 +04:00
dm_pool_destroy(mem);
2001-10-09 21:44:58 +04:00
return 1;
2001-10-10 14:05:29 +04:00
bad:
dm_pool_destroy(mem);
2001-10-10 14:05:29 +04:00
return 0;
2001-10-09 21:44:58 +04:00
}
static int _format1_vg_setup(struct format_instance *fid, struct volume_group *vg)
2001-10-12 18:25:53 +04:00
{
/* just check max_pv and max_lv */
if (!vg->max_lv || vg->max_lv >= MAX_LV)
2001-10-15 22:39:40 +04:00
vg->max_lv = MAX_LV - 1;
2001-10-12 18:25:53 +04:00
if (!vg->max_pv || vg->max_pv >= MAX_PV)
2001-10-15 22:39:40 +04:00
vg->max_pv = MAX_PV - 1;
2001-10-12 18:25:53 +04:00
2001-10-16 00:29:15 +04:00
if (vg->extent_size > MAX_PE_SIZE || vg->extent_size < MIN_PE_SIZE) {
log_error("Extent size must be between %s and %s",
display_size(fid->fmt->cmd, (uint64_t) MIN_PE_SIZE),
display_size(fid->fmt->cmd, (uint64_t) MAX_PE_SIZE));
2001-10-16 00:29:15 +04:00
return 0;
}
if (vg->extent_size % MIN_PE_SIZE) {
log_error("Extent size must be multiple of %s",
display_size(fid->fmt->cmd, (uint64_t) MIN_PE_SIZE));
2001-10-16 00:29:15 +04:00
return 0;
}
/* Redundant? */
if (vg->extent_size & (vg->extent_size - 1)) {
log_error("Extent size must be power of 2");
return 0;
}
2001-10-12 18:25:53 +04:00
return 1;
}
2001-10-09 21:44:58 +04:00
static int _format1_segtype_supported(struct format_instance *fid,
const struct segment_type *segtype)
{
if (!(segtype->flags & SEG_FORMAT1_SUPPORT)) {
stack;
return 0;
}
return 1;
}
2002-11-18 17:01:16 +03:00
static struct metadata_area_ops _metadata_format1_ops = {
.vg_read = _format1_vg_read,
.vg_write = _format1_vg_write,
2002-11-18 17:01:16 +03:00
};
static struct format_instance *_format1_create_instance(const struct format_type *fmt,
const char *vgname,
const char *vgid,
void *private)
{
struct format_instance *fid;
struct metadata_area *mda;
if (!(fid = dm_pool_alloc(fmt->cmd->mem, sizeof(*fid)))) {
stack;
return NULL;
}
fid->fmt = fmt;
list_init(&fid->metadata_areas);
/* Define a NULL metadata area */
if (!(mda = dm_pool_alloc(fmt->cmd->mem, sizeof(*mda)))) {
stack;
dm_pool_free(fmt->cmd->mem, fid);
return NULL;
}
2002-11-18 17:01:16 +03:00
mda->ops = &_metadata_format1_ops;
mda->metadata_locn = NULL;
list_add(&fid->metadata_areas, &mda->list);
return fid;
}
static void _format1_destroy_instance(struct format_instance *fid)
2001-10-04 21:48:55 +04:00
{
return;
2001-10-08 16:11:33 +04:00
}
static void _format1_destroy(const struct format_type *fmt)
{
dm_free((void *) fmt);
}
2001-10-08 16:11:33 +04:00
static struct format_handler _format1_ops = {
.pv_read = _format1_pv_read,
.pv_setup = _format1_pv_setup,
.pv_write = _format1_pv_write,
.lv_setup = _format1_lv_setup,
.vg_setup = _format1_vg_setup,
.segtype_supported = _format1_segtype_supported,
.create_instance = _format1_create_instance,
.destroy_instance = _format1_destroy_instance,
.destroy = _format1_destroy,
};
2001-10-30 20:53:21 +03:00
2002-11-18 17:01:16 +03:00
#ifdef LVM1_INTERNAL
struct format_type *init_lvm1_format(struct cmd_context *cmd)
#else /* Shared */
2003-04-15 17:24:42 +04:00
struct format_type *init_format(struct cmd_context *cmd);
2002-11-18 17:01:16 +03:00
struct format_type *init_format(struct cmd_context *cmd)
#endif
{
struct format_type *fmt = dm_malloc(sizeof(*fmt));
2001-10-08 16:11:33 +04:00
if (!fmt) {
2001-10-08 16:11:33 +04:00
stack;
return NULL;
2001-10-08 16:11:33 +04:00
}
fmt->cmd = cmd;
fmt->ops = &_format1_ops;
fmt->name = FMT_LVM1_NAME;
2002-11-18 17:01:16 +03:00
fmt->alias = NULL;
fmt->features = FMT_RESTRICTED_LVIDS | FMT_ORPHAN_ALLOCATABLE;
fmt->private = NULL;
2001-10-08 16:11:33 +04:00
2002-11-18 17:01:16 +03:00
if (!(fmt->labeller = lvm1_labeller_create(fmt))) {
log_error("Couldn't create lvm1 label handler.");
return NULL;
}
if (!(label_register_handler(FMT_LVM1_NAME, fmt->labeller))) {
log_error("Couldn't register lvm1 label handler.");
return NULL;
}
log_very_verbose("Initialised format: %s", fmt->name);
return fmt;
2001-10-04 21:48:55 +04:00
}