mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-24 06:04:19 +03:00
3b45ebe86b
Currently we never fail with 64byte uuid buffer, but just stay consitent with rest of the code and check for result.
678 lines
16 KiB
C
678 lines
16 KiB
C
#include "lib.h"
|
|
#include "toolcontext.h"
|
|
#include "metadata.h"
|
|
#include "device.h"
|
|
#include "lvmetad.h"
|
|
#include "lvmcache.h"
|
|
#include "lvmetad-client.h"
|
|
#include "format-text.h" // TODO for disk_locn, used as a DA representation
|
|
#include "filter.h"
|
|
|
|
static int _using_lvmetad = 0;
|
|
static daemon_handle _lvmetad;
|
|
|
|
void lvmetad_init(void)
|
|
{
|
|
const char *socket = getenv("LVM_LVMETAD_SOCKET");
|
|
if (_using_lvmetad) { /* configured by the toolcontext */
|
|
_lvmetad = lvmetad_open(socket ?: DEFAULT_RUN_DIR "/lvmetad.socket");
|
|
if (_lvmetad.socket_fd < 0 || _lvmetad.error) {
|
|
log_warn("Failed to connect to lvmetad: %s. Falling back to scanning.", strerror(_lvmetad.error));
|
|
_using_lvmetad = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Helper; evaluate the reply from lvmetad, check for errors, print diagnostics
|
|
* and return a summary success/failure exit code. Frees up the reply resources
|
|
* as well.
|
|
*/
|
|
static int _lvmetad_handle_reply(daemon_reply reply, const char *action, const char *object) {
|
|
if (reply.error || strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
log_error("Request to %s %s in lvmetad has failed. Reason: %s",
|
|
action, object, reply.error ? strerror(reply.error) :
|
|
daemon_reply_str(reply, "reason", "Unknown."));
|
|
daemon_reply_destroy(reply);
|
|
return 0;
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
static int _read_mda(struct lvmcache_info *info,
|
|
struct format_type *fmt,
|
|
const struct dm_config_node *cn)
|
|
{
|
|
struct metadata_area_ops *ops;
|
|
|
|
dm_list_iterate_items(ops, &fmt->mda_ops)
|
|
if (ops->mda_import_text && ops->mda_import_text(info, cn))
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct lvmcache_info *_pv_populate_lvmcache(
|
|
struct cmd_context *cmd, struct dm_config_node *cn, dev_t fallback)
|
|
{
|
|
struct device *device;
|
|
struct id pvid, vgid;
|
|
char mda_id[32];
|
|
char da_id[32];
|
|
int i = 0;
|
|
struct dm_config_node *mda = NULL;
|
|
struct dm_config_node *da = NULL;
|
|
uint64_t offset, size;
|
|
struct lvmcache_info *info;
|
|
const char *pvid_txt = dm_config_find_str(cn->child, "id", NULL),
|
|
*vgid_txt = dm_config_find_str(cn->child, "vgid", NULL),
|
|
*vgname = dm_config_find_str(cn->child, "vgname", NULL),
|
|
*fmt_name = dm_config_find_str(cn->child, "format", NULL);
|
|
dev_t devt = dm_config_find_int(cn->child, "device", 0);
|
|
uint64_t devsize = dm_config_find_int(cn->child, "dev_size", 0),
|
|
label_sector = dm_config_find_int(cn->child, "label_sector", 0);
|
|
|
|
struct format_type *fmt = fmt_name ? get_format_by_name(cmd, fmt_name) : NULL;
|
|
|
|
if (!fmt) {
|
|
log_warn("No format for PV %s. It is probably missing.", pvid_txt);
|
|
return NULL;
|
|
}
|
|
|
|
device = dev_cache_get_by_devt(devt, cmd->filter);
|
|
if (!device && fallback)
|
|
device = dev_cache_get_by_devt(fallback, cmd->filter);
|
|
|
|
if (!device) {
|
|
log_warn("No device for PV %s.", pvid_txt);
|
|
return NULL;
|
|
}
|
|
|
|
if (!pvid_txt || !id_read_format(&pvid, pvid_txt)) {
|
|
log_warn("Missing or ill-formatted PVID for PV: %s.", pvid_txt);
|
|
return NULL;
|
|
}
|
|
|
|
if (vgid_txt)
|
|
id_read_format(&vgid, vgid_txt);
|
|
else
|
|
strcpy((char*)&vgid, fmt->orphan_vg_name);
|
|
|
|
if (!vgname)
|
|
vgname = fmt->orphan_vg_name;
|
|
|
|
info = lvmcache_add(fmt->labeller, (const char *)&pvid, device,
|
|
vgname, (const char *)&vgid, 0);
|
|
|
|
lvmcache_get_label(info)->sector = label_sector;
|
|
lvmcache_set_device_size(info, devsize);
|
|
lvmcache_del_das(info);
|
|
lvmcache_del_mdas(info);
|
|
|
|
do {
|
|
sprintf(mda_id, "mda%d", i);
|
|
mda = dm_config_find_node(cn->child, mda_id);
|
|
if (mda)
|
|
_read_mda(info, fmt, mda);
|
|
++i;
|
|
} while (mda);
|
|
|
|
i = 0;
|
|
do {
|
|
sprintf(da_id, "da%d", i);
|
|
da = dm_config_find_node(cn->child, da_id);
|
|
if (da) {
|
|
if (!dm_config_get_uint64(da->child, "offset", &offset)) return_0;
|
|
if (!dm_config_get_uint64(da->child, "size", &size)) return_0;
|
|
lvmcache_add_da(info, offset, size);
|
|
}
|
|
++i;
|
|
} while (da);
|
|
|
|
return info;
|
|
}
|
|
|
|
struct volume_group *lvmetad_vg_lookup(struct cmd_context *cmd, const char *vgname, const char *vgid)
|
|
{
|
|
struct volume_group *vg = NULL;
|
|
daemon_reply reply;
|
|
char uuid[64];
|
|
struct format_instance *fid;
|
|
struct format_instance_ctx fic;
|
|
struct dm_config_node *top;
|
|
const char *name;
|
|
const char *fmt_name;
|
|
struct format_type *fmt;
|
|
struct dm_config_node *pvcn;
|
|
struct pv_list *pvl;
|
|
struct lvmcache_info *info;
|
|
|
|
if (!_using_lvmetad)
|
|
return NULL;
|
|
|
|
if (vgid) {
|
|
if (!id_write_format((const struct id*)vgid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
reply = daemon_send_simple(_lvmetad, "vg_lookup", "uuid = %s", uuid, NULL);
|
|
} else {
|
|
if (!vgname)
|
|
log_error(INTERNAL_ERROR "VG name required (VGID not available)");
|
|
reply = daemon_send_simple(_lvmetad, "vg_lookup", "name = %s", vgname, NULL);
|
|
}
|
|
|
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
|
|
top = dm_config_find_node(reply.cft->root, "metadata");
|
|
name = daemon_reply_str(reply, "name", NULL);
|
|
|
|
/* fall back to lvm2 if we don't know better */
|
|
fmt_name = dm_config_find_str(top, "metadata/format", "lvm2");
|
|
if (!(fmt = get_format_by_name(cmd, fmt_name))) {
|
|
log_error(INTERNAL_ERROR
|
|
"We do not know the format (%s) reported by lvmetad.",
|
|
fmt_name);
|
|
return NULL;
|
|
}
|
|
|
|
fic.type = FMT_INSTANCE_MDAS | FMT_INSTANCE_AUX_MDAS;
|
|
fic.context.vg_ref.vg_name = name;
|
|
fic.context.vg_ref.vg_id = vgid;
|
|
|
|
if (!(fid = fmt->ops->create_instance(fmt, &fic)))
|
|
return_NULL;
|
|
|
|
pvcn = dm_config_find_node(top, "metadata/physical_volumes")->child;
|
|
while (pvcn) {
|
|
_pv_populate_lvmcache(cmd, pvcn, 0);
|
|
pvcn = pvcn->sib;
|
|
}
|
|
|
|
top->key = name;
|
|
vg = import_vg_from_config_tree(reply.cft, fid);
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if ((info = lvmcache_info_from_pvid((const char *)&pvl->pv->id, 0))) {
|
|
pvl->pv->label_sector = lvmcache_get_label(info)->sector;
|
|
pvl->pv->dev = lvmcache_device(info);
|
|
lvmcache_fid_add_mdas_pv(info, fid);
|
|
} /* else probably missing */
|
|
}
|
|
|
|
lvmcache_update_vg(vg, 0);
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return vg;
|
|
}
|
|
|
|
struct _fixup_baton {
|
|
int i;
|
|
int find;
|
|
int ignore;
|
|
};
|
|
|
|
static int _fixup_ignored(struct metadata_area *mda, void *baton) {
|
|
struct _fixup_baton *b = baton;
|
|
if (b->i == b->find)
|
|
mda_set_ignored(mda, b->ignore);
|
|
b->i ++;
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_update(struct volume_group *vg)
|
|
{
|
|
char *buf = NULL;
|
|
daemon_reply reply;
|
|
struct dm_hash_node *n;
|
|
struct metadata_area *mda;
|
|
char mda_id[128], *num;
|
|
struct pv_list *pvl;
|
|
struct lvmcache_info *info;
|
|
struct _fixup_baton baton;
|
|
|
|
if (!vg)
|
|
return 0;
|
|
|
|
if (!_using_lvmetad)
|
|
return 1; /* fake it */
|
|
|
|
/* TODO. This is not entirely correct, since export_vg_to_buffer
|
|
* adds trailing nodes to the buffer. We may need to use
|
|
* export_vg_to_config_tree and format the buffer ourselves. It
|
|
* does, however, work for now, since the garbage is well
|
|
* formatted and has no conflicting keys with the rest of the
|
|
* request. */
|
|
if (!export_vg_to_buffer(vg, &buf)) {
|
|
log_error("Could not format VG metadata.");
|
|
return 0;
|
|
}
|
|
|
|
reply = daemon_send_simple(_lvmetad, "vg_update", "vgname = %s", vg->name,
|
|
"metadata = %b", strchr(buf, '{'),
|
|
NULL);
|
|
dm_free(buf);
|
|
|
|
if (!_lvmetad_handle_reply(reply, "update VG", vg->name))
|
|
return 0;
|
|
|
|
n = (vg->fid && vg->fid->metadata_areas_index) ?
|
|
dm_hash_get_first(vg->fid->metadata_areas_index) : NULL;
|
|
while (n) {
|
|
mda = dm_hash_get_data(vg->fid->metadata_areas_index, n);
|
|
strcpy(mda_id, dm_hash_get_key(vg->fid->metadata_areas_index, n));
|
|
if ((num = strchr(mda_id, '_'))) {
|
|
*num = 0;
|
|
++num;
|
|
if ((info = lvmcache_info_from_pvid(mda_id, 0))) {
|
|
memset(&baton, 0, sizeof(baton));
|
|
baton.find = atoi(num);
|
|
baton.ignore = mda_is_ignored(mda);
|
|
lvmcache_foreach_mda(info, _fixup_ignored, &baton);
|
|
}
|
|
}
|
|
n = dm_hash_get_next(vg->fid->metadata_areas_index, n);
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
/* NB. the PV fmt pointer is sometimes wrong during vgconvert */
|
|
if (pvl->pv->dev && !lvmetad_pv_found(pvl->pv->id, pvl->pv->dev,
|
|
vg->fid ? vg->fid->fmt : pvl->pv->fmt,
|
|
pvl->pv->label_sector, NULL))
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_remove(struct volume_group *vg)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
|
|
if (!_using_lvmetad)
|
|
return 1; /* just fake it */
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "vg_remove", "uuid = %s", uuid, NULL);
|
|
|
|
return _lvmetad_handle_reply(reply, "remove VG", vg->name);
|
|
}
|
|
|
|
int lvmetad_pv_lookup(struct cmd_context *cmd, struct id pvid)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
int result = 1;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!_using_lvmetad)
|
|
return_0;
|
|
|
|
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "pv_lookup", "uuid = %s", uuid, NULL);
|
|
|
|
if (reply.error || strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
_lvmetad_handle_reply(reply, "lookup PVs", "");
|
|
return_0;
|
|
}
|
|
|
|
cn = dm_config_find_node(reply.cft->root, "physical_volume");
|
|
if (!_pv_populate_lvmcache(cmd, cn, 0))
|
|
result = 0;
|
|
|
|
daemon_reply_destroy(reply);
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_lookup_by_devt(struct cmd_context *cmd, dev_t device)
|
|
{
|
|
int result = 1;
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!_using_lvmetad)
|
|
return_0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "pv_lookup", "device = %d", device, NULL);
|
|
|
|
if (reply.error || strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
_lvmetad_handle_reply(reply, "lookup PVs", "");
|
|
return_0;
|
|
}
|
|
|
|
cn = dm_config_find_node(reply.cft->root, "physical_volume");
|
|
if (!_pv_populate_lvmcache(cmd, cn, device))
|
|
result = 0;
|
|
|
|
daemon_reply_destroy(reply);
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_list_to_lvmcache(struct cmd_context *cmd)
|
|
{
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!_using_lvmetad)
|
|
return_0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "pv_list", NULL);
|
|
|
|
if (reply.error || strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
_lvmetad_handle_reply(reply, "list PVs", "");
|
|
return_0;
|
|
}
|
|
|
|
cn = dm_config_find_node(reply.cft->root, "physical_volumes")->child;
|
|
while (cn) {
|
|
_pv_populate_lvmcache(cmd, cn, 0);
|
|
cn = cn->sib;
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_list_to_lvmcache(struct cmd_context *cmd)
|
|
{
|
|
struct volume_group *tmp;
|
|
struct id vgid;
|
|
const char *vgid_txt;
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!_using_lvmetad)
|
|
return_0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "vg_list", NULL);
|
|
if (reply.error || strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
_lvmetad_handle_reply(reply, "list VGs", "");
|
|
return_0;
|
|
}
|
|
|
|
cn = dm_config_find_node(reply.cft->root, "volume_groups")->child;
|
|
while (cn) {
|
|
vgid_txt = cn->key;
|
|
id_read_format(&vgid, vgid_txt);
|
|
cn = cn->sib;
|
|
|
|
/* the call to lvmetad_vg_lookup will poke the VG into lvmcache */
|
|
tmp = lvmetad_vg_lookup(cmd, NULL, (const char*)&vgid);
|
|
release_vg(tmp);
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
struct _print_mda_baton {
|
|
int i;
|
|
char *buffer;
|
|
};
|
|
|
|
static int _print_mda(struct metadata_area *mda, void *baton)
|
|
{
|
|
int result = 0;
|
|
struct _print_mda_baton *b = baton;
|
|
char *buf, *mda_txt;
|
|
|
|
if (!mda->ops->mda_export_text) /* do nothing */
|
|
return 1;
|
|
|
|
buf = b->buffer;
|
|
mda_txt = mda->ops->mda_export_text(mda);
|
|
if (!dm_asprintf(&b->buffer, "%s mda%i { %s }", b->buffer ?: "", b->i, mda_txt))
|
|
goto_out;
|
|
b->i ++;
|
|
result = 1;
|
|
out:
|
|
dm_free(mda_txt);
|
|
dm_free(buf);
|
|
return result;
|
|
}
|
|
|
|
static int _print_da(struct disk_locn *da, void *baton)
|
|
{
|
|
struct _print_mda_baton *b;
|
|
char *buf;
|
|
|
|
if (!da)
|
|
return 1;
|
|
|
|
b = baton;
|
|
buf = b->buffer;
|
|
if (!dm_asprintf(&b->buffer, "%s da%i { offset = %" PRIu64
|
|
" size = %" PRIu64 " }",
|
|
b->buffer ?: "", b->i, da->offset, da->size))
|
|
{
|
|
dm_free(buf);
|
|
return_0;
|
|
}
|
|
b->i ++;
|
|
dm_free(buf);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static const char *_print_mdas(struct lvmcache_info *info)
|
|
{
|
|
struct _print_mda_baton baton = { .i = 0, .buffer = NULL };
|
|
|
|
if (!lvmcache_foreach_mda(info, &_print_mda, &baton))
|
|
return NULL;
|
|
baton.i = 0;
|
|
if (!lvmcache_foreach_da(info, &_print_da, &baton))
|
|
return NULL;
|
|
|
|
return baton.buffer;
|
|
}
|
|
|
|
int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_type *fmt,
|
|
uint64_t label_sector, struct volume_group *vg)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
struct lvmcache_info *info;
|
|
const char *mdas = NULL;
|
|
char *pvmeta;
|
|
char *buf = NULL;
|
|
|
|
if (!_using_lvmetad)
|
|
return 1;
|
|
|
|
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
/* FIXME A more direct route would be much preferable. */
|
|
if ((info = lvmcache_info_from_pvid((const char *)&pvid, 0)))
|
|
mdas = _print_mdas(info);
|
|
|
|
if (!dm_asprintf(&pvmeta,
|
|
"{ device = %" PRIu64 "\n"
|
|
" dev_size = %" PRIu64 "\n"
|
|
" format = \"%s\"\n"
|
|
" label_sector = %" PRIu64 "\n"
|
|
" id = \"%s\"\n"
|
|
" %s"
|
|
"}", device->dev,
|
|
info ? lvmcache_device_size(info) : 0,
|
|
fmt->name, label_sector, uuid, mdas ?: "")) {
|
|
dm_free((char *)mdas);
|
|
return_0;
|
|
}
|
|
|
|
dm_free((char *)mdas);
|
|
|
|
if (vg) {
|
|
/*
|
|
* TODO. This is not entirely correct, since export_vg_to_buffer
|
|
* adds trailing garbage to the buffer. We may need to use
|
|
* export_vg_to_config_tree and format the buffer ourselves. It
|
|
* does, however, work for now, since the garbage is well
|
|
* formatted and has no conflicting keys with the rest of the
|
|
* request.
|
|
*/
|
|
export_vg_to_buffer(vg, &buf);
|
|
reply = daemon_send_simple(_lvmetad,
|
|
"pv_found",
|
|
"pvmeta = %b", pvmeta,
|
|
"vgname = %s", vg->name,
|
|
"metadata = %b", strchr(buf, '{'),
|
|
NULL);
|
|
} else {
|
|
/* There are no MDAs on this PV. */
|
|
reply = daemon_send_simple(_lvmetad,
|
|
"pv_found",
|
|
"pvmeta = %b", pvmeta,
|
|
NULL);
|
|
}
|
|
|
|
dm_free(pvmeta);
|
|
return _lvmetad_handle_reply(reply, "update PV", uuid);
|
|
}
|
|
|
|
int lvmetad_pv_gone(dev_t device)
|
|
{
|
|
if (!_using_lvmetad)
|
|
return 1;
|
|
|
|
daemon_reply reply =
|
|
daemon_send_simple(_lvmetad, "pv_gone", "device = %d", device, NULL);
|
|
|
|
return _lvmetad_handle_reply(reply, "drop PV", "");
|
|
}
|
|
|
|
int lvmetad_active(void)
|
|
{
|
|
return _using_lvmetad;
|
|
}
|
|
|
|
void lvmetad_set_active(int active)
|
|
{
|
|
_using_lvmetad = active;
|
|
}
|
|
|
|
/*
|
|
* The following code implements pvscan --lvmetad.
|
|
*/
|
|
|
|
struct _pvscan_lvmetad_baton {
|
|
struct volume_group *vg;
|
|
struct format_instance *fid;
|
|
};
|
|
|
|
static int _pvscan_lvmetad_single(struct metadata_area *mda, void *baton)
|
|
{
|
|
struct _pvscan_lvmetad_baton *b = baton;
|
|
struct volume_group *this = mda->ops->vg_read(b->fid, "", mda);
|
|
if (!b->vg || this->seqno > b->vg->seqno)
|
|
b->vg = this;
|
|
else if (b->vg)
|
|
release_vg(this);
|
|
return 1;
|
|
}
|
|
|
|
static dev_t _parse_devt(const char *str) { /* Oh. */
|
|
char *where = (char *) str;
|
|
int major = strtol(str, &where, 10);
|
|
int minor;
|
|
|
|
if (where == str)
|
|
return -1;
|
|
|
|
if (*where != ':')
|
|
return -1;
|
|
|
|
str = ++where;
|
|
minor = strtol(str, &where, 10);
|
|
|
|
if (where == str)
|
|
return -1;
|
|
|
|
if (*where)
|
|
return -1;
|
|
|
|
return MKDEV(major, minor);
|
|
}
|
|
|
|
int pvscan_lvmetad(struct cmd_context *cmd, int argc, char **argv)
|
|
{
|
|
struct device *dev;
|
|
struct label *label;
|
|
struct lvmcache_info *info;
|
|
struct physical_volume pv;
|
|
struct _pvscan_lvmetad_baton baton;
|
|
/* Create a dummy instance. */
|
|
struct format_instance_ctx fic = { .type = 0 };
|
|
|
|
if (argc != 1) {
|
|
log_error("Exactly one device parameter required.");
|
|
return 0;
|
|
}
|
|
|
|
if (!lvmetad_active()) {
|
|
log_error("Cannot proceed since lvmetad is not active.");
|
|
return 0;
|
|
}
|
|
|
|
dev = dev_cache_get(argv[0], NULL);
|
|
if (!dev && _parse_devt(argv[0]) != -1)
|
|
dev = dev_cache_get_by_devt(_parse_devt(argv[0]), NULL);
|
|
|
|
if (!dev) {
|
|
if (_parse_devt(argv[0]) == -1) {
|
|
log_error("For devices that do not exist, we need a MAJOR:MINOR pair.");
|
|
return 0;
|
|
}
|
|
|
|
if (!lvmetad_pv_gone(_parse_devt(argv[0])))
|
|
goto fatal;
|
|
|
|
log_info("Device %s not found and was wiped from lvmetad.", argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
if (!label_read(dev, &label, 0)) {
|
|
log_warn("No PV label found on %s.", dev_name(dev));
|
|
if (!lvmetad_pv_gone(dev->dev))
|
|
goto fatal;
|
|
return 1;
|
|
}
|
|
|
|
info = (struct lvmcache_info *) label->info;
|
|
memset(&pv, 0, sizeof(pv));
|
|
|
|
baton.vg = NULL;
|
|
baton.fid = lvmcache_fmt(info)->ops->create_instance(lvmcache_fmt(info),
|
|
&fic);
|
|
|
|
lvmcache_foreach_mda(info, _pvscan_lvmetad_single, &baton);
|
|
if (!baton.vg)
|
|
lvmcache_fmt(info)->ops->destroy_instance(baton.fid);
|
|
|
|
/*
|
|
* NB. If this command failed and we are relying on lvmetad to have an
|
|
* *exact* image of the system, the lvmetad instance that went out of
|
|
* sync needs to be killed.
|
|
*/
|
|
if (!lvmetad_pv_found(*(struct id *)dev->pvid, dev, lvmcache_fmt(info),
|
|
label->sector, baton.vg))
|
|
goto fatal;
|
|
|
|
release_vg(baton.vg);
|
|
return 1;
|
|
fatal:
|
|
release_vg(baton.vg);
|
|
/* FIXME kill lvmetad automatically if we can */
|
|
log_error("Update of lvmetad failed. This is a serious problem.\n "
|
|
"It is strongly recommended that you restart lvmetad immediately.");
|
|
return 0;
|
|
}
|
|
|