mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-11 09:18:25 +03:00
Optimise PV segments search.
The function find_peg_by_pe is incredibly inefficient for Pvs with many segments. In shiny future there should be binary (or interval) tree instead of sorted linked list (volunteers?). Anyway, for now, we can use dirty trick here to optimise this case: - Allocations are usually applied from the beginning of PV (we have no alloocation policy which allocates areas "backwards") - The only user of find_peg_by_pe is pv_split_segment() call. In *most* cases it need to split *last* PV segment. So if we search sorted pv segment list backwards, we hit the requested segment immediatelly. This patch applies this tiny change. (and saves >30% of processing time when >3000LVs segments are on one PV!) To discourage using this inefficient function from other code, it is moved to pv_manip.c and used static for now:-)
This commit is contained in:
parent
4e8859d851
commit
a0074aa07e
@ -1,5 +1,6 @@
|
|||||||
Version 2.02.63 -
|
Version 2.02.63 -
|
||||||
================================
|
================================
|
||||||
|
Optimise PV segments search for the most last segment search case.
|
||||||
Remove vg_validate call when parsing cached metadata.
|
Remove vg_validate call when parsing cached metadata.
|
||||||
Use hash table of LVs to speed up parsing of text metadata with many LVs.
|
Use hash table of LVs to speed up parsing of text metadata with many LVs.
|
||||||
Fix two messages, add a whitespace and parentheses
|
Fix two messages, add a whitespace and parentheses
|
||||||
|
@ -1799,18 +1799,6 @@ struct lv_segment *first_seg(const struct logical_volume *lv)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Find segment at a given physical extent in a PV */
|
|
||||||
struct pv_segment *find_peg_by_pe(const struct physical_volume *pv, uint32_t pe)
|
|
||||||
{
|
|
||||||
struct pv_segment *peg;
|
|
||||||
|
|
||||||
dm_list_iterate_items(peg, &pv->segments)
|
|
||||||
if (pe >= peg->pe && pe < peg->pe + peg->len)
|
|
||||||
return peg;
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
int vg_remove_mdas(struct volume_group *vg)
|
int vg_remove_mdas(struct volume_group *vg)
|
||||||
{
|
{
|
||||||
struct metadata_area *mda;
|
struct metadata_area *mda;
|
||||||
|
@ -308,9 +308,6 @@ struct pv_list *find_pv_in_pv_list(const struct dm_list *pl,
|
|||||||
/* Find LV segment containing given LE */
|
/* Find LV segment containing given LE */
|
||||||
struct lv_segment *find_seg_by_le(const struct logical_volume *lv, uint32_t le);
|
struct lv_segment *find_seg_by_le(const struct logical_volume *lv, uint32_t le);
|
||||||
|
|
||||||
/* Find PV segment containing given LE */
|
|
||||||
struct pv_segment *find_peg_by_pe(const struct physical_volume *pv, uint32_t pe);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Remove a dev_dir if present.
|
* Remove a dev_dir if present.
|
||||||
*/
|
*/
|
||||||
|
@ -78,6 +78,20 @@ int peg_dup(struct dm_pool *mem, struct dm_list *peg_new, struct dm_list *peg_ol
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Find segment at a given physical extent in a PV */
|
||||||
|
static struct pv_segment *find_peg_by_pe(const struct physical_volume *pv,
|
||||||
|
uint32_t pe)
|
||||||
|
{
|
||||||
|
struct pv_segment *pvseg;
|
||||||
|
|
||||||
|
/* search backwards to optimise mostly used last segment split */
|
||||||
|
dm_list_iterate_back_items(pvseg, &pv->segments)
|
||||||
|
if (pe >= pvseg->pe && pe < pvseg->pe + pvseg->len)
|
||||||
|
return pvseg;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Split peg at given extent.
|
* Split peg at given extent.
|
||||||
* Second part is always deallocated.
|
* Second part is always deallocated.
|
||||||
|
Loading…
Reference in New Issue
Block a user