From 13aea49489a44f28dc87f25b229dd0b1044f93e0 Mon Sep 17 00:00:00 2001 From: Zdenek Kabelac Date: Sat, 26 Oct 2024 22:37:00 +0200 Subject: [PATCH] vg: add radix_tree for lv uuids When searching for committed LV by uuid, this search can be expensive for commands like 'vgremove' - so for this part introduce 'lv_uuids' radix_tree that is build with first access to lv_committed(). --- lib/metadata/metadata.c | 15 +++++++++++++++ lib/metadata/vg.c | 3 +++ lib/metadata/vg.h | 1 + 3 files changed, 19 insertions(+) diff --git a/lib/metadata/metadata.c b/lib/metadata/metadata.c index 8232e9c34..38d7ae99a 100644 --- a/lib/metadata/metadata.c +++ b/lib/metadata/metadata.c @@ -1667,6 +1667,10 @@ struct logical_volume *find_lv_in_vg_by_lvid(const struct volume_group *vg, if (memcmp(&lvid->id[0], &vg->id, ID_LEN)) return NULL; /* Check VG does not match */ + if (vg->lv_uuids) + return radix_tree_lookup_ptr(vg->lv_uuids, &lvid->id[1], + sizeof(lvid->id[1])); + dm_list_iterate_items(lvl, &vg->lvs) if (!memcmp(&lvid->id[1], &lvl->lv->lvid.id[1], sizeof(lvid->id[1]))) return lvl->lv; /* LV uuid match */ @@ -4344,6 +4348,7 @@ const struct logical_volume *lv_committed(const struct logical_volume *lv) { struct volume_group *vg; const struct logical_volume *found_lv; + struct lv_list *lvl; if (!lv) return NULL; @@ -4353,6 +4358,16 @@ const struct logical_volume *lv_committed(const struct logical_volume *lv) vg = lv->vg->vg_committed; + if (!vg->lv_uuids && + (vg->lv_uuids = radix_tree_create(NULL, NULL))) + dm_list_iterate_items(lvl, &vg->lvs) { + if (!radix_tree_insert_ptr(vg->lv_uuids, + &lvl->lv->lvid.id[1], + sizeof(lvl->lv->lvid.id[1]), + lvl->lv)) + return NULL; + } + if (!(found_lv = find_lv_in_vg_by_lvid(vg, &lv->lvid))) { log_error(INTERNAL_ERROR "LV %s (UUID %s) not found in committed metadata.", display_lvname(lv), lv->lvid.s); diff --git a/lib/metadata/vg.c b/lib/metadata/vg.c index 49dfe0266..d78022bb7 100644 --- a/lib/metadata/vg.c +++ b/lib/metadata/vg.c @@ -81,6 +81,9 @@ static void _free_vg(struct volume_group *vg) if (vg->lv_names) radix_tree_destroy(vg->lv_names); + if (vg->lv_uuids) + radix_tree_destroy(vg->lv_uuids); + if (vg->pv_names) radix_tree_destroy(vg->pv_names); diff --git a/lib/metadata/vg.h b/lib/metadata/vg.h index 4117932bd..92a15ef2f 100644 --- a/lib/metadata/vg.h +++ b/lib/metadata/vg.h @@ -65,6 +65,7 @@ struct volume_group { uint64_t status; struct radix_tree *lv_names; /* maintained tree for LV names within VG */ + struct radix_tree *lv_uuids; /* LV uuid (when searching committed metadata) */ struct radix_tree *pv_names; /* PV names used for metadata import */ struct id id;