summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPeter Rajnoha <prajnoha@redhat.com>2015-08-26 14:12:24 +0200
committerPeter Rajnoha <prajnoha@redhat.com>2016-02-19 14:40:22 +0100
commit9ddbe44d26293b01777971a0ffde07f902bbff72 (patch)
tree9a5d2c4c0a30d86cc2c8007f2471330aa5474812
parent17d0fc1c1a3b8961a76a559490b5376a99e9e860 (diff)
downloadlvm2-9ddbe44d26293b01777971a0ffde07f902bbff72.tar.gz
metadata: add add_glv_to_indirect_glvs and remove_glv_from_indirect_glvs fns
The add_glv_to_indirect_glvs is a helper function that registers a volume represented by struct generic_logical_volume instance ("glv") as an indirect user of another volume ("origin_glv") and vice versa - it also registers the other volume ("origin_glv") as indirect_origin of user volume ("glv"). The remove_glv_from_indirect_glvs does the opposite.
-rw-r--r--lib/metadata/lv_manip.c58
-rw-r--r--lib/metadata/metadata.h6
2 files changed, 64 insertions, 0 deletions
diff --git a/lib/metadata/lv_manip.c b/lib/metadata/lv_manip.c
index 02efcb9c8..999084f00 100644
--- a/lib/metadata/lv_manip.c
+++ b/lib/metadata/lv_manip.c
@@ -5341,6 +5341,64 @@ struct glv_list *get_or_create_glvl(struct dm_pool *mem, struct logical_volume *
return glvl;
}
+int add_glv_to_indirect_glvs(struct dm_pool *mem,
+ struct generic_logical_volume *origin_glv,
+ struct generic_logical_volume *glv)
+{
+ struct glv_list *glvl;
+
+ if (!(glvl = dm_pool_zalloc(mem, sizeof(struct glv_list)))) {
+ log_error("Failed to allocate generic volume list item "
+ "for indirect glv %s", glv->is_former ? glv->former->name
+ : glv->live->name);
+ return 0;
+ }
+
+ glvl->glv = glv;
+
+ if (glv->is_former)
+ glv->former->indirect_origin = origin_glv;
+ else
+ first_seg(glv->live)->indirect_origin = origin_glv;
+
+ if (origin_glv) {
+ if (origin_glv->is_former)
+ dm_list_add(&origin_glv->former->indirect_glvs, &glvl->list);
+ else
+ dm_list_add(&origin_glv->live->indirect_glvs, &glvl->list);
+ }
+
+ return 1;
+}
+
+int remove_glv_from_indirect_glvs(struct generic_logical_volume *origin_glv,
+ struct generic_logical_volume *glv)
+{
+ struct glv_list *glvl, *tglvl;
+ struct dm_list *list = origin_glv->is_former ? &origin_glv->former->indirect_glvs
+ : &origin_glv->live->indirect_glvs;
+
+ dm_list_iterate_items_safe(glvl, tglvl, list) {
+ if (glvl->glv != glv)
+ continue;
+
+ dm_list_del(&glvl->list);
+
+ if (glvl->glv->is_former)
+ glvl->glv->former->indirect_origin = NULL;
+ else
+ first_seg(glvl->glv->live)->indirect_origin = NULL;
+
+ return 1;
+ }
+
+ log_error(INTERNAL_ERROR "%s logical volume %s is not a user of %s.",
+ glv->is_former ? "Former" : "Live",
+ glv->is_former ? glv->former->dname : glv->live->name,
+ origin_glv->is_former ? origin_glv->former->name : origin_glv->live->name);
+ return 0;
+}
+
struct logical_volume *alloc_lv(struct dm_pool *mem)
{
struct logical_volume *lv;
diff --git a/lib/metadata/metadata.h b/lib/metadata/metadata.h
index f2b62e477..785fdfed7 100644
--- a/lib/metadata/metadata.h
+++ b/lib/metadata/metadata.h
@@ -443,6 +443,12 @@ int lv_split_segment(struct logical_volume *lv, uint32_t le);
int add_seg_to_segs_using_this_lv(struct logical_volume *lv, struct lv_segment *seg);
int remove_seg_from_segs_using_this_lv(struct logical_volume *lv, struct lv_segment *seg);
+int add_glv_to_indirect_glvs(struct dm_pool *mem,
+ struct generic_logical_volume *origin_glv,
+ struct generic_logical_volume *user_glv);
+int remove_glv_from_indirect_glvs(struct generic_logical_volume *glv,
+ struct generic_logical_volume *user_glv);
+
int for_each_sub_lv_except_pools(struct logical_volume *lv,
int (*fn)(struct logical_volume *lv, void *data),
void *data);