/* * Copyright (C) 2001 Sistina Software * * This file is released under the LGPL. */ #include "lib.h" #include "pv_map.h" #include "hash.h" static int _create_maps(struct pool *mem, struct list *pvs, struct list *maps) { struct list *tmp; struct pv_map *pvm; struct pv_list *pvl; list_iterate(tmp, pvs) { pvl = list_item(tmp, struct pv_list); if (!(pvl->pv->status & ALLOCATABLE_PV)) continue; if (!(pvm = pool_zalloc(mem, sizeof(*pvm)))) { stack; return 0; } pvm->pvl = pvl; if (!(pvm->allocated_extents = bitset_create(mem, pvl->pv->pe_count))) { stack; return 0; } list_init(&pvm->areas); list_add(maps, &pvm->list); } return 1; } static int _set_allocd(struct hash_table *hash, struct physical_volume *pv, uint32_t pe) { struct pv_map *pvm; if (!(pvm = (struct pv_map *) hash_lookup(hash, dev_name(pv->dev)))) { /* * it doesn't matter that this fails, it just * means this part of the lv is on a pv that * we're not interested in allocating to. */ return 1; } /* sanity check */ if (bit(pvm->allocated_extents, pe)) { log_error("Physical extent %d of %s referenced by more than " "one logical volume", pe, dev_name(pv->dev)); return 0; } bit_set(pvm->allocated_extents, pe); return 1; } static int _fill_bitsets(struct volume_group *vg, struct list *maps) { struct list *lvh, *pvmh, *segh; struct logical_volume *lv; struct pv_map *pvm; uint32_t s, pe; struct hash_table *hash; struct lv_segment *seg; int r = 0; if (!(hash = hash_create(128))) { log_err("Couldn't create hash table for pv maps."); return 0; } /* populate the hash table */ list_iterate(pvmh, maps) { pvm = list_item(pvmh, struct pv_map); if (!hash_insert(hash, dev_name(pvm->pvl->pv->dev), pvm)) { stack; goto out; } } /* iterate through all the lv's setting bit's for used pe's */ list_iterate(lvh, &vg->lvs) { lv = list_item(lvh, struct lv_list)->lv; list_iterate(segh, &lv->segments) { seg = list_item(segh, struct lv_segment); for (s = 0u; s < seg->area_count; s++) { for (pe = 0u; pe < seg->area_len; pe++) { if (seg->area[s].type != AREA_PV) continue; if (!_set_allocd(hash, seg->area[s].u.pv.pv, seg->area[s].u.pv.pe + pe)) { stack; goto out; } } } } } r = 1; out: hash_destroy(hash); return r; } /* * Areas are maintained in size order. */ static void _insert_area(struct list *head, struct pv_area *a) { struct list *pvah; struct pv_area *pva = NULL; if (list_empty(head)) { list_add(head, &a->list); return; } list_iterate(pvah, head) { pva = list_item(pvah, struct pv_area); if (pva->count < a->count) break; } list_add_h(&pva->list, &a->list); } static int _create_single_area(struct pool *mem, struct pv_map *pvm, uint32_t end, uint32_t *extent) { uint32_t e = *extent, b; struct pv_area *pva; while (e <= end && bit(pvm->allocated_extents, e)) e++; if (e > end) { *extent = e; return 1; } b = e++; while (e <= end && !bit(pvm->allocated_extents, e)) e++; if (!(pva = pool_zalloc(mem, sizeof(*pva)))) { stack; return 0; } log_debug("Allowing allocation on %s start PE %" PRIu32 " length %" PRIu32, dev_name(pvm->pvl->pv->dev), b, e - b); pva->map = pvm; pva->start = b; pva->count = e - b; _insert_area(&pvm->areas, pva); *extent = e; return 1; } static int _create_areas(struct pool *mem, struct pv_map *pvm, uint32_t start, uint32_t count) { uint32_t pe, end; end = start + count - 1; if (end > pvm->pvl->pv->pe_count - 1) end = pvm->pvl->pv->pe_count - 1; pe = start; while (pe <= end) if (!_create_single_area(mem, pvm, end, &pe)) { stack; return 0; } return 1; } static int _create_allocatable_areas(struct pool *mem, struct pv_map *pvm) { struct list *alloc_areas, *aah; struct alloc_area *aa; alloc_areas = pvm->pvl->alloc_areas; if (alloc_areas) { list_iterate(aah, alloc_areas) { aa = list_item(aah, struct alloc_area); if (!_create_areas(mem, pvm, aa->start, aa->count)) { stack; return 0; } } } else { /* Use whole PV */ if (!_create_areas(mem, pvm, UINT32_C(0), pvm->pvl->pv->pe_count)) { stack; return 0; } } return 1; } static int _create_all_areas(struct pool *mem, struct list *maps, struct list *pvs) { struct list *tmp; struct pv_map *pvm; list_iterate(tmp, maps) { pvm = list_item(tmp, struct pv_map); if (!_create_allocatable_areas(mem, pvm)) { stack; return 0; } } return 1; } struct list *create_pv_maps(struct pool *mem, struct volume_group *vg, struct list *pvs) { struct list *maps = pool_zalloc(mem, sizeof(*maps)); if (!maps) { stack; return NULL; } list_init(maps); if (!_create_maps(mem, pvs, maps)) { log_error("Couldn't create physical volume maps in %s", vg->name); goto bad; } if (!_fill_bitsets(vg, maps)) { log_error("Couldn't fill extent allocation bitmaps in %s", vg->name); goto bad; } if (!_create_all_areas(mem, maps, pvs)) { log_error("Couldn't create area maps in %s", vg->name); goto bad; } return maps; bad: pool_free(mem, maps); return NULL; } void consume_pv_area(struct pv_area *pva, uint32_t to_go) { list_del(&pva->list); assert(to_go <= pva->count); if (to_go < pva->count) { /* split the area */ pva->start += to_go; pva->count -= to_go; _insert_area(&pva->map->areas, pva); } }