mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-31 21:18:26 +03:00
0a13815e68
Introduce FMT_OBSOLETE to identify pool metadata and use it and FMT_MDAS instead of hard-coded format names. Explain device accesses on pvscan --cache man page.
340 lines
8.2 KiB
C
340 lines
8.2 KiB
C
/*
|
|
* Copyright (C) 1997-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2012 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 "label.h"
|
|
#include "metadata.h"
|
|
#include "limits.h"
|
|
#include "display.h"
|
|
#include "toolcontext.h"
|
|
#include "lvmcache.h"
|
|
#include "disk_rep.h"
|
|
#include "format_pool.h"
|
|
#include "pool_label.h"
|
|
|
|
/* Must be called after pvs are imported */
|
|
static struct user_subpool *_build_usp(struct dm_list *pls, struct dm_pool *mem,
|
|
int *sps)
|
|
{
|
|
struct pool_list *pl;
|
|
struct user_subpool *usp = NULL, *cur_sp = NULL;
|
|
struct user_device *cur_dev = NULL;
|
|
|
|
/*
|
|
* FIXME: Need to do some checks here - I'm tempted to add a
|
|
* user_pool structure and build the entire thing to check against.
|
|
*/
|
|
dm_list_iterate_items(pl, pls) {
|
|
*sps = pl->pd.pl_subpools;
|
|
if (!usp && (!(usp = dm_pool_zalloc(mem, sizeof(*usp) * (*sps))))) {
|
|
log_error("Unable to allocate %d subpool structures",
|
|
*sps);
|
|
return 0;
|
|
}
|
|
|
|
if (cur_sp != &usp[pl->pd.pl_sp_id]) {
|
|
cur_sp = &usp[pl->pd.pl_sp_id];
|
|
|
|
cur_sp->id = pl->pd.pl_sp_id;
|
|
cur_sp->striping = pl->pd.pl_striping;
|
|
cur_sp->num_devs = pl->pd.pl_sp_devs;
|
|
cur_sp->type = pl->pd.pl_sp_type;
|
|
cur_sp->initialized = 1;
|
|
}
|
|
|
|
if (!cur_sp->devs &&
|
|
(!(cur_sp->devs =
|
|
dm_pool_zalloc(mem,
|
|
sizeof(*usp->devs) * pl->pd.pl_sp_devs)))) {
|
|
|
|
log_error("Unable to allocate %d pool_device "
|
|
"structures", pl->pd.pl_sp_devs);
|
|
return 0;
|
|
}
|
|
|
|
cur_dev = &cur_sp->devs[pl->pd.pl_sp_devid];
|
|
cur_dev->sp_id = cur_sp->id;
|
|
cur_dev->devid = pl->pd.pl_sp_id;
|
|
cur_dev->blocks = pl->pd.pl_blocks;
|
|
cur_dev->pv = pl->pv;
|
|
cur_dev->initialized = 1;
|
|
}
|
|
|
|
return usp;
|
|
}
|
|
|
|
static int _check_usp(const char *vgname, struct user_subpool *usp, int sp_count)
|
|
{
|
|
int i;
|
|
unsigned j;
|
|
|
|
for (i = 0; i < sp_count; i++) {
|
|
if (!usp[i].initialized) {
|
|
log_error("Missing subpool %d in pool %s", i, vgname);
|
|
return 0;
|
|
}
|
|
for (j = 0; j < usp[i].num_devs; j++) {
|
|
if (!usp[i].devs[j].initialized) {
|
|
log_error("Missing device %u for subpool %d"
|
|
" in pool %s", j, i, vgname);
|
|
return 0;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static struct volume_group *_pool_vg_read(struct format_instance *fid,
|
|
const char *vg_name,
|
|
struct metadata_area *mda __attribute__((unused)),
|
|
int single_device __attribute__((unused)))
|
|
{
|
|
struct volume_group *vg;
|
|
struct user_subpool *usp;
|
|
int sp_count;
|
|
DM_LIST_INIT(pds);
|
|
|
|
/* We can safely ignore the mda passed in */
|
|
|
|
/* Strip dev_dir if present */
|
|
if (vg_name)
|
|
vg_name = strip_dir(vg_name, fid->fmt->cmd->dev_dir);
|
|
|
|
/* Set vg_name through read_pool_pds() */
|
|
if (!(vg = alloc_vg("pool_vg_read", fid->fmt->cmd, NULL)))
|
|
return_NULL;
|
|
|
|
/* Read all the pvs in the vg */
|
|
if (!read_pool_pds(fid->fmt, vg_name, vg->vgmem, &pds))
|
|
goto_bad;
|
|
|
|
/* Setting pool seqno to 1 because the code always did this,
|
|
* although we don't think it's needed. */
|
|
vg->seqno = 1;
|
|
|
|
if (!import_pool_vg(vg, vg->vgmem, &pds))
|
|
goto_bad;
|
|
|
|
if (!import_pool_pvs(fid->fmt, vg, vg->vgmem, &pds))
|
|
goto_bad;
|
|
|
|
if (!import_pool_lvs(vg, vg->vgmem, &pds))
|
|
goto_bad;
|
|
|
|
/*
|
|
* I need an intermediate subpool structure that contains all the
|
|
* relevant info for this. Then i can iterate through the subpool
|
|
* structures for checking, and create the segments
|
|
*/
|
|
if (!(usp = _build_usp(&pds, vg->vgmem, &sp_count)))
|
|
goto_bad;
|
|
|
|
/*
|
|
* check the subpool structures - we can't handle partial VGs in
|
|
* the pool format, so this will error out if we're missing PVs
|
|
*/
|
|
if (!_check_usp(vg->name, usp, sp_count))
|
|
goto_bad;
|
|
|
|
if (!import_pool_segments(&vg->lvs, vg->vgmem, usp, sp_count))
|
|
goto_bad;
|
|
|
|
vg_set_fid(vg, fid);
|
|
|
|
return vg;
|
|
|
|
bad:
|
|
release_vg(vg);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int _pool_pv_initialise(const struct format_type *fmt __attribute__((unused)),
|
|
int64_t label_sector __attribute__((unused)),
|
|
unsigned long data_alignment __attribute__((unused)),
|
|
unsigned long data_alignment_offset __attribute__((unused)),
|
|
struct pvcreate_restorable_params *rp __attribute__((unused)),
|
|
struct physical_volume *pv __attribute__((unused)))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int _pool_pv_setup(const struct format_type *fmt __attribute__((unused)),
|
|
struct physical_volume *pv __attribute__((unused)),
|
|
struct volume_group *vg __attribute__((unused)))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int _pool_pv_read(const struct format_type *fmt, const char *pv_name,
|
|
struct physical_volume *pv,
|
|
int scan_label_only __attribute__((unused)))
|
|
{
|
|
struct dm_pool *mem = dm_pool_create("pool pv_read", 1024);
|
|
struct pool_list *pl;
|
|
struct device *dev;
|
|
int r = 0;
|
|
|
|
log_very_verbose("Reading physical volume data %s from disk", pv_name);
|
|
|
|
if (!mem)
|
|
return_0;
|
|
|
|
if (!(dev = dev_cache_get(pv_name, fmt->cmd->filter)))
|
|
goto_out;
|
|
|
|
/*
|
|
* I need to read the disk and populate a pv structure here
|
|
* I'll probably need to abstract some of this later for the
|
|
* vg_read code
|
|
*/
|
|
if (!(pl = read_pool_disk(fmt, dev, mem, NULL)))
|
|
goto_out;
|
|
|
|
if (!import_pool_pv(fmt, fmt->cmd->mem, NULL, pv, pl))
|
|
goto_out;
|
|
|
|
pv->fmt = fmt;
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_pool_destroy(mem);
|
|
return r;
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
static struct metadata_area_ops _metadata_format_pool_ops = {
|
|
.vg_read = _pool_vg_read,
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
static struct format_instance *_pool_create_instance(const struct format_type *fmt,
|
|
const struct format_instance_ctx *fic)
|
|
{
|
|
struct format_instance *fid;
|
|
struct metadata_area *mda;
|
|
|
|
if (!(fid = alloc_fid(fmt, fic)))
|
|
return_NULL;
|
|
|
|
/* Define a NULL metadata area */
|
|
if (!(mda = dm_pool_zalloc(fid->mem, sizeof(*mda)))) {
|
|
log_error("Unable to allocate metadata area structure "
|
|
"for pool format");
|
|
goto bad;
|
|
}
|
|
|
|
mda->ops = &_metadata_format_pool_ops;
|
|
mda->metadata_locn = NULL;
|
|
mda->status = 0;
|
|
dm_list_add(&fid->metadata_areas_in_use, &mda->list);
|
|
|
|
return fid;
|
|
|
|
bad:
|
|
dm_pool_destroy(fid->mem);
|
|
return NULL;
|
|
}
|
|
|
|
static void _pool_destroy_instance(struct format_instance *fid)
|
|
{
|
|
if (--fid->ref_count <= 1)
|
|
dm_pool_destroy(fid->mem);
|
|
}
|
|
|
|
static void _pool_destroy(struct format_type *fmt)
|
|
{
|
|
if (fmt->orphan_vg)
|
|
free_orphan_vg(fmt->orphan_vg);
|
|
|
|
dm_free(fmt);
|
|
}
|
|
|
|
/* *INDENT-OFF* */
|
|
static struct format_handler _format_pool_ops = {
|
|
.pv_read = _pool_pv_read,
|
|
.pv_initialise = _pool_pv_initialise,
|
|
.pv_setup = _pool_pv_setup,
|
|
.create_instance = _pool_create_instance,
|
|
.destroy_instance = _pool_destroy_instance,
|
|
.destroy = _pool_destroy,
|
|
};
|
|
/* *INDENT-ON */
|
|
|
|
#ifdef POOL_INTERNAL
|
|
struct format_type *init_pool_format(struct cmd_context *cmd)
|
|
#else /* Shared */
|
|
struct format_type *init_format(struct cmd_context *cmd);
|
|
struct format_type *init_format(struct cmd_context *cmd)
|
|
#endif
|
|
{
|
|
struct format_type *fmt = dm_malloc(sizeof(*fmt));
|
|
struct format_instance_ctx fic;
|
|
struct format_instance *fid;
|
|
|
|
if (!fmt) {
|
|
log_error("Unable to allocate format type structure for pool "
|
|
"format");
|
|
return NULL;
|
|
}
|
|
|
|
fmt->cmd = cmd;
|
|
fmt->ops = &_format_pool_ops;
|
|
fmt->name = FMT_POOL_NAME;
|
|
fmt->alias = NULL;
|
|
fmt->orphan_vg_name = FMT_POOL_ORPHAN_VG_NAME;
|
|
fmt->features = FMT_OBSOLETE;
|
|
fmt->private = NULL;
|
|
|
|
dm_list_init(&fmt->mda_ops);
|
|
|
|
if (!(fmt->labeller = pool_labeller_create(fmt))) {
|
|
log_error("Couldn't create pool label handler.");
|
|
dm_free(fmt);
|
|
return NULL;
|
|
}
|
|
|
|
if (!(label_register_handler(fmt->labeller))) {
|
|
log_error("Couldn't register pool label handler.");
|
|
fmt->labeller->ops->destroy(fmt->labeller);
|
|
dm_free(fmt);
|
|
return NULL;
|
|
}
|
|
|
|
if (!(fmt->orphan_vg = alloc_vg("pool_orphan", cmd, fmt->orphan_vg_name))) {
|
|
log_error("Couldn't create pool orphan VG.");
|
|
dm_free(fmt);
|
|
return NULL;
|
|
}
|
|
|
|
fic.type = FMT_INSTANCE_AUX_MDAS;
|
|
fic.context.vg_ref.vg_name = fmt->orphan_vg_name;
|
|
fic.context.vg_ref.vg_id = NULL;
|
|
|
|
if (!(fid = _pool_create_instance(fmt, &fic))) {
|
|
_pool_destroy(fmt);
|
|
return NULL;
|
|
}
|
|
|
|
vg_set_fid(fmt->orphan_vg, fid);
|
|
|
|
log_very_verbose("Initialised format: %s", fmt->name);
|
|
|
|
return fmt;
|
|
}
|