diff options
author | Peter Rajnoha <prajnoha@redhat.com> | 2015-08-26 14:12:24 +0200 |
---|---|---|
committer | Peter Rajnoha <prajnoha@redhat.com> | 2016-02-19 14:40:22 +0100 |
commit | 9ddbe44d26293b01777971a0ffde07f902bbff72 (patch) | |
tree | 9a5d2c4c0a30d86cc2c8007f2471330aa5474812 | |
parent | 17d0fc1c1a3b8961a76a559490b5376a99e9e860 (diff) | |
download | lvm2-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.c | 58 | ||||
-rw-r--r-- | lib/metadata/metadata.h | 6 |
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); |