summaryrefslogtreecommitdiffstats
path: root/lib/metadata/metadata.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/metadata/metadata.c')
-rw-r--r--lib/metadata/metadata.c305
1 files changed, 275 insertions, 30 deletions
diff --git a/lib/metadata/metadata.c b/lib/metadata/metadata.c
index a8c7399f..b8a55e65 100644
--- a/lib/metadata/metadata.c
+++ b/lib/metadata/metadata.c
@@ -12,6 +12,7 @@
#include "toolcontext.h"
#include "lvm-string.h"
#include "uuid.h"
+#include "vgcache.h"
#include <string.h>
@@ -20,17 +21,17 @@ int _add_pv_to_vg(struct format_instance *fi, struct volume_group *vg,
{
struct pv_list *pvl;
struct physical_volume *pv;
- struct pool *mem = fi->cmd->mem;
+ struct pool *mem = fi->fmt->cmd->mem;
log_verbose("Adding physical volume '%s' to volume group '%s'",
pv_name, vg->name);
- if (!(pvl = pool_alloc(mem, sizeof (*pvl)))) {
+ if (!(pvl = pool_alloc(mem, sizeof(*pvl)))) {
log_error("pv_list allocation for '%s' failed", pv_name);
return 0;
}
- if (!(pv = fi->ops->pv_read(fi, pv_name))) {
+ if (!(pv = pv_read(fi->fmt->cmd, pv_name))) {
log_error("Failed to read existing physical volume '%s'",
pv_name);
return 0;
@@ -50,16 +51,20 @@ int _add_pv_to_vg(struct format_instance *fi, struct volume_group *vg,
/* Units of 512-byte sectors */
pv->pe_size = vg->extent_size;
+ /* FIXME Do proper rounding-up alignment? */
+ /* Reserved space for label; this holds 0 for PVs created by LVM1 */
+ if (pv->pe_start < PE_ALIGN)
+ pv->pe_start = PE_ALIGN;
+
/*
* The next two fields should be corrected
* by fi->pv_setup.
*/
- pv->pe_start = 0;
- pv->pe_count = pv->size / pv->pe_size;
+ pv->pe_count = (pv->size - pv->pe_start) / pv->pe_size;
- pv->pe_allocated = 0;
+ pv->pe_alloc_count = 0;
- if (!fi->ops->pv_setup(fi, pv, vg)) {
+ if (!fi->fmt->ops->pv_setup(fi, pv, vg)) {
log_error("Format-specific setup of physical volume '%s' "
"failed.", pv_name);
return 0;
@@ -113,42 +118,43 @@ const char *strip_dir(const char *vg_name, const char *dev_dir)
return vg_name;
}
-struct volume_group *vg_create(struct format_instance *fi, const char *vg_name,
+struct volume_group *vg_create(struct cmd_context *cmd, const char *vg_name,
uint32_t extent_size, int max_pv, int max_lv,
int pv_count, char **pv_names)
{
struct volume_group *vg;
- struct pool *mem = fi->cmd->mem;
+ struct pool *mem = cmd->mem;
- if (!(vg = pool_alloc(mem, sizeof (*vg)))) {
+ if (!(vg = pool_zalloc(mem, sizeof(*vg)))) {
stack;
return NULL;
}
/* is this vg name already in use ? */
init_partial(1);
- if (fi->ops->vg_read(fi, vg_name)) {
+ if (vg_read(cmd, vg_name)) {
log_err("A volume group called '%s' already exists.", vg_name);
goto bad;
}
init_partial(0);
if (!id_create(&vg->id)) {
- log_err("Couldn't create uuid for volume group '%s'.",
- vg_name);
+ log_err("Couldn't create uuid for volume group '%s'.", vg_name);
goto bad;
}
/* Strip dev_dir if present */
- vg_name = strip_dir(vg_name, fi->cmd->dev_dir);
+ vg_name = strip_dir(vg_name, cmd->dev_dir);
- vg->cmd = fi->cmd;
+ vg->cmd = cmd;
if (!(vg->name = pool_strdup(mem, vg_name))) {
stack;
goto bad;
}
+ vg->seqno = 0;
+
vg->status = (RESIZEABLE_VG | LVM_READ | LVM_WRITE);
vg->system_id = pool_alloc(mem, NAME_LEN);
*vg->system_id = '\0';
@@ -169,30 +175,35 @@ struct volume_group *vg_create(struct format_instance *fi, const char *vg_name,
vg->snapshot_count = 0;
list_init(&vg->snapshots);
- if (!fi->ops->vg_setup(fi, vg)) {
+ if (!(vg->fid = cmd->fmt->ops->create_instance(cmd->fmt, vg_name,
+ NULL))) {
+ log_error("Failed to create format instance");
+ goto bad;
+ }
+
+ if (!vg->fid->fmt->ops->vg_setup(vg->fid, vg)) {
log_error("Format specific setup of volume group '%s' failed.",
vg_name);
goto bad;
}
/* attach the pv's */
- if (!vg_extend(fi, vg, pv_count, pv_names))
+ if (!vg_extend(vg->fid, vg, pv_count, pv_names))
goto bad;
return vg;
- bad:
+ bad:
pool_free(mem, vg);
return NULL;
}
-struct physical_volume *pv_create(struct format_instance *fi,
+struct physical_volume *pv_create(struct format_instance *fid,
const char *name,
- struct id *id,
- uint64_t size)
+ struct id *id, uint64_t size)
{
- struct pool *mem = fi->cmd->mem;
- struct physical_volume *pv = pool_alloc(mem, sizeof (*pv));
+ struct pool *mem = fid->fmt->cmd->mem;
+ struct physical_volume *pv = pool_alloc(mem, sizeof(*pv));
if (!pv) {
stack;
@@ -204,7 +215,7 @@ struct physical_volume *pv_create(struct format_instance *fi,
else
memcpy(&pv->id, id, sizeof(*id));
- if (!(pv->dev = dev_cache_get(name, fi->cmd->filter))) {
+ if (!(pv->dev = dev_cache_get(name, fid->fmt->cmd->filter))) {
log_error("%s: Couldn't find device.", name);
goto bad;
}
@@ -225,14 +236,14 @@ struct physical_volume *pv_create(struct format_instance *fi,
if (size) {
if (size > pv->size)
log_print("WARNING: %s: Overriding real size. "
- "You could lose data.", name);
- log_verbose("%s: Pretending size is %" PRIu64 " sectors.",
+ "You could lose data.", name);
+ log_verbose("%s: Pretending size is %" PRIu64 " sectors.",
name, size);
pv->size = size;
}
-
+
if (pv->size < PV_MIN_SIZE) {
- log_error("%s: Size must exceed minimum of %lu sectors.",
+ log_error("%s: Size must exceed minimum of %lu sectors.",
name, PV_MIN_SIZE);
goto bad;
}
@@ -240,9 +251,10 @@ struct physical_volume *pv_create(struct format_instance *fi,
pv->pe_size = 0;
pv->pe_start = 0;
pv->pe_count = 0;
- pv->pe_allocated = 0;
+ pv->pe_alloc_count = 0;
+ pv->fid = fid;
- if (!fi->ops->pv_setup(fi, pv, NULL)) {
+ if (!fid->fmt->ops->pv_setup(fid, pv, NULL)) {
log_error("%s: Format-specific setup of physical volume "
"failed.", name);
goto bad;
@@ -324,3 +336,236 @@ struct physical_volume *find_pv(struct volume_group *vg, struct device *dev)
return NULL;
}
+int vg_remove(struct volume_group *vg)
+{
+ struct list *mdah;
+ void *mdl;
+
+ if (!vg->fid->fmt->ops->vg_remove)
+ return 1;
+
+ /* FIXME Improve recovery situation? */
+ /* Remove each copy of the metadata */
+ list_iterate(mdah, &vg->fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!vg->fid->fmt->ops->vg_remove(vg->fid, vg, mdl)) {
+ stack;
+ return 0;
+ }
+ }
+
+ return 1;
+}
+
+int vg_write(struct volume_group *vg)
+{
+ struct list *mdah;
+ void *mdl;
+
+ vg->seqno++;
+
+ /* Write to each copy of the metadata area */
+ list_iterate(mdah, &vg->fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!vg->fid->fmt->ops->vg_write(vg->fid, vg, mdl)) {
+ stack;
+ return 0;
+ }
+ }
+
+ if (!vg->fid->fmt->ops->vg_commit)
+ return 1;
+
+ /* Commit to each copy of the metadata area */
+ list_iterate(mdah, &vg->fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!vg->fid->fmt->ops->vg_commit(vg->fid, vg, mdl)) {
+ stack;
+ return 0;
+ }
+ }
+
+ return 1;
+}
+
+struct volume_group *vg_read(struct cmd_context *cmd, const char *vg_name)
+{
+ struct format_instance *fid;
+ struct format_type *fmt;
+ struct volume_group *vg, *correct_vg;
+ struct list *mdah, *names;
+ void *mdl;
+ int inconsistent = 0, first_time = 1;
+
+ /* create format instance with appropriate metadata area */
+ if (!(fmt = vgcache_find_format(vg_name))) {
+ /* Do full scan */
+ if (!(names = get_vgs(cmd))) {
+ stack;
+ return NULL;
+ }
+ pool_free(cmd->mem, names);
+ if (!(fmt = vgcache_find_format(vg_name))) {
+ stack;
+ return NULL;
+ }
+ }
+
+ if (!(fid = fmt->ops->create_instance(fmt, vg_name, NULL))) {
+ log_error("Failed to create format instance");
+ return NULL;
+ }
+
+ /* Ensure contents of all metadata areas match - else do recovery */
+ list_iterate(mdah, &fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!(vg = fid->fmt->ops->vg_read(fid, vg_name, mdl))) {
+ stack;
+ return NULL;
+ }
+ if (first_time) {
+ correct_vg = vg;
+ first_time = 0;
+ continue;
+ }
+ if (correct_vg->seqno != vg->seqno) {
+ inconsistent = 1;
+ if (vg->seqno > correct_vg->seqno)
+ correct_vg = vg;
+ }
+ }
+
+ if (inconsistent) {
+ log_print("Inconsistent metadata copies found - updating "
+ "to use version %u", correct_vg->seqno);
+ if (!vg_write(correct_vg)) {
+ log_error("Automatic metadata correction failed");
+ return NULL;
+ }
+ }
+
+ vgcache_add(vg_name, correct_vg->id.uuid, NULL, fmt);
+
+ return vg;
+}
+
+struct volume_group *vg_read_by_vgid(struct cmd_context *cmd, const char *vgid)
+{
+ char *vgname;
+ struct list *vgs, *vgh;
+ struct volume_group *vg;
+
+ if (!(vgs = get_vgs(cmd))) {
+ log_error("vg_read_by_vgid: get_vgs failed");
+ return NULL;
+ }
+
+ list_iterate(vgh, vgs) {
+ vgname = list_item(vgh, struct name_list)->name;
+ if ((vg = vg_read(cmd, vgname)) &&
+ !strncmp(vg->id.uuid, vgid, ID_LEN)) return vg;
+ }
+
+ pool_free(cmd->mem, vgs);
+ return NULL;
+}
+
+/* FIXME Use label functions instead of PV functions? */
+struct physical_volume *pv_read(struct cmd_context *cmd, const char *pv_name)
+{
+ struct physical_volume *pv;
+
+ if (!(pv = pool_zalloc(cmd->mem, sizeof(*pv)))) {
+ log_error("pv_list allocation for '%s' failed", pv_name);
+ return 0;
+ }
+
+ /* Member of a format1 VG? */
+ if (!(cmd->fmt1->ops->pv_read(cmd->fmt1, pv_name, pv))) {
+ log_error("Failed to read existing physical volume '%s'",
+ pv_name);
+ return 0;
+ }
+
+ /* Member of a format_text VG? */
+ if (!(cmd->fmtt->ops->pv_read(cmd->fmtt, pv_name, pv))) {
+ log_error("Failed to read existing physical volume '%s'",
+ pv_name);
+ return 0;
+ }
+
+ if (!pv->size)
+ return NULL;
+ else
+ return pv;
+}
+
+struct list *get_vgs(struct cmd_context *cmd)
+{
+ struct list *names;
+
+ if (!(names = pool_alloc(cmd->mem, sizeof(*names)))) {
+ log_error("VG name list allocation failed");
+ return NULL;
+ }
+
+ list_init(names);
+
+ if (!cmd->fmt1->ops->get_vgs(cmd->fmt1, names) ||
+ !cmd->fmtt->ops->get_vgs(cmd->fmtt, names)) {
+ pool_free(cmd->mem, names);
+ return NULL;
+ }
+
+ return names;
+}
+
+struct list *get_pvs(struct cmd_context *cmd)
+{
+ struct list *results;
+
+ if (!(results = pool_alloc(cmd->mem, sizeof(*results)))) {
+ log_error("PV list allocation failed");
+ return NULL;
+ }
+
+ list_init(results);
+
+ /* fmtt modifies fmt1 output */
+ if (!cmd->fmt1->ops->get_pvs(cmd->fmt1, results) ||
+ !cmd->fmtt->ops->get_pvs(cmd->fmtt, results)) {
+ pool_free(cmd->mem, results);
+ return NULL;
+ }
+
+ return results;
+}
+
+int pv_write(struct cmd_context *cmd, struct physical_volume *pv)
+{
+ struct list *mdah;
+ void *mdl;
+
+ /* Write to each copy of the metadata area */
+ list_iterate(mdah, &pv->fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!pv->fid->fmt->ops->pv_write(pv->fid, pv, mdl)) {
+ stack;
+ return 0;
+ }
+ }
+
+ if (!pv->fid->fmt->ops->pv_commit)
+ return 1;
+
+ /* Commit to each copy of the metadata area */
+ list_iterate(mdah, &pv->fid->metadata_areas) {
+ mdl = list_item(mdah, struct metadata_area)->metadata_locn;
+ if (!pv->fid->fmt->ops->pv_commit(pv->fid, pv, mdl)) {
+ stack;
+ return 0;
+ }
+ }
+
+ return 1;
+}