diff --git a/lib/format_text/format-text.c b/lib/format_text/format-text.c index 0204a651c..e11ab52c3 100644 --- a/lib/format_text/format-text.c +++ b/lib/format_text/format-text.c @@ -2028,6 +2028,250 @@ out: return 1; } +int add_metadata_area_to_pv(struct physical_volume *pv, + unsigned mda_index, + uint64_t mda_start, + uint64_t mda_size, + unsigned mda_ignored) +{ + struct metadata_area *mda; + struct mda_context *mdac; + struct mda_lists *mda_lists = (struct mda_lists *) pv->fmt->private; + + if (mda_index >= FMT_TEXT_MAX_MDAS_PER_PV) { + log_error(INTERNAL_ERROR "can't add metadata area with " + "index %u to PV %s. Metadata " + "layout not supported by %s format.", + mda_index, dev_name(pv->dev), + pv->fmt->name); + } + + if (!(mda = dm_malloc(sizeof(struct metadata_area)))) { + log_error("struct metadata_area allocation failed"); + return 0; + } + + if (!(mdac = dm_malloc(sizeof(struct mda_context)))) { + log_error("struct mda_context allocation failed"); + dm_free(mda); + return 0; + } + + mda->ops = mda_lists->raw_ops; + mda->metadata_locn = mdac; + mda->status = 0; + + mdac->area.dev = pv->dev; + mdac->area.start = mda_start; + mdac->area.size = mda_size; + mdac->free_sectors = UINT64_C(0); + memset(&mdac->rlocn, 0, sizeof(mdac->rlocn)); + mda_set_ignored(mda, mda_ignored); + + fid_add_mda(pv->fid, mda, (char *) &pv->id, ID_LEN, mda_index); + + return 1; +} + +static int _text_pv_add_metadata_area(const struct format_type *fmt, + struct physical_volume *pv, + int pe_start_locked, + unsigned mda_index, + uint64_t mda_size, + unsigned mda_ignored) +{ + struct format_instance *fid = pv->fid; + const char *pvid = (char *) &pv->id; + uint64_t pe_start, pe_end; + uint64_t alignment, alignment_offset; + uint64_t disk_size; + uint64_t mda_start; + uint64_t adjustment, limit; + uint64_t wipe_size = 8 << SECTOR_SHIFT; + size_t page_size = lvm_getpagesize(); + struct metadata_area *mda; + struct mda_context *mdac; + + if (mda_index >= FMT_TEXT_MAX_MDAS_PER_PV) { + log_error(INTERNAL_ERROR "invalid index of value %u used " + "while trying to add metadata area on PV %s. " + "Metadata layout not supported by %s format.", + mda_index, pv_dev_name(pv), fmt->name); + return 0; + } + + pe_start = pv->pe_start << SECTOR_SHIFT; + alignment = pv->pe_align << SECTOR_SHIFT; + alignment_offset = pv->pe_align_offset << SECTOR_SHIFT; + disk_size = pv->size << SECTOR_SHIFT; + mda_size = mda_size << SECTOR_SHIFT; + + if (fid_get_mda_indexed(fid, pvid, ID_LEN, mda_index)) { + log_error(INTERNAL_ERROR "metadata area with index %u already " + "exists on PV %s.", mda_index, pv_dev_name(pv)); + return 0; + } + + /* First metadata area at the start of the device. */ + if (mda_index == 0) { + /* + * Try to fit MDA0 end within given pe_start limit if its value + * is locked. If it's not locked, count with any existing MDA1. + * If there's no MDA1, just use disk size as the limit. + */ + if (pe_start_locked) + limit = pe_start; + else if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 1)) && + (mdac = mda->metadata_locn)) + limit = mdac->area.start; + else + limit = disk_size; + + if (limit > disk_size) + goto bad; + + mda_start = LABEL_SCAN_SIZE; + + /* Align MDA0 start with page size if possible. */ + if (limit - mda_start >= MDA_SIZE_MIN) { + if ((adjustment = mda_start % page_size)) + mda_start += (page_size - adjustment); + } + + /* Align MDA0 end position with given alignment if possible. */ + if (alignment) { + if ((adjustment = (mda_start + mda_size) % alignment)) { + mda_size += (alignment - adjustment); + if (mda_start + mda_size > limit) + mda_size -= (alignment - adjustment); + } + } + + /* Align MDA0 end position with given alignment offset if possible. */ + if (alignment_offset && + (((mda_start + mda_size) % alignment) == 0)) { + mda_size += alignment_offset; + if (mda_start + mda_size > limit) + mda_size -= alignment_offset; + } + + if (mda_start + mda_size > limit) { + log_warn("WARNING: metadata area size outreaches " + "a limit on PV %s specified by its %s. " + "Trying to adjust metadata area size.", + pv_dev_name(pv), + pe_start_locked ? "PE start" : "disk size"); + + /* + * Try to decrease the MDA0 size with twice the + * alignment and then align with given alignment. + * If pe_start is locked, skip this type of + * alignment since it would be useless. + * Check first whether we can apply that! + */ + if (!pe_start_locked && + ((limit - mda_start) > alignment * 2)) { + mda_size = limit - mda_start - alignment * 2; + + if ((adjustment = (mda_start + mda_size) % alignment)) + mda_size += (alignment - adjustment); + + /* Still too much? Then there's nothing else to do. */ + if (mda_start + mda_size > limit) + goto bad; + } + /* Otherwise, give up and take any usable space. */ + /* FIXME: We should probably check for some minimum MDA size here. */ + else + mda_size = limit - mda_start; + } + + /* + * If PV's pe_start is not locked, update pe_start value with the + * start of the area that follows the MDA0 we've just calculated. + */ + if (!pe_start_locked) { + pe_start = mda_start + mda_size; + pv->pe_start = pe_start >> SECTOR_SHIFT; + } + } + /* Second metadata area at the end of the device. */ + else { + /* + * Try to fit MDA1 start within given pe_end or pe_start limit + * if it's locked. If pe_start and pe_end are not defined yet, + * count with any existing MDA0 and pe_start. If MDA0 does not + * exist, just use LABEL_SCAN_SIZE. + */ + pe_end = pv->pe_count ? (pv->pe_start + + pv->pe_count * pv->pe_size - 1) << SECTOR_SHIFT + : 0; + if (pe_start_locked) + limit = pe_end ? pe_end : pe_start; + else if (pe_start) + limit = pe_start; + /* + * Normally MDA0's start + size should be pe_start. + * The statemet here is probably useless since the + * situation is covered by previous statement. + */ + else if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 0)) && + (mdac = mda->metadata_locn)) + limit = mdac->area.start + mdac->area.size; + else + limit = LABEL_SCAN_SIZE; + + if (limit > disk_size || mda_size > disk_size) + goto bad; + + mda_start = disk_size - mda_size; + + if (alignment) { + adjustment = mda_start % alignment; + if (adjustment) + mda_size += adjustment; + } + + if (disk_size - mda_size < limit) + mda_size = disk_size - limit; + + /* + * If PV's pe_end not set yet, set it to the end of the + * area that precedes the MDA1 we've just calculated. + * FIXME: do we need to set this? Isn't it always set before? + */ + /*if (!pe_end) { + pe_end = mda_start; + pv->pe_end = pe_end >> SECTOR_SHIFT; + }*/ + } + + if (mda_size) { + /* Wipe metadata area with zeroes. */ + if (!dev_set((struct device *) pv->dev, mda_start, + (size_t) ((mda_size > wipe_size) ? + wipe_size : mda_size), 0)) { + log_error("Failed to wipe new metadata area " + "at the %s of the %s", + mda_index ? "end" : "start", + pv_dev_name(pv)); + return 0; + } + + /* Finally, add new metadata area to PV's format instance. */ + if (!add_metadata_area_to_pv(pv, mda_index, mda_start, + mda_size, mda_ignored)) + return_0; + } + + return 1; + +bad: + log_error("Not enough space available for metadata area " + "with index %u on PV %s.", mda_index, pv_dev_name(pv)); + return 0; +} + /* NULL vgname means use only the supplied context e.g. an archive file */ static struct format_instance *_text_create_text_instance(const struct format_type *fmt, const struct format_instance_ctx *fic) @@ -2102,6 +2346,7 @@ static struct format_handler _text_handler = { .scan = _text_scan, .pv_read = _text_pv_read, .pv_setup = _text_pv_setup, + .pv_add_metadata_area = _text_pv_add_metadata_area, .pv_write = _text_pv_write, .vg_setup = _text_vg_setup, .lv_setup = _text_lv_setup, diff --git a/lib/format_text/format-text.h b/lib/format_text/format-text.h index f3cf4f540..f65be745d 100644 --- a/lib/format_text/format-text.h +++ b/lib/format_text/format-text.h @@ -56,6 +56,11 @@ int add_da(struct dm_pool *mem, struct dm_list *das, uint64_t start, uint64_t size); void del_das(struct dm_list *das); +int add_metadata_area_to_pv(struct physical_volume *pv, + unsigned mda_index, + uint64_t mda_start, + uint64_t mda_size, + unsigned mda_ignored); int add_mda(const struct format_type *fmt, struct dm_pool *mem, struct dm_list *mdas, struct device *dev, uint64_t start, uint64_t size, unsigned ignored); void del_mdas(struct dm_list *mdas); diff --git a/lib/metadata/metadata.h b/lib/metadata/metadata.h index 00c07aa26..e112007a0 100644 --- a/lib/metadata/metadata.h +++ b/lib/metadata/metadata.h @@ -35,6 +35,7 @@ //#define MAX_RESTRICTED_LVS 255 /* Used by FMT_RESTRICTED_LVIDS */ #define MIRROR_LOG_OFFSET 2 /* sectors */ #define VG_MEMPOOL_CHUNK 10240 /* in bytes, hint only */ +#define PV_PE_START_CALC ((uint64_t) -1) /* Calculate pe_start value */ /* * Ceiling(n / sz) @@ -266,6 +267,16 @@ struct format_handler { unsigned metadataignore, struct dm_list * mdas, struct physical_volume * pv, struct volume_group * vg); + /* + * Add metadata area to a PV. Changes will take effect on pv_write. + */ + int (*pv_add_metadata_area) (const struct format_type * fmt, + struct physical_volume * pv, + int pe_start_locked, + unsigned metadata_index, + uint64_t metadata_size, + unsigned metadata_ignored); + /* * Write a PV structure to disk. Fails if the PV is in a VG ie * pv->vg_name must be a valid orphan VG name