1
0
mirror of git://sourceware.org/git/lvm2.git synced 2024-12-31 21:18:26 +03:00
lvm2/lib/format1/layout.c

164 lines
3.6 KiB
C
Raw Normal View History

/*
* Copyright (C) 2001 Sistina Software (UK) Limited.
*
2001-10-31 15:47:01 +03:00
* This file is released under the LGPL.
*/
2002-11-18 17:01:16 +03:00
#include "lib.h"
#include "disk-rep.h"
/*
* Only works with powers of 2.
*/
2003-04-15 17:24:42 +04:00
static inline uint32_t _round_up(uint32_t n, uint32_t size)
{
size--;
return (n + size) & ~size;
}
2003-04-15 17:24:42 +04:00
static inline uint32_t _div_up(uint32_t n, uint32_t size)
{
return _round_up(n, size) / size;
}
/*
* Each chunk of metadata should be aligned to
* METADATA_ALIGN.
*/
static uint32_t _next_base(struct data_area *area)
{
return _round_up(area->base + area->size, METADATA_ALIGN);
}
/*
* Quick calculation based on pe_start.
*/
static int _adjust_pe_on_disk(struct pv_disk *pvd)
{
2002-11-18 17:01:16 +03:00
uint32_t pe_start = pvd->pe_start << SECTOR_SHIFT;
if (pe_start < pvd->pe_on_disk.base + pvd->pe_on_disk.size)
return 0;
pvd->pe_on_disk.size = pe_start - pvd->pe_on_disk.base;
return 1;
}
static void _calc_simple_layout(struct pv_disk *pvd)
{
pvd->pv_on_disk.base = METADATA_BASE;
pvd->pv_on_disk.size = PV_SIZE;
pvd->vg_on_disk.base = _next_base(&pvd->pv_on_disk);
pvd->vg_on_disk.size = VG_SIZE;
pvd->pv_uuidlist_on_disk.base = _next_base(&pvd->vg_on_disk);
pvd->pv_uuidlist_on_disk.size = MAX_PV * NAME_LEN;
pvd->lv_on_disk.base = _next_base(&pvd->pv_uuidlist_on_disk);
pvd->lv_on_disk.size = MAX_LV * sizeof(struct lv_disk);
pvd->pe_on_disk.base = _next_base(&pvd->lv_on_disk);
pvd->pe_on_disk.size = pvd->pe_total * sizeof(struct pe_disk);
}
static int _check_vg_limits(struct disk_list *dl)
2001-10-16 20:25:28 +04:00
{
if (dl->vgd.lv_max > MAX_LV) {
2001-10-16 20:25:28 +04:00
log_error("MaxLogicalVolumes of %d exceeds format limit of %d "
2002-01-25 16:41:13 +03:00
"for VG '%s'", dl->vgd.lv_max, MAX_LV - 1,
dl->pvd.vg_name);
2001-10-16 20:25:28 +04:00
return 0;
}
if (dl->vgd.pv_max > MAX_PV) {
2001-10-16 20:25:28 +04:00
log_error("MaxPhysicalVolumes of %d exceeds format limit of %d "
2002-01-25 16:41:13 +03:00
"for VG '%s'", dl->vgd.pv_max, MAX_PV - 1,
dl->pvd.vg_name);
2001-10-16 20:25:28 +04:00
return 0;
}
return 1;
}
/*
* This assumes pe_count and pe_start have already
* been calculated correctly.
*/
int calculate_layout(struct disk_list *dl)
{
struct pv_disk *pvd = &dl->pvd;
_calc_simple_layout(pvd);
if (!_adjust_pe_on_disk(pvd)) {
2001-10-16 20:25:28 +04:00
log_error("Insufficient space for metadata and PE's.");
return 0;
}
2001-10-16 20:25:28 +04:00
if (!_check_vg_limits(dl))
return 0;
return 1;
}
/*
2002-11-18 17:01:16 +03:00
* The number of extents that can fit on a disk is metadata format dependant.
* pe_start is any existing value for pe_start
*/
2002-11-18 17:01:16 +03:00
int calculate_extent_count(struct physical_volume *pv, uint32_t extent_size,
uint32_t max_extent_count, uint64_t pe_start)
{
struct pv_disk *pvd = dbg_malloc(sizeof(*pvd));
uint32_t end;
if (!pvd) {
stack;
return 0;
}
/*
2002-01-31 18:28:30 +03:00
* Guess how many extents will fit, bearing in mind that
* one is going to be knocked off at the start of the
* next loop.
*/
2002-11-18 17:01:16 +03:00
if (max_extent_count)
pvd->pe_total = max_extent_count + 1;
else
pvd->pe_total = (pv->size / extent_size);
2001-10-16 00:29:15 +04:00
if (pvd->pe_total < PE_SIZE_PV_SIZE_REL) {
2002-11-18 17:01:16 +03:00
log_error("Too few extents on %s. Try smaller extent size.",
dev_name(pv->dev));
2002-01-24 20:24:32 +03:00
dbg_free(pvd);
2001-10-16 00:29:15 +04:00
return 0;
}
do {
pvd->pe_total--;
_calc_simple_layout(pvd);
end = ((pvd->pe_on_disk.base + pvd->pe_on_disk.size +
2002-11-18 17:01:16 +03:00
SECTOR_SIZE - 1) >> SECTOR_SHIFT);
if (pe_start && end < pe_start)
end = pe_start;
pvd->pe_start = _round_up(end, PE_ALIGN);
2002-11-18 17:01:16 +03:00
} while ((pvd->pe_start + (pvd->pe_total * extent_size))
> pv->size);
if (pvd->pe_total > MAX_PE_TOTAL) {
log_error("Metadata extent limit (%u) exceeded for %s - "
"%u required", MAX_PE_TOTAL, dev_name(pv->dev),
pvd->pe_total);
dbg_free(pvd);
return 0;
}
pv->pe_count = pvd->pe_total;
pv->pe_start = pvd->pe_start;
2002-11-18 17:01:16 +03:00
/* We can't set pe_size here without breaking LVM1 compatibility */
dbg_free(pvd);
return 1;
}