mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-06 17:18:29 +03:00
0c1316cda8
After the VG lock is taken for vg_read, reread the mda_header and compare the metadata text offset and checksum to what was seen during label scan. If it is unchanged, then the metadata has not changed since the label scan, and the metadata does not need to be reread under the lock for command processing. For commands that do not make changes (e.g. reporting), the mda_header is reread and checked on one mda to decide if the full metadata rereading can be skipped. For other commands (e.g. modifying the vg) the mda_header is reread and checked from all PVs. (These could probably just check one mda also.)
2725 lines
82 KiB
C
2725 lines
82 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2012 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "lib/misc/lib.h"
|
|
#include "import-export.h"
|
|
#include "format-text.h"
|
|
#include "layout.h"
|
|
#include "lib/device/device.h"
|
|
#include "lib/misc/lvm-file.h"
|
|
#include "lib/config/config.h"
|
|
#include "lib/display/display.h"
|
|
#include "lib/commands/toolcontext.h"
|
|
#include "lib/misc/lvm-string.h"
|
|
#include "lib/uuid/uuid.h"
|
|
#include "lib/misc/crc.h"
|
|
#include "lib/mm/xlate.h"
|
|
#include "lib/label/label.h"
|
|
#include "lib/cache/lvmcache.h"
|
|
|
|
#include <unistd.h>
|
|
#include <limits.h>
|
|
#include <dirent.h>
|
|
#include <ctype.h>
|
|
|
|
static struct format_instance *_text_create_text_instance(const struct format_type *fmt,
|
|
const struct format_instance_ctx *fic);
|
|
|
|
struct text_fid_context {
|
|
char *write_buf; /* buffer containing metadata text to write to disk */
|
|
uint32_t write_buf_size; /* mem size of write_buf, increases in 64K multiples */
|
|
uint32_t new_metadata_size; /* size of text metadata in buf */
|
|
unsigned preserve:1;
|
|
};
|
|
|
|
void preserve_text_fidtc(struct volume_group *vg)
|
|
{
|
|
struct format_instance *fid = vg->fid;
|
|
struct text_fid_context *fidtc = (struct text_fid_context *)fid->private;
|
|
|
|
if (fidtc)
|
|
fidtc->preserve = 1;
|
|
}
|
|
|
|
void free_text_fidtc(struct volume_group *vg)
|
|
{
|
|
struct format_instance *fid = vg->fid;
|
|
struct text_fid_context *fidtc = (struct text_fid_context *)fid->private;
|
|
|
|
if (!fidtc)
|
|
return;
|
|
|
|
fidtc->preserve = 0;
|
|
|
|
if (fidtc->write_buf)
|
|
free(fidtc->write_buf);
|
|
fidtc->write_buf = NULL;
|
|
fidtc->write_buf_size = 0;
|
|
fidtc->new_metadata_size = 0;
|
|
}
|
|
|
|
int rlocn_is_ignored(const struct raw_locn *rlocn)
|
|
{
|
|
return (rlocn->flags & RAW_LOCN_IGNORED ? 1 : 0);
|
|
}
|
|
|
|
void rlocn_set_ignored(struct raw_locn *rlocn, unsigned mda_ignored)
|
|
{
|
|
if (mda_ignored)
|
|
rlocn->flags |= RAW_LOCN_IGNORED;
|
|
else
|
|
rlocn->flags &= ~RAW_LOCN_IGNORED;
|
|
}
|
|
|
|
/*
|
|
* NOTE: Currently there can be only one vg per text file.
|
|
*/
|
|
|
|
/*
|
|
* Only used by vgcreate.
|
|
*/
|
|
static int _text_vg_setup(struct format_instance *fid,
|
|
struct volume_group *vg)
|
|
{
|
|
if (!vg_check_new_extent_size(vg->fid->fmt, vg->extent_size))
|
|
return_0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static uint64_t _mda_free_sectors_raw(struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
|
|
return mdac->free_sectors;
|
|
}
|
|
|
|
static uint64_t _mda_total_sectors_raw(struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
|
|
return mdac->area.size >> SECTOR_SHIFT;
|
|
}
|
|
|
|
/*
|
|
* Check if metadata area belongs to vg
|
|
*/
|
|
static int _mda_in_vg_raw(struct format_instance *fid __attribute__((unused)),
|
|
struct volume_group *vg, struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct pv_list *pvl;
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs)
|
|
if (pvl->pv->dev == mdac->area.dev)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned _mda_locns_match_raw(struct metadata_area *mda1,
|
|
struct metadata_area *mda2)
|
|
{
|
|
struct mda_context *mda1c = (struct mda_context *) mda1->metadata_locn;
|
|
struct mda_context *mda2c = (struct mda_context *) mda2->metadata_locn;
|
|
|
|
if ((mda1c->area.dev == mda2c->area.dev) &&
|
|
(mda1c->area.start == mda2c->area.start) &&
|
|
(mda1c->area.size == mda2c->area.size))
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct device *_mda_get_device_raw(struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
return mdac->area.dev;
|
|
}
|
|
|
|
static int _text_lv_setup(struct format_instance *fid __attribute__((unused)),
|
|
struct logical_volume *lv)
|
|
{
|
|
/******** FIXME Any LV size restriction?
|
|
uint64_t max_size = UINT_MAX;
|
|
|
|
if (lv->size > max_size) {
|
|
char *dummy = display_size(max_size);
|
|
log_error("logical volumes cannot be larger than %s", dummy);
|
|
free(dummy);
|
|
return 0;
|
|
}
|
|
*/
|
|
|
|
if (!*lv->lvid.s && !lvid_create(&lv->lvid, &lv->vg->id)) {
|
|
log_error("Random lvid creation failed for %s/%s.",
|
|
lv->vg->name, lv->name);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _xlate_mdah(struct mda_header *mdah)
|
|
{
|
|
struct raw_locn *rl;
|
|
|
|
mdah->version = xlate32(mdah->version);
|
|
mdah->start = xlate64(mdah->start);
|
|
mdah->size = xlate64(mdah->size);
|
|
|
|
rl = &mdah->raw_locns[0];
|
|
while (rl->offset) {
|
|
rl->checksum = xlate32(rl->checksum);
|
|
rl->offset = xlate64(rl->offset);
|
|
rl->size = xlate64(rl->size);
|
|
rl++;
|
|
}
|
|
}
|
|
|
|
static int _raw_read_mda_header(struct mda_header *mdah, struct device_area *dev_area,
|
|
int primary_mda, uint32_t ignore_bad_fields, uint32_t *bad_fields)
|
|
{
|
|
log_debug_metadata("Reading mda header sector from %s at %llu",
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
|
|
if (!dev_read_bytes(dev_area->dev, dev_area->start, MDA_HEADER_SIZE, mdah)) {
|
|
log_error("Failed to read metadata area header on %s at %llu",
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
*bad_fields |= BAD_MDA_READ;
|
|
return 0;
|
|
}
|
|
|
|
if (mdah->checksum_xl != xlate32(calc_crc(INITIAL_CRC, (uint8_t *)mdah->magic,
|
|
MDA_HEADER_SIZE -
|
|
sizeof(mdah->checksum_xl)))) {
|
|
log_warn("WARNING: wrong checksum %x in mda header on %s at %llu",
|
|
mdah->checksum_xl,
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
*bad_fields |= BAD_MDA_CHECKSUM;
|
|
}
|
|
|
|
_xlate_mdah(mdah);
|
|
|
|
if (memcmp(mdah->magic, FMTT_MAGIC, sizeof(mdah->magic))) {
|
|
log_warn("WARNING: wrong magic number in mda header on %s at %llu",
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
*bad_fields |= BAD_MDA_MAGIC;
|
|
}
|
|
|
|
if (mdah->version != FMTT_VERSION) {
|
|
log_warn("WARNING: wrong version %u in mda header on %s at %llu",
|
|
mdah->version,
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
*bad_fields |= BAD_MDA_VERSION;
|
|
}
|
|
|
|
if (mdah->start != dev_area->start) {
|
|
log_warn("WARNING: wrong start sector %llu in mda header on %s at %llu",
|
|
(unsigned long long)mdah->start,
|
|
dev_name(dev_area->dev), (unsigned long long)dev_area->start);
|
|
*bad_fields |= BAD_MDA_START;
|
|
}
|
|
|
|
*bad_fields &= ~ignore_bad_fields;
|
|
|
|
if (*bad_fields)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct mda_header *raw_read_mda_header(const struct format_type *fmt,
|
|
struct device_area *dev_area,
|
|
int primary_mda, uint32_t ignore_bad_fields, uint32_t *bad_fields)
|
|
{
|
|
struct mda_header *mdah;
|
|
|
|
if (!(mdah = dm_pool_alloc(fmt->cmd->mem, MDA_HEADER_SIZE))) {
|
|
log_error("struct mda_header allocation failed");
|
|
*bad_fields |= BAD_MDA_INTERNAL;
|
|
return NULL;
|
|
}
|
|
|
|
if (!_raw_read_mda_header(mdah, dev_area, primary_mda, ignore_bad_fields, bad_fields)) {
|
|
dm_pool_free(fmt->cmd->mem, mdah);
|
|
return NULL;
|
|
}
|
|
|
|
return mdah;
|
|
}
|
|
|
|
static int _raw_write_mda_header(const struct format_type *fmt,
|
|
struct device *dev, int primary_mda,
|
|
uint64_t start_byte, struct mda_header *mdah)
|
|
{
|
|
memcpy(mdah->magic, FMTT_MAGIC, sizeof(mdah->magic));
|
|
mdah->version = FMTT_VERSION;
|
|
mdah->start = start_byte;
|
|
|
|
_xlate_mdah(mdah);
|
|
mdah->checksum_xl = xlate32(calc_crc(INITIAL_CRC, (uint8_t *)mdah->magic,
|
|
MDA_HEADER_SIZE -
|
|
sizeof(mdah->checksum_xl)));
|
|
|
|
dev_set_last_byte(dev, start_byte + MDA_HEADER_SIZE);
|
|
|
|
if (!dev_write_bytes(dev, start_byte, MDA_HEADER_SIZE, mdah)) {
|
|
log_error("Failed to write mda header to %s fd %d", dev_name(dev), dev->bcache_fd);
|
|
return 0;
|
|
}
|
|
dev_unset_last_byte(dev);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* FIXME: unify this with read_metadata_location() which is used
|
|
* in the label scanning path.
|
|
*/
|
|
|
|
static struct raw_locn *_read_metadata_location_vg(struct device_area *dev_area,
|
|
struct mda_header *mdah, int primary_mda,
|
|
const char *vgname,
|
|
int *precommitted)
|
|
{
|
|
size_t len;
|
|
char vgnamebuf[NAME_LEN + 2] __attribute__((aligned(8)));
|
|
struct raw_locn *rlocn, *rlocn_precommitted;
|
|
struct lvmcache_info *info;
|
|
struct lvmcache_vgsummary vgsummary_orphan = {
|
|
.vgname = FMT_TEXT_ORPHAN_VG_NAME,
|
|
};
|
|
int rlocn_was_ignored;
|
|
|
|
dm_list_init(&vgsummary_orphan.pvsummaries);
|
|
|
|
memcpy(&vgsummary_orphan.vgid, FMT_TEXT_ORPHAN_VG_NAME, sizeof(FMT_TEXT_ORPHAN_VG_NAME));
|
|
|
|
rlocn = mdah->raw_locns; /* Slot 0 */
|
|
rlocn_precommitted = rlocn + 1; /* Slot 1 */
|
|
|
|
rlocn_was_ignored = rlocn_is_ignored(rlocn);
|
|
|
|
/* Should we use precommitted metadata? */
|
|
if (*precommitted && rlocn_precommitted->size &&
|
|
(rlocn_precommitted->offset != rlocn->offset)) {
|
|
rlocn = rlocn_precommitted;
|
|
log_debug_metadata("VG %s metadata check %s mda %llu slot1 offset %llu size %llu",
|
|
vgname ?: "",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)dev_area->start,
|
|
(unsigned long long)rlocn->offset,
|
|
(unsigned long long)rlocn->size);
|
|
} else {
|
|
*precommitted = 0;
|
|
log_debug_metadata("VG %s metadata check %s mda %llu slot0 offset %llu size %llu",
|
|
vgname ?: "",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)dev_area->start,
|
|
(unsigned long long)rlocn->offset,
|
|
(unsigned long long)rlocn->size);
|
|
}
|
|
|
|
/* Do not check non-existent metadata. */
|
|
if (!rlocn->offset && !rlocn->size)
|
|
return NULL;
|
|
|
|
/*
|
|
* Don't try to check existing metadata
|
|
* if given vgname is an empty string.
|
|
*/
|
|
if (!vgname || !*vgname)
|
|
return rlocn;
|
|
|
|
/*
|
|
* If live rlocn has ignored flag, data will be out-of-date so skip further checks.
|
|
*/
|
|
if (rlocn_was_ignored)
|
|
return rlocn;
|
|
|
|
/*
|
|
* Verify that the VG metadata pointed to by the rlocn
|
|
* begins with a valid vgname.
|
|
*/
|
|
memset(vgnamebuf, 0, sizeof(vgnamebuf));
|
|
|
|
if (!dev_read_bytes(dev_area->dev, dev_area->start + rlocn->offset, NAME_LEN, vgnamebuf))
|
|
goto fail;
|
|
|
|
if (!strncmp(vgnamebuf, vgname, len = strlen(vgname)) &&
|
|
(isspace(vgnamebuf[len]) || vgnamebuf[len] == '{'))
|
|
return rlocn;
|
|
fail:
|
|
log_error("Metadata on %s at %llu has wrong VG name \"%s\" expected %s.",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset),
|
|
vgnamebuf, vgname);
|
|
|
|
if ((info = lvmcache_info_from_pvid(dev_area->dev->pvid, dev_area->dev, 0)) &&
|
|
!lvmcache_update_vgname_and_id(info, &vgsummary_orphan))
|
|
stack;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Determine offset for new metadata
|
|
*
|
|
* The rounding can have a negative effect: when the current metadata
|
|
* text size is just below the max, a command to remove something, that
|
|
* *reduces* the text metadata size, can still be rejected for being too large,
|
|
* even though it's smaller than the current size. In this case, the user
|
|
* would need to find something in the VG to remove that uses more text space
|
|
* to compensate for the increase due to rounding.
|
|
* Update: I think that the new max_size restriction avoids this problem.
|
|
*/
|
|
|
|
static uint64_t _next_rlocn_offset(struct volume_group *vg, struct raw_locn *rlocn_old, uint64_t old_last, struct mda_header *mdah, uint64_t mdac_area_start, uint64_t alignment)
|
|
{
|
|
uint64_t next_start;
|
|
uint64_t new_start;
|
|
uint64_t adjust = 0;
|
|
|
|
/* This has only been designed to work with 512. */
|
|
if (alignment != 512)
|
|
log_warn("WARNING: metadata alignment should be 512 not %llu",
|
|
(unsigned long long)alignment);
|
|
|
|
/*
|
|
* No metadata has been written yet, begin at MDA_HEADER_SIZE offset
|
|
* from the start of the area.
|
|
*/
|
|
if (!rlocn_old)
|
|
return MDA_HEADER_SIZE;
|
|
|
|
/*
|
|
* If new start would be less than alignment bytes from the end of the
|
|
* metadata area, then start at beginning.
|
|
*/
|
|
if (mdah->size - old_last < alignment) {
|
|
log_debug_metadata("VG %s %u new metadata start align from %llu to beginning %u",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)(old_last + 1), MDA_HEADER_SIZE);
|
|
return MDA_HEADER_SIZE;
|
|
}
|
|
|
|
/*
|
|
* New metadata begins after the old, rounded up to alignment.
|
|
*/
|
|
|
|
next_start = old_last + 1;
|
|
|
|
if (next_start % alignment)
|
|
adjust = alignment - (next_start % alignment);
|
|
|
|
new_start = next_start + adjust;
|
|
|
|
log_debug_metadata("VG %s %u new metadata start align from %llu to %llu (+%llu)",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)next_start,
|
|
(unsigned long long)new_start,
|
|
(unsigned long long)adjust);
|
|
|
|
/*
|
|
* If new_start is beyond the end of the metadata area or within
|
|
* alignment bytes of the end, then start at the beginning.
|
|
*/
|
|
if (new_start > mdah->size - alignment) {
|
|
log_debug_metadata("VG %s %u new metadata start align from %llu to beginning %u",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)new_start, MDA_HEADER_SIZE);
|
|
return MDA_HEADER_SIZE;
|
|
}
|
|
|
|
return new_start;
|
|
}
|
|
|
|
static struct volume_group *_vg_read_raw_area(struct format_instance *fid,
|
|
const char *vgname,
|
|
struct device_area *area,
|
|
struct cached_vg_fmtdata **vg_fmtdata,
|
|
unsigned *use_previous_vg,
|
|
int precommitted,
|
|
int primary_mda)
|
|
{
|
|
struct volume_group *vg = NULL;
|
|
struct raw_locn *rlocn;
|
|
struct mda_header *mdah;
|
|
time_t when;
|
|
char *desc;
|
|
uint32_t wrap = 0;
|
|
uint32_t bad_fields = 0;
|
|
|
|
if (!(mdah = raw_read_mda_header(fid->fmt, area, primary_mda, 0, &bad_fields))) {
|
|
log_error("Failed to read vg %s from %s", vgname, dev_name(area->dev));
|
|
goto out;
|
|
}
|
|
|
|
if (!(rlocn = _read_metadata_location_vg(area, mdah, primary_mda, vgname, &precommitted))) {
|
|
log_debug_metadata("VG %s not found on %s", vgname, dev_name(area->dev));
|
|
goto out;
|
|
}
|
|
|
|
if (rlocn->offset + rlocn->size > mdah->size)
|
|
wrap = (uint32_t) ((rlocn->offset + rlocn->size) - mdah->size);
|
|
|
|
vg = text_read_metadata(fid, NULL, vg_fmtdata, use_previous_vg, area->dev, primary_mda,
|
|
(off_t) (area->start + rlocn->offset),
|
|
(uint32_t) (rlocn->size - wrap),
|
|
(off_t) (area->start + MDA_HEADER_SIZE),
|
|
wrap,
|
|
calc_crc,
|
|
rlocn->checksum,
|
|
&when, &desc);
|
|
|
|
if (!vg) {
|
|
/* FIXME: detect and handle errors, and distinguish from the optimization
|
|
that skips parsing the metadata which also returns NULL. */
|
|
}
|
|
|
|
log_debug_metadata("Found metadata on %s at %llu size %llu for VG %s",
|
|
dev_name(area->dev),
|
|
(unsigned long long)(area->start + rlocn->offset),
|
|
(unsigned long long)rlocn->size,
|
|
vgname);
|
|
|
|
if (vg && precommitted)
|
|
vg->status |= PRECOMMITTED;
|
|
|
|
out:
|
|
return vg;
|
|
}
|
|
|
|
static struct volume_group *_vg_read_raw(struct format_instance *fid,
|
|
const char *vgname,
|
|
struct metadata_area *mda,
|
|
struct cached_vg_fmtdata **vg_fmtdata,
|
|
unsigned *use_previous_vg)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct volume_group *vg;
|
|
|
|
vg = _vg_read_raw_area(fid, vgname, &mdac->area, vg_fmtdata, use_previous_vg, 0, mda_is_primary(mda));
|
|
|
|
return vg;
|
|
}
|
|
|
|
static struct volume_group *_vg_read_precommit_raw(struct format_instance *fid,
|
|
const char *vgname,
|
|
struct metadata_area *mda,
|
|
struct cached_vg_fmtdata **vg_fmtdata,
|
|
unsigned *use_previous_vg)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct volume_group *vg;
|
|
|
|
vg = _vg_read_raw_area(fid, vgname, &mdac->area, vg_fmtdata, use_previous_vg, 1, mda_is_primary(mda));
|
|
|
|
return vg;
|
|
}
|
|
|
|
/*
|
|
* VG metadata updates:
|
|
*
|
|
* [mda_header] [raw_locn_0] [raw_locn_1] [text metadata circular buffer]
|
|
*
|
|
* raw_locn.offset points into the metadata circular buffer to the
|
|
* start of metadata.
|
|
*
|
|
* When vg_read wants to read metadata from disk, it looks at the
|
|
* raw_locn_0 offset and reads the text metadata from that location
|
|
* in the circular buffer.
|
|
*
|
|
* Two full copies of the text metadata always exist in the circular
|
|
* buffer. When new metadata needs to be written, the following
|
|
* process is followed:
|
|
*
|
|
* - vg_write is called and writes the new text metadata into the
|
|
* circular buffer after the end of the current copy. vg_write saves
|
|
* an in-memory raw_locn struct (mdac->rlocn) pointing to the new
|
|
* metadata in the buffer. No raw_locn structs are written to disk.
|
|
*
|
|
* - vg_precommit is called and writes the in-memory raw_locn struct that
|
|
* was saved by vg_write into raw_locn_1 (slot 1, the "precommit" slot.)
|
|
* raw_locn_0 still points to the old metadata, and raw_locn_1 points
|
|
* to the new metadata.
|
|
*
|
|
* - vg_commit is called and writes the new raw_locn struct into raw_locn_0
|
|
* (slot 0, the "committed" slot).
|
|
*/
|
|
|
|
/*
|
|
* Writes new text metadata into the circular metadata buffer following the
|
|
* current (old) text metadata that's already in the metadata buffer.
|
|
*
|
|
* vg_write does *not* write new raw_locn fields pointing to the new metadata.
|
|
* The new raw_locn fields for the new metadata are saved in mdac->rlocn and
|
|
* are written later by both vg_precommit and vg_commit. vg_precommit will
|
|
* write the new raw_locn into slot 1 and vg_commit will write the new raw_locn
|
|
* into slot 0.
|
|
*/
|
|
|
|
static int _vg_write_raw(struct format_instance *fid, struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
char desc[2048];
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct text_fid_context *fidtc = (struct text_fid_context *) fid->private;
|
|
struct raw_locn *rlocn_old;
|
|
struct raw_locn *rlocn_new;
|
|
struct mda_header *mdah;
|
|
struct pv_list *pvl;
|
|
uint64_t mda_start = mdac->area.start;
|
|
uint64_t max_size;
|
|
uint64_t old_start = 0, old_last = 0, old_size = 0, old_wrap = 0;
|
|
uint64_t new_start = 0, new_last = 0, new_size = 0, new_wrap = 0;
|
|
uint64_t write1_start = 0, write1_last = 0, write1_size = 0;
|
|
uint64_t write2_start = 0, write2_last = 0, write2_size = 0;
|
|
uint32_t write1_over = 0, write2_over = 0;
|
|
uint32_t write_buf_size;
|
|
uint32_t extra_size;
|
|
uint32_t bad_fields = 0;
|
|
char *write_buf = NULL;
|
|
const char *devname = dev_name(mdac->area.dev);
|
|
bool overlap;
|
|
int found = 0;
|
|
int r = 0;
|
|
|
|
/*
|
|
* old_start/old_last/new_start/new_last are relative to the
|
|
* start of the metadata area (mda_start), and specify the first
|
|
* and last bytes of old/new metadata copies in the metadata area.
|
|
*
|
|
* write1_start/write1_last/write2_start/write2_last are
|
|
* relative to the start of the disk, and specify the
|
|
* first/last bytes written to disk when writing a new
|
|
* copy of metadata. (Will generally be larger than the
|
|
* size of the metadata since the write is extended past
|
|
* the end of the new metadata to end on a 512 byte boundary.)
|
|
*
|
|
* So, write1_start == mda_start + new_start.
|
|
*
|
|
* "last" values are inclusive, so last - start + 1 = size.
|
|
* old_last/new_last are the last bytes containing metadata.
|
|
* write1_last/write2_last are the last bytes written.
|
|
* The next copy of metadata will be written beginning at
|
|
* write1_last+1.
|
|
*/
|
|
|
|
/* Ignore any mda on a PV outside the VG. vgsplit relies on this */
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (pvl->pv->dev == mdac->area.dev) {
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
return 1;
|
|
|
|
if (!(mdah = raw_read_mda_header(fid->fmt, &mdac->area, mda_is_primary(mda), mda->ignore_bad_fields, &bad_fields)))
|
|
goto_out;
|
|
|
|
/*
|
|
* Create a text metadata representation of struct vg in buffer.
|
|
* This buffer is written to disk below. This function is called
|
|
* to write metadata to each device/mda in the VG. The first time
|
|
* the metadata text is saved in write_buf and subsequent
|
|
* mdas use that.
|
|
*
|
|
* write_buf_size is increased in 64K increments, so will generally
|
|
* be larger than new_size. The extra space in write_buf (after
|
|
* new_size) is zeroed. More than new_size can be written from
|
|
* write_buf to zero data on disk following the new text metadata,
|
|
* up to the next 512 byte boundary.
|
|
*/
|
|
if (fidtc->write_buf) {
|
|
write_buf = fidtc->write_buf;
|
|
write_buf_size = fidtc->write_buf_size;
|
|
new_size = fidtc->new_metadata_size;
|
|
} else {
|
|
if (!vg->write_count++)
|
|
(void) dm_snprintf(desc, sizeof(desc), "Write from %s.", vg->cmd->cmd_line);
|
|
else
|
|
(void) dm_snprintf(desc, sizeof(desc), "Write[%u] from %s.", vg->write_count, vg->cmd->cmd_line);
|
|
|
|
new_size = text_vg_export_raw(vg, desc, &write_buf, &write_buf_size);
|
|
fidtc->write_buf = write_buf;
|
|
fidtc->write_buf_size = write_buf_size;
|
|
fidtc->new_metadata_size = new_size;
|
|
}
|
|
|
|
if (!new_size || !write_buf) {
|
|
log_error("VG %s metadata writing failed", vg->name);
|
|
goto out;
|
|
}
|
|
|
|
log_debug_metadata("VG %s seqno %u metadata write to %s mda_start %llu mda_size %llu mda_last %llu",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)mda_start,
|
|
(unsigned long long)mdah->size,
|
|
(unsigned long long)(mda_start + mdah->size - 1));
|
|
|
|
/*
|
|
* The max size of a single copy of text metadata.
|
|
*
|
|
* The space available for all text metadata is the size of the
|
|
* metadata area (mdah->size) minus the sector used for the header.
|
|
* Two copies of the text metadata must fit in this space, so it is
|
|
* divided in two. This result is then reduced by 512 because any
|
|
* single copy of metadata is rounded to begin on a sector boundary.
|
|
*/
|
|
|
|
max_size = ((mdah->size - MDA_HEADER_SIZE) / 2) - 512;
|
|
|
|
if (new_size > max_size) {
|
|
log_error("VG %s %u metadata on %s (%llu bytes) exceeds maximum metadata size (%llu bytes)",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)new_size,
|
|
(unsigned long long)max_size);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* rlocn_old is the current, committed, raw_locn data in slot0 on disk.
|
|
*
|
|
* rlocn_new (mdac->rlocn) is the new, in-memory, raw_locn data for the
|
|
* new metadata. rlocn_new is in-memory only, not yet written to disk.
|
|
*
|
|
* rlocn_new is not written to disk by vg_write. vg_write only writes
|
|
* the new text metadata into the circular buffer, it does not update any
|
|
* raw_locn slot to point to that new metadata. vg_write saves raw_locn
|
|
* values for the new metadata in memory at mdac->rlocn so that
|
|
* vg_precommit and vg_commit can find it later and write it to disk.
|
|
*
|
|
* rlocn/raw_locn values, old_start, old_last, old_size, new_start,
|
|
* new_last, new_size, are all in bytes, and are all relative to the
|
|
* the start of the metadata area (not to the start of the disk.)
|
|
*
|
|
* The start and last values are the first and last bytes that hold
|
|
* the metadata inclusively, e.g.
|
|
* metadata_v1 start = 512, last = 611, size = 100
|
|
* metadata_v2 start = 612, last = 711, size = 100
|
|
*
|
|
* {old,new}_{start,last} values are all offset values from the
|
|
* beginning of the metadata area mdac->area.start. At the beginning
|
|
* of the metadata area (area.start), the first 512 bytes
|
|
* (MDA_HEADER_SIZE) is reserved for the mda_header/raw_locn structs,
|
|
* after which the circular buffer of text metadata begins.
|
|
* So, the when the text metadata wraps around, it starts again at
|
|
* area.start + MDA_HEADER_SIZE.
|
|
*
|
|
* When pe_start is at 1MB (the default), and mda_start is at 4KB,
|
|
* there will be 1MB - 4KB - 512 bytes of circular buffer space for
|
|
* text metadata.
|
|
*/
|
|
|
|
rlocn_old = &mdah->raw_locns[0]; /* slot0, committed metadata */
|
|
|
|
if (rlocn_is_ignored(rlocn_old))
|
|
rlocn_old = NULL;
|
|
|
|
else if (!rlocn_old->offset && !rlocn_old->size)
|
|
rlocn_old = NULL;
|
|
|
|
else {
|
|
old_start = rlocn_old->offset;
|
|
old_size = rlocn_old->size;
|
|
|
|
if (rlocn_old->offset + rlocn_old->size > mdah->size) {
|
|
old_wrap = (old_start + old_size) - mdah->size;
|
|
old_last = old_wrap + MDA_HEADER_SIZE - 1;
|
|
} else {
|
|
old_wrap = 0;
|
|
old_last = old_start + old_size - 1;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* _next_rlocn_offset returns the new offset to use for the new
|
|
* metadata. It is set to follow the end of the old metadata, plus
|
|
* some adjustment to start the new metadata on a 512 byte alignment.
|
|
* If the new metadata would start beyond the end of the metadata area,
|
|
* or would start less than 512 bytes before the end of the metadata
|
|
* area, then the new start is set back at the beginning
|
|
* (metadata begins MDA_HEADER_SIZE after start of metadata area).
|
|
*/
|
|
new_start = _next_rlocn_offset(vg, rlocn_old, old_last, mdah, mda_start, MDA_ORIGINAL_ALIGNMENT);
|
|
|
|
if (new_start + new_size > mdah->size) {
|
|
new_wrap = (new_start + new_size) - mdah->size;
|
|
new_last = new_wrap + MDA_HEADER_SIZE - 1;
|
|
|
|
log_debug_metadata("VG %s %u wrapping metadata new_start %llu new_size %llu to size1 %llu size2 %llu",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)new_start,
|
|
(unsigned long long)new_size,
|
|
(unsigned long long)(new_size - new_wrap),
|
|
(unsigned long long)new_wrap);
|
|
} else {
|
|
new_wrap = 0;
|
|
new_last = new_start + new_size - 1;
|
|
}
|
|
|
|
/*
|
|
* Save the new metadata location in memory for vg_precommit and
|
|
* vg_commit. The new location is not written to disk here.
|
|
*/
|
|
rlocn_new = &mdac->rlocn;
|
|
rlocn_new->offset = new_start;
|
|
rlocn_new->size = new_size;
|
|
|
|
log_debug_metadata("VG %s %u metadata area location old start %llu last %llu size %llu wrap %llu",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)old_start,
|
|
(unsigned long long)old_last,
|
|
(unsigned long long)old_size,
|
|
(unsigned long long)old_wrap);
|
|
|
|
log_debug_metadata("VG %s %u metadata area location new start %llu last %llu size %llu wrap %llu",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)new_start,
|
|
(unsigned long long)new_last,
|
|
(unsigned long long)new_size,
|
|
(unsigned long long)new_wrap);
|
|
|
|
/*
|
|
* If the new copy of the metadata would overlap the old copy of the
|
|
* metadata, it means that the circular metadata buffer is full.
|
|
*
|
|
* Given the max_size restriction above, two copies of metadata should
|
|
* never overlap, so these overlap checks should not be technically
|
|
* necessary, and a failure should not occur here. It's left as a
|
|
* sanity check. For some unknown time, lvm did not enforce a
|
|
* max_size, but rather detected the too-large failure by checking for
|
|
* overlap between old and new.
|
|
*/
|
|
|
|
if (new_wrap && old_wrap) {
|
|
|
|
/* old and new can't both wrap without overlapping */
|
|
overlap = true;
|
|
|
|
} else if (!new_wrap && !old_wrap &&
|
|
(new_start > old_last) && (new_last > new_start)) {
|
|
|
|
/* new metadata is located entirely after the old metadata */
|
|
overlap = false;
|
|
|
|
} else if (!new_wrap && !old_wrap &&
|
|
(new_start < old_start) && (new_last < old_start)) {
|
|
|
|
/* new metadata is located entirely before the old metadata */
|
|
overlap = false;
|
|
|
|
} else if (old_wrap && !new_wrap &&
|
|
(old_last < new_start) && (new_start < new_last) && (new_last < old_start)) {
|
|
|
|
/* when old wraps and the new doesn't, then no overlap is:
|
|
old_last followed by new_start followed by new_last
|
|
followed by old_start */
|
|
overlap = false;
|
|
|
|
} else if (new_wrap && !old_wrap &&
|
|
(new_last < old_start) && (old_start < old_last) && (old_last < new_start)) {
|
|
|
|
/* when new wraps and the old doesn't, then no overlap is:
|
|
new_last followed by old_start followed by old_last
|
|
followed by new_start. */
|
|
overlap = false;
|
|
|
|
} else {
|
|
overlap = true;
|
|
}
|
|
|
|
if (overlap) {
|
|
log_error("VG %s %u metadata on %s (%llu bytes) too large for circular buffer (%llu bytes with %llu used)",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)new_size,
|
|
(unsigned long long)(mdah->size - MDA_HEADER_SIZE),
|
|
(unsigned long long)old_size);
|
|
goto out;
|
|
}
|
|
|
|
if (!new_wrap) {
|
|
write1_start = mda_start + new_start;
|
|
write1_size = new_size;
|
|
write1_last = write1_start + write1_size - 1;
|
|
write1_over = (write1_last + 1) % 512;
|
|
write2_start = 0;
|
|
write2_size = 0;
|
|
write2_last = 0;
|
|
write2_over = 0;
|
|
} else {
|
|
write1_start = mda_start + new_start;
|
|
write1_size = new_size - new_wrap;
|
|
write1_last = write1_start + write1_size - 1;
|
|
write1_over = 0;
|
|
write2_start = mda_start + MDA_HEADER_SIZE;
|
|
write2_size = new_wrap;
|
|
write2_last = write2_start + write2_size - 1;
|
|
write2_over = (write2_last + 1) % 512;
|
|
}
|
|
|
|
if (!new_wrap)
|
|
log_debug_metadata("VG %s %u metadata disk location start %llu size %llu last %llu",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write1_start,
|
|
(unsigned long long)write1_size,
|
|
(unsigned long long)write1_last);
|
|
else
|
|
log_debug_metadata("VG %s %u metadata disk location write1 start %llu size %llu last %llu write2 start %llu size %llu last %llu",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write1_start,
|
|
(unsigned long long)write1_size,
|
|
(unsigned long long)write1_last,
|
|
(unsigned long long)write2_start,
|
|
(unsigned long long)write2_size,
|
|
(unsigned long long)write2_last);
|
|
|
|
/*
|
|
* Write more than the size of the new metadata, up to the next
|
|
* 512 byte boundary so that the space between this copy and the
|
|
* subsequent copy of metadata will be zeroed.
|
|
*
|
|
* Extend write1_size so that write1_last+1 is a 512 byte multiple.
|
|
* The next metadata write should follow immediately after the
|
|
* extended write1_last since new metadata tries to begin on a 512
|
|
* byte boundary.
|
|
*
|
|
* write1_size can be extended up to write_buf_size which is the size
|
|
* of write_buf (new_size is the portion of write_buf used by the new
|
|
* metadata.)
|
|
*
|
|
* If this metadata write will wrap, the first write is written
|
|
* all the way to the end of the metadata area, and it's the
|
|
* second wrapped write that is extended up to a 512 byte boundary.
|
|
*/
|
|
|
|
if (write1_over) {
|
|
extra_size = 512 - write1_over; /* this many extra zero bytes written after metadata text */
|
|
write1_size += extra_size;
|
|
write1_last = write1_start + write1_size - 1;
|
|
|
|
log_debug_metadata("VG %s %u metadata last align from %llu to %llu (+%u)",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write1_last - extra_size,
|
|
(unsigned long long)write1_last, extra_size);
|
|
|
|
if (write1_size > write_buf_size) {
|
|
/* sanity check, shouldn't happen */
|
|
log_error("VG %s %u %s adjusted metadata end %llu extra %u larger than write buffer %llu",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)write1_size, extra_size,
|
|
(unsigned long long)write_buf_size);
|
|
write1_size -= extra_size;
|
|
}
|
|
}
|
|
|
|
if (write2_over) {
|
|
extra_size = 512 - write2_over; /* this many extra zero bytes written after metadata text */
|
|
write2_size += extra_size;
|
|
write2_last = write2_start + write2_size - 1;
|
|
|
|
log_debug_metadata("VG %s %u metadata last align from %llu to %llu (+%u) (wrapped)",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write2_last - extra_size,
|
|
(unsigned long long)write2_last, extra_size);
|
|
|
|
if (write1_size + write2_size > write_buf_size) {
|
|
/* sanity check, shouldn't happen */
|
|
log_error("VG %s %u %s adjusted metadata end %llu wrap %llu extra %u larger than write buffer %llu",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)write1_size,
|
|
(unsigned long long)write2_size, extra_size,
|
|
(unsigned long long)write_buf_size);
|
|
write2_size -= extra_size;
|
|
}
|
|
}
|
|
|
|
if ((write1_size > write_buf_size) || (write2_size > write_buf_size)) {
|
|
/* sanity check, shouldn't happen */
|
|
log_error("VG %s %u %s metadata write size %llu %llu larger than buffer %llu",
|
|
vg->name, vg->seqno, devname,
|
|
(unsigned long long)write1_size,
|
|
(unsigned long long)write2_size,
|
|
(unsigned long long)write_buf_size);
|
|
goto out;
|
|
}
|
|
|
|
dev_set_last_byte(mdac->area.dev, mda_start + mdah->size);
|
|
|
|
log_debug_metadata("VG %s %u metadata write at %llu size %llu (wrap %llu)",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write1_start,
|
|
(unsigned long long)write1_size,
|
|
(unsigned long long)write2_size);
|
|
|
|
if (!dev_write_bytes(mdac->area.dev, write1_start, (size_t)write1_size, write_buf)) {
|
|
log_error("Failed to write metadata to %s fd %d", devname, mdac->area.dev->bcache_fd);
|
|
goto out;
|
|
}
|
|
|
|
if (write2_size) {
|
|
log_debug_metadata("VG %s %u metadata write at %llu size %llu (wrapped)",
|
|
vg->name, vg->seqno,
|
|
(unsigned long long)write2_start,
|
|
(unsigned long long)write2_size);
|
|
|
|
if (!dev_write_bytes(mdac->area.dev, write2_start, write2_size,
|
|
write_buf + new_size - new_wrap)) {
|
|
log_error("Failed to write metadata wrap to %s fd %d", devname, mdac->area.dev->bcache_fd);
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
dev_unset_last_byte(mdac->area.dev);
|
|
|
|
rlocn_new->checksum = calc_crc(INITIAL_CRC,
|
|
(uint8_t *)write_buf,
|
|
(uint32_t)(new_size - new_wrap));
|
|
if (new_wrap)
|
|
rlocn_new->checksum = calc_crc(rlocn_new->checksum,
|
|
(uint8_t *)write_buf + new_size - new_wrap,
|
|
(uint32_t)new_wrap);
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
if (!r) {
|
|
free(fidtc->write_buf);
|
|
fidtc->write_buf = NULL;
|
|
fidtc->write_buf_size = 0;
|
|
fidtc->new_metadata_size = 0;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
/*
|
|
* Writes new raw_locn to disk that was saved by vg_write_raw (in mdac->rlocn).
|
|
* The new raw_locn points to the new metadata that was written by vg_write_raw.
|
|
*
|
|
* After vg_write writes the new text metadata into the circular buffer,
|
|
* vg_precommit writes the new raw_locn (pointing to the new metadata)
|
|
* into slot1 (raw_locns[1]). Then vg_commit writes the same raw_locn
|
|
* values again, but into slot0 (raw_locns[0]). slot0 is the committed
|
|
* slot, and once slot0 is written, subsequent vg_reads will see the new
|
|
* metadata.
|
|
*/
|
|
|
|
static int _vg_commit_raw_rlocn(struct format_instance *fid,
|
|
struct volume_group *vg,
|
|
struct metadata_area *mda,
|
|
int precommit)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct text_fid_context *fidtc = (struct text_fid_context *) fid->private;
|
|
struct mda_header *mdab;
|
|
struct raw_locn *rlocn_slot0;
|
|
struct raw_locn *rlocn_slot1;
|
|
struct raw_locn *rlocn_new;
|
|
struct pv_list *pvl;
|
|
uint32_t bad_fields = 0;
|
|
int r = 0;
|
|
int found = 0;
|
|
|
|
/* Ignore any mda on a PV outside the VG. vgsplit relies on this */
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (pvl->pv->dev == mdac->area.dev) {
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
return 1;
|
|
|
|
/*
|
|
* Data is read into the mdab buffer, the mdab buffer is then modified
|
|
* with new raw_locn values, then the mdab buffer is written. Note
|
|
* this is different than _vg_write_raw, where data is read into the
|
|
* mdah buffer, but the mdah buffer is not modified and mdac->rlocn is
|
|
* modified.
|
|
*/
|
|
if (!(mdab = raw_read_mda_header(fid->fmt, &mdac->area, mda_is_primary(mda), mda->ignore_bad_fields, &bad_fields)))
|
|
goto_out;
|
|
|
|
/*
|
|
* rlocn_slot0/rlocn_slot1 point into mdab which is the buffer that
|
|
* will be modified and written.
|
|
*/
|
|
rlocn_slot0 = &mdab->raw_locns[0];
|
|
rlocn_slot1 = &mdab->raw_locns[1];
|
|
|
|
if (rlocn_is_ignored(rlocn_slot0) || (!rlocn_slot0->offset && !rlocn_slot0->size)) {
|
|
rlocn_slot0->offset = 0;
|
|
rlocn_slot0->size = 0;
|
|
rlocn_slot0->checksum = 0;
|
|
rlocn_slot1->offset = 0;
|
|
rlocn_slot1->size = 0;
|
|
rlocn_slot1->checksum = 0;
|
|
}
|
|
|
|
/*
|
|
* mdac->rlocn is the in-memory copy of the new metadata's location on
|
|
* disk. mdac->rlocn was saved by vg_write after it wrote the new text
|
|
* metadata to disk. This location of the new metadata is now written
|
|
* to disk by vg_precommit and vg_commit. vg_precommit writes the new
|
|
* location into the precommit slot (slot1 / raw_locns[1]) and
|
|
* vg_commit writes the new location into committed slot (slot0 /
|
|
* raw_locns[0]).
|
|
*
|
|
* vg_revert sets the size of the im-memory mdac->rlocn to 0 and calls
|
|
* this function to clear the precommit slot.
|
|
*/
|
|
|
|
rlocn_new = &mdac->rlocn;
|
|
|
|
if (!rlocn_new->size) {
|
|
/*
|
|
* When there is no new metadata, the precommit slot is
|
|
* cleared and the committed slot is left alone. (see revert)
|
|
*/
|
|
rlocn_slot1->offset = 0;
|
|
rlocn_slot1->size = 0;
|
|
rlocn_slot1->checksum = 0;
|
|
|
|
} else if (precommit) {
|
|
/*
|
|
* vg_precommit writes the new raw_locn into slot 1,
|
|
* and keeps the existing committed raw_locn in slot 0.
|
|
*/
|
|
rlocn_slot1->offset = rlocn_new->offset;
|
|
rlocn_slot1->size = rlocn_new->size;
|
|
rlocn_slot1->checksum = rlocn_new->checksum;
|
|
} else {
|
|
/*
|
|
* vg_commit writes the new raw_locn into slot 0,
|
|
* and zeros the precommitted values in slot 1.
|
|
*/
|
|
rlocn_slot0->offset = rlocn_new->offset;
|
|
rlocn_slot0->size = rlocn_new->size;
|
|
rlocn_slot0->checksum = rlocn_new->checksum;
|
|
|
|
rlocn_slot1->offset = 0;
|
|
rlocn_slot1->size = 0;
|
|
rlocn_slot1->checksum = 0;
|
|
}
|
|
|
|
rlocn_set_ignored(rlocn_slot0, mda_is_ignored(mda));
|
|
|
|
if (mdac->rlocn.size) {
|
|
if (precommit) {
|
|
log_debug_metadata("VG %s metadata precommit seq %u on %s mda header at %llu %s",
|
|
vg->name, vg->seqno, dev_name(mdac->area.dev),
|
|
(unsigned long long)mdac->area.start,
|
|
mda_is_ignored(mda) ? "(ignored)" : "(used)");
|
|
|
|
log_debug_metadata("VG %s metadata precommit slot0 offset %llu size %llu slot1 offset %llu size %llu",
|
|
vg->name,
|
|
(unsigned long long)mdab->raw_locns[0].offset,
|
|
(unsigned long long)mdab->raw_locns[0].size,
|
|
(unsigned long long)mdab->raw_locns[1].offset,
|
|
(unsigned long long)mdab->raw_locns[1].size);
|
|
|
|
} else {
|
|
log_debug_metadata("VG %s metadata commit seq %u on %s mda header at %llu %s",
|
|
vg->name, vg->seqno, dev_name(mdac->area.dev),
|
|
(unsigned long long)mdac->area.start,
|
|
mda_is_ignored(mda) ? "(ignored)" : "(used)");
|
|
|
|
log_debug_metadata("VG %s metadata commit slot0 offset %llu size %llu slot1 offset %llu size %llu",
|
|
vg->name,
|
|
(unsigned long long)mdab->raw_locns[0].offset,
|
|
(unsigned long long)mdab->raw_locns[0].size,
|
|
(unsigned long long)mdab->raw_locns[1].offset,
|
|
(unsigned long long)mdab->raw_locns[1].size);
|
|
}
|
|
} else {
|
|
if (precommit) {
|
|
log_debug_metadata("VG %s metadata precommit empty seq %u on %s mda header at %llu %s",
|
|
vg->name, vg->seqno, dev_name(mdac->area.dev),
|
|
(unsigned long long)mdac->area.start,
|
|
mda_is_ignored(mda) ? "(ignored)" : "(used)");
|
|
|
|
log_debug_metadata("VG %s metadata precommit empty slot0 offset %llu size %llu slot1 offset %llu size %llu",
|
|
vg->name,
|
|
(unsigned long long)mdab->raw_locns[0].offset,
|
|
(unsigned long long)mdab->raw_locns[0].size,
|
|
(unsigned long long)mdab->raw_locns[1].offset,
|
|
(unsigned long long)mdab->raw_locns[1].size);
|
|
|
|
} else {
|
|
log_debug_metadata("VG %s metadata commit empty seq %u on %s mda header at %llu %s",
|
|
vg->name, vg->seqno, dev_name(mdac->area.dev),
|
|
(unsigned long long)mdac->area.start,
|
|
mda_is_ignored(mda) ? "(ignored)" : "(used)");
|
|
|
|
log_debug_metadata("VG %s metadata commit empty slot0 offset %llu size %llu slot1 offset %llu size %llu",
|
|
vg->name,
|
|
(unsigned long long)mdab->raw_locns[0].offset,
|
|
(unsigned long long)mdab->raw_locns[0].size,
|
|
(unsigned long long)mdab->raw_locns[1].offset,
|
|
(unsigned long long)mdab->raw_locns[1].size);
|
|
}
|
|
}
|
|
|
|
rlocn_set_ignored(mdab->raw_locns, mda_is_ignored(mda));
|
|
|
|
if (!_raw_write_mda_header(fid->fmt, mdac->area.dev, mda_is_primary(mda), mdac->area.start,
|
|
mdab)) {
|
|
dm_pool_free(fid->fmt->cmd->mem, mdab);
|
|
log_error("Failed to write metadata area header");
|
|
goto out;
|
|
}
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
if (!precommit && !fidtc->preserve) {
|
|
free(fidtc->write_buf);
|
|
fidtc->write_buf = NULL;
|
|
fidtc->write_buf_size = 0;
|
|
fidtc->new_metadata_size = 0;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static int _vg_commit_raw(struct format_instance *fid, struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
return _vg_commit_raw_rlocn(fid, vg, mda, 0);
|
|
}
|
|
|
|
static int _vg_precommit_raw(struct format_instance *fid,
|
|
struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
return _vg_commit_raw_rlocn(fid, vg, mda, 1);
|
|
}
|
|
|
|
/* Close metadata area devices */
|
|
static int _vg_revert_raw(struct format_instance *fid, struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct pv_list *pvl;
|
|
int found = 0;
|
|
|
|
/* Ignore any mda on a PV outside the VG. vgsplit relies on this */
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (pvl->pv->dev == mdac->area.dev) {
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
return 1;
|
|
|
|
/* Wipe pre-committed metadata */
|
|
mdac->rlocn.size = 0;
|
|
return _vg_commit_raw_rlocn(fid, vg, mda, 0);
|
|
}
|
|
|
|
/*
|
|
* vg_remove clears the two raw_locn slots but leaves the circular metadata
|
|
* buffer alone.
|
|
*/
|
|
|
|
static int _vg_remove_raw(struct format_instance *fid, struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) mda->metadata_locn;
|
|
struct mda_header *mdah;
|
|
struct raw_locn *rlocn_slot0;
|
|
struct raw_locn *rlocn_slot1;
|
|
uint32_t bad_fields = 0;
|
|
int r = 0;
|
|
|
|
if (!(mdah = dm_pool_alloc(fid->fmt->cmd->mem, MDA_HEADER_SIZE))) {
|
|
log_error("struct mda_header allocation failed");
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* FIXME: what's the point of reading the mda_header and metadata,
|
|
* since we zero the rlocn fields whether we can read them or not.
|
|
* Just to print the warning?
|
|
*/
|
|
|
|
if (!_raw_read_mda_header(mdah, &mdac->area, mda_is_primary(mda), 0, &bad_fields))
|
|
log_warn("WARNING: Removing metadata location on %s with bad mda header.",
|
|
dev_name(mdac->area.dev));
|
|
|
|
rlocn_slot0 = &mdah->raw_locns[0];
|
|
rlocn_slot1 = &mdah->raw_locns[1];
|
|
|
|
rlocn_slot0->offset = 0;
|
|
rlocn_slot0->size = 0;
|
|
rlocn_slot0->checksum = 0;
|
|
rlocn_set_ignored(rlocn_slot0, mda_is_ignored(mda));
|
|
|
|
rlocn_slot1->offset = 0;
|
|
rlocn_slot1->size = 0;
|
|
rlocn_slot1->checksum = 0;
|
|
|
|
if (!_raw_write_mda_header(fid->fmt, mdac->area.dev, mda_is_primary(mda), mdac->area.start,
|
|
mdah)) {
|
|
dm_pool_free(fid->fmt->cmd->mem, mdah);
|
|
log_error("Failed to write metadata area header");
|
|
goto out;
|
|
}
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
return r;
|
|
}
|
|
|
|
static struct volume_group *_vg_read_file_name(struct format_instance *fid,
|
|
const char *vgname,
|
|
const char *read_path)
|
|
{
|
|
struct volume_group *vg;
|
|
time_t when;
|
|
char *desc;
|
|
|
|
if (!(vg = text_read_metadata_file(fid, read_path, &when, &desc))) {
|
|
log_error("Failed to read VG %s from %s", vgname, read_path);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Currently you can only have a single volume group per
|
|
* text file (this restriction may remain). We need to
|
|
* check that it contains the correct volume group.
|
|
*/
|
|
if (vgname && strcmp(vgname, vg->name)) {
|
|
fid->ref_count++; /* Preserve FID after vg release */
|
|
release_vg(vg);
|
|
log_error("'%s' does not contain volume group '%s'.",
|
|
read_path, vgname);
|
|
return NULL;
|
|
}
|
|
|
|
log_debug_metadata("Read volume group %s from %s", vg->name, read_path);
|
|
|
|
return vg;
|
|
}
|
|
|
|
static struct volume_group *_vg_read_file(struct format_instance *fid,
|
|
const char *vgname,
|
|
struct metadata_area *mda,
|
|
struct cached_vg_fmtdata **vg_fmtdata,
|
|
unsigned *use_previous_vg __attribute__((unused)))
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
|
|
return _vg_read_file_name(fid, vgname, tc->path_live);
|
|
}
|
|
|
|
static struct volume_group *_vg_read_precommit_file(struct format_instance *fid,
|
|
const char *vgname,
|
|
struct metadata_area *mda,
|
|
struct cached_vg_fmtdata **vg_fmtdata,
|
|
unsigned *use_previous_vg __attribute__((unused)))
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
struct volume_group *vg;
|
|
|
|
if ((vg = _vg_read_file_name(fid, vgname, tc->path_edit)))
|
|
vg->status |= PRECOMMITTED;
|
|
else
|
|
vg = _vg_read_file_name(fid, vgname, tc->path_live);
|
|
|
|
return vg;
|
|
}
|
|
|
|
static int _vg_write_file(struct format_instance *fid __attribute__((unused)),
|
|
struct volume_group *vg, struct metadata_area *mda)
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
|
|
FILE *fp;
|
|
int fd;
|
|
char *slash;
|
|
char temp_file[PATH_MAX], temp_dir[PATH_MAX];
|
|
|
|
slash = strrchr(tc->path_edit, '/');
|
|
|
|
if (slash == 0)
|
|
strcpy(temp_dir, ".");
|
|
else if (slash - tc->path_edit < PATH_MAX) {
|
|
(void) dm_strncpy(temp_dir, tc->path_edit,
|
|
(size_t) (slash - tc->path_edit + 1));
|
|
} else {
|
|
log_error("Text format failed to determine directory.");
|
|
return 0;
|
|
}
|
|
|
|
if (!create_temp_name(temp_dir, temp_file, sizeof(temp_file), &fd,
|
|
&vg->cmd->rand_seed)) {
|
|
log_error("Couldn't create temporary text file name.");
|
|
return 0;
|
|
}
|
|
|
|
if (!(fp = fdopen(fd, "w"))) {
|
|
log_sys_error("fdopen", temp_file);
|
|
if (close(fd))
|
|
log_sys_error("fclose", temp_file);
|
|
return 0;
|
|
}
|
|
|
|
log_debug_metadata("Writing %s metadata to %s", vg->name, temp_file);
|
|
|
|
if (!text_vg_export_file(vg, tc->desc, fp)) {
|
|
log_error("Failed to write metadata to %s.", temp_file);
|
|
if (fclose(fp))
|
|
log_sys_error("fclose", temp_file);
|
|
return 0;
|
|
}
|
|
|
|
if (fsync(fd) && (errno != EROFS) && (errno != EINVAL)) {
|
|
log_sys_error("fsync", tc->path_edit);
|
|
if (fclose(fp))
|
|
log_sys_error("fclose", tc->path_edit);
|
|
return 0;
|
|
}
|
|
|
|
if (lvm_fclose(fp, tc->path_edit))
|
|
return_0;
|
|
|
|
log_debug_metadata("Renaming %s to %s", temp_file, tc->path_edit);
|
|
if (rename(temp_file, tc->path_edit)) {
|
|
log_error("%s: rename to %s failed: %s", temp_file,
|
|
tc->path_edit, strerror(errno));
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _vg_commit_file_backup(struct format_instance *fid __attribute__((unused)),
|
|
struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
|
|
if (test_mode()) {
|
|
log_verbose("Test mode: Skipping committing %s metadata (%u)",
|
|
vg->name, vg->seqno);
|
|
if (unlink(tc->path_edit)) {
|
|
log_debug_metadata("Unlinking %s", tc->path_edit);
|
|
log_sys_error("unlink", tc->path_edit);
|
|
return 0;
|
|
}
|
|
} else {
|
|
log_debug_metadata("Committing file %s metadata (%u)", vg->name, vg->seqno);
|
|
log_debug_metadata("Renaming %s to %s", tc->path_edit, tc->path_live);
|
|
if (rename(tc->path_edit, tc->path_live)) {
|
|
log_error("%s: rename to %s failed: %s", tc->path_edit,
|
|
tc->path_live, strerror(errno));
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
sync_dir(tc->path_edit);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _vg_commit_file(struct format_instance *fid, struct volume_group *vg,
|
|
struct metadata_area *mda)
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
const char *slash;
|
|
char new_name[PATH_MAX];
|
|
size_t len;
|
|
|
|
if (!_vg_commit_file_backup(fid, vg, mda))
|
|
return 0;
|
|
|
|
/* vgrename? */
|
|
if ((slash = strrchr(tc->path_live, '/')))
|
|
slash = slash + 1;
|
|
else
|
|
slash = tc->path_live;
|
|
|
|
if (strcmp(slash, vg->name)) {
|
|
len = slash - tc->path_live;
|
|
if ((len + strlen(vg->name)) > (sizeof(new_name) - 1)) {
|
|
log_error("Renaming path %s is too long for VG %s.",
|
|
tc->path_live, vg->name);
|
|
return 0;
|
|
}
|
|
strncpy(new_name, tc->path_live, len);
|
|
strcpy(new_name + len, vg->name);
|
|
log_debug_metadata("Renaming %s to %s", tc->path_live, new_name);
|
|
if (test_mode())
|
|
log_verbose("Test mode: Skipping rename");
|
|
else {
|
|
if (rename(tc->path_live, new_name)) {
|
|
log_error("%s: rename to %s failed: %s",
|
|
tc->path_live, new_name,
|
|
strerror(errno));
|
|
sync_dir(new_name);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _vg_remove_file(struct format_instance *fid __attribute__((unused)),
|
|
struct volume_group *vg __attribute__((unused)),
|
|
struct metadata_area *mda)
|
|
{
|
|
struct text_context *tc = (struct text_context *) mda->metadata_locn;
|
|
|
|
if (path_exists(tc->path_edit) && unlink(tc->path_edit)) {
|
|
log_sys_error("unlink", tc->path_edit);
|
|
return 0;
|
|
}
|
|
|
|
if (path_exists(tc->path_live) && unlink(tc->path_live)) {
|
|
log_sys_error("unlink", tc->path_live);
|
|
return 0;
|
|
}
|
|
|
|
sync_dir(tc->path_live);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int read_metadata_location_summary(const struct format_type *fmt,
|
|
struct metadata_area *mda,
|
|
struct mda_header *mdah, int primary_mda, struct device_area *dev_area,
|
|
struct lvmcache_vgsummary *vgsummary, uint64_t *mda_free_sectors)
|
|
{
|
|
struct raw_locn *rlocn;
|
|
uint32_t wrap = 0;
|
|
unsigned int len = 0;
|
|
char namebuf[NAME_LEN + 1] __attribute__((aligned(8)));
|
|
uint64_t max_size;
|
|
|
|
if (!mdah) {
|
|
log_error(INTERNAL_ERROR "read_metadata_location_summary called with NULL pointer for mda_header");
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* For the case where the metadata area is unused, half is available.
|
|
*/
|
|
if (mda_free_sectors) {
|
|
max_size = ((mdah->size - MDA_HEADER_SIZE) / 2) - 512;
|
|
*mda_free_sectors = max_size >> SECTOR_SHIFT;
|
|
}
|
|
|
|
rlocn = mdah->raw_locns; /* slot0, committed metadata */
|
|
|
|
/*
|
|
* If no valid offset, do not try to search for vgname
|
|
*/
|
|
if (!rlocn->offset) {
|
|
log_debug_metadata("Metadata location on %s at %llu has offset 0.",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset));
|
|
vgsummary->zero_offset = 1;
|
|
return 0;
|
|
}
|
|
|
|
memset(namebuf, 0, sizeof(namebuf));
|
|
|
|
if (!dev_read_bytes(dev_area->dev, dev_area->start + rlocn->offset, NAME_LEN, namebuf))
|
|
stack;
|
|
|
|
while (namebuf[len] && !isspace(namebuf[len]) && namebuf[len] != '{' &&
|
|
len < (NAME_LEN - 1))
|
|
len++;
|
|
|
|
namebuf[len] = '\0';
|
|
|
|
/*
|
|
* Check that the text metadata in the circular buffer begins with a
|
|
* valid vg name.
|
|
*/
|
|
if (!validate_name(namebuf)) {
|
|
log_warn("WARNING: Metadata location on %s at %llu begins with invalid VG name.",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset));
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* This function is used to read the vg summary during label scan.
|
|
* Save the text start location and checksum during scan. After the VG
|
|
* lock is acquired in vg_read, we can reread the mda_header, and
|
|
* compare rlocn->offset,checksum to what was saved during scan. If
|
|
* unchanged, it means that the metadata was not changed between scan
|
|
* and the read.
|
|
*/
|
|
mda->scan_text_offset = rlocn->offset;
|
|
mda->scan_text_checksum = rlocn->checksum;
|
|
|
|
/*
|
|
* When the current metadata wraps around the end of the metadata area
|
|
* (so some is located at the end and some is located at the
|
|
* beginning), then "wrap" is the number of bytes that was written back
|
|
* at the beginning. The end of this wrapped metadata is located at an
|
|
* offset of wrap+MDA_HEADER_SIZE from area.start.
|
|
*/
|
|
if (rlocn->offset + rlocn->size > mdah->size)
|
|
wrap = (uint32_t) ((rlocn->offset + rlocn->size) - mdah->size);
|
|
|
|
/*
|
|
* Did we see this metadata before?
|
|
* Look in lvmcache to see if there is vg info matching
|
|
* the checksum/size that we see in the mda_header (rlocn)
|
|
* on this device. If so, then vgsummary->name is is set
|
|
* and controls if the "checksum_only" flag passed to
|
|
* text_read_metadata_summary() is 1 or 0.
|
|
*
|
|
* If checksum_only = 1, then text_read_metadata_summary()
|
|
* will read the metadata from this device, and run the
|
|
* checksum function on it. If the calculated checksum
|
|
* of the metadata matches the checksum in the mda_header,
|
|
* which also matches the checksum saved in vginfo from
|
|
* another device, then it skips parsing the metadata into
|
|
* a config tree, which saves considerable cpu time.
|
|
*
|
|
* (NB. there can be different VGs with different metadata
|
|
* and checksums, but with the same name.)
|
|
*
|
|
* FIXME: handle the case where mda_header checksum is bad
|
|
* but metadata checksum is good.
|
|
*/
|
|
|
|
/*
|
|
* If the checksum we compute of the metadata differs from
|
|
* the checksum from mda_header that we save here, then we
|
|
* ignore the device. FIXME: we need to classify a device
|
|
* with errors like this as defective.
|
|
*
|
|
* If the checksum from mda_header and computed from metadata
|
|
* does not match the checksum saved in lvmcache from a prev
|
|
* device, then we do not skip parsing/saving metadata from
|
|
* this dev. It's parsed, fields saved in vgsummary, which
|
|
* is passed into lvmcache (update_vgname_and_id), and
|
|
* there we'll see a checksum mismatch.
|
|
*/
|
|
vgsummary->mda_checksum = rlocn->checksum;
|
|
vgsummary->mda_size = rlocn->size;
|
|
|
|
/* Keep track of largest metadata size we find. */
|
|
lvmcache_save_metadata_size(rlocn->size);
|
|
|
|
lvmcache_lookup_mda(vgsummary);
|
|
|
|
if (!text_read_metadata_summary(fmt, dev_area->dev, MDA_CONTENT_REASON(primary_mda),
|
|
(off_t) (dev_area->start + rlocn->offset),
|
|
(uint32_t) (rlocn->size - wrap),
|
|
(off_t) (dev_area->start + MDA_HEADER_SIZE),
|
|
wrap, calc_crc, vgsummary->vgname ? 1 : 0,
|
|
vgsummary)) {
|
|
log_warn("WARNING: metadata on %s at %llu has invalid summary for VG.",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset));
|
|
return 0;
|
|
}
|
|
|
|
/* Ignore this entry if the characters aren't permissible */
|
|
if (!validate_name(vgsummary->vgname)) {
|
|
log_warn("WARNING: metadata on %s at %llu has invalid VG name.",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset));
|
|
return 0;
|
|
}
|
|
|
|
log_debug_metadata("Found metadata summary on %s at %llu size %llu for VG %s",
|
|
dev_name(dev_area->dev),
|
|
(unsigned long long)(dev_area->start + rlocn->offset),
|
|
(unsigned long long)rlocn->size,
|
|
vgsummary->vgname);
|
|
|
|
if (mda_free_sectors) {
|
|
/*
|
|
* Report remaining space given that a single copy of metadata
|
|
* can be as large as half the total metadata space, minus 512
|
|
* because each copy is rounded to begin on a sector boundary.
|
|
*/
|
|
max_size = ((mdah->size - MDA_HEADER_SIZE) / 2) - 512;
|
|
|
|
if (rlocn->size >= max_size)
|
|
*mda_free_sectors = UINT64_C(0);
|
|
else
|
|
*mda_free_sectors = (max_size - rlocn->size) >> SECTOR_SHIFT;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct _write_single_mda_baton {
|
|
const struct format_type *fmt;
|
|
struct physical_volume *pv;
|
|
};
|
|
|
|
static int _write_single_mda(struct metadata_area *mda, void *baton)
|
|
{
|
|
struct _write_single_mda_baton *p = baton;
|
|
struct mda_context *mdac;
|
|
|
|
char buf[MDA_HEADER_SIZE] __attribute__((aligned(8))) = { 0 };
|
|
struct mda_header *mdah = (struct mda_header *) buf;
|
|
|
|
mdac = mda->metadata_locn;
|
|
mdah->size = mdac->area.size;
|
|
rlocn_set_ignored(mdah->raw_locns, mda_is_ignored(mda));
|
|
|
|
if (!_raw_write_mda_header(p->fmt, mdac->area.dev, mda_is_primary(mda),
|
|
mdac->area.start, mdah)) {
|
|
return_0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int _set_ext_flags(struct physical_volume *pv, struct lvmcache_info *info)
|
|
{
|
|
uint32_t ext_flags = lvmcache_ext_flags(info);
|
|
|
|
if (is_orphan(pv))
|
|
ext_flags &= ~PV_EXT_USED;
|
|
else
|
|
ext_flags |= PV_EXT_USED;
|
|
|
|
lvmcache_set_ext_version(info, PV_HEADER_EXTENSION_VSN);
|
|
lvmcache_set_ext_flags(info, ext_flags);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Only for orphans - FIXME That's not true any more */
|
|
static int _text_pv_write(const struct format_type *fmt, struct physical_volume *pv)
|
|
{
|
|
struct format_instance *fid = pv->fid;
|
|
const char *pvid = (const char *) (*pv->old_id.uuid ? &pv->old_id : &pv->id);
|
|
struct label *label;
|
|
struct lvmcache_info *info;
|
|
struct mda_context *mdac;
|
|
struct metadata_area *mda;
|
|
struct _write_single_mda_baton baton;
|
|
unsigned mda_index;
|
|
|
|
/* Add a new cache entry with PV info or update existing one. */
|
|
if (!(info = lvmcache_add(fmt->labeller, (const char *) &pv->id,
|
|
pv->dev, pv->label_sector, pv->vg_name,
|
|
is_orphan_vg(pv->vg_name) ? pv->vg_name : pv->vg ? (const char *) &pv->vg->id : NULL, 0, NULL)))
|
|
return_0;
|
|
|
|
/* lvmcache_add() creates info and info->label structs for the dev, get info->label. */
|
|
label = lvmcache_get_label(info);
|
|
|
|
lvmcache_update_pv(info, pv, fmt);
|
|
|
|
/* Flush all cached metadata areas, we will reenter new/modified ones. */
|
|
lvmcache_del_mdas(info);
|
|
|
|
/*
|
|
* Add all new or modified metadata areas for this PV stored in
|
|
* its format instance. If this PV is not part of a VG yet,
|
|
* pv->fid will be used. Otherwise pv->vg->fid will be used.
|
|
* The fid_get_mda_indexed fn can handle that transparently,
|
|
* just pass the right format_instance in.
|
|
*/
|
|
for (mda_index = 0; mda_index < FMT_TEXT_MAX_MDAS_PER_PV; mda_index++) {
|
|
if (!(mda = fid_get_mda_indexed(fid, pvid, ID_LEN, mda_index)))
|
|
continue;
|
|
|
|
mdac = (struct mda_context *) mda->metadata_locn;
|
|
log_debug_metadata("Creating metadata area on %s at sector "
|
|
FMTu64 " size " FMTu64 " sectors",
|
|
dev_name(mdac->area.dev),
|
|
mdac->area.start >> SECTOR_SHIFT,
|
|
mdac->area.size >> SECTOR_SHIFT);
|
|
|
|
// if fmt is not the same as info->fmt we are in trouble
|
|
if (!lvmcache_add_mda(info, mdac->area.dev,
|
|
mdac->area.start, mdac->area.size,
|
|
mda_is_ignored(mda), NULL))
|
|
return_0;
|
|
}
|
|
|
|
if (!lvmcache_update_bas(info, pv))
|
|
return_0;
|
|
|
|
/*
|
|
* FIXME: Allow writing zero offset/size data area to disk.
|
|
* This requires defining a special value since we can't
|
|
* write offset/size that is 0/0 - this is already reserved
|
|
* as a delimiter in data/metadata area area list in PV header
|
|
* (needs exploring compatibility with older lvm2).
|
|
*/
|
|
|
|
/*
|
|
* We can't actually write pe_start = 0 (a data area offset)
|
|
* in PV header now. We need to replace this value here. This can
|
|
* happen with vgcfgrestore with redefined pe_start or
|
|
* pvcreate --restorefile. However, we can can have this value in
|
|
* metadata which will override the value in the PV header.
|
|
*/
|
|
|
|
if (!lvmcache_update_das(info, pv))
|
|
return_0;
|
|
|
|
baton.pv = pv;
|
|
baton.fmt = fmt;
|
|
|
|
if (!lvmcache_foreach_mda(info, _write_single_mda, &baton))
|
|
return_0;
|
|
|
|
if (!_set_ext_flags(pv, info))
|
|
return_0;
|
|
|
|
if (!label_write(pv->dev, label)) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* FIXME: We should probably use the format instance's metadata
|
|
* areas for label_write and only if it's successful,
|
|
* update the cache afterwards?
|
|
*/
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _text_pv_needs_rewrite(const struct format_type *fmt, struct physical_volume *pv,
|
|
int *needs_rewrite)
|
|
{
|
|
struct lvmcache_info *info;
|
|
uint32_t ext_vsn;
|
|
uint32_t ext_flags;
|
|
|
|
*needs_rewrite = 0;
|
|
|
|
if (!pv->is_labelled)
|
|
return 1;
|
|
|
|
if (!pv->dev)
|
|
return 1;
|
|
|
|
if (!(info = lvmcache_info_from_pvid((const char *)&pv->id, pv->dev, 0))) {
|
|
log_error("Failed to find cached info for PV %s.", pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
ext_vsn = lvmcache_ext_version(info);
|
|
|
|
if (ext_vsn < PV_HEADER_EXTENSION_VSN) {
|
|
log_debug("PV %s header needs rewrite for new ext version", dev_name(pv->dev));
|
|
*needs_rewrite = 1;
|
|
}
|
|
|
|
ext_flags = lvmcache_ext_flags(info);
|
|
if (!(ext_flags & PV_EXT_USED)) {
|
|
log_debug("PV %s header needs rewrite to set ext used", dev_name(pv->dev));
|
|
*needs_rewrite = 1;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Copy constructor for a metadata_locn.
|
|
*/
|
|
static void *_metadata_locn_copy_raw(struct dm_pool *mem, void *metadata_locn)
|
|
{
|
|
struct mda_context *mdac, *mdac_new;
|
|
|
|
mdac = (struct mda_context *) metadata_locn;
|
|
if (!(mdac_new = dm_pool_alloc(mem, sizeof(*mdac_new)))) {
|
|
log_error("mda_context allocation failed");
|
|
return NULL;
|
|
}
|
|
memcpy(mdac_new, mdac, sizeof(*mdac));
|
|
|
|
return mdac_new;
|
|
}
|
|
|
|
/*
|
|
* Return a string description of the metadata location.
|
|
*/
|
|
static const char *_metadata_locn_name_raw(void *metadata_locn)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) metadata_locn;
|
|
|
|
return dev_name(mdac->area.dev);
|
|
}
|
|
|
|
static uint64_t _metadata_locn_offset_raw(void *metadata_locn)
|
|
{
|
|
struct mda_context *mdac = (struct mda_context *) metadata_locn;
|
|
|
|
return mdac->area.start;
|
|
}
|
|
|
|
static int _text_pv_initialise(const struct format_type *fmt,
|
|
struct pv_create_args *pva,
|
|
struct physical_volume *pv)
|
|
{
|
|
uint64_t data_alignment_sectors = pva->data_alignment;
|
|
uint64_t data_alignment_offset_sectors = pva->data_alignment_offset;
|
|
uint64_t adjustment;
|
|
uint64_t final_alignment_sectors = 0;
|
|
|
|
log_debug("PV init requested data_alignment_sectors %llu data_alignment_offset_sectors %llu",
|
|
(unsigned long long)data_alignment_sectors, (unsigned long long)data_alignment_offset_sectors);
|
|
|
|
if (!data_alignment_sectors) {
|
|
data_alignment_sectors = find_config_tree_int(pv->fmt->cmd, devices_data_alignment_CFG, NULL) * 2;
|
|
if (data_alignment_sectors)
|
|
log_debug("PV init config data_alignment_sectors %llu",
|
|
(unsigned long long)data_alignment_sectors);
|
|
}
|
|
|
|
/* sets pv->pe_align */
|
|
set_pe_align(pv, data_alignment_sectors);
|
|
|
|
/* sets pv->pe_align_offset */
|
|
set_pe_align_offset(pv, data_alignment_offset_sectors);
|
|
|
|
if (pv->pe_align < pv->pe_align_offset) {
|
|
log_error("%s: pe_align (%llu sectors) must not be less than pe_align_offset (%llu sectors)",
|
|
pv_dev_name(pv), (unsigned long long)pv->pe_align, (unsigned long long)pv->pe_align_offset);
|
|
return 0;
|
|
}
|
|
|
|
final_alignment_sectors = pv->pe_align + pv->pe_align_offset;
|
|
|
|
log_debug("PV init final alignment %llu sectors from align %llu align_offset %llu",
|
|
(unsigned long long)final_alignment_sectors,
|
|
(unsigned long long)pv->pe_align,
|
|
(unsigned long long)pv->pe_align_offset);
|
|
|
|
if (pv->size < final_alignment_sectors) {
|
|
log_error("%s: Data alignment must not exceed device size.",
|
|
pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
if (pv->size < final_alignment_sectors + pva->ba_size) {
|
|
log_error("%s: Bootloader area with data-aligned start must "
|
|
"not exceed device size.", pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
if (pva->pe_start == PV_PE_START_CALC) {
|
|
/*
|
|
* Calculate new PE start and bootloader area start value.
|
|
* Make sure both are properly aligned!
|
|
* If PE start can't be aligned because BA is taking
|
|
* the whole space, make PE start equal to the PV size
|
|
* which effectively disables DA - it will have zero size.
|
|
* This needs to be done as we can't have a PV without any DA.
|
|
* But we still want to support a PV with BA only!
|
|
*/
|
|
if (pva->ba_size) {
|
|
pv->ba_start = final_alignment_sectors;
|
|
pv->ba_size = pva->ba_size;
|
|
if ((adjustment = pva->ba_size % pv->pe_align))
|
|
pv->ba_size += pv->pe_align - adjustment;
|
|
if (pv->size < pv->ba_start + pv->ba_size)
|
|
pv->ba_size = pv->size - pv->ba_start;
|
|
pv->pe_start = pv->ba_start + pv->ba_size;
|
|
log_debug("Setting pe start to %llu sectors after ba start %llu size %llu for %s",
|
|
(unsigned long long)pv->pe_start,
|
|
(unsigned long long)pv->ba_start,
|
|
(unsigned long long)pv->ba_size,
|
|
pv_dev_name(pv));
|
|
} else {
|
|
pv->pe_start = final_alignment_sectors;
|
|
log_debug("Setting PE start to %llu sectors for %s",
|
|
(unsigned long long)pv->pe_start, pv_dev_name(pv));
|
|
}
|
|
} else {
|
|
/*
|
|
* Try to keep the value of PE start set to a firm value if
|
|
* requested. This is useful when restoring existing PE start
|
|
* value (e.g. backups). Also, if creating a BA, try to place
|
|
* it in between the final alignment and existing PE start
|
|
* if possible.
|
|
*/
|
|
pv->pe_start = pva->pe_start;
|
|
|
|
log_debug("Setting pe start to requested %llu sectors for %s",
|
|
(unsigned long long)pv->pe_start, pv_dev_name(pv));
|
|
|
|
if (pva->ba_size) {
|
|
if ((pva->ba_start && pva->ba_start + pva->ba_size > pva->pe_start) ||
|
|
(pva->pe_start <= final_alignment_sectors) ||
|
|
(pva->pe_start - final_alignment_sectors < pva->ba_size)) {
|
|
log_error("%s: Bootloader area would overlap data area.", pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
pv->ba_start = pva->ba_start ? : final_alignment_sectors;
|
|
pv->ba_size = pva->ba_size;
|
|
}
|
|
}
|
|
|
|
if (pva->extent_size)
|
|
pv->pe_size = pva->extent_size;
|
|
|
|
if (pva->extent_count)
|
|
pv->pe_count = pva->extent_count;
|
|
|
|
if ((pv->pe_start + pv->pe_count * (uint64_t)pv->pe_size - 1) > pv->size) {
|
|
log_error("Physical extents end beyond end of device %s.",
|
|
pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
if (pva->label_sector != -1)
|
|
pv->label_sector = pva->label_sector;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _text_destroy_instance(struct format_instance *fid)
|
|
{
|
|
if (--fid->ref_count <= 1) {
|
|
if (fid->metadata_areas_index)
|
|
dm_hash_destroy(fid->metadata_areas_index);
|
|
dm_pool_destroy(fid->mem);
|
|
}
|
|
}
|
|
|
|
static void _text_destroy(struct format_type *fmt)
|
|
{
|
|
if (fmt->orphan_vg)
|
|
free_orphan_vg(fmt->orphan_vg);
|
|
|
|
if (fmt->private)
|
|
free(fmt->private);
|
|
|
|
free(fmt);
|
|
}
|
|
|
|
static struct metadata_area_ops _metadata_text_file_ops = {
|
|
.vg_read = _vg_read_file,
|
|
.vg_read_precommit = _vg_read_precommit_file,
|
|
.vg_write = _vg_write_file,
|
|
.vg_remove = _vg_remove_file,
|
|
.vg_commit = _vg_commit_file
|
|
};
|
|
|
|
static struct metadata_area_ops _metadata_text_file_backup_ops = {
|
|
.vg_read = _vg_read_file,
|
|
.vg_write = _vg_write_file,
|
|
.vg_remove = _vg_remove_file,
|
|
.vg_commit = _vg_commit_file_backup
|
|
};
|
|
|
|
static struct metadata_area_ops _metadata_text_raw_ops = {
|
|
.vg_read = _vg_read_raw,
|
|
.vg_read_precommit = _vg_read_precommit_raw,
|
|
.vg_write = _vg_write_raw,
|
|
.vg_remove = _vg_remove_raw,
|
|
.vg_precommit = _vg_precommit_raw,
|
|
.vg_commit = _vg_commit_raw,
|
|
.vg_revert = _vg_revert_raw,
|
|
.mda_metadata_locn_copy = _metadata_locn_copy_raw,
|
|
.mda_metadata_locn_name = _metadata_locn_name_raw,
|
|
.mda_metadata_locn_offset = _metadata_locn_offset_raw,
|
|
.mda_free_sectors = _mda_free_sectors_raw,
|
|
.mda_total_sectors = _mda_total_sectors_raw,
|
|
.mda_in_vg = _mda_in_vg_raw,
|
|
.mda_locns_match = _mda_locns_match_raw,
|
|
.mda_get_device = _mda_get_device_raw,
|
|
};
|
|
|
|
static int _text_pv_setup(const struct format_type *fmt,
|
|
struct physical_volume *pv,
|
|
struct volume_group *vg)
|
|
{
|
|
struct format_instance *fid = pv->fid;
|
|
const char *pvid = (const char *) (*pv->old_id.uuid ? &pv->old_id : &pv->id);
|
|
struct lvmcache_info *info;
|
|
unsigned mda_index;
|
|
struct metadata_area *pv_mda, *pv_mda_copy;
|
|
struct mda_context *pv_mdac;
|
|
uint64_t pe_count;
|
|
uint64_t size_reduction = 0;
|
|
|
|
/* If PV has its own format instance, add mdas from pv->fid to vg->fid. */
|
|
if (pv->fid != vg->fid) {
|
|
for (mda_index = 0; mda_index < FMT_TEXT_MAX_MDAS_PER_PV; mda_index++) {
|
|
if (!(pv_mda = fid_get_mda_indexed(fid, pvid, ID_LEN, mda_index)))
|
|
continue;
|
|
|
|
/* Be sure it's not already in VG's format instance! */
|
|
if (!fid_get_mda_indexed(vg->fid, pvid, ID_LEN, mda_index)) {
|
|
if (!(pv_mda_copy = mda_copy(vg->fid->mem, pv_mda)))
|
|
return_0;
|
|
fid_add_mda(vg->fid, pv_mda_copy, pvid, ID_LEN, mda_index);
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
* Otherwise, if the PV is already a part of the VG (pv->fid == vg->fid),
|
|
* reread PV mda information from the cache and add it to vg->fid.
|
|
*/
|
|
else {
|
|
if (!pv->dev ||
|
|
!(info = lvmcache_info_from_pvid(pv->dev->pvid, pv->dev, 0))) {
|
|
log_error("PV %s missing from cache", pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
|
|
if (!lvmcache_check_format(info, fmt))
|
|
return_0;
|
|
|
|
if (!lvmcache_fid_add_mdas_pv(info, fid))
|
|
return_0;
|
|
}
|
|
|
|
/* If there's the 2nd mda, we need to reduce
|
|
* usable size for further pe_count calculation! */
|
|
if ((pv_mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 1)) &&
|
|
(pv_mdac = pv_mda->metadata_locn))
|
|
size_reduction = pv_mdac->area.size >> SECTOR_SHIFT;
|
|
|
|
/* From now on, VG format instance will be used. */
|
|
pv_set_fid(pv, vg->fid);
|
|
|
|
/* FIXME Cope with genuine pe_count 0 */
|
|
|
|
/* If missing, estimate pv->size from file-based metadata */
|
|
if (!pv->size && pv->pe_count)
|
|
pv->size = pv->pe_count * (uint64_t) vg->extent_size +
|
|
pv->pe_start + size_reduction;
|
|
|
|
/* Recalculate number of extents that will fit */
|
|
if (!pv->pe_count && vg->extent_size) {
|
|
pe_count = (pv->size - pv->pe_start - size_reduction) /
|
|
vg->extent_size;
|
|
if (pe_count > UINT32_MAX) {
|
|
log_error("PV %s too large for extent size %s.",
|
|
pv_dev_name(pv),
|
|
display_size(vg->cmd, (uint64_t) vg->extent_size));
|
|
return 0;
|
|
}
|
|
pv->pe_count = (uint32_t) pe_count;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void *_create_text_context(struct dm_pool *mem, struct text_context *tc)
|
|
{
|
|
struct text_context *new_tc;
|
|
const char *path;
|
|
char *tmp;
|
|
|
|
if (!tc)
|
|
return NULL;
|
|
|
|
path = tc->path_live;
|
|
|
|
if ((tmp = strstr(path, ".tmp")) && (tmp == path + strlen(path) - 4)) {
|
|
log_error("%s: Volume group filename may not end in .tmp",
|
|
path);
|
|
return NULL;
|
|
}
|
|
|
|
if (!(new_tc = dm_pool_alloc(mem, sizeof(*new_tc))))
|
|
return_NULL;
|
|
|
|
if (!(new_tc->path_live = dm_pool_strdup(mem, path)))
|
|
goto_bad;
|
|
|
|
/* If path_edit not defined, create one from path_live with .tmp suffix. */
|
|
if (!tc->path_edit) {
|
|
if (!(tmp = dm_pool_alloc(mem, strlen(path) + 5)))
|
|
goto_bad;
|
|
sprintf(tmp, "%s.tmp", path);
|
|
new_tc->path_edit = tmp;
|
|
}
|
|
else if (!(new_tc->path_edit = dm_pool_strdup(mem, tc->path_edit)))
|
|
goto_bad;
|
|
|
|
if (!(new_tc->desc = tc->desc ? dm_pool_strdup(mem, tc->desc)
|
|
: dm_pool_strdup(mem, "")))
|
|
goto_bad;
|
|
|
|
return (void *) new_tc;
|
|
|
|
bad:
|
|
dm_pool_free(mem, new_tc);
|
|
|
|
log_error("Couldn't allocate text format context object.");
|
|
return NULL;
|
|
}
|
|
|
|
static int _create_vg_text_instance(struct format_instance *fid,
|
|
const struct format_instance_ctx *fic)
|
|
{
|
|
uint32_t type = fic->type;
|
|
struct text_fid_context *fidtc;
|
|
struct metadata_area *mda;
|
|
struct lvmcache_vginfo *vginfo;
|
|
const char *vg_name, *vg_id;
|
|
|
|
if (!(fidtc = (struct text_fid_context *)
|
|
dm_pool_zalloc(fid->mem, sizeof(*fidtc)))) {
|
|
log_error("Couldn't allocate text_fid_context.");
|
|
return 0;
|
|
}
|
|
|
|
fid->private = (void *) fidtc;
|
|
|
|
if (type & FMT_INSTANCE_PRIVATE_MDAS) {
|
|
if (!(mda = dm_pool_zalloc(fid->mem, sizeof(*mda))))
|
|
return_0;
|
|
mda->ops = &_metadata_text_file_backup_ops;
|
|
mda->metadata_locn = _create_text_context(fid->mem, fic->context.private);
|
|
mda->status = 0;
|
|
fid->metadata_areas_index = NULL;
|
|
fid_add_mda(fid, mda, NULL, 0, 0);
|
|
} else {
|
|
vg_name = fic->context.vg_ref.vg_name;
|
|
vg_id = fic->context.vg_ref.vg_id;
|
|
|
|
if (!(fid->metadata_areas_index = dm_hash_create(128))) {
|
|
log_error("Couldn't create metadata index for format "
|
|
"instance of VG %s.", vg_name);
|
|
return 0;
|
|
}
|
|
|
|
if (type & FMT_INSTANCE_MDAS) {
|
|
if (!(vginfo = lvmcache_vginfo_from_vgname(vg_name, vg_id)))
|
|
goto_out;
|
|
if (!lvmcache_fid_add_mdas_vg(vginfo, fid))
|
|
goto_out;
|
|
}
|
|
}
|
|
|
|
out:
|
|
return 1;
|
|
}
|
|
|
|
static 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_pool_zalloc(pv->fid->mem, sizeof(struct metadata_area)))) {
|
|
log_error("struct metadata_area allocation failed");
|
|
return 0;
|
|
}
|
|
|
|
if (!(mdac = dm_pool_zalloc(pv->fid->mem, sizeof(struct mda_context)))) {
|
|
log_error("struct mda_context allocation failed");
|
|
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_remove_metadata_area(const struct format_type *fmt,
|
|
struct physical_volume *pv,
|
|
unsigned mda_index);
|
|
|
|
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 = (const char *) (*pv->old_id.uuid ? &pv->old_id : &pv->id);
|
|
uint64_t ba_size, pe_start, first_unallocated;
|
|
uint64_t alignment, alignment_offset;
|
|
uint64_t disk_size;
|
|
uint64_t mda_start;
|
|
uint64_t adjustment, limit, tmp_mda_size;
|
|
uint64_t wipe_size = 8 << SECTOR_SHIFT;
|
|
uint64_t zero_len;
|
|
size_t page_size = lvm_getpagesize();
|
|
struct metadata_area *mda;
|
|
struct mda_context *mdac;
|
|
const char *limit_name;
|
|
int limit_applied = 0;
|
|
|
|
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;
|
|
ba_size = pv->ba_size << 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)) {
|
|
if (!_text_pv_remove_metadata_area(fmt, pv, mda_index)) {
|
|
log_error(INTERNAL_ERROR "metadata area with index %u already "
|
|
"exists on PV %s and removal failed.",
|
|
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;
|
|
limit_name = "pe_start";
|
|
}
|
|
else if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 1)) &&
|
|
(mdac = mda->metadata_locn)) {
|
|
limit = mdac->area.start;
|
|
limit_name = "MDA1 start";
|
|
}
|
|
else {
|
|
limit = disk_size;
|
|
limit_name = "disk size";
|
|
}
|
|
|
|
/* Adjust limits for bootloader area if present. */
|
|
if (ba_size) {
|
|
limit -= ba_size;
|
|
limit_name = "ba_start";
|
|
}
|
|
|
|
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 &&
|
|
(adjustment = (mda_start + mda_size) % alignment)) {
|
|
tmp_mda_size = mda_size + alignment - adjustment;
|
|
if (mda_start + tmp_mda_size <= limit)
|
|
mda_size = tmp_mda_size;
|
|
}
|
|
|
|
/* Align MDA0 end position with given alignment offset if possible. */
|
|
if (alignment && alignment_offset &&
|
|
(((mda_start + mda_size) % alignment) == 0)) {
|
|
tmp_mda_size = mda_size + alignment_offset;
|
|
if (mda_start + tmp_mda_size <= limit)
|
|
mda_size = tmp_mda_size;
|
|
}
|
|
|
|
if (mda_start + mda_size > limit) {
|
|
/*
|
|
* 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 && alignment &&
|
|
((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. */
|
|
else
|
|
mda_size = limit - mda_start;
|
|
|
|
limit_applied = 1;
|
|
}
|
|
|
|
/*
|
|
* 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) {
|
|
if (ba_size) {
|
|
pv->ba_start = (mda_start + mda_size) >> SECTOR_SHIFT;
|
|
pv->pe_start = pv->ba_start + pv->ba_size;
|
|
} else
|
|
pv->pe_start = (mda_start + mda_size) >> 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 defined or locked. If pe_start is not defined yet, count
|
|
* with any existing MDA0. If MDA0 does not exist, just use
|
|
* LABEL_SCAN_SIZE.
|
|
*
|
|
* The first_unallocated here is the first unallocated byte
|
|
* beyond existing pe_end if there is any preallocated data area
|
|
* reserved already so we can take that as lower limit for our MDA1
|
|
* start calculation. If data area is not reserved yet, we set
|
|
* first_unallocated to 0, meaning this is not our limiting factor
|
|
* and we will look at other limiting factors if they exist.
|
|
* Of course, if we have preallocated data area, we also must
|
|
* have pe_start assigned too (simply, data area needs its start
|
|
* and end specification).
|
|
*/
|
|
first_unallocated = pv->pe_count ? (pv->pe_start + pv->pe_count *
|
|
(uint64_t)pv->pe_size) << SECTOR_SHIFT
|
|
: 0;
|
|
|
|
if (pe_start || pe_start_locked) {
|
|
limit = first_unallocated ? first_unallocated : pe_start;
|
|
limit_name = first_unallocated ? "pe_end" : "pe_start";
|
|
} else {
|
|
if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 0)) &&
|
|
(mdac = mda->metadata_locn)) {
|
|
limit = mdac->area.start + mdac->area.size;
|
|
limit_name = "MDA0 end";
|
|
}
|
|
else {
|
|
limit = LABEL_SCAN_SIZE;
|
|
limit_name = "label scan size";
|
|
}
|
|
|
|
/* Adjust limits for bootloader area if present. */
|
|
if (ba_size) {
|
|
limit += ba_size;
|
|
limit_name = "ba_end";
|
|
}
|
|
}
|
|
|
|
if (limit >= disk_size)
|
|
goto bad;
|
|
|
|
if (mda_size > disk_size) {
|
|
mda_size = disk_size - limit;
|
|
limit_applied = 1;
|
|
}
|
|
|
|
mda_start = disk_size - mda_size;
|
|
|
|
/* If MDA1 size is too big, just take any usable space. */
|
|
if (disk_size - mda_size < limit) {
|
|
mda_size = disk_size - limit;
|
|
mda_start = disk_size - mda_size;
|
|
limit_applied = 1;
|
|
}
|
|
/* Otherwise, try to align MDA1 start if possible. */
|
|
else if (alignment &&
|
|
(adjustment = mda_start % alignment)) {
|
|
tmp_mda_size = mda_size + adjustment;
|
|
if (tmp_mda_size < disk_size &&
|
|
disk_size - tmp_mda_size >= limit) {
|
|
mda_size = tmp_mda_size;
|
|
mda_start = disk_size - mda_size;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (limit_applied)
|
|
log_very_verbose("Using limited metadata area size on %s "
|
|
"with value " FMTu64 " (limited by %s of "
|
|
FMTu64 ").", pv_dev_name(pv),
|
|
mda_size, limit_name, limit);
|
|
|
|
if (mda_size) {
|
|
if (mda_size < MDA_SIZE_MIN) {
|
|
log_error("Metadata area size too small: " FMTu64 " bytes. "
|
|
"It must be at least %u bytes.", mda_size, MDA_SIZE_MIN);
|
|
goto bad;
|
|
}
|
|
|
|
/* Wipe metadata area with zeroes. */
|
|
|
|
zero_len = (mda_size > wipe_size) ? wipe_size : mda_size;
|
|
|
|
if (!dev_write_zeros(pv->dev, mda_start, zero_len)) {
|
|
log_error("Failed to wipe new metadata area on %s at %llu len %llu",
|
|
pv_dev_name(pv),
|
|
(unsigned long long)mda_start,
|
|
(unsigned long long)zero_len);
|
|
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;
|
|
}
|
|
|
|
static int _remove_metadata_area_from_pv(struct physical_volume *pv,
|
|
unsigned mda_index)
|
|
{
|
|
if (mda_index >= FMT_TEXT_MAX_MDAS_PER_PV) {
|
|
log_error(INTERNAL_ERROR "can't remove metadata area with "
|
|
"index %u from PV %s. Metadata "
|
|
"layou not supported by %s format.",
|
|
mda_index, dev_name(pv->dev),
|
|
pv->fmt->name);
|
|
return 0;
|
|
}
|
|
|
|
return fid_remove_mda(pv->fid, NULL, (const char *) &pv->id,
|
|
ID_LEN, mda_index);
|
|
}
|
|
|
|
static int _text_pv_remove_metadata_area(const struct format_type *fmt,
|
|
struct physical_volume *pv,
|
|
unsigned mda_index)
|
|
{
|
|
return _remove_metadata_area_from_pv(pv, mda_index);
|
|
}
|
|
|
|
static int _text_pv_resize(const struct format_type *fmt,
|
|
struct physical_volume *pv,
|
|
struct volume_group *vg,
|
|
uint64_t size)
|
|
{
|
|
struct format_instance *fid = pv->fid;
|
|
const char *pvid = (const char *) (*pv->old_id.uuid ? &pv->old_id : &pv->id);
|
|
struct metadata_area *mda;
|
|
struct mda_context *mdac;
|
|
uint64_t size_reduction;
|
|
uint64_t mda_size;
|
|
unsigned mda_ignored;
|
|
|
|
/*
|
|
* First, set the new size and update the cache and reset pe_count.
|
|
* (pe_count must be reset otherwise it would be considered as
|
|
* a limiting factor while moving the mda!)
|
|
*/
|
|
pv->size = size;
|
|
pv->pe_count = 0;
|
|
|
|
/* If there's an mda at the end, move it to a new position. */
|
|
if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 1)) &&
|
|
(mdac = mda->metadata_locn)) {
|
|
/* FIXME: Maybe MDA0 size would be better? */
|
|
mda_size = mdac->area.size >> SECTOR_SHIFT;
|
|
mda_ignored = mda_is_ignored(mda);
|
|
|
|
if (!_text_pv_remove_metadata_area(fmt, pv, 1) ||
|
|
!_text_pv_add_metadata_area(fmt, pv, 1, 1, mda_size,
|
|
mda_ignored)) {
|
|
log_error("Failed to move metadata area with index 1 "
|
|
"while resizing PV %s.", pv_dev_name(pv));
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* If there's a VG, reduce size by counting in pe_start and metadata areas. */
|
|
if (vg && !is_orphan_vg(vg->name)) {
|
|
size_reduction = pv_pe_start(pv);
|
|
if ((mda = fid_get_mda_indexed(fid, pvid, ID_LEN, 1)) &&
|
|
(mdac = mda->metadata_locn))
|
|
size_reduction += mdac->area.size >> SECTOR_SHIFT;
|
|
pv->size -= size_reduction;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static struct format_instance *_text_create_text_instance(const struct format_type *fmt,
|
|
const struct format_instance_ctx *fic)
|
|
{
|
|
struct format_instance *fid;
|
|
|
|
if (!(fid = alloc_fid(fmt, fic)))
|
|
return_NULL;
|
|
|
|
if (!_create_vg_text_instance(fid, fic)) {
|
|
dm_pool_destroy(fid->mem);
|
|
return_NULL;
|
|
}
|
|
|
|
return fid;
|
|
}
|
|
|
|
static struct format_handler _text_handler = {
|
|
.pv_initialise = _text_pv_initialise,
|
|
.pv_setup = _text_pv_setup,
|
|
.pv_add_metadata_area = _text_pv_add_metadata_area,
|
|
.pv_remove_metadata_area = _text_pv_remove_metadata_area,
|
|
.pv_resize = _text_pv_resize,
|
|
.pv_write = _text_pv_write,
|
|
.pv_needs_rewrite = _text_pv_needs_rewrite,
|
|
.vg_setup = _text_vg_setup,
|
|
.lv_setup = _text_lv_setup,
|
|
.create_instance = _text_create_text_instance,
|
|
.destroy_instance = _text_destroy_instance,
|
|
.destroy = _text_destroy
|
|
};
|
|
|
|
struct format_type *create_text_format(struct cmd_context *cmd)
|
|
{
|
|
struct format_instance_ctx fic;
|
|
struct format_instance *fid;
|
|
struct format_type *fmt;
|
|
struct mda_lists *mda_lists;
|
|
|
|
if (!(fmt = malloc(sizeof(*fmt)))) {
|
|
log_error("Failed to allocate text format type structure.");
|
|
return NULL;
|
|
}
|
|
|
|
fmt->cmd = cmd;
|
|
fmt->ops = &_text_handler;
|
|
fmt->name = FMT_TEXT_NAME;
|
|
fmt->alias = FMT_TEXT_ALIAS;
|
|
fmt->orphan_vg_name = ORPHAN_VG_NAME(FMT_TEXT_NAME);
|
|
fmt->features = FMT_SEGMENTS | FMT_TAGS | FMT_PRECOMMIT |
|
|
FMT_UNLIMITED_VOLS | FMT_RESIZE_PV |
|
|
FMT_UNLIMITED_STRIPESIZE | FMT_CONFIG_PROFILE |
|
|
FMT_NON_POWER2_EXTENTS | FMT_PV_FLAGS;
|
|
|
|
if (!(mda_lists = malloc(sizeof(struct mda_lists)))) {
|
|
log_error("Failed to allocate dir_list");
|
|
free(fmt);
|
|
return NULL;
|
|
}
|
|
|
|
mda_lists->file_ops = &_metadata_text_file_ops;
|
|
mda_lists->raw_ops = &_metadata_text_raw_ops;
|
|
fmt->private = (void *) mda_lists;
|
|
|
|
dm_list_init(&fmt->mda_ops);
|
|
dm_list_add(&fmt->mda_ops, &_metadata_text_raw_ops.list);
|
|
|
|
if (!(fmt->labeller = text_labeller_create(fmt))) {
|
|
log_error("Couldn't create text label handler.");
|
|
goto bad;
|
|
}
|
|
|
|
if (!(label_register_handler(fmt->labeller))) {
|
|
log_error("Couldn't register text label handler.");
|
|
fmt->labeller->ops->destroy(fmt->labeller);
|
|
goto bad;
|
|
}
|
|
|
|
if (!(fmt->orphan_vg = alloc_vg("text_orphan", cmd, fmt->orphan_vg_name)))
|
|
goto_bad;
|
|
|
|
fic.type = FMT_INSTANCE_AUX_MDAS;
|
|
fic.context.vg_ref.vg_name = fmt->orphan_vg_name;
|
|
fic.context.vg_ref.vg_id = NULL;
|
|
if (!(fid = _text_create_text_instance(fmt, &fic)))
|
|
goto_bad;
|
|
|
|
vg_set_fid(fmt->orphan_vg, fid);
|
|
|
|
log_very_verbose("Initialised format: %s", fmt->name);
|
|
|
|
return fmt;
|
|
bad:
|
|
_text_destroy(fmt);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int text_wipe_outdated_pv_mda(struct cmd_context *cmd, struct device *dev,
|
|
struct metadata_area *mda)
|
|
{
|
|
struct mda_context *mdac = mda->metadata_locn;
|
|
uint64_t start_byte = mdac->area.start;
|
|
struct mda_header *mdab;
|
|
struct raw_locn *rlocn_slot0;
|
|
struct raw_locn *rlocn_slot1;
|
|
uint32_t bad_fields = 0;
|
|
|
|
if (!(mdab = raw_read_mda_header(cmd->fmt, &mdac->area, mda_is_primary(mda), 0, &bad_fields))) {
|
|
log_error("Failed to read outdated pv mda header on %s", dev_name(dev));
|
|
return 0;
|
|
}
|
|
|
|
rlocn_slot0 = &mdab->raw_locns[0];
|
|
rlocn_slot1 = &mdab->raw_locns[1];
|
|
|
|
rlocn_slot0->offset = 0;
|
|
rlocn_slot0->size = 0;
|
|
rlocn_slot0->checksum = 0;
|
|
rlocn_slot1->offset = 0;
|
|
rlocn_slot1->size = 0;
|
|
rlocn_slot1->checksum = 0;
|
|
|
|
if (!_raw_write_mda_header(cmd->fmt, dev, mda_is_primary(mda), start_byte, mdab)) {
|
|
log_error("Failed to write outdated pv mda header on %s", dev_name(dev));
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|