mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-30 17:18:21 +03:00
6ea250e2d0
In some cases, the command will update VG metadata in lvmetad without writing it. In these cases there is no vg->vg_committed and it should use 'vg' directly. This happens when the command finds that the lvmetad VG has been invalidated, rereads the metadata from disk, then updates lvmetad with that metadata. This happens often with lvmlockd or foreign VGs, and can happen without lvmlockd if a previous command fails after invalidating the VG in lvmetad.
2860 lines
82 KiB
C
2860 lines
82 KiB
C
/*
|
|
* Copyright (C) 2012 Red Hat, Inc.
|
|
*
|
|
* 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.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 "crc.h"
|
|
#include "lvm-signal.h"
|
|
#include "lvmlockd.h"
|
|
#include "str_list.h"
|
|
|
|
#include <time.h>
|
|
|
|
static daemon_handle _lvmetad = { .error = 0 };
|
|
static int _lvmetad_use = 0;
|
|
static int _lvmetad_connected = 0;
|
|
static int _lvmetad_daemon_pid = 0;
|
|
|
|
static char *_lvmetad_token = NULL;
|
|
static const char *_lvmetad_socket = NULL;
|
|
static struct cmd_context *_lvmetad_cmd = NULL;
|
|
static int64_t _lvmetad_update_timeout;
|
|
|
|
static int _found_lvm1_metadata = 0;
|
|
|
|
static struct volume_group *lvmetad_pvscan_vg(struct cmd_context *cmd, struct volume_group *vg);
|
|
|
|
static uint64_t _monotonic_seconds(void)
|
|
{
|
|
struct timespec ts;
|
|
|
|
if (clock_gettime(CLOCK_MONOTONIC, &ts) < 0)
|
|
return 0;
|
|
return ts.tv_sec;
|
|
}
|
|
|
|
static int _log_debug_inequality(const char *name, struct dm_config_node *a, struct dm_config_node *b)
|
|
{
|
|
int result = 0;
|
|
int final_result = 0;
|
|
|
|
if (a->v && b->v) {
|
|
result = compare_value(a->v, b->v);
|
|
if (result) {
|
|
struct dm_config_value *av = a->v;
|
|
struct dm_config_value *bv = b->v;
|
|
|
|
if (!strcmp(a->key, b->key)) {
|
|
if (a->v->type == DM_CFG_STRING && b->v->type == DM_CFG_STRING)
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s: %s / %s",
|
|
name, a->key, b->key, av->v.str, bv->v.str);
|
|
else if (a->v->type == DM_CFG_INT && b->v->type == DM_CFG_INT)
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s: " FMTi64 " / " FMTi64,
|
|
name, a->key, b->key, av->v.i, bv->v.i);
|
|
else
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s: type %d / type %d",
|
|
name, a->key, b->key, av->type, bv->type);
|
|
} else {
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s", name, a->key, b->key);
|
|
}
|
|
final_result = result;
|
|
}
|
|
}
|
|
|
|
if (a->v && !b->v) {
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s", name, a->key, b->key);
|
|
final_result = 1;
|
|
}
|
|
|
|
if (!a->v && b->v) {
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s", name, a->key, b->key);
|
|
final_result = -1;
|
|
}
|
|
|
|
if (a->child && b->child) {
|
|
result = _log_debug_inequality(name, a->child, b->child);
|
|
if (result)
|
|
final_result = result;
|
|
}
|
|
|
|
if (a->sib && b->sib) {
|
|
result = _log_debug_inequality(name, a->sib, b->sib);
|
|
if (result)
|
|
final_result = result;
|
|
}
|
|
|
|
|
|
if (a->sib && !b->sib) {
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s", name, a->key, b->key);
|
|
final_result = 1;
|
|
}
|
|
|
|
if (!a->sib && b->sib) {
|
|
log_debug_lvmetad("VG %s metadata inequality at %s / %s", name, a->key, b->key);
|
|
final_result = -1;
|
|
}
|
|
|
|
return final_result;
|
|
}
|
|
|
|
void lvmetad_disconnect(void)
|
|
{
|
|
if (_lvmetad_connected)
|
|
daemon_close(_lvmetad);
|
|
|
|
_lvmetad_connected = 0;
|
|
_lvmetad_use = 0;
|
|
_lvmetad_cmd = NULL;
|
|
}
|
|
|
|
int lvmetad_connect(struct cmd_context *cmd)
|
|
{
|
|
if (!lvmetad_socket_present()) {
|
|
log_debug_lvmetad("Failed to connect to lvmetad: socket not present.");
|
|
_lvmetad_connected = 0;
|
|
_lvmetad_use = 0;
|
|
_lvmetad_cmd = NULL;
|
|
return 0;
|
|
}
|
|
|
|
_lvmetad_update_timeout = find_config_tree_int(cmd, global_lvmetad_update_wait_time_CFG, NULL);
|
|
|
|
_lvmetad = lvmetad_open(_lvmetad_socket);
|
|
|
|
if (_lvmetad.socket_fd >= 0 && !_lvmetad.error) {
|
|
log_debug_lvmetad("Successfully connected to lvmetad on fd %d.",
|
|
_lvmetad.socket_fd);
|
|
_lvmetad_connected = 1;
|
|
_lvmetad_use = 1;
|
|
_lvmetad_cmd = cmd;
|
|
return 1;
|
|
} else {
|
|
log_debug_lvmetad("Failed to connect to lvmetad: %s", strerror(_lvmetad.error));
|
|
_lvmetad_connected = 0;
|
|
_lvmetad_use = 0;
|
|
_lvmetad_cmd = NULL;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int lvmetad_used(void)
|
|
{
|
|
return _lvmetad_use;
|
|
}
|
|
|
|
void lvmetad_make_unused(struct cmd_context *cmd)
|
|
{
|
|
lvmetad_disconnect();
|
|
|
|
if (cmd && !refresh_filters(cmd))
|
|
stack;
|
|
}
|
|
|
|
int lvmetad_pidfile_present(void)
|
|
{
|
|
const char *pidfile = getenv("LVM_LVMETAD_PIDFILE") ?: LVMETAD_PIDFILE;
|
|
|
|
return !access(pidfile, F_OK);
|
|
}
|
|
|
|
int lvmetad_socket_present(void)
|
|
{
|
|
const char *socket = _lvmetad_socket ?: LVMETAD_SOCKET;
|
|
int r;
|
|
|
|
if ((r = access(socket, F_OK)) && errno != ENOENT)
|
|
log_sys_error("access", socket);
|
|
|
|
return !r;
|
|
}
|
|
|
|
void lvmetad_set_socket(const char *sock)
|
|
{
|
|
_lvmetad_socket = sock;
|
|
}
|
|
|
|
/*
|
|
* Use a crc of the strings in the filter as the lvmetad token.
|
|
*/
|
|
void lvmetad_set_token(const struct dm_config_value *filter)
|
|
{
|
|
int ft = 0;
|
|
|
|
dm_free(_lvmetad_token);
|
|
|
|
while (filter && filter->type == DM_CFG_STRING) {
|
|
ft = calc_crc(ft, (const uint8_t *) filter->v.str, strlen(filter->v.str));
|
|
filter = filter->next;
|
|
}
|
|
|
|
if (dm_asprintf(&_lvmetad_token, "filter:%u", ft) < 0)
|
|
log_warn("WARNING: Failed to set lvmetad token. Out of memory?");
|
|
}
|
|
|
|
void lvmetad_release_token(void)
|
|
{
|
|
dm_free(_lvmetad_token);
|
|
_lvmetad_token = NULL;
|
|
}
|
|
|
|
/*
|
|
* Check if lvmetad's token matches our token. The token is a hash of the
|
|
* global filter used to populate lvmetad. The lvmetad token was set by the
|
|
* last command to populate lvmetad, and it was set to the hash of the global
|
|
* filter that command used when scanning to populate lvmetad.
|
|
*
|
|
* Our token is a hash of the global filter this command is using.
|
|
*
|
|
* If the lvmetad token is not set (or "none"), then lvmetad has not been
|
|
* populated. If the lvmetad token is "update in progress", then lvmetad is
|
|
* currently being populated -- this should be temporary, so wait for a while
|
|
* for the current update to finish and then compare our token with the new one
|
|
* (hopefully it will match). If the lvmetad token otherwise differs from
|
|
* ours, then lvmetad was populated using a different global filter that we are
|
|
* using.
|
|
*
|
|
* Return 1 if the lvmetad token matches ours. We can use it as is.
|
|
*
|
|
* Return 0 if the lvmetad token does not match ours (lvmetad is empty or
|
|
* populated using a different global filter). The caller will repopulate
|
|
* lvmetad (via lvmetad_pvscan_all_devs) before using lvmetad.
|
|
*
|
|
* If we time out waiting for an lvmetad update to finish, then disable this
|
|
* command's use of lvmetad and return 0.
|
|
*/
|
|
|
|
int lvmetad_token_matches(struct cmd_context *cmd)
|
|
{
|
|
daemon_reply reply;
|
|
const char *daemon_token;
|
|
unsigned int delay_usec = 0;
|
|
unsigned int wait_sec = 0;
|
|
uint64_t now = 0, wait_start = 0;
|
|
int ret = 1;
|
|
|
|
wait_sec = (unsigned int)_lvmetad_update_timeout;
|
|
|
|
retry:
|
|
log_debug_lvmetad("Sending lvmetad get_global_info");
|
|
|
|
reply = daemon_send_simple(_lvmetad, "get_global_info",
|
|
"token = %s", "skip",
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
if (reply.error) {
|
|
log_warn("WARNING: Not using lvmetad after send error (%d).", reply.error);
|
|
goto fail;
|
|
}
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
log_warn("WARNING: Not using lvmetad after response error.");
|
|
goto fail;
|
|
}
|
|
|
|
if (!(daemon_token = daemon_reply_str(reply, "token", NULL))) {
|
|
log_warn("WARNING: Not using lvmetad with older version.");
|
|
goto fail;
|
|
}
|
|
|
|
_lvmetad_daemon_pid = (int)daemon_reply_int(reply, "daemon_pid", 0);
|
|
|
|
/*
|
|
* If lvmetad is being updated by another command, then sleep and retry
|
|
* until the token shows the update is done, and go on to the token
|
|
* comparison.
|
|
*
|
|
* Between retries, sleep for a random period between 1 and 2 seconds.
|
|
* Retry in this way for up to a configurable period of time.
|
|
*/
|
|
if (!strcmp(daemon_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS)) {
|
|
if (!(now = _monotonic_seconds()))
|
|
goto fail;
|
|
|
|
if (!wait_start)
|
|
wait_start = now;
|
|
|
|
if (now - wait_start > wait_sec) {
|
|
log_warn("WARNING: Not using lvmetad after %u sec lvmetad_update_wait_time.", wait_sec);
|
|
goto fail;
|
|
}
|
|
|
|
log_warn("WARNING: lvmetad is being updated, retrying (setup) for %u more seconds.",
|
|
wait_sec - (unsigned int)(now - wait_start));
|
|
|
|
/* Delay a random period between 1 and 2 seconds. */
|
|
delay_usec = 1000000 + lvm_even_rand(&_lvmetad_cmd->rand_seed, 1000000);
|
|
usleep(delay_usec);
|
|
daemon_reply_destroy(reply);
|
|
goto retry;
|
|
}
|
|
|
|
/*
|
|
* lvmetad is empty, not yet populated.
|
|
* The caller should do a disk scan to populate lvmetad.
|
|
*/
|
|
if (!strcmp(daemon_token, "none")) {
|
|
ret = 0;
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* lvmetad has an unmatching token; it was last populated using
|
|
* a different global filter.
|
|
* The caller should do a disk scan to populate lvmetad with
|
|
* our global filter.
|
|
*/
|
|
if (strcmp(daemon_token, _lvmetad_token)) {
|
|
ret = 0;
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
daemon_reply_destroy(reply);
|
|
return ret;
|
|
|
|
fail:
|
|
daemon_reply_destroy(reply);
|
|
/* The command will not use lvmetad and will revert to scanning. */
|
|
lvmetad_make_unused(cmd);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Wait up to lvmetad_update_wait_time for the lvmetad updating state to be
|
|
* finished.
|
|
*
|
|
* Return 0 if lvmetad is not updating or there's an error and we can't tell.
|
|
* Return 1 if lvmetad is updating.
|
|
*/
|
|
static int _lvmetad_is_updating(struct cmd_context *cmd, int do_wait)
|
|
{
|
|
daemon_reply reply;
|
|
const char *daemon_token;
|
|
unsigned int wait_sec = 0;
|
|
uint64_t now = 0, wait_start = 0;
|
|
int ret = 0;
|
|
|
|
wait_sec = (unsigned int)_lvmetad_update_timeout;
|
|
retry:
|
|
log_debug_lvmetad("Sending lvmetad get_global_info");
|
|
|
|
reply = daemon_send_simple(_lvmetad, "get_global_info",
|
|
"token = %s", "skip",
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
if (reply.error)
|
|
goto out;
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK"))
|
|
goto out;
|
|
|
|
if (!(daemon_token = daemon_reply_str(reply, "token", NULL)))
|
|
goto out;
|
|
|
|
if (!strcmp(daemon_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS)) {
|
|
ret = 1;
|
|
|
|
if (!do_wait)
|
|
goto out;
|
|
|
|
if (!(now = _monotonic_seconds()))
|
|
goto out;
|
|
|
|
if (!wait_start)
|
|
wait_start = now;
|
|
|
|
if (now - wait_start >= wait_sec)
|
|
goto out;
|
|
|
|
log_warn("WARNING: lvmetad is being updated, waiting for %u more seconds.",
|
|
wait_sec - (unsigned int)(now - wait_start));
|
|
|
|
usleep(1000000);
|
|
daemon_reply_destroy(reply);
|
|
goto retry;
|
|
} else {
|
|
ret = 0;
|
|
}
|
|
|
|
out:
|
|
daemon_reply_destroy(reply);
|
|
return ret;
|
|
}
|
|
|
|
static daemon_reply _lvmetad_send(struct cmd_context *cmd, const char *id, ...)
|
|
{
|
|
va_list ap;
|
|
daemon_reply reply = { 0 };
|
|
daemon_request req;
|
|
const char *token_expected;
|
|
unsigned int delay_usec;
|
|
unsigned int wait_sec = 0;
|
|
uint64_t now = 0, wait_start = 0;
|
|
int daemon_in_update;
|
|
int we_are_in_update;
|
|
|
|
if (!_lvmetad_connected || !_lvmetad_use) {
|
|
reply.error = ECONNRESET;
|
|
return reply;
|
|
}
|
|
|
|
wait_sec = (unsigned int)_lvmetad_update_timeout;
|
|
retry:
|
|
req = daemon_request_make(id);
|
|
|
|
if (!daemon_request_extend(req,
|
|
"token = %s", _lvmetad_token ?: "none",
|
|
"update_timeout = " FMTd64, (int64_t)wait_sec,
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL)) {
|
|
reply.error = ENOMEM;
|
|
return reply;
|
|
}
|
|
|
|
va_start(ap, id);
|
|
daemon_request_extend_v(req, ap);
|
|
va_end(ap);
|
|
|
|
reply = daemon_send(_lvmetad, req);
|
|
|
|
daemon_request_destroy(req);
|
|
|
|
if (reply.error == ECONNRESET)
|
|
log_warn("WARNING: lvmetad connection failed, cannot reconnect.");
|
|
|
|
/*
|
|
* For the "token_update" message, the result is handled entirely
|
|
* by the _token_update() function, so return the reply immediately.
|
|
*/
|
|
if (!strcmp(id, "token_update"))
|
|
return reply;
|
|
|
|
/*
|
|
* For other messages it may be useful to retry and resend the
|
|
* message, so check for that case before returning the reply.
|
|
* The reply will be checked further in lvmetad_handle_reply.
|
|
*/
|
|
|
|
if (reply.error)
|
|
return reply;
|
|
|
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "token_mismatch")) {
|
|
token_expected = daemon_reply_str(reply, "expected", "");
|
|
daemon_in_update = !strcmp(token_expected, LVMETAD_TOKEN_UPDATE_IN_PROGRESS);
|
|
we_are_in_update = !strcmp(_lvmetad_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS);
|
|
|
|
if (daemon_in_update && !we_are_in_update) {
|
|
/*
|
|
* Another command is updating lvmetad, and we cannot
|
|
* use lvmetad until the update is finished. Retry our
|
|
* request for a while; the update should finish
|
|
* shortly. This should not usually happen because
|
|
* this command already checked that the token is
|
|
* usable in lvmetad_token_matches(), but it's possible
|
|
* for another command's rescan to slip in between the
|
|
* time we call lvmetad_token_matches() and the time we
|
|
* get here to lvmetad_send().
|
|
*/
|
|
|
|
if (!(now = _monotonic_seconds()))
|
|
goto out;
|
|
|
|
if (!wait_start)
|
|
wait_start = now;
|
|
|
|
if (!wait_sec || (now - wait_start >= wait_sec)) {
|
|
log_warn("WARNING: Cannot use lvmetad after %u sec lvmetad_update_wait_time.", wait_sec);
|
|
goto out;
|
|
}
|
|
|
|
log_warn("WARNING: lvmetad is being updated, retrying (%s) for %u more seconds.",
|
|
id, wait_sec - (unsigned int)(now - wait_start));
|
|
|
|
/* Delay a random period between 1 and 2 seconds. */
|
|
delay_usec = 1000000 + lvm_even_rand(&_lvmetad_cmd->rand_seed, 1000000);
|
|
usleep(delay_usec);
|
|
daemon_reply_destroy(reply);
|
|
goto retry;
|
|
|
|
} else {
|
|
/* See lvmetad_handle_reply for handling other cases. */
|
|
}
|
|
}
|
|
out:
|
|
return reply;
|
|
}
|
|
|
|
/*
|
|
* token_update happens when starting or ending an lvmetad update.
|
|
* When starting we set the token to "update in progress".
|
|
* When ending we set the token to our filter:<hash>.
|
|
*
|
|
* From the perspective of a command, the lvmetad state is one of:
|
|
* "none" - the lvmetad cache is not populated and an update is required.
|
|
* "filter:<matching_hash>" - the command with can use the lvmetad cache.
|
|
* "filter:<unmatching_hash>" - the lvmetad cache must be updated to be used.
|
|
* "update in progress" - a command is updating the lvmetad cache.
|
|
*
|
|
* . If none, the command will update (scan and populate lvmetad),
|
|
* then use the cache.
|
|
*
|
|
* . If filter is matching, the command will use the cache.
|
|
*
|
|
* . If filter is unmatching, the command will update (scan and
|
|
* populate lvmetad), then use the cache.
|
|
*
|
|
* . If update in progress, the command will wait for a while for the state
|
|
* to become non-updating. If it changes, see above, if it doesn't change,
|
|
* then the command either reverts to not using lvmetad, or does an update
|
|
* (scan and populate lvmetad) and then uses the cache.
|
|
*
|
|
* A command that is explicitly intended to update the cache will always do
|
|
* that (it may wait for a while first to allow a current update to complete).
|
|
* A command that is not explicitly intended to update the cache may choose
|
|
* to revert to scanning and not use lvmetad.
|
|
*
|
|
* Because two different updates from two commands can potentially overlap,
|
|
* lvmetad saves the pid of the latest update to start, so it can reject messages
|
|
* from preempted updates. This prevents an invalid mix of two different updates.
|
|
* (The command makes use of the update_pid to print more informative messages.)
|
|
*
|
|
* If lvmetad detects that a command doing an update is taking too long, it will
|
|
* change the token from "update in progress" to "none", which means a new update
|
|
* is required, causing the next command to do an update. This effectively
|
|
* cancels/preempts a slow/stuck update, and helps to automatically resolve
|
|
* some failure cases.
|
|
*/
|
|
|
|
static int _token_update(int *replaced_update)
|
|
{
|
|
daemon_reply reply;
|
|
const char *token_expected;
|
|
const char *prev_token;
|
|
int update_pid;
|
|
int ending_our_update;
|
|
|
|
log_debug_lvmetad("Sending lvmetad token_update %s", _lvmetad_token);
|
|
reply = _lvmetad_send(NULL, "token_update", NULL);
|
|
|
|
if (replaced_update)
|
|
*replaced_update = 0;
|
|
|
|
if (reply.error) {
|
|
log_warn("WARNING: lvmetad token update error: %s", strerror(reply.error));
|
|
daemon_reply_destroy(reply);
|
|
return 0;
|
|
}
|
|
|
|
update_pid = (int)daemon_reply_int(reply, "update_pid", 0);
|
|
|
|
/*
|
|
* A mismatch can only happen when this command attempts to set the
|
|
* token to filter:<hash> at the end of its update, but the update has
|
|
* been preempted in lvmetad by a new one (from update_pid).
|
|
*/
|
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "token_mismatch")) {
|
|
token_expected = daemon_reply_str(reply, "expected", "");
|
|
|
|
ending_our_update = strcmp(_lvmetad_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS);
|
|
|
|
log_debug_lvmetad("Received token update mismatch expected \"%s\" our token \"%s\" update_pid %d our pid %d",
|
|
token_expected, _lvmetad_token, update_pid, getpid());
|
|
|
|
if (ending_our_update && (update_pid != getpid())) {
|
|
log_warn("WARNING: lvmetad was updated by another command (pid %d).", update_pid);
|
|
} else {
|
|
/*
|
|
* Shouldn't happen.
|
|
* If we're ending our update and our pid matches the update_pid,
|
|
* then there would not be a mismatch.
|
|
* If we're starting a new update, lvmetad never returns a
|
|
* token mismatch.
|
|
* In any case, it doesn't hurt to just return an error here.
|
|
*/
|
|
log_error(INTERNAL_ERROR "lvmetad token update mismatch pid %d matches our own pid %d", update_pid, getpid());
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 0;
|
|
}
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
log_error("Failed response from lvmetad for token update.");
|
|
daemon_reply_destroy(reply);
|
|
return 0;
|
|
}
|
|
|
|
if ((prev_token = daemon_reply_str(reply, "prev_token", NULL))) {
|
|
if (!strcmp(prev_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS))
|
|
if (replaced_update && (update_pid != getpid()))
|
|
*replaced_update = 1;
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Helper; evaluate the reply from lvmetad, check for errors, print diagnostics
|
|
* and return a summary success/failure exit code.
|
|
*
|
|
* If found is set, *found indicates whether or not device exists,
|
|
* and missing device is not treated as an error.
|
|
*/
|
|
static int _lvmetad_handle_reply(daemon_reply reply, const char *id, const char *object, int *found)
|
|
{
|
|
const char *token_expected;
|
|
const char *action;
|
|
int action_modifies = 0;
|
|
int daemon_in_update;
|
|
int we_are_in_update;
|
|
int update_pid;
|
|
|
|
if (!id)
|
|
action = "<none>";
|
|
else if (!strcmp(id, "pv_list"))
|
|
action = "list PVs";
|
|
else if (!strcmp(id, "vg_list"))
|
|
action = "list VGs";
|
|
else if (!strcmp(id, "vg_lookup"))
|
|
action = "lookup VG";
|
|
else if (!strcmp(id, "pv_lookup"))
|
|
action = "lookup PV";
|
|
else if (!strcmp(id, "pv_clear_all"))
|
|
action = "clear info about all PVs";
|
|
else if (!strcmp(id, "vg_clear_outdated_pvs"))
|
|
action = "clear the list of outdated PVs";
|
|
else if (!strcmp(id, "set_vg_info"))
|
|
action = "set VG info";
|
|
else if (!strcmp(id, "vg_update"))
|
|
action = "update VG";
|
|
else if (!strcmp(id, "vg_remove"))
|
|
action = "remove VG";
|
|
else if (!strcmp(id, "pv_found")) {
|
|
action = "update PV";
|
|
action_modifies = 1;
|
|
} else if (!strcmp(id, "pv_gone")) {
|
|
action = "drop PV";
|
|
action_modifies = 1;
|
|
} else {
|
|
log_error(INTERNAL_ERROR "Unchecked lvmetad message %s.", id);
|
|
action = "action unknown";
|
|
}
|
|
|
|
if (reply.error) {
|
|
log_warn("WARNING: lvmetad cannot be used due to error: %s", strerror(reply.error));
|
|
goto fail;
|
|
}
|
|
|
|
/*
|
|
* Errors related to token mismatch.
|
|
*/
|
|
|
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "token_mismatch")) {
|
|
|
|
token_expected = daemon_reply_str(reply, "expected", "");
|
|
update_pid = (int)daemon_reply_int(reply, "update_pid", 0);
|
|
|
|
log_debug("lvmetad token mismatch, expected \"%s\" our token \"%s\"",
|
|
token_expected, _lvmetad_token);
|
|
|
|
daemon_in_update = !strcmp(token_expected, LVMETAD_TOKEN_UPDATE_IN_PROGRESS);
|
|
we_are_in_update = !strcmp(_lvmetad_token, LVMETAD_TOKEN_UPDATE_IN_PROGRESS);
|
|
|
|
if (daemon_in_update && we_are_in_update) {
|
|
|
|
/*
|
|
* When we do not match the update_pid, it means our
|
|
* update was cancelled and another process is now
|
|
* updating the cache.
|
|
*/
|
|
|
|
if (update_pid != getpid()) {
|
|
log_warn("WARNING: lvmetad is being updated by another command (pid %d).", update_pid);
|
|
} else {
|
|
/* Shouldn't happen */
|
|
log_error(INTERNAL_ERROR "lvmetad update by pid %d matches our own pid %d", update_pid, getpid());
|
|
}
|
|
/* We don't care if the action was modifying during a token update. */
|
|
action_modifies = 0;
|
|
goto fail;
|
|
|
|
} else if (daemon_in_update && !we_are_in_update) {
|
|
|
|
/*
|
|
* Another command is updating lvmetad, and we cannot
|
|
* use lvmetad until the update is finished.
|
|
* lvmetad_send resent this message up to the limit and
|
|
* eventually gave up. The caller may choose to not
|
|
* use lvmetad at this point and revert to scanning.
|
|
*/
|
|
|
|
log_warn("WARNING: lvmetad is being updated and cannot be used.");
|
|
goto fail;
|
|
|
|
} else if (!daemon_in_update && we_are_in_update) {
|
|
|
|
/*
|
|
* We are updating lvmetad after setting the token to
|
|
* "update in progress", but lvmetad has a non-update
|
|
* token and is rejecting our update messages. This
|
|
* must mean that lvmetad cancelled our update (we were
|
|
* probably too slow, taking longer than the timeout),
|
|
* so another command completed an update and set the
|
|
* token based on its filter. Here we've attempt to
|
|
* continue our cache update, and find we've been
|
|
* preempted, so we should just abort our failed
|
|
* update.
|
|
*/
|
|
|
|
log_warn("WARNING: lvmetad was updated by another command.");
|
|
/* We don't care if the action was modifying during a token update. */
|
|
action_modifies = 0;
|
|
goto fail;
|
|
|
|
} else if (!daemon_in_update && !we_are_in_update) {
|
|
|
|
/*
|
|
* Another command has updated the lvmetad cache, and
|
|
* has done so using a different device filter from our
|
|
* own, which has made the lvmetad token and our token
|
|
* not match. This should not usually happen because
|
|
* this command has already checked for a matching token
|
|
* in lvmetad_token_matches(), but it's possible for
|
|
* another command's rescan to slip in between the time
|
|
* we call lvmetad_token_matches() and the time we get
|
|
* here to lvmetad_send(). With a mismatched token
|
|
* (different set of devices), we cannot use the lvmetad
|
|
* cache.
|
|
*
|
|
* FIXME: it would be nice to have this command ignore
|
|
* lvmetad at this point and revert to disk scanning,
|
|
* but the layers above lvmetad_send are not yet able
|
|
* to switch modes in the middle of processing.
|
|
*
|
|
* (The advantage of lvmetad_check_token is that it
|
|
* can rescan to get the token in sync, or if that
|
|
* fails it can make the command revert to scanning
|
|
* from the start.)
|
|
*/
|
|
|
|
log_warn("WARNING: Cannot use lvmetad while it caches different devices.");
|
|
goto fail;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Non-token-mismatch related error checking.
|
|
*/
|
|
|
|
/* All OK? */
|
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
if (found)
|
|
*found = 1;
|
|
return 1;
|
|
}
|
|
|
|
/* Unknown device permitted? */
|
|
if (found && !strcmp(daemon_reply_str(reply, "response", ""), "unknown")) {
|
|
log_very_verbose("Request to %s %s%sin lvmetad did not find any matching object.",
|
|
action, object, *object ? " " : "");
|
|
*found = 0;
|
|
return 1;
|
|
}
|
|
|
|
/* Multiple VGs with the same name were found. */
|
|
if (found && !strcmp(daemon_reply_str(reply, "response", ""), "multiple")) {
|
|
log_very_verbose("Request to %s %s%sin lvmetad found multiple matching objects.",
|
|
action, object, *object ? " " : "");
|
|
if (found)
|
|
*found = 2;
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Generic error message for error cases not specifically checked above.
|
|
*/
|
|
log_error("Request to %s %s%sin lvmetad gave response %s. Reason: %s",
|
|
action, object, *object ? " " : "",
|
|
daemon_reply_str(reply, "response", "<missing>"),
|
|
daemon_reply_str(reply, "reason", "<missing>"));
|
|
fail:
|
|
/*
|
|
* If the failed lvmetad message was updating lvmetad with new metadata
|
|
* that has been changed by this command, it is important to restart
|
|
* lvmetad (or at least rescan.) (An lvmetad update that is just
|
|
* scanning disks to populate the cache is not a problem, so we try to
|
|
* avoid printing a "corruption" warning in that case.)
|
|
*/
|
|
|
|
if (action_modifies) {
|
|
/*
|
|
* FIXME: experiment with killing the lvmetad process here, e.g.
|
|
* kill(_lvmetad_daemon_pid, SIGKILL);
|
|
*/
|
|
log_warn("WARNING: To avoid corruption, restart lvmetad (or disable with use_lvmetad=0).");
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
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 int _pv_populate_lvmcache(struct cmd_context *cmd,
|
|
struct dm_config_node *cn,
|
|
struct format_type *fmt, dev_t fallback)
|
|
{
|
|
struct device *dev;
|
|
struct id pvid, vgid;
|
|
char mda_id[32];
|
|
char da_id[32];
|
|
int i = 0;
|
|
struct dm_config_node *mda, *da;
|
|
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_int64(cn->child, "dev_size", 0),
|
|
label_sector = dm_config_find_int64(cn->child, "label_sector", 0);
|
|
uint32_t ext_flags = (uint32_t) dm_config_find_int64(cn->child, "ext_flags", 0);
|
|
uint32_t ext_version = (uint32_t) dm_config_find_int64(cn->child, "ext_version", 0);
|
|
|
|
if (!fmt && fmt_name)
|
|
fmt = get_format_by_name(cmd, fmt_name);
|
|
|
|
if (!fmt) {
|
|
log_error("PV %s not recognised. Is the device missing?", pvid_txt);
|
|
return 0;
|
|
}
|
|
|
|
dev = dev_cache_get_by_devt(devt, cmd->filter);
|
|
if (!dev && fallback)
|
|
dev = dev_cache_get_by_devt(fallback, cmd->filter);
|
|
|
|
if (!dev) {
|
|
log_warn("WARNING: Device for PV %s not found or rejected by a filter.", pvid_txt);
|
|
return 0;
|
|
}
|
|
|
|
if (!pvid_txt || !id_read_format(&pvid, pvid_txt)) {
|
|
log_error("Missing or ill-formatted PVID for PV: %s.", pvid_txt);
|
|
return 0;
|
|
}
|
|
|
|
if (vgid_txt) {
|
|
if (!id_read_format(&vgid, vgid_txt))
|
|
return_0;
|
|
} else
|
|
/* NB uuid is short and NUL-terminated. */
|
|
(void) dm_strncpy((char*)&vgid, fmt->orphan_vg_name, sizeof(vgid));
|
|
|
|
if (!vgname)
|
|
vgname = fmt->orphan_vg_name;
|
|
|
|
if (!(info = lvmcache_add(fmt->labeller, (const char *)&pvid, dev,
|
|
vgname, (const char *)&vgid, 0)))
|
|
return_0;
|
|
|
|
lvmcache_get_label(info)->sector = label_sector;
|
|
lvmcache_get_label(info)->dev = dev;
|
|
lvmcache_set_device_size(info, devsize);
|
|
lvmcache_del_das(info);
|
|
lvmcache_del_mdas(info);
|
|
lvmcache_del_bas(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);
|
|
|
|
i = 0;
|
|
do {
|
|
sprintf(da_id, "ba%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_ba(info, offset, size);
|
|
}
|
|
++i;
|
|
} while (da);
|
|
|
|
lvmcache_set_ext_flags(info, ext_flags);
|
|
lvmcache_set_ext_version(info, ext_version);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _pv_update_struct_pv(struct physical_volume *pv, struct format_instance *fid)
|
|
{
|
|
struct lvmcache_info *info;
|
|
|
|
if ((info = lvmcache_info_from_pvid((const char *)&pv->id, pv->dev, 0))) {
|
|
pv->label_sector = lvmcache_get_label(info)->sector;
|
|
pv->dev = lvmcache_device(info);
|
|
if (!pv->dev)
|
|
pv->status |= MISSING_PV;
|
|
if (!lvmcache_fid_add_mdas_pv(info, fid))
|
|
return_0;
|
|
pv->fid = fid;
|
|
} else
|
|
pv->status |= MISSING_PV; /* probably missing */
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct volume_group *lvmetad_vg_lookup(struct cmd_context *cmd, const char *vgname, const char *vgid)
|
|
{
|
|
struct volume_group *vg = NULL;
|
|
struct volume_group *vg2 = NULL;
|
|
daemon_reply reply;
|
|
int found;
|
|
char uuid[64];
|
|
struct format_instance *fid = NULL;
|
|
struct format_instance_ctx fic;
|
|
struct dm_config_node *top;
|
|
const char *name, *diag_name;
|
|
const char *fmt_name;
|
|
struct format_type *fmt;
|
|
struct dm_config_node *pvcn;
|
|
struct pv_list *pvl;
|
|
int rescan = 0;
|
|
|
|
if (!lvmetad_used())
|
|
return NULL;
|
|
|
|
if (vgid) {
|
|
if (!id_write_format((const struct id*)vgid, uuid, sizeof(uuid)))
|
|
return_NULL;
|
|
}
|
|
|
|
if (vgid && vgname) {
|
|
log_debug_lvmetad("Asking lvmetad for VG %s %s", uuid, vgname);
|
|
reply = _lvmetad_send(cmd, "vg_lookup",
|
|
"uuid = %s", uuid,
|
|
"name = %s", vgname,
|
|
NULL);
|
|
diag_name = uuid;
|
|
|
|
} else if (vgid) {
|
|
log_debug_lvmetad("Asking lvmetad for VG vgid %s", uuid);
|
|
reply = _lvmetad_send(cmd, "vg_lookup", "uuid = %s", uuid, NULL);
|
|
diag_name = uuid;
|
|
|
|
} else if (vgname) {
|
|
log_debug_lvmetad("Asking lvmetad for VG %s", vgname);
|
|
reply = _lvmetad_send(cmd, "vg_lookup", "name = %s", vgname, NULL);
|
|
diag_name = vgname;
|
|
|
|
} else {
|
|
log_error(INTERNAL_ERROR "VG name required (VGID not available)");
|
|
return NULL;
|
|
}
|
|
|
|
if (_lvmetad_handle_reply(reply, "vg_lookup", diag_name, &found) && found) {
|
|
|
|
if ((found == 2) && vgname) {
|
|
log_error("Multiple VGs found with the same name: %s.", vgname);
|
|
log_error("See the --select option with VG UUID (vg_uuid).");
|
|
goto out;
|
|
}
|
|
|
|
if (!(top = dm_config_find_node(reply.cft->root, "metadata"))) {
|
|
log_error(INTERNAL_ERROR "metadata config node not found.");
|
|
goto out;
|
|
}
|
|
|
|
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);
|
|
goto out;
|
|
}
|
|
|
|
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)))
|
|
goto_out;
|
|
|
|
if ((pvcn = dm_config_find_node(top, "metadata/physical_volumes")))
|
|
for (pvcn = pvcn->child; pvcn; pvcn = pvcn->sib)
|
|
_pv_populate_lvmcache(cmd, pvcn, fmt, 0);
|
|
|
|
if ((pvcn = dm_config_find_node(top, "metadata/outdated_pvs")))
|
|
for (pvcn = pvcn->child; pvcn; pvcn = pvcn->sib)
|
|
_pv_populate_lvmcache(cmd, pvcn, fmt, 0);
|
|
|
|
top->key = name;
|
|
if (!(vg = import_vg_from_lvmetad_config_tree(reply.cft, fid)))
|
|
goto_out;
|
|
|
|
/*
|
|
* Read the VG from disk, ignoring the lvmetad copy in these
|
|
* cases:
|
|
*
|
|
* 1. The host is not using lvmlockd, but is reading lockd VGs
|
|
* using the --shared option. The shared option is meant to
|
|
* let hosts not running lvmlockd look at lockd VGs, like the
|
|
* foreign option allows hosts to look at foreign VGs. When
|
|
* --foreign is used, the code forces a rescan since the local
|
|
* lvmetad cache of foreign VGs is likely stale. Similarly,
|
|
* for --shared, have the code reading the shared VGs below
|
|
* not use the cached copy from lvmetad but to rescan the VG.
|
|
*
|
|
* 2. The host failed to acquire the VG lock from lvmlockd for
|
|
* the lockd VG. In this case, the usual mechanisms for
|
|
* updating the lvmetad copy of the VG have been missed. Since
|
|
* we don't know if the cached copy is valid, assume it's not.
|
|
*
|
|
* 3. lvmetad has returned the "vg_invalid" flag, which is the
|
|
* usual mechanism used by lvmlockd/lvmetad to cause a host to
|
|
* reread a VG from disk that has been modified from another
|
|
* host.
|
|
*/
|
|
|
|
if (is_lockd_type(vg->lock_type) && cmd->include_shared_vgs) {
|
|
log_debug_lvmetad("Rescan VG %s because including shared", vgname);
|
|
rescan = 1;
|
|
} else if (is_lockd_type(vg->lock_type) && cmd->lockd_vg_rescan) {
|
|
log_debug_lvmetad("Rescan VG %s because no lvmlockd lock is held", vgname);
|
|
rescan = 1;
|
|
} else if (dm_config_find_node(reply.cft->root, "vg_invalid")) {
|
|
if (!is_lockd_type(vg->lock_type)) {
|
|
/* Can happen if a previous command failed/crashed without updating lvmetad. */
|
|
log_warn("WARNING: Reading VG %s from disk because lvmetad metadata is invalid.", vgname);
|
|
} else {
|
|
/* This is normal when the VG was modified by another host. */
|
|
log_debug_lvmetad("Rescan VG %s because lvmetad returned invalid", vgname);
|
|
}
|
|
rescan = 1;
|
|
}
|
|
|
|
/*
|
|
* locking may have detected a newer vg version and
|
|
* invalidated the cached vg.
|
|
*/
|
|
if (rescan) {
|
|
if (!(vg2 = lvmetad_pvscan_vg(cmd, vg))) {
|
|
log_debug_lvmetad("VG %s from lvmetad not found during rescan.", vgname);
|
|
fid = NULL;
|
|
release_vg(vg);
|
|
vg = NULL;
|
|
goto out;
|
|
}
|
|
release_vg(vg);
|
|
vg = vg2;
|
|
fid = vg2->fid;
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (!_pv_update_struct_pv(pvl->pv, fid)) {
|
|
vg = NULL;
|
|
goto_out; /* FIXME error path */
|
|
}
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &vg->pvs_outdated) {
|
|
if (!_pv_update_struct_pv(pvl->pv, fid)) {
|
|
vg = NULL;
|
|
goto_out; /* FIXME error path */
|
|
}
|
|
}
|
|
|
|
lvmcache_update_vg(vg, 0);
|
|
vg_mark_partial_lvs(vg, 1);
|
|
}
|
|
|
|
out:
|
|
if (!vg && fid)
|
|
fid->fmt->ops->destroy_instance(fid);
|
|
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;
|
|
}
|
|
|
|
/*
|
|
* After the VG is written to disk, but before it's committed,
|
|
* lvmetad is told the new seqno. lvmetad sets the INVALID
|
|
* flag on the cached VG and saves the new seqno.
|
|
*
|
|
* After the VG is committed on disk, the command sends the
|
|
* new VG metadata, containing the new seqno. lvmetad sees
|
|
* that it has the updated metadata and clears the INVALID
|
|
* flag on the cached VG.
|
|
*
|
|
* If the command fails after committing the metadata on disk
|
|
* but before sending the new metadata to lvmetad, then the
|
|
* next command that asks lvmetad for the metadata will get
|
|
* back the INVALID flag. That command will then read the
|
|
* VG metadata from disk to use, and will send the latest
|
|
* metadata from disk to lvmetad which will clear the
|
|
* INVALID flag.
|
|
*/
|
|
|
|
int lvmetad_vg_update_pending(struct volume_group *vg)
|
|
{
|
|
char uuid[64] __attribute__((aligned(8)));
|
|
daemon_reply reply;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1; /* fake it */
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Sending lvmetad pending VG %s (seqno %" PRIu32 ")", vg->name, vg->seqno);
|
|
reply = _lvmetad_send(vg->cmd, "set_vg_info",
|
|
"name = %s", vg->name,
|
|
"uuid = %s", uuid,
|
|
"version = %"PRId64, (int64_t)vg->seqno,
|
|
NULL);
|
|
|
|
if (!_lvmetad_handle_reply(reply, "set_vg_info", vg->name, NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
vg->lvmetad_update_pending = 1;
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_update_finish(struct volume_group *vg)
|
|
{
|
|
char uuid[64] __attribute__((aligned(8)));
|
|
daemon_reply reply;
|
|
struct dm_hash_node *n;
|
|
struct metadata_area *mda;
|
|
char mda_id[128], *num;
|
|
struct volume_group *vgu;
|
|
struct dm_config_tree *vgmeta;
|
|
struct pv_list *pvl;
|
|
struct lvmcache_info *info;
|
|
struct _fixup_baton baton;
|
|
|
|
if (!vg->lvmetad_update_pending)
|
|
return 1;
|
|
|
|
if (!(vg->fid->fmt->features & FMT_PRECOMMIT))
|
|
return 1;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1; /* fake it */
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
/*
|
|
* vg->vg_committted is the state of the VG metadata when vg_commit()
|
|
* was called. Since then, 'vg' may have been partially modified and
|
|
* not committed. We only want to send committed metadata to lvmetad.
|
|
*
|
|
* lvmetad is sometimes updated in cases where the VG is not written
|
|
* (no vg_committed). In those cases 'vg' has just been read from
|
|
* disk, and we can send 'vg' to lvmetad. This happens when the
|
|
* command finds the lvmetad cache invalid, so the VG has been read
|
|
* from disk and is then sent to lvmetad.
|
|
*/
|
|
|
|
vgu = vg->vg_committed ? vg->vg_committed : vg;
|
|
|
|
if (!(vgmeta = export_vg_to_config_tree(vgu))) {
|
|
log_error("Failed to export VG to config tree.");
|
|
return 0;
|
|
}
|
|
|
|
log_debug_lvmetad("Sending lvmetad updated VG %s (seqno %" PRIu32 ")", vg->name, vg->seqno);
|
|
reply = _lvmetad_send(vg->cmd, "vg_update",
|
|
"vgname = %s", vg->name,
|
|
"metadata = %t", vgmeta,
|
|
NULL);
|
|
|
|
dm_config_destroy(vgmeta);
|
|
|
|
if (!_lvmetad_handle_reply(reply, "vg_update", vg->name, NULL)) {
|
|
/*
|
|
* In this failure case, the VG cached in lvmetad remains in
|
|
* the INVALID state (from lvmetad_vg_update_pending).
|
|
* A subsequent command will see INVALID, ignore the cached
|
|
* copy, read the VG from disk, and update the cached copy.
|
|
*/
|
|
daemon_reply_destroy(reply);
|
|
return 0;
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
n = (vgu->fid && vgu->fid->metadata_areas_index) ?
|
|
dm_hash_get_first(vgu->fid->metadata_areas_index) : NULL;
|
|
while (n) {
|
|
mda = dm_hash_get_data(vgu->fid->metadata_areas_index, n);
|
|
(void) dm_strncpy(mda_id, dm_hash_get_key(vgu->fid->metadata_areas_index, n), sizeof(mda_id));
|
|
if ((num = strchr(mda_id, '_'))) {
|
|
*num = 0;
|
|
++num;
|
|
if ((info = lvmcache_info_from_pvid(mda_id, NULL, 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(vgu->fid->metadata_areas_index, n);
|
|
}
|
|
|
|
dm_list_iterate_items(pvl, &vgu->pvs) {
|
|
/* NB. the PV fmt pointer is sometimes wrong during vgconvert */
|
|
if (pvl->pv->dev && !lvmetad_pv_found(vg->cmd, &pvl->pv->id, pvl->pv->dev,
|
|
vgu->fid ? vgu->fid->fmt : pvl->pv->fmt,
|
|
pvl->pv->label_sector, NULL, NULL, NULL))
|
|
return 0;
|
|
}
|
|
|
|
vg->lvmetad_update_pending = 0;
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_remove_pending(struct volume_group *vg)
|
|
{
|
|
char uuid[64] __attribute__((aligned(8)));
|
|
daemon_reply reply;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1; /* fake it */
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
/* Sending version/seqno 0 in set_vg_info will set the INVALID flag. */
|
|
|
|
log_debug_lvmetad("Sending lvmetad pending remove VG %s", vg->name);
|
|
reply = _lvmetad_send(vg->cmd, "set_vg_info",
|
|
"name = %s", vg->name,
|
|
"uuid = %s", uuid,
|
|
"version = %d", 0,
|
|
NULL);
|
|
|
|
if (!_lvmetad_handle_reply(reply, "set_vg_info", vg->name, NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_vg_remove_finish(struct volume_group *vg)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
int result;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1; /* just fake it */
|
|
|
|
vg->lvmetad_update_pending = 0;
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Telling lvmetad to remove VGID %s (%s)", uuid, vg->name);
|
|
reply = _lvmetad_send(vg->cmd, "vg_remove", "uuid = %s", uuid, NULL);
|
|
result = _lvmetad_handle_reply(reply, "vg_remove", vg->name, NULL);
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_lookup(struct cmd_context *cmd, struct id pvid, int *found)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
int result = 0;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!lvmetad_used())
|
|
return_0;
|
|
|
|
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for PV %s", uuid);
|
|
reply = _lvmetad_send(cmd, "pv_lookup", "uuid = %s", uuid, NULL);
|
|
if (!_lvmetad_handle_reply(reply, "pv_lookup", "", found))
|
|
goto_out;
|
|
|
|
if (found && !*found)
|
|
goto out_success;
|
|
|
|
if (!(cn = dm_config_find_node(reply.cft->root, "physical_volume")))
|
|
goto_out;
|
|
else if (!_pv_populate_lvmcache(cmd, cn, NULL, 0))
|
|
goto_out;
|
|
|
|
out_success:
|
|
result = 1;
|
|
|
|
out:
|
|
daemon_reply_destroy(reply);
|
|
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_lookup_by_dev(struct cmd_context *cmd, struct device *dev, int *found)
|
|
{
|
|
int result = 0;
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
|
|
if (!lvmetad_used())
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for PV on %s", dev_name(dev));
|
|
reply = _lvmetad_send(cmd, "pv_lookup", "device = %" PRId64, (int64_t) dev->dev, NULL);
|
|
if (!_lvmetad_handle_reply(reply, "pv_lookup", dev_name(dev), found))
|
|
goto_out;
|
|
|
|
if (found && !*found)
|
|
goto out_success;
|
|
|
|
cn = dm_config_find_node(reply.cft->root, "physical_volume");
|
|
if (!cn || !_pv_populate_lvmcache(cmd, cn, NULL, dev->dev))
|
|
goto_out;
|
|
|
|
out_success:
|
|
result = 1;
|
|
|
|
out:
|
|
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 (!lvmetad_used())
|
|
return 1;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for complete list of known PVs");
|
|
reply = _lvmetad_send(cmd, "pv_list", NULL);
|
|
if (!_lvmetad_handle_reply(reply, "pv_list", "", NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
if ((cn = dm_config_find_node(reply.cft->root, "physical_volumes")))
|
|
for (cn = cn->child; cn; cn = cn->sib)
|
|
_pv_populate_lvmcache(cmd, cn, NULL, 0);
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_get_vgnameids(struct cmd_context *cmd, struct dm_list *vgnameids)
|
|
{
|
|
struct vgnameid_list *vgnl;
|
|
struct id vgid;
|
|
const char *vgid_txt;
|
|
const char *vg_name;
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for complete list of known VG ids/names");
|
|
reply = _lvmetad_send(cmd, "vg_list", NULL);
|
|
if (!_lvmetad_handle_reply(reply, "vg_list", "", NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
if ((cn = dm_config_find_node(reply.cft->root, "volume_groups"))) {
|
|
for (cn = cn->child; cn; cn = cn->sib) {
|
|
vgid_txt = cn->key;
|
|
if (!id_read_format(&vgid, vgid_txt)) {
|
|
stack;
|
|
continue;
|
|
}
|
|
|
|
if (!(vgnl = dm_pool_alloc(cmd->mem, sizeof(*vgnl)))) {
|
|
log_error("vgnameid_list allocation failed.");
|
|
return 0;
|
|
}
|
|
|
|
if (!(vg_name = dm_config_find_str(cn->child, "name", NULL))) {
|
|
log_error("vg_list no name found.");
|
|
return 0;
|
|
}
|
|
|
|
vgnl->vgid = dm_pool_strdup(cmd->mem, (char *)&vgid);
|
|
vgnl->vg_name = dm_pool_strdup(cmd->mem, vg_name);
|
|
|
|
if (!vgnl->vgid || !vgnl->vg_name) {
|
|
log_error("vgnameid_list member allocation failed.");
|
|
return 0;
|
|
}
|
|
|
|
dm_list_add(vgnameids, &vgnl->list);
|
|
}
|
|
}
|
|
|
|
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 (!lvmetad_used())
|
|
return 1;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for complete list of known VGs");
|
|
reply = _lvmetad_send(cmd, "vg_list", NULL);
|
|
if (!_lvmetad_handle_reply(reply, "vg_list", "", NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
if ((cn = dm_config_find_node(reply.cft->root, "volume_groups")))
|
|
for (cn = cn->child; cn; cn = cn->sib) {
|
|
vgid_txt = cn->key;
|
|
if (!id_read_format(&vgid, vgid_txt)) {
|
|
stack;
|
|
continue;
|
|
}
|
|
|
|
/* 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 _extract_dl_baton {
|
|
int i;
|
|
struct dm_config_tree *cft;
|
|
struct dm_config_node *pre_sib;
|
|
};
|
|
|
|
static int _extract_mda(struct metadata_area *mda, void *baton)
|
|
{
|
|
struct _extract_dl_baton *b = baton;
|
|
struct dm_config_node *cn;
|
|
char id[32];
|
|
|
|
if (!mda->ops->mda_export_text) /* do nothing */
|
|
return 1;
|
|
|
|
(void) dm_snprintf(id, 32, "mda%d", b->i);
|
|
if (!(cn = make_config_node(b->cft, id, b->cft->root, b->pre_sib)))
|
|
return 0;
|
|
if (!mda->ops->mda_export_text(mda, b->cft, cn))
|
|
return 0;
|
|
|
|
b->i ++;
|
|
b->pre_sib = cn; /* for efficiency */
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _extract_disk_location(const char *name, struct disk_locn *dl, void *baton)
|
|
{
|
|
struct _extract_dl_baton *b = baton;
|
|
struct dm_config_node *cn;
|
|
char id[32];
|
|
|
|
if (!dl)
|
|
return 1;
|
|
|
|
(void) dm_snprintf(id, 32, "%s%d", name, b->i);
|
|
if (!(cn = make_config_node(b->cft, id, b->cft->root, b->pre_sib)))
|
|
return 0;
|
|
if (!config_make_nodes(b->cft, cn, NULL,
|
|
"offset = %"PRId64, (int64_t) dl->offset,
|
|
"size = %"PRId64, (int64_t) dl->size,
|
|
NULL))
|
|
return 0;
|
|
|
|
b->i ++;
|
|
b->pre_sib = cn; /* for efficiency */
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _extract_da(struct disk_locn *da, void *baton)
|
|
{
|
|
return _extract_disk_location("da", da, baton);
|
|
}
|
|
|
|
static int _extract_ba(struct disk_locn *ba, void *baton)
|
|
{
|
|
return _extract_disk_location("ba", ba, baton);
|
|
}
|
|
|
|
static int _extract_mdas(struct lvmcache_info *info, struct dm_config_tree *cft,
|
|
struct dm_config_node *pre_sib)
|
|
{
|
|
struct _extract_dl_baton baton = { .cft = cft };
|
|
|
|
if (!lvmcache_foreach_mda(info, &_extract_mda, &baton))
|
|
return 0;
|
|
|
|
baton.i = 0;
|
|
if (!lvmcache_foreach_da(info, &_extract_da, &baton))
|
|
return 0;
|
|
|
|
baton.i = 0;
|
|
if (!lvmcache_foreach_ba(info, &_extract_ba, &baton))
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
int lvmetad_pv_found(struct cmd_context *cmd, const struct id *pvid, struct device *dev, const struct format_type *fmt,
|
|
uint64_t label_sector, struct volume_group *vg,
|
|
struct dm_list *found_vgnames,
|
|
struct dm_list *changed_vgnames)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
struct lvmcache_info *info;
|
|
struct dm_config_tree *pvmeta, *vgmeta;
|
|
const char *status = NULL, *vgname = NULL;
|
|
int64_t changed = 0;
|
|
int result;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1;
|
|
|
|
if (!id_write_format(pvid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
pvmeta = dm_config_create();
|
|
if (!pvmeta)
|
|
return_0;
|
|
|
|
info = lvmcache_info_from_pvid((const char *)pvid, dev, 0);
|
|
|
|
if (!(pvmeta->root = make_config_node(pvmeta, "pv", NULL, NULL))) {
|
|
dm_config_destroy(pvmeta);
|
|
return_0;
|
|
}
|
|
|
|
/* TODO: resolve what does it actually mean 'info == NULL'
|
|
* missing info is likely an INTERNAL_ERROR */
|
|
if (!config_make_nodes(pvmeta, pvmeta->root, NULL,
|
|
"device = %"PRId64, (int64_t) dev->dev,
|
|
"dev_size = %"PRId64, (int64_t) (info ? lvmcache_device_size(info) : 0),
|
|
"format = %s", fmt->name,
|
|
"label_sector = %"PRId64, (int64_t) label_sector,
|
|
"id = %s", uuid,
|
|
"ext_version = %"PRId64, (int64_t) (info ? lvmcache_ext_version(info) : 0),
|
|
"ext_flags = %"PRId64, (int64_t) (info ? lvmcache_ext_flags(info) : 0),
|
|
NULL))
|
|
{
|
|
dm_config_destroy(pvmeta);
|
|
return_0;
|
|
}
|
|
|
|
if (info)
|
|
/* FIXME A more direct route would be much preferable. */
|
|
_extract_mdas(info, pvmeta, pvmeta->root);
|
|
|
|
if (vg) {
|
|
if (!(vgmeta = export_vg_to_config_tree(vg))) {
|
|
dm_config_destroy(pvmeta);
|
|
return_0;
|
|
}
|
|
|
|
log_debug_lvmetad("Telling lvmetad to store PV %s (%s) in VG %s", dev_name(dev), uuid, vg->name);
|
|
reply = _lvmetad_send(cmd, "pv_found",
|
|
"pvmeta = %t", pvmeta,
|
|
"vgname = %s", vg->name,
|
|
"metadata = %t", vgmeta,
|
|
NULL);
|
|
dm_config_destroy(vgmeta);
|
|
} else {
|
|
/*
|
|
* There is no VG metadata stored on this PV.
|
|
* It might or might not be an orphan.
|
|
*/
|
|
log_debug_lvmetad("Telling lvmetad to store PV %s (%s)", dev_name(dev), uuid);
|
|
reply = _lvmetad_send(NULL, "pv_found", "pvmeta = %t", pvmeta, NULL);
|
|
}
|
|
|
|
dm_config_destroy(pvmeta);
|
|
|
|
result = _lvmetad_handle_reply(reply, "pv_found", uuid, NULL);
|
|
|
|
if (vg && result &&
|
|
(daemon_reply_int(reply, "seqno_after", -1) != vg->seqno ||
|
|
daemon_reply_int(reply, "seqno_after", -1) != daemon_reply_int(reply, "seqno_before", -1)))
|
|
log_warn("WARNING: Inconsistent metadata found for VG %s", vg->name);
|
|
|
|
if (result && found_vgnames) {
|
|
status = daemon_reply_str(reply, "status", NULL);
|
|
vgname = daemon_reply_str(reply, "vgname", NULL);
|
|
changed = daemon_reply_int(reply, "changed", 0);
|
|
}
|
|
|
|
/*
|
|
* If lvmetad now sees all PVs in the VG, it returned the
|
|
* "complete" status string. Add this VG name to the list
|
|
* of found VGs so that the caller can do autoactivation.
|
|
*
|
|
* If there was a problem notifying lvmetad about the new
|
|
* PV, e.g. lvmetad was disabled due to a duplicate, then
|
|
* no autoactivation is attempted.
|
|
*
|
|
* FIXME: there was a previous fixme indicating that
|
|
* autoactivation might also be done for VGs with the
|
|
* "partial" status.
|
|
*
|
|
* If the VG has "changed" by finding the PV, lvmetad returns
|
|
* the "changed" flag. The names of "changed" VGs are saved
|
|
* in the changed_vgnames lists, which is used during autoactivation.
|
|
* If a VG is changed, then autoactivation refreshes LVs in the VG.
|
|
*/
|
|
|
|
if (found_vgnames && vgname && status && !strcmp(status, "complete")) {
|
|
log_debug("VG %s is complete in lvmetad with dev %s.", vgname, dev_name(dev));
|
|
if (!str_list_add(cmd->mem, found_vgnames, dm_pool_strdup(cmd->mem, vgname)))
|
|
log_error("str_list_add failed");
|
|
|
|
if (changed_vgnames && changed) {
|
|
log_debug("VG %s is changed in lvmetad.", vgname);
|
|
if (!str_list_add(cmd->mem, changed_vgnames, dm_pool_strdup(cmd->mem, vgname)))
|
|
log_error("str_list_add failed");
|
|
}
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_gone(dev_t devno, const char *pv_name)
|
|
{
|
|
daemon_reply reply;
|
|
int result;
|
|
int found;
|
|
|
|
if (!lvmetad_used() || test_mode())
|
|
return 1;
|
|
|
|
/*
|
|
* TODO: automatic volume deactivation takes place here *before*
|
|
* all cached info is gone - call handler. Also, consider
|
|
* integrating existing deactivation script that deactivates
|
|
* the whole stack from top to bottom (not yet upstream).
|
|
*/
|
|
|
|
log_debug_lvmetad("Telling lvmetad to forget any PV on %s", pv_name);
|
|
reply = _lvmetad_send(NULL, "pv_gone", "device = %" PRId64, (int64_t) devno, NULL);
|
|
|
|
result = _lvmetad_handle_reply(reply, "pv_gone", pv_name, &found);
|
|
/* We don't care whether or not the daemon had the PV cached. */
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return result;
|
|
}
|
|
|
|
int lvmetad_pv_gone_by_dev(struct device *dev)
|
|
{
|
|
return lvmetad_pv_gone(dev->dev, dev_name(dev));
|
|
}
|
|
|
|
/*
|
|
* The following code implements pvscan --cache.
|
|
*/
|
|
|
|
struct _lvmetad_pvscan_baton {
|
|
struct volume_group *vg;
|
|
struct format_instance *fid;
|
|
};
|
|
|
|
static int _lvmetad_pvscan_single(struct metadata_area *mda, void *baton)
|
|
{
|
|
struct _lvmetad_pvscan_baton *b = baton;
|
|
struct volume_group *vg;
|
|
|
|
if (mda_is_ignored(mda) ||
|
|
!(vg = mda->ops->vg_read(b->fid, "", mda, NULL, NULL, 1)))
|
|
return 1;
|
|
|
|
/* FIXME Also ensure contents match etc. */
|
|
if (!b->vg || vg->seqno > b->vg->seqno)
|
|
b->vg = vg;
|
|
else if (b->vg)
|
|
release_vg(vg);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* The lock manager may detect that the vg cached in lvmetad is out of date,
|
|
* due to something like an lvcreate from another host.
|
|
* This is limited to changes that only affect the vg (not global state like
|
|
* orphan PVs), so we only need to reread mdas on the vg's existing pvs.
|
|
* But, a previous PV in the VG may have been removed since we last read
|
|
* the VG, and that PV may have been reused for another VG.
|
|
*/
|
|
|
|
static struct volume_group *lvmetad_pvscan_vg(struct cmd_context *cmd, struct volume_group *vg)
|
|
{
|
|
char pvid_s[ID_LEN + 1] __attribute__((aligned(8)));
|
|
char uuid[64] __attribute__((aligned(8)));
|
|
struct label *label;
|
|
struct volume_group *vg_ret = NULL;
|
|
struct dm_config_tree *vgmeta_ret = NULL;
|
|
struct dm_config_tree *vgmeta;
|
|
struct pv_list *pvl, *pvl_new;
|
|
struct device_list *devl, *devl_new, *devlsafe;
|
|
struct dm_list pvs_scan;
|
|
struct dm_list pvs_drop;
|
|
struct dm_list pvs_new;
|
|
struct lvmcache_info *info = NULL;
|
|
struct format_instance *fid;
|
|
struct format_instance_ctx fic = { .type = 0 };
|
|
struct _lvmetad_pvscan_baton baton;
|
|
struct device *save_dev = NULL;
|
|
uint32_t save_seqno = 0;
|
|
int missing_devs = 0;
|
|
int check_new_pvs = 0;
|
|
int found;
|
|
|
|
dm_list_init(&pvs_scan);
|
|
dm_list_init(&pvs_drop);
|
|
dm_list_init(&pvs_new);
|
|
|
|
log_debug_lvmetad("Rescanning VG %s (seqno %u).", vg->name, vg->seqno);
|
|
|
|
/*
|
|
* Another host may have added a PV to the VG, and some
|
|
* commands do not always populate their lvmcache with
|
|
* all devs from lvmetad, so they would fail to find
|
|
* the new PV when scanning the VG. So make sure this
|
|
* command knows about all PVs from lvmetad.
|
|
*/
|
|
lvmcache_seed_infos_from_lvmetad(cmd);
|
|
|
|
/*
|
|
* Start with the list of PVs that we last saw in the VG.
|
|
* Some may now be gone, and some new PVs may have been added.
|
|
*/
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (!(devl = dm_pool_zalloc(cmd->mem, sizeof(*devl))))
|
|
return_NULL;
|
|
devl->dev = pvl->pv->dev;
|
|
dm_list_add(&pvs_scan, &devl->list);
|
|
}
|
|
|
|
scan_more:
|
|
|
|
/*
|
|
* Run the equivalent of lvmetad_pvscan_single on each dev in the VG.
|
|
*/
|
|
dm_list_iterate_items_safe(devl, devlsafe, &pvs_scan) {
|
|
if (!devl->dev)
|
|
continue;
|
|
|
|
log_debug_lvmetad("Rescan VG %s scanning %s.", vg->name, dev_name(devl->dev));
|
|
|
|
if (!label_read(devl->dev, &label, 0)) {
|
|
/* Another host removed this PV from the VG. */
|
|
log_debug_lvmetad("Rescan VG %s found %s was removed.", vg->name, dev_name(devl->dev));
|
|
|
|
if ((info = lvmcache_info_from_pvid(devl->dev->pvid, NULL, 0)))
|
|
lvmcache_del(info);
|
|
|
|
dm_list_move(&pvs_drop, &devl->list);
|
|
continue;
|
|
}
|
|
|
|
info = (struct lvmcache_info *) label->info;
|
|
|
|
baton.vg = NULL;
|
|
baton.fid = lvmcache_fmt(info)->ops->create_instance(lvmcache_fmt(info), &fic);
|
|
if (!baton.fid)
|
|
return_NULL;
|
|
|
|
if (baton.fid->fmt->features & FMT_OBSOLETE) {
|
|
log_debug_lvmetad("Ignoring obsolete format on PV %s in VG %s.", dev_name(devl->dev), vg->name);
|
|
lvmcache_fmt(info)->ops->destroy_instance(baton.fid);
|
|
dm_list_move(&pvs_drop, &devl->list);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Read VG metadata from this dev's mdas.
|
|
*/
|
|
lvmcache_foreach_mda(info, _lvmetad_pvscan_single, &baton);
|
|
|
|
/*
|
|
* The PV may have been removed from the VG by another host
|
|
* since we last read the VG.
|
|
*/
|
|
if (!baton.vg) {
|
|
log_debug_lvmetad("Rescan VG %s did not find %s.", vg->name, dev_name(devl->dev));
|
|
lvmcache_fmt(info)->ops->destroy_instance(baton.fid);
|
|
dm_list_move(&pvs_drop, &devl->list);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* The PV may have been removed from the VG and used for a
|
|
* different VG since we last read the VG.
|
|
*/
|
|
if (strcmp(baton.vg->name, vg->name)) {
|
|
log_debug_lvmetad("Rescan VG %s found different VG %s on PV %s.",
|
|
vg->name, baton.vg->name, dev_name(devl->dev));
|
|
release_vg(baton.vg);
|
|
dm_list_move(&pvs_drop, &devl->list);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* The VG metadata read from each dev should match. Save the
|
|
* metadata from the first dev, and compare it to the metadata
|
|
* read from each other dev.
|
|
*/
|
|
|
|
if (!save_seqno)
|
|
save_seqno = baton.vg->seqno;
|
|
|
|
if (!(vgmeta = export_vg_to_config_tree(baton.vg))) {
|
|
log_error("VG export to config tree failed");
|
|
release_vg(baton.vg);
|
|
return NULL;
|
|
}
|
|
|
|
if (!vgmeta_ret) {
|
|
vgmeta_ret = vgmeta;
|
|
save_dev = devl->dev;
|
|
} else {
|
|
if (compare_config(vgmeta_ret->root, vgmeta->root)) {
|
|
log_error("VG %s metadata comparison failed for device %s vs %s",
|
|
vg->name, dev_name(devl->dev), save_dev ? dev_name(save_dev) : "none");
|
|
_log_debug_inequality(vg->name, vgmeta_ret->root, vgmeta->root);
|
|
dm_config_destroy(vgmeta);
|
|
dm_config_destroy(vgmeta_ret);
|
|
release_vg(baton.vg);
|
|
return NULL;
|
|
}
|
|
dm_config_destroy(vgmeta);
|
|
}
|
|
|
|
/*
|
|
* Look for any new PVs in the VG metadata that were not in our
|
|
* previous version of the VG. Add them to pvs_new to be
|
|
* scanned in this loop just like the old PVs.
|
|
*/
|
|
if (!check_new_pvs) {
|
|
check_new_pvs = 1;
|
|
dm_list_iterate_items(pvl_new, &baton.vg->pvs) {
|
|
found = 0;
|
|
dm_list_iterate_items(pvl, &vg->pvs) {
|
|
if (pvl_new->pv->dev != pvl->pv->dev)
|
|
continue;
|
|
found = 1;
|
|
break;
|
|
}
|
|
if (found)
|
|
continue;
|
|
if (!pvl_new->pv->dev) {
|
|
strncpy(pvid_s, (char *) &pvl_new->pv->id, sizeof(pvid_s) - 1);
|
|
if (!id_write_format((const struct id *)&pvid_s, uuid, sizeof(uuid)))
|
|
stack;
|
|
log_error("Device not found for PV %s in VG %s", uuid, vg->name);
|
|
missing_devs++;
|
|
continue;
|
|
}
|
|
if (!(devl_new = dm_pool_zalloc(cmd->mem, sizeof(*devl_new))))
|
|
return_NULL;
|
|
devl_new->dev = pvl_new->pv->dev;
|
|
dm_list_add(&pvs_new, &devl_new->list);
|
|
log_debug_lvmetad("Rescan VG %s found %s was added.", vg->name, dev_name(devl_new->dev));
|
|
}
|
|
}
|
|
|
|
release_vg(baton.vg);
|
|
}
|
|
|
|
/*
|
|
* Do the same scanning above for any new PVs.
|
|
*/
|
|
if (!dm_list_empty(&pvs_new)) {
|
|
dm_list_init(&pvs_scan);
|
|
dm_list_splice(&pvs_scan, &pvs_new);
|
|
dm_list_init(&pvs_new);
|
|
log_debug_lvmetad("Rescan VG %s found new PVs to scan.", vg->name);
|
|
goto scan_more;
|
|
}
|
|
|
|
if (missing_devs) {
|
|
if (vgmeta_ret)
|
|
dm_config_destroy(vgmeta_ret);
|
|
return_NULL;
|
|
}
|
|
|
|
/*
|
|
* Remove pvs_drop entries from lvmetad.
|
|
*/
|
|
dm_list_iterate_items(devl, &pvs_drop) {
|
|
if (!devl->dev)
|
|
continue;
|
|
log_debug_lvmetad("Rescan VG %s dropping %s.", vg->name, dev_name(devl->dev));
|
|
if (!lvmetad_pv_gone_by_dev(devl->dev))
|
|
return_NULL;
|
|
}
|
|
|
|
/*
|
|
* Update the VG in lvmetad.
|
|
*/
|
|
if (vgmeta_ret) {
|
|
fid = lvmcache_fmt(info)->ops->create_instance(lvmcache_fmt(info), &fic);
|
|
if (!(vg_ret = import_vg_from_config_tree(vgmeta_ret, fid))) {
|
|
log_error("VG import from config tree failed");
|
|
lvmcache_fmt(info)->ops->destroy_instance(fid);
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Update lvmetad with the newly read version of the VG.
|
|
* When the seqno is unchanged the cached VG can be left.
|
|
*/
|
|
if (save_seqno != vg->seqno) {
|
|
dm_list_iterate_items(devl, &pvs_scan) {
|
|
if (!devl->dev)
|
|
continue;
|
|
log_debug_lvmetad("Rescan VG %s dropping to replace %s.", vg->name, dev_name(devl->dev));
|
|
if (!lvmetad_pv_gone_by_dev(devl->dev))
|
|
return_NULL;
|
|
}
|
|
|
|
log_debug_lvmetad("Rescan VG %s updating lvmetad from seqno %u to seqno %u.",
|
|
vg->name, vg->seqno, save_seqno);
|
|
|
|
/*
|
|
* If this vg_update fails the cached metadata in
|
|
* lvmetad will remain invalid.
|
|
*/
|
|
vg_ret->lvmetad_update_pending = 1;
|
|
if (!lvmetad_vg_update_finish(vg_ret))
|
|
log_error("Failed to update lvmetad with new VG meta");
|
|
}
|
|
dm_config_destroy(vgmeta_ret);
|
|
}
|
|
out:
|
|
log_debug_lvmetad("Rescan VG %s done (seqno %u).", vg_ret->name, vg_ret->seqno);
|
|
return vg_ret;
|
|
}
|
|
|
|
int lvmetad_pvscan_single(struct cmd_context *cmd, struct device *dev,
|
|
struct dm_list *found_vgnames,
|
|
struct dm_list *changed_vgnames)
|
|
{
|
|
struct label *label;
|
|
struct lvmcache_info *info;
|
|
struct _lvmetad_pvscan_baton baton;
|
|
/* Create a dummy instance. */
|
|
struct format_instance_ctx fic = { .type = 0 };
|
|
|
|
if (!lvmetad_used()) {
|
|
log_error("Cannot proceed since lvmetad is not active.");
|
|
return 0;
|
|
}
|
|
|
|
if (!label_read(dev, &label, 0)) {
|
|
log_print_unless_silent("No PV label found on %s.", dev_name(dev));
|
|
if (!lvmetad_pv_gone_by_dev(dev))
|
|
goto_bad;
|
|
return 1;
|
|
}
|
|
|
|
info = (struct lvmcache_info *) label->info;
|
|
|
|
baton.vg = NULL;
|
|
baton.fid = lvmcache_fmt(info)->ops->create_instance(lvmcache_fmt(info), &fic);
|
|
|
|
if (!baton.fid)
|
|
goto_bad;
|
|
|
|
if (baton.fid->fmt->features & FMT_OBSOLETE) {
|
|
lvmcache_fmt(info)->ops->destroy_instance(baton.fid);
|
|
log_warn("WARNING: Disabling lvmetad cache which does not support obsolete (lvm1) metadata.");
|
|
lvmetad_set_disabled(cmd, LVMETAD_DISABLE_REASON_LVM1);
|
|
_found_lvm1_metadata = 1;
|
|
/*
|
|
* return 1 (success) so that we'll continue to populate lvmetad
|
|
* instead of leaving the update incomplete.
|
|
*/
|
|
return 1;
|
|
}
|
|
|
|
lvmcache_foreach_mda(info, _lvmetad_pvscan_single, &baton);
|
|
|
|
if (!baton.vg)
|
|
lvmcache_fmt(info)->ops->destroy_instance(baton.fid);
|
|
|
|
if (!lvmetad_pv_found(cmd, (const struct id *) &dev->pvid, dev, lvmcache_fmt(info),
|
|
label->sector, baton.vg, found_vgnames, changed_vgnames)) {
|
|
release_vg(baton.vg);
|
|
goto_bad;
|
|
}
|
|
|
|
release_vg(baton.vg);
|
|
return 1;
|
|
|
|
bad:
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Update the lvmetad cache: clear the current lvmetad cache, and scan all
|
|
* devs, sending all info from the devs to lvmetad.
|
|
*
|
|
* We want only one command to be doing this at a time. When do_wait is set,
|
|
* this will first check if lvmetad is currently being updated by another
|
|
* command, and if so it will delay until that update is finished, or until a
|
|
* timeout, at which point it will go ahead and do the lvmetad update.
|
|
*
|
|
* Callers that have already checked and waited for the updating state, e.g. by
|
|
* using lvmetad_token_matches(), will generaly set do_wait to 0. Callers that
|
|
* have not checked for the updating state yet will generally set do_wait to 1.
|
|
*
|
|
* If another command doing an update failed, it left lvmetad in the "update in
|
|
* progess" state, so we can't just wait until that state has cleared, but have
|
|
* to go ahead after a timeout.
|
|
*
|
|
* The _lvmetad_is_updating check avoids most races to update lvmetad from
|
|
* multiple commands (which shouldn't generally happen anway) but does not
|
|
* eliminate them. If an update race happens, the second will see that the
|
|
* previous token was "update in progress" when it calls _token_update(). It
|
|
* will then fail, and the command calling lvmetad_pvscan_all_devs() will
|
|
* generally revert disk scanning and not use lvmetad.
|
|
*/
|
|
|
|
int lvmetad_pvscan_all_devs(struct cmd_context *cmd, int do_wait)
|
|
{
|
|
struct dev_iter *iter;
|
|
struct device *dev;
|
|
daemon_reply reply;
|
|
char *future_token;
|
|
const char *reason;
|
|
int was_silent;
|
|
int replacing_other_update = 0;
|
|
int replaced_update = 0;
|
|
int retries = 0;
|
|
int ret = 1;
|
|
|
|
if (!lvmetad_used()) {
|
|
log_error("Cannot proceed since lvmetad is not active.");
|
|
return 0;
|
|
}
|
|
|
|
retry:
|
|
/*
|
|
* If another update is in progress, delay to allow it to finish,
|
|
* rather than interrupting it with our own update.
|
|
*/
|
|
if (do_wait && _lvmetad_is_updating(cmd, 1)) {
|
|
log_warn("WARNING: lvmetad update is interrupting another update in progress.");
|
|
replacing_other_update = 1;
|
|
}
|
|
|
|
log_verbose("Scanning all devices to update lvmetad.");
|
|
|
|
if (!(iter = dev_iter_create(cmd->lvmetad_filter, 1))) {
|
|
log_error("dev_iter creation failed");
|
|
return 0;
|
|
}
|
|
|
|
future_token = _lvmetad_token;
|
|
_lvmetad_token = (char *) LVMETAD_TOKEN_UPDATE_IN_PROGRESS;
|
|
|
|
if (!_token_update(&replaced_update)) {
|
|
log_error("Failed to update lvmetad which had an update in progress.");
|
|
dev_iter_destroy(iter);
|
|
_lvmetad_token = future_token;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* if _token_update() sets replaced_update to 1, it means that we set
|
|
* "update in progress" when the lvmetad was already set to "udpate in
|
|
* progress". This detects a race between two commands doing updates
|
|
* at once. The attempt above to avoid this race using
|
|
* _lvmetad_is_updating isn't perfect.
|
|
*/
|
|
if (!replacing_other_update && replaced_update) {
|
|
if (do_wait && !retries) {
|
|
retries = 1;
|
|
log_warn("WARNING: lvmetad update in progress, retrying update.");
|
|
dev_iter_destroy(iter);
|
|
_lvmetad_token = future_token;
|
|
goto retry;
|
|
}
|
|
log_warn("WARNING: lvmetad update in progress, skipping update.");
|
|
dev_iter_destroy(iter);
|
|
_lvmetad_token = future_token;
|
|
return 0;
|
|
}
|
|
|
|
log_debug_lvmetad("Telling lvmetad to clear its cache");
|
|
reply = _lvmetad_send(cmd, "pv_clear_all", NULL);
|
|
if (!_lvmetad_handle_reply(reply, "pv_clear_all", "", NULL))
|
|
ret = 0;
|
|
daemon_reply_destroy(reply);
|
|
|
|
was_silent = silent_mode();
|
|
init_silent(1);
|
|
|
|
while ((dev = dev_iter_get(iter))) {
|
|
if (sigint_caught()) {
|
|
ret = 0;
|
|
stack;
|
|
break;
|
|
}
|
|
|
|
if (!lvmetad_pvscan_single(cmd, dev, NULL, NULL)) {
|
|
ret = 0;
|
|
stack;
|
|
break;
|
|
}
|
|
}
|
|
|
|
init_silent(was_silent);
|
|
|
|
dev_iter_destroy(iter);
|
|
|
|
_lvmetad_token = future_token;
|
|
|
|
/*
|
|
* If we failed to fully and successfully populate lvmetad just leave
|
|
* the existing "update in progress" token in place so lvmetad will
|
|
* time out our update and force another command to do it.
|
|
* (We could try to set the token to empty here, but that doesn't
|
|
* help much.)
|
|
*/
|
|
if (!ret)
|
|
return 0;
|
|
|
|
if (!_token_update(NULL)) {
|
|
log_error("Failed to update lvmetad token after device scan.");
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* If lvmetad is disabled, and no lvm1 metadata was seen and no
|
|
* duplicate PVs were seen, then re-enable lvmetad.
|
|
*/
|
|
if (lvmetad_is_disabled(cmd, &reason) &&
|
|
!lvmcache_found_duplicate_pvs() && !_found_lvm1_metadata) {
|
|
log_debug_lvmetad("Enabling lvmetad which was previously disabled.");
|
|
lvmetad_clear_disabled(cmd);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int lvmetad_vg_clear_outdated_pvs(struct volume_group *vg)
|
|
{
|
|
char uuid[64];
|
|
daemon_reply reply;
|
|
int result;
|
|
|
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Sending lvmetad vg_clear_outdated_pvs");
|
|
reply = _lvmetad_send(vg->cmd, "vg_clear_outdated_pvs", "vgid = %s", uuid, NULL);
|
|
result = _lvmetad_handle_reply(reply, "vg_clear_outdated_pvs", vg->name, NULL);
|
|
daemon_reply_destroy(reply);
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Records the state of cached PVs in lvmetad so we can look for changes
|
|
* after rescanning.
|
|
*/
|
|
struct pv_cache_list {
|
|
struct dm_list list;
|
|
dev_t devt;
|
|
struct id pvid;
|
|
const char *vgid;
|
|
unsigned found : 1;
|
|
unsigned update_udev : 1;
|
|
};
|
|
|
|
/*
|
|
* Get the list of PVs known to lvmetad.
|
|
*/
|
|
static int _lvmetad_get_pv_cache_list(struct cmd_context *cmd, struct dm_list *pvc_list)
|
|
{
|
|
daemon_reply reply;
|
|
struct dm_config_node *cn;
|
|
struct pv_cache_list *pvcl;
|
|
const char *pvid_txt;
|
|
const char *vgid;
|
|
|
|
if (!lvmetad_used())
|
|
return 1;
|
|
|
|
log_debug_lvmetad("Asking lvmetad for complete list of known PVs");
|
|
|
|
reply = _lvmetad_send(cmd, "pv_list", NULL);
|
|
if (!_lvmetad_handle_reply(reply, "pv_list", "", NULL)) {
|
|
daemon_reply_destroy(reply);
|
|
return_0;
|
|
}
|
|
|
|
if ((cn = dm_config_find_node(reply.cft->root, "physical_volumes"))) {
|
|
for (cn = cn->child; cn; cn = cn->sib) {
|
|
if (!(pvcl = dm_pool_zalloc(cmd->mem, sizeof(*pvcl)))) {
|
|
log_error("pv_cache_list allocation failed.");
|
|
return 0;
|
|
}
|
|
|
|
pvid_txt = cn->key;
|
|
if (!id_read_format(&pvcl->pvid, pvid_txt)) {
|
|
stack;
|
|
continue;
|
|
}
|
|
|
|
pvcl->devt = dm_config_find_int(cn->child, "device", 0);
|
|
|
|
if ((vgid = dm_config_find_str(cn->child, "vgid", NULL)))
|
|
pvcl->vgid = dm_pool_strdup(cmd->mem, vgid);
|
|
|
|
dm_list_add(pvc_list, &pvcl->list);
|
|
}
|
|
}
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Opening the device RDWR should trigger a udev db update.
|
|
* FIXME: is there a better way to update the udev db than
|
|
* doing an open/close of the device? - For example writing
|
|
* "change" to /sys/block/<device>/uevent?
|
|
*/
|
|
static void _update_pv_in_udev(struct cmd_context *cmd, dev_t devt)
|
|
{
|
|
struct device *dev;
|
|
|
|
log_debug_devs("device %d:%d open to update udev",
|
|
(int)MAJOR(devt), (int)MINOR(devt));
|
|
|
|
if (!(dev = dev_cache_get_by_devt(devt, cmd->lvmetad_filter))) {
|
|
log_error("_update_pv_in_udev no dev found");
|
|
return;
|
|
}
|
|
|
|
if (!dev_open(dev)) {
|
|
stack;
|
|
return;
|
|
}
|
|
|
|
if (!dev_close(dev))
|
|
stack;
|
|
}
|
|
|
|
/*
|
|
* Compare before and after PV lists from before/after rescanning,
|
|
* and update udev db for changes.
|
|
*
|
|
* For PVs that have changed pvid or vgid in lvmetad from rescanning,
|
|
* there may be information in the udev database to update, so open
|
|
* these devices to trigger a udev update.
|
|
*
|
|
* "before" refers to the list of pvs from lvmetad before rescanning
|
|
* "after" refers to the list of pvs from lvmetad after rescanning
|
|
*
|
|
* Comparing both lists, we can see which PVs changed (pvid or vgid),
|
|
* and trigger a udev db update for those.
|
|
*/
|
|
static void _update_changed_pvs_in_udev(struct cmd_context *cmd,
|
|
struct dm_list *pvc_before,
|
|
struct dm_list *pvc_after)
|
|
{
|
|
struct pv_cache_list *before;
|
|
struct pv_cache_list *after;
|
|
char id_before[ID_LEN + 1];
|
|
char id_after[ID_LEN + 1];
|
|
int found;
|
|
|
|
dm_list_iterate_items(before, pvc_before) {
|
|
found = 0;
|
|
|
|
dm_list_iterate_items(after, pvc_after) {
|
|
if (after->found)
|
|
continue;
|
|
|
|
if (before->devt != after->devt)
|
|
continue;
|
|
|
|
if (!id_equal(&before->pvid, &after->pvid)) {
|
|
(void) dm_strncpy(id_before, (char *) &before->pvid, sizeof(id_before));
|
|
(void) dm_strncpy(id_after, (char *) &after->pvid, sizeof(id_after));
|
|
|
|
log_debug_devs("device %d:%d changed pvid from %s to %s",
|
|
(int)MAJOR(before->devt), (int)MINOR(before->devt),
|
|
id_before, id_after);
|
|
|
|
before->update_udev = 1;
|
|
|
|
} else if ((before->vgid && !after->vgid) ||
|
|
(after->vgid && !before->vgid) ||
|
|
(before->vgid && after->vgid && strcmp(before->vgid, after->vgid))) {
|
|
|
|
log_debug_devs("device %d:%d changed vg from %s to %s",
|
|
(int)MAJOR(before->devt), (int)MINOR(before->devt),
|
|
before->vgid ?: "none", after->vgid ?: "none");
|
|
|
|
before->update_udev = 1;
|
|
}
|
|
|
|
after->found = 1;
|
|
before->found = 1;
|
|
found = 1;
|
|
break;
|
|
}
|
|
|
|
if (!found) {
|
|
(void) dm_strncpy(id_before, (char *) &before->pvid, sizeof(id_before));
|
|
|
|
log_debug_devs("device %d:%d pvid %s vg %s is gone",
|
|
(int)MAJOR(before->devt), (int)MINOR(before->devt),
|
|
id_before, before->vgid ? before->vgid : "none");
|
|
|
|
before->update_udev = 1;
|
|
}
|
|
}
|
|
|
|
dm_list_iterate_items(before, pvc_before) {
|
|
if (before->update_udev)
|
|
_update_pv_in_udev(cmd, before->devt);
|
|
}
|
|
|
|
dm_list_iterate_items(after, pvc_after) {
|
|
if (after->update_udev)
|
|
_update_pv_in_udev(cmd, after->devt);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Before this command was run, some external entity may have
|
|
* invalidated lvmetad's cache of global information, e.g. lvmlockd.
|
|
*
|
|
* The global information includes things like a new VG, a
|
|
* VG that was removed, the assignment of a PV to a VG;
|
|
* any change that is not isolated within a single VG.
|
|
*
|
|
* The external entity, like a lock manager, would invalidate
|
|
* the lvmetad global cache if it detected that the global
|
|
* information had been changed on disk by something other
|
|
* than a local lvm command, e.g. an lvm command on another
|
|
* host with access to the same devices. (How it detects
|
|
* the change is specific to lock manager or other entity.)
|
|
*
|
|
* The effect is that metadata on disk is newer than the metadata
|
|
* in the local lvmetad daemon, and the local lvmetad's cache
|
|
* should be updated from disk before this command uses it.
|
|
*
|
|
* So, using this function, a command checks if lvmetad's global
|
|
* cache is valid. If so, it does nothing. If not, it rescans
|
|
* devices to update the lvmetad cache, then it notifies lvmetad
|
|
* that it's cache is valid again (consistent with what's on disk.)
|
|
* This command can then go ahead and use the newly refreshed metadata.
|
|
*
|
|
* 1. Check if the lvmetad global cache is invalid.
|
|
* 2. If so, reread metadata from all devices and update the lvmetad cache.
|
|
* 3. Tell lvmetad that the global cache is now valid.
|
|
*/
|
|
|
|
void lvmetad_validate_global_cache(struct cmd_context *cmd, int force)
|
|
{
|
|
struct dm_list pvc_before; /* pv_cache_list */
|
|
struct dm_list pvc_after; /* pv_cache_list */
|
|
const char *reason = NULL;
|
|
daemon_reply reply;
|
|
int global_invalid;
|
|
|
|
dm_list_init(&pvc_before);
|
|
dm_list_init(&pvc_after);
|
|
|
|
if (!lvmlockd_use()) {
|
|
log_error(INTERNAL_ERROR "validate global cache without lvmlockd");
|
|
return;
|
|
}
|
|
|
|
if (!lvmetad_used())
|
|
return;
|
|
|
|
log_debug_lvmetad("Validating global lvmetad cache");
|
|
|
|
if (force)
|
|
goto do_scan;
|
|
|
|
log_debug_lvmetad("lvmetad validate send get_global_info");
|
|
|
|
reply = daemon_send_simple(_lvmetad, "get_global_info",
|
|
"token = %s", "skip",
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
|
|
if (reply.error) {
|
|
log_error("lvmetad_validate_global_cache get_global_info error %d", reply.error);
|
|
goto do_scan;
|
|
}
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
log_error("lvmetad_validate_global_cache get_global_info not ok");
|
|
goto do_scan;
|
|
}
|
|
|
|
global_invalid = daemon_reply_int(reply, "global_invalid", -1);
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
if (!global_invalid)
|
|
return; /* cache is valid */
|
|
|
|
do_scan:
|
|
/*
|
|
* Save the current state of pvs from lvmetad so after devices are
|
|
* scanned, we can compare to the new state to see if pvs changed.
|
|
*/
|
|
_lvmetad_get_pv_cache_list(cmd, &pvc_before);
|
|
|
|
/*
|
|
* Update the local lvmetad cache so it correctly reflects any
|
|
* changes made on remote hosts. (It's possible that this command
|
|
* already refreshed the local lvmetad because of a token change,
|
|
* but we need to do it again here since we now hold the global
|
|
* lock. Another host may have changed things between the time
|
|
* we rescanned for the token, and the time we acquired the global
|
|
* lock.)
|
|
*/
|
|
if (!lvmetad_pvscan_all_devs(cmd, 1)) {
|
|
log_warn("WARNING: Not using lvmetad because cache update failed.");
|
|
lvmetad_make_unused(cmd);
|
|
return;
|
|
}
|
|
|
|
if (lvmetad_is_disabled(cmd, &reason)) {
|
|
log_warn("WARNING: Not using lvmetad because %s.", reason);
|
|
lvmetad_make_unused(cmd);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Clear the global_invalid flag in lvmetad.
|
|
* Subsequent local commands that read global state
|
|
* from lvmetad will not see global_invalid until
|
|
* another host makes another global change.
|
|
*/
|
|
log_debug_lvmetad("lvmetad validate send set_global_info");
|
|
|
|
reply = daemon_send_simple(_lvmetad, "set_global_info",
|
|
"token = %s", "skip",
|
|
"global_invalid = " FMTd64, INT64_C(0),
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
if (reply.error)
|
|
log_error("lvmetad_validate_global_cache set_global_info error %d", reply.error);
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK"))
|
|
log_error("lvmetad_validate_global_cache set_global_info not ok");
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
/*
|
|
* Populate this command's lvmcache structures from lvmetad.
|
|
*/
|
|
lvmcache_seed_infos_from_lvmetad(cmd);
|
|
|
|
/*
|
|
* Update the local udev database to reflect PV changes from
|
|
* other hosts.
|
|
*
|
|
* Compare the before and after PV lists, and if a PV's
|
|
* pvid or vgid has changed, then open that device to trigger
|
|
* a uevent to update the udev db.
|
|
*
|
|
* This has no direct benefit to lvm, but is just a best effort
|
|
* attempt to keep the udev db updated and reflecting current
|
|
* lvm information.
|
|
*
|
|
* FIXME: lvmcache_seed_infos_from_lvmetad() and _lvmetad_get_pv_cache_list()
|
|
* each get pv_list from lvmetad, and they could share a single pv_list reply.
|
|
*/
|
|
if (!dm_list_empty(&pvc_before)) {
|
|
_lvmetad_get_pv_cache_list(cmd, &pvc_after);
|
|
_update_changed_pvs_in_udev(cmd, &pvc_before, &pvc_after);
|
|
}
|
|
|
|
log_debug_lvmetad("Validating global lvmetad cache finished");
|
|
}
|
|
|
|
int lvmetad_vg_is_foreign(struct cmd_context *cmd, const char *vgname, const char *vgid)
|
|
{
|
|
daemon_reply reply;
|
|
struct dm_config_node *top;
|
|
const char *system_id = NULL;
|
|
char uuid[64];
|
|
int ret;
|
|
|
|
if (!id_write_format((const struct id*)vgid, uuid, sizeof(uuid)))
|
|
return_0;
|
|
|
|
log_debug_lvmetad("Sending lvmetad vg_clear_outdated_pvs");
|
|
reply = _lvmetad_send(cmd, "vg_lookup",
|
|
"uuid = %s", uuid,
|
|
"name = %s", vgname,
|
|
NULL);
|
|
|
|
if ((top = dm_config_find_node(reply.cft->root, "metadata")))
|
|
system_id = dm_config_find_str(top, "metadata/system_id", NULL);
|
|
|
|
ret = !is_system_id_allowed(cmd, system_id);
|
|
|
|
daemon_reply_destroy(reply);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* lvmetad has a disabled state in which it continues running,
|
|
* and returns the "disabled" flag in a get_global_info query.
|
|
*
|
|
* Case 1
|
|
* ------
|
|
* When "normal" commands start, (those not specifically
|
|
* intended to rescan devs) they begin by checking lvmetad's
|
|
* token and global info:
|
|
*
|
|
* - If the token doesn't match (should be uncommon), the
|
|
* command first rescans devices to repopulate lvmetad with
|
|
* the global_filter it is using. After rescanning, the
|
|
* lvmetad disabled state is set or cleared depending on
|
|
* what the scan saw.
|
|
*
|
|
* An unmatching token occurs when:
|
|
* . lvmetad was just started and has not been populated yet.
|
|
* . The global_filter has been changed in lvm.conf since the
|
|
* last command was run.
|
|
* . The global_filter is overriden on the command line.
|
|
* (There's little point in using lvmetad if global_filter
|
|
* is often changed/overridden.)
|
|
*
|
|
* - If the token does match (common case), the command and
|
|
* lvmetad are using the same global_filter and the command
|
|
* does not rescan devs to repopulate lvmetad, or change the
|
|
* lvmetad disabled state.
|
|
*
|
|
* - After the token check/sync, the command checks if the
|
|
* disabled flag is set in lvmetad. If it is, the command will
|
|
* not use the lvmetad cache and will revert to scanning, i.e.
|
|
* it runs the same as if use_lvmetad=0.
|
|
*
|
|
* So, "normal" commands try to use the lvmetad cache to avoid
|
|
* scanning devices. In the uncommon case when the token doesn't
|
|
* match, these commands will first rescan devs to repopulate the
|
|
* lvmetad cache, and then attempt to use the lvmetad cache.
|
|
* In the uncommon case where lvmetad is disabled (by a previous
|
|
* command), the common commands do not rescan devs to repopulate
|
|
* lvmetad, but revert the equivalent of use_lvmetad=0, reading
|
|
* from disk instead of the cache.
|
|
* The combination of those two uncommon cases means that a command
|
|
* could begin by rescanning devs because of a token mismatch, then
|
|
* disable lvmetad as a result of that scan, and continue without
|
|
* using lvmetad.
|
|
*
|
|
* Case 2
|
|
* ------
|
|
* Commands that are meant to scan devices to repopulate the
|
|
* lvmetad cache, e.g. pvscan --cache, will always rescan
|
|
* devices and then set/clear the disabled state according to
|
|
* what they found when scanning. The global_filter is always
|
|
* used when choosing which devices to scan to populate lvmetad.
|
|
* The command-specific filter is never used when choosing
|
|
* which devices to scan for repopulating the lvmetad cache.
|
|
*
|
|
* During a scan repopulating the lvmetad cache, a command looks
|
|
* for PVs with lvm1 metadata, or duplicate PVs (two devices with
|
|
* the same PVID). If either of those are found during the scan,
|
|
* the command sets the disabled state in lvmetad. If none are
|
|
* found, the command clears the disabled state in lvmetad.
|
|
* (Other problems scanning may also cause the command to set the
|
|
* disabled state.)
|
|
*
|
|
* Case 3
|
|
* ------
|
|
* The special command 'pvscan --cache <dev>' is meant to only
|
|
* scan the specified device and send info from the dev to
|
|
* lvmetad. This single-dev pvscan will not detect duplicate PVs
|
|
* since it only sees the one device. If lvmetad already knows
|
|
* about the same PV on another device, then lvmetad will be the
|
|
* first to discover that a duplicate PV exists. In this case,
|
|
* lvmetad sets the disabled state for itself.
|
|
*
|
|
* Duplicates
|
|
* ----------
|
|
* The most common reasons for duplicate PVs to exist are:
|
|
*
|
|
* 1. Multipath. When multipath is running, it creates a new
|
|
* mpath device for the underlying "duplicate" devs. lvm has
|
|
* built in, automatic filtering that will hide the duplicate
|
|
* devs of the underlying mpath dev, so the duplicates will
|
|
* be skipping during scanning (multipath_component_detection).
|
|
*
|
|
* If multipath_component_detection=0, or if multipathd is not
|
|
* running, or multipath is not set up to handle a particular
|
|
* set of devs, then lvm will see the multipath paths as
|
|
* duplicates. lvm will choose one of them to use, consider
|
|
* the other a duplicate, and disable lvmetad. multipathd
|
|
* should be configured and running to resolve these duplicates,
|
|
* and multipath_component_detection enabled.
|
|
*
|
|
* 2. Cloning by copying. One device is copied over another, e.g.
|
|
* with dd. This is a more concerning case because using the
|
|
* wrong device could lead to corruption. LVM will attempt to
|
|
* choose the best device as the PV, but it may not always
|
|
* be the right one. In this case, lvmetad is disabled.
|
|
* vgimportclone should be used on the new copy to resolve the
|
|
* duplicates.
|
|
*
|
|
* 3. Cloning by hardware. A LUN is cloned/snapshotted on
|
|
* a hardware device. The description here is the same as
|
|
* cloning by copying.
|
|
*
|
|
* 4. Creating LVM snapshots of LVs being used as PVs.
|
|
* If pvcreate is run on an LV, and lvcreate is used to
|
|
* create a snapshot of that LV, then the two LVs will
|
|
* appear to be duplicate PVs.
|
|
*
|
|
* Filtering duplicates
|
|
* --------------------
|
|
*
|
|
* If all but one copy of a PV is added to the global_filter,
|
|
* then duplicates will not be seen when scanning to populate
|
|
* the lvmetad cache. Neither common commands nor scanning
|
|
* commands will see the duplicates, and lvmetad will not be
|
|
* disabled.
|
|
*
|
|
* If the global_filter is *not* used to hide duplicates,
|
|
* then lvmetad will be disabled when they are scanned, but
|
|
* common commands can use the command filter to hide the
|
|
* duplicates and work with a selected instance of the PV.
|
|
* The command will not use lvmetad in this case, but will
|
|
* not see duplicate PVs itself because its command filter
|
|
* is more restrictive than the global_filter and has hidden
|
|
* the duplicates.
|
|
*/
|
|
|
|
/*
|
|
* FIXME: if we fail to disable lvmetad, then other commands could
|
|
* potentially use incorrect cache data from lvmetad. Should we
|
|
* do something more severe if the disable messages fails, like
|
|
* sending SIGKILL to the lvmetad pid?
|
|
*
|
|
* FIXME: log something in syslog any time we disable lvmetad?
|
|
* At a minimum if we fail to disable lvmetad.
|
|
*/
|
|
void lvmetad_set_disabled(struct cmd_context *cmd, const char *reason)
|
|
{
|
|
daemon_reply reply;
|
|
|
|
if (!_lvmetad_use)
|
|
return;
|
|
|
|
log_debug_lvmetad("Sending lvmetad disabled %s", reason);
|
|
|
|
reply = daemon_send_simple(_lvmetad, "set_global_info",
|
|
"token = %s", "skip",
|
|
"global_disable = " FMTd64, (int64_t)1,
|
|
"disable_reason = %s", reason,
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
if (reply.error)
|
|
log_error("Failed to send message to lvmetad %d", reply.error);
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK"))
|
|
log_error("Failed response from lvmetad.");
|
|
|
|
daemon_reply_destroy(reply);
|
|
}
|
|
|
|
void lvmetad_clear_disabled(struct cmd_context *cmd)
|
|
{
|
|
daemon_reply reply;
|
|
|
|
if (!_lvmetad_use)
|
|
return;
|
|
|
|
log_debug_lvmetad("Sending lvmetad disabled 0");
|
|
|
|
reply = daemon_send_simple(_lvmetad, "set_global_info",
|
|
"token = %s", "skip",
|
|
"global_disable = " FMTd64, (int64_t)0,
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
if (reply.error)
|
|
log_error("Failed to send message to lvmetad %d", reply.error);
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK"))
|
|
log_error("Failed response from lvmetad.");
|
|
|
|
daemon_reply_destroy(reply);
|
|
}
|
|
|
|
int lvmetad_is_disabled(struct cmd_context *cmd, const char **reason)
|
|
{
|
|
daemon_reply reply;
|
|
const char *reply_reason;
|
|
int ret = 0;
|
|
|
|
reply = daemon_send_simple(_lvmetad, "get_global_info",
|
|
"token = %s", "skip",
|
|
"pid = " FMTd64, (int64_t)getpid(),
|
|
"cmd = %s", get_cmd_name(),
|
|
NULL);
|
|
|
|
if (reply.error) {
|
|
*reason = "send error";
|
|
ret = 1;
|
|
goto out;
|
|
}
|
|
|
|
if (strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
|
*reason = "response error";
|
|
ret = 1;
|
|
goto out;
|
|
}
|
|
|
|
if (daemon_reply_int(reply, "global_disable", 0)) {
|
|
ret = 1;
|
|
|
|
reply_reason = daemon_reply_str(reply, "disable_reason", NULL);
|
|
|
|
if (!reply_reason) {
|
|
*reason = "<not set>";
|
|
|
|
} else if (strstr(reply_reason, LVMETAD_DISABLE_REASON_DIRECT)) {
|
|
*reason = "the disable flag was set directly";
|
|
|
|
} else if (strstr(reply_reason, LVMETAD_DISABLE_REASON_LVM1)) {
|
|
*reason = "LVM1 metadata was found";
|
|
|
|
} else if (strstr(reply_reason, LVMETAD_DISABLE_REASON_DUPLICATES)) {
|
|
*reason = "duplicate PVs were found";
|
|
|
|
} else if (strstr(reply_reason, LVMETAD_DISABLE_REASON_VGRESTORE)) {
|
|
*reason = "vgcfgrestore is restoring VG metadata";
|
|
|
|
} else {
|
|
*reason = "<unknown>";
|
|
}
|
|
}
|
|
out:
|
|
daemon_reply_destroy(reply);
|
|
return ret;
|
|
}
|
|
|