mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-10 05:18:36 +03:00
c407d2bd11
preserve exlusive LV locks.
948 lines
22 KiB
C
948 lines
22 KiB
C
/*
|
|
* Copyright (C) 2002-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2010 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU General Public License v.2.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#define _FILE_OFFSET_BITS 64
|
|
|
|
#include <configure.h>
|
|
#include <pthread.h>
|
|
#include <sys/types.h>
|
|
#include <sys/utsname.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <fcntl.h>
|
|
#include <string.h>
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <syslog.h>
|
|
#include <assert.h>
|
|
#include <libdevmapper.h>
|
|
#include <libdlm.h>
|
|
|
|
#include "lvm-types.h"
|
|
#include "clvm.h"
|
|
#include "clvmd-comms.h"
|
|
#include "clvmd.h"
|
|
#include "lvm-functions.h"
|
|
|
|
/* LVM2 headers */
|
|
#include "toolcontext.h"
|
|
#include "lvmcache.h"
|
|
#include "lvm-logging.h"
|
|
#include "lvm-globals.h"
|
|
#include "activate.h"
|
|
#include "locking.h"
|
|
#include "archiver.h"
|
|
#include "defaults.h"
|
|
#include "memlock.h"
|
|
|
|
static struct cmd_context *cmd = NULL;
|
|
static struct dm_hash_table *lv_hash = NULL;
|
|
static pthread_mutex_t lv_hash_lock;
|
|
static pthread_mutex_t lvm_lock;
|
|
static char last_error[1024];
|
|
static int suspended = 0;
|
|
|
|
struct lv_info {
|
|
int lock_id;
|
|
int lock_mode;
|
|
};
|
|
|
|
static const char *decode_locking_cmd(unsigned char cmdl)
|
|
{
|
|
static char buf[128];
|
|
const char *type;
|
|
const char *scope;
|
|
const char *command;
|
|
|
|
switch (cmdl & LCK_TYPE_MASK) {
|
|
case LCK_NULL:
|
|
type = "NULL";
|
|
break;
|
|
case LCK_READ:
|
|
type = "READ";
|
|
break;
|
|
case LCK_PREAD:
|
|
type = "PREAD";
|
|
break;
|
|
case LCK_WRITE:
|
|
type = "WRITE";
|
|
break;
|
|
case LCK_EXCL:
|
|
type = "EXCL";
|
|
break;
|
|
case LCK_UNLOCK:
|
|
type = "UNLOCK";
|
|
break;
|
|
default:
|
|
type = "unknown";
|
|
break;
|
|
}
|
|
|
|
switch (cmdl & LCK_SCOPE_MASK) {
|
|
case LCK_VG:
|
|
scope = "VG";
|
|
command = "LCK_VG";
|
|
break;
|
|
case LCK_LV:
|
|
scope = "LV";
|
|
switch (cmdl & LCK_MASK) {
|
|
case LCK_LV_EXCLUSIVE & LCK_MASK:
|
|
command = "LCK_LV_EXCLUSIVE";
|
|
break;
|
|
case LCK_LV_SUSPEND & LCK_MASK:
|
|
command = "LCK_LV_SUSPEND";
|
|
break;
|
|
case LCK_LV_RESUME & LCK_MASK:
|
|
command = "LCK_LV_RESUME";
|
|
break;
|
|
case LCK_LV_ACTIVATE & LCK_MASK:
|
|
command = "LCK_LV_ACTIVATE";
|
|
break;
|
|
case LCK_LV_DEACTIVATE & LCK_MASK:
|
|
command = "LCK_LV_DEACTIVATE";
|
|
break;
|
|
default:
|
|
command = "unknown";
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
scope = "unknown";
|
|
command = "unknown";
|
|
break;
|
|
}
|
|
|
|
sprintf(buf, "0x%x %s (%s|%s%s%s%s%s%s)", cmdl, command, type, scope,
|
|
cmdl & LCK_NONBLOCK ? "|NONBLOCK" : "",
|
|
cmdl & LCK_HOLD ? "|HOLD" : "",
|
|
cmdl & LCK_LOCAL ? "|LOCAL" : "",
|
|
cmdl & LCK_CLUSTER_VG ? "|CLUSTER_VG" : "",
|
|
cmdl & LCK_CACHE ? "|CACHE" : "");
|
|
|
|
return buf;
|
|
}
|
|
|
|
static const char *decode_flags(unsigned char flags)
|
|
{
|
|
static char buf[128];
|
|
|
|
sprintf(buf, "0x%x (%s%s%s%s)", flags,
|
|
flags & LCK_PARTIAL_MODE ? "PARTIAL_MODE " : "",
|
|
flags & LCK_MIRROR_NOSYNC_MODE ? "MIRROR_NOSYNC " : "",
|
|
flags & LCK_DMEVENTD_MONITOR_MODE ? "DMEVENTD_MONITOR " : "",
|
|
flags & LCK_CONVERT ? "CONVERT " : "");
|
|
|
|
return buf;
|
|
}
|
|
|
|
char *get_last_lvm_error()
|
|
{
|
|
return last_error;
|
|
}
|
|
|
|
/*
|
|
* Hash lock info helpers
|
|
*/
|
|
static struct lv_info *lookup_info(const char *resource)
|
|
{
|
|
struct lv_info *lvi;
|
|
|
|
pthread_mutex_lock(&lv_hash_lock);
|
|
lvi = dm_hash_lookup(lv_hash, resource);
|
|
pthread_mutex_unlock(&lv_hash_lock);
|
|
|
|
return lvi;
|
|
}
|
|
|
|
static void insert_info(const char *resource, struct lv_info *lvi)
|
|
{
|
|
pthread_mutex_lock(&lv_hash_lock);
|
|
dm_hash_insert(lv_hash, resource, lvi);
|
|
pthread_mutex_unlock(&lv_hash_lock);
|
|
}
|
|
|
|
static void remove_info(const char *resource)
|
|
{
|
|
pthread_mutex_lock(&lv_hash_lock);
|
|
dm_hash_remove(lv_hash, resource);
|
|
pthread_mutex_unlock(&lv_hash_lock);
|
|
}
|
|
|
|
/*
|
|
* Return the mode a lock is currently held at (or -1 if not held)
|
|
*/
|
|
static int get_current_lock(char *resource)
|
|
{
|
|
struct lv_info *lvi;
|
|
|
|
if ((lvi = lookup_info(resource)))
|
|
return lvi->lock_mode;
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
void init_lvhash()
|
|
{
|
|
/* Create hash table for keeping LV locks & status */
|
|
lv_hash = dm_hash_create(100);
|
|
pthread_mutex_init(&lv_hash_lock, NULL);
|
|
pthread_mutex_init(&lvm_lock, NULL);
|
|
}
|
|
|
|
/* Called at shutdown to tidy the lockspace */
|
|
void destroy_lvhash()
|
|
{
|
|
struct dm_hash_node *v;
|
|
struct lv_info *lvi;
|
|
char *resource;
|
|
int status;
|
|
|
|
pthread_mutex_lock(&lv_hash_lock);
|
|
|
|
dm_hash_iterate(v, lv_hash) {
|
|
lvi = dm_hash_get_data(lv_hash, v);
|
|
resource = dm_hash_get_key(lv_hash, v);
|
|
|
|
if ((status = sync_unlock(resource, lvi->lock_id)))
|
|
DEBUGLOG("unlock_all. unlock failed(%d): %s\n",
|
|
status, strerror(errno));
|
|
free(lvi);
|
|
}
|
|
|
|
dm_hash_destroy(lv_hash);
|
|
lv_hash = NULL;
|
|
|
|
pthread_mutex_unlock(&lv_hash_lock);
|
|
}
|
|
|
|
/* Gets a real lock and keeps the info in the hash table */
|
|
int hold_lock(char *resource, int mode, int flags)
|
|
{
|
|
int status;
|
|
int saved_errno;
|
|
struct lv_info *lvi;
|
|
|
|
/* Mask off invalid options */
|
|
flags &= LKF_NOQUEUE | LKF_CONVERT;
|
|
|
|
lvi = lookup_info(resource);
|
|
|
|
if (lvi && lvi->lock_mode == mode) {
|
|
DEBUGLOG("hold_lock, lock mode %d already held\n", mode);
|
|
return 0;
|
|
}
|
|
|
|
/* Only allow explicit conversions */
|
|
if (lvi && !(flags & LKF_CONVERT)) {
|
|
errno = EBUSY;
|
|
return -1;
|
|
}
|
|
if (lvi) {
|
|
/* Already exists - convert it */
|
|
status =
|
|
sync_lock(resource, mode, flags, &lvi->lock_id);
|
|
saved_errno = errno;
|
|
if (!status)
|
|
lvi->lock_mode = mode;
|
|
|
|
if (status) {
|
|
DEBUGLOG("hold_lock. convert to %d failed: %s\n", mode,
|
|
strerror(errno));
|
|
}
|
|
errno = saved_errno;
|
|
} else {
|
|
lvi = malloc(sizeof(struct lv_info));
|
|
if (!lvi)
|
|
return -1;
|
|
|
|
lvi->lock_mode = mode;
|
|
status = sync_lock(resource, mode, flags & ~LKF_CONVERT, &lvi->lock_id);
|
|
saved_errno = errno;
|
|
if (status) {
|
|
free(lvi);
|
|
DEBUGLOG("hold_lock. lock at %d failed: %s\n", mode,
|
|
strerror(errno));
|
|
} else
|
|
insert_info(resource, lvi);
|
|
|
|
errno = saved_errno;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
/* Unlock and remove it from the hash table */
|
|
int hold_unlock(char *resource)
|
|
{
|
|
struct lv_info *lvi;
|
|
int status;
|
|
int saved_errno;
|
|
|
|
if (!(lvi = lookup_info(resource))) {
|
|
DEBUGLOG("hold_unlock, lock not already held\n");
|
|
return 0;
|
|
}
|
|
|
|
status = sync_unlock(resource, lvi->lock_id);
|
|
saved_errno = errno;
|
|
if (!status) {
|
|
remove_info(resource);
|
|
free(lvi);
|
|
} else {
|
|
DEBUGLOG("hold_unlock. unlock failed(%d): %s\n", status,
|
|
strerror(errno));
|
|
}
|
|
|
|
errno = saved_errno;
|
|
return status;
|
|
}
|
|
|
|
/* Watch the return codes here.
|
|
liblvm API functions return 1(true) for success, 0(false) for failure and don't set errno.
|
|
libdlm API functions return 0 for success, -1 for failure and do set errno.
|
|
These functions here return 0 for success or >0 for failure (where the retcode is errno)
|
|
*/
|
|
|
|
/* Activate LV exclusive or non-exclusive */
|
|
static int do_activate_lv(char *resource, unsigned char lock_flags, int mode)
|
|
{
|
|
int oldmode;
|
|
int status;
|
|
int activate_lv;
|
|
int exclusive = 0;
|
|
struct lvinfo lvi;
|
|
|
|
/* Is it already open ? */
|
|
oldmode = get_current_lock(resource);
|
|
if (oldmode == mode && (lock_flags & LCK_CLUSTER_VG)) {
|
|
DEBUGLOG("do_activate_lv, lock already held at %d\n", oldmode);
|
|
return 0; /* Nothing to do */
|
|
}
|
|
|
|
/* Does the config file want us to activate this LV ? */
|
|
if (!lv_activation_filter(cmd, resource, &activate_lv))
|
|
return EIO;
|
|
|
|
if (!activate_lv)
|
|
return 0; /* Success, we did nothing! */
|
|
|
|
/* Do we need to activate exclusively? */
|
|
if ((activate_lv == 2) || (mode == LKM_EXMODE)) {
|
|
exclusive = 1;
|
|
mode = LKM_EXMODE;
|
|
}
|
|
|
|
/*
|
|
* Try to get the lock if it's a clustered volume group.
|
|
* Use lock conversion only if requested, to prevent implicit conversion
|
|
* of exclusive lock to shared one during activation.
|
|
*/
|
|
if (lock_flags & LCK_CLUSTER_VG) {
|
|
status = hold_lock(resource, mode, LKF_NOQUEUE | (lock_flags & LCK_CONVERT?LKF_CONVERT:0));
|
|
if (status) {
|
|
/* Return an LVM-sensible error for this.
|
|
* Forcing EIO makes the upper level return this text
|
|
* rather than the strerror text for EAGAIN.
|
|
*/
|
|
if (errno == EAGAIN) {
|
|
sprintf(last_error, "Volume is busy on another node");
|
|
errno = EIO;
|
|
}
|
|
return errno;
|
|
}
|
|
}
|
|
|
|
/* If it's suspended then resume it */
|
|
if (!lv_info_by_lvid(cmd, resource, &lvi, 0, 0))
|
|
goto error;
|
|
|
|
if (lvi.suspended)
|
|
if (!lv_resume(cmd, resource))
|
|
goto error;
|
|
|
|
/* Now activate it */
|
|
if (!lv_activate(cmd, resource, exclusive))
|
|
goto error;
|
|
|
|
return 0;
|
|
|
|
error:
|
|
if (oldmode == -1 || oldmode != mode)
|
|
(void)hold_unlock(resource);
|
|
return EIO;
|
|
}
|
|
|
|
/* Resume the LV if it was active */
|
|
static int do_resume_lv(char *resource, unsigned char lock_flags)
|
|
{
|
|
int oldmode;
|
|
|
|
/* Is it open ? */
|
|
oldmode = get_current_lock(resource);
|
|
if (oldmode == -1 && (lock_flags & LCK_CLUSTER_VG)) {
|
|
DEBUGLOG("do_resume_lv, lock not already held\n");
|
|
return 0; /* We don't need to do anything */
|
|
}
|
|
|
|
if (!lv_resume_if_active(cmd, resource))
|
|
return EIO;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Suspend the device if active */
|
|
static int do_suspend_lv(char *resource, unsigned char lock_flags)
|
|
{
|
|
int oldmode;
|
|
struct lvinfo lvi;
|
|
|
|
/* Is it open ? */
|
|
oldmode = get_current_lock(resource);
|
|
if (oldmode == -1 && (lock_flags & LCK_CLUSTER_VG)) {
|
|
DEBUGLOG("do_suspend_lv, lock not already held\n");
|
|
return 0; /* Not active, so it's OK */
|
|
}
|
|
|
|
/* Only suspend it if it exists */
|
|
if (!lv_info_by_lvid(cmd, resource, &lvi, 0, 0))
|
|
return EIO;
|
|
|
|
if (lvi.exists) {
|
|
if (!lv_suspend_if_active(cmd, resource)) {
|
|
return EIO;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int do_deactivate_lv(char *resource, unsigned char lock_flags)
|
|
{
|
|
int oldmode;
|
|
int status;
|
|
|
|
/* Is it open ? */
|
|
oldmode = get_current_lock(resource);
|
|
if (oldmode == -1 && (lock_flags & LCK_CLUSTER_VG)) {
|
|
DEBUGLOG("do_deactivate_lock, lock not already held\n");
|
|
return 0; /* We don't need to do anything */
|
|
}
|
|
|
|
if (!lv_deactivate(cmd, resource))
|
|
return EIO;
|
|
|
|
if (lock_flags & LCK_CLUSTER_VG) {
|
|
status = hold_unlock(resource);
|
|
if (status)
|
|
return errno;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
const char *do_lock_query(char *resource)
|
|
{
|
|
int mode;
|
|
const char *type = NULL;
|
|
|
|
mode = get_current_lock(resource);
|
|
switch (mode) {
|
|
case LKM_NLMODE: type = "NL"; break;
|
|
case LKM_CRMODE: type = "CR"; break;
|
|
case LKM_CWMODE: type = "CW"; break;
|
|
case LKM_PRMODE: type = "PR"; break;
|
|
case LKM_PWMODE: type = "PW"; break;
|
|
case LKM_EXMODE: type = "EX"; break;
|
|
}
|
|
|
|
DEBUGLOG("do_lock_query: resource '%s', mode %i (%s)\n", resource, mode, type ?: "?");
|
|
|
|
return type;
|
|
}
|
|
|
|
/* This is the LOCK_LV part that happens on all nodes in the cluster -
|
|
it is responsible for the interaction with device-mapper and LVM */
|
|
int do_lock_lv(unsigned char command, unsigned char lock_flags, char *resource)
|
|
{
|
|
int status = 0;
|
|
|
|
DEBUGLOG("do_lock_lv: resource '%s', cmd = %s, flags = %s, memlock = %d\n",
|
|
resource, decode_locking_cmd(command), decode_flags(lock_flags), memlock());
|
|
|
|
if (!cmd->config_valid || config_files_changed(cmd)) {
|
|
/* Reinitialise various settings inc. logging, filters */
|
|
if (do_refresh_cache()) {
|
|
log_error("Updated config file invalid. Aborting.");
|
|
return EINVAL;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_lock(&lvm_lock);
|
|
if (lock_flags & LCK_MIRROR_NOSYNC_MODE)
|
|
init_mirror_in_sync(1);
|
|
|
|
if (lock_flags & LCK_DMEVENTD_MONITOR_MODE)
|
|
init_dmeventd_monitor(1);
|
|
else
|
|
init_dmeventd_monitor(0);
|
|
|
|
cmd->partial_activation = (lock_flags & LCK_PARTIAL_MODE) ? 1 : 0;
|
|
|
|
/* clvmd should never try to read suspended device */
|
|
init_ignore_suspended_devices(1);
|
|
|
|
switch (command & LCK_MASK) {
|
|
case LCK_LV_EXCLUSIVE:
|
|
status = do_activate_lv(resource, lock_flags, LKM_EXMODE);
|
|
break;
|
|
|
|
case LCK_LV_SUSPEND:
|
|
status = do_suspend_lv(resource, lock_flags);
|
|
if (!status)
|
|
suspended++;
|
|
break;
|
|
|
|
case LCK_UNLOCK:
|
|
case LCK_LV_RESUME: /* if active */
|
|
status = do_resume_lv(resource, lock_flags);
|
|
if (!status)
|
|
suspended--;
|
|
break;
|
|
|
|
case LCK_LV_ACTIVATE:
|
|
status = do_activate_lv(resource, lock_flags, LKM_CRMODE);
|
|
break;
|
|
|
|
case LCK_LV_DEACTIVATE:
|
|
status = do_deactivate_lv(resource, lock_flags);
|
|
break;
|
|
|
|
default:
|
|
DEBUGLOG("Invalid LV command 0x%x\n", command);
|
|
status = EINVAL;
|
|
break;
|
|
}
|
|
|
|
if (lock_flags & LCK_MIRROR_NOSYNC_MODE)
|
|
init_mirror_in_sync(0);
|
|
|
|
cmd->partial_activation = 0;
|
|
|
|
/* clean the pool for another command */
|
|
dm_pool_empty(cmd->mem);
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
|
|
DEBUGLOG("Command return is %d, memlock is %d\n", status, memlock());
|
|
return status;
|
|
}
|
|
|
|
/* Functions to do on the local node only BEFORE the cluster-wide stuff above happens */
|
|
int pre_lock_lv(unsigned char command, unsigned char lock_flags, char *resource)
|
|
{
|
|
/* Nearly all the stuff happens cluster-wide. Apart from SUSPEND. Here we get the
|
|
lock out on this node (because we are the node modifying the metadata)
|
|
before suspending cluster-wide.
|
|
LKF_CONVERT is used always, local node is going to modify metadata
|
|
*/
|
|
if ((command & (LCK_SCOPE_MASK | LCK_TYPE_MASK)) == LCK_LV_SUSPEND &&
|
|
(lock_flags & LCK_CLUSTER_VG)) {
|
|
DEBUGLOG("pre_lock_lv: resource '%s', cmd = %s, flags = %s\n",
|
|
resource, decode_locking_cmd(command), decode_flags(lock_flags));
|
|
|
|
if (hold_lock(resource, LKM_PWMODE, LKF_NOQUEUE | LKF_CONVERT))
|
|
return errno;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Functions to do on the local node only AFTER the cluster-wide stuff above happens */
|
|
int post_lock_lv(unsigned char command, unsigned char lock_flags,
|
|
char *resource)
|
|
{
|
|
int status;
|
|
|
|
/* Opposite of above, done on resume after a metadata update */
|
|
if ((command & (LCK_SCOPE_MASK | LCK_TYPE_MASK)) == LCK_LV_RESUME &&
|
|
(lock_flags & LCK_CLUSTER_VG)) {
|
|
int oldmode;
|
|
|
|
DEBUGLOG
|
|
("post_lock_lv: resource '%s', cmd = %s, flags = %s\n",
|
|
resource, decode_locking_cmd(command), decode_flags(lock_flags));
|
|
|
|
/* If the lock state is PW then restore it to what it was */
|
|
oldmode = get_current_lock(resource);
|
|
if (oldmode == LKM_PWMODE) {
|
|
struct lvinfo lvi;
|
|
|
|
pthread_mutex_lock(&lvm_lock);
|
|
status = lv_info_by_lvid(cmd, resource, &lvi, 0, 0);
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
if (!status)
|
|
return EIO;
|
|
|
|
if (lvi.exists) {
|
|
if (hold_lock(resource, LKM_CRMODE, LKF_CONVERT))
|
|
return errno;
|
|
} else {
|
|
if (hold_unlock(resource))
|
|
return errno;
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Check if a VG is in use by LVM1 so we don't stomp on it */
|
|
int do_check_lvm1(const char *vgname)
|
|
{
|
|
int status;
|
|
|
|
status = check_lvm1_vg_inactive(cmd, vgname);
|
|
|
|
return status == 1 ? 0 : EBUSY;
|
|
}
|
|
|
|
int do_refresh_cache()
|
|
{
|
|
DEBUGLOG("Refreshing context\n");
|
|
log_notice("Refreshing context");
|
|
|
|
pthread_mutex_lock(&lvm_lock);
|
|
|
|
if (!refresh_toolcontext(cmd)) {
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
return -1;
|
|
}
|
|
|
|
init_full_scan_done(0);
|
|
init_ignore_suspended_devices(1);
|
|
lvmcache_label_scan(cmd, 2);
|
|
dm_pool_empty(cmd->mem);
|
|
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Only called at gulm startup. Drop any leftover VG or P_orphan locks
|
|
that might be hanging around if we died for any reason
|
|
*/
|
|
static void drop_vg_locks()
|
|
{
|
|
char vg[128];
|
|
char line[255];
|
|
FILE *vgs =
|
|
popen
|
|
("lvm pvs --config 'log{command_names=0 prefix=\"\"}' --nolocking --noheadings -o vg_name", "r");
|
|
|
|
sync_unlock("P_" VG_ORPHANS, LCK_EXCL);
|
|
sync_unlock("P_" VG_GLOBAL, LCK_EXCL);
|
|
|
|
if (!vgs)
|
|
return;
|
|
|
|
while (fgets(line, sizeof(line), vgs)) {
|
|
char *vgend;
|
|
char *vgstart;
|
|
|
|
if (line[strlen(line)-1] == '\n')
|
|
line[strlen(line)-1] = '\0';
|
|
|
|
vgstart = line + strspn(line, " ");
|
|
vgend = vgstart + strcspn(vgstart, " ");
|
|
*vgend = '\0';
|
|
|
|
if (strncmp(vgstart, "WARNING:", 8) == 0)
|
|
continue;
|
|
|
|
sprintf(vg, "V_%s", vgstart);
|
|
sync_unlock(vg, LCK_EXCL);
|
|
|
|
}
|
|
if (fclose(vgs))
|
|
DEBUGLOG("vgs fclose failed: %s\n", strerror(errno));
|
|
}
|
|
|
|
/*
|
|
* Handle VG lock - drop metadata or update lvmcache state
|
|
*/
|
|
void do_lock_vg(unsigned char command, unsigned char lock_flags, char *resource)
|
|
{
|
|
uint32_t lock_cmd = command;
|
|
char *vgname = resource + 2;
|
|
|
|
DEBUGLOG("do_lock_vg: resource '%s', cmd = %s, flags = %s, memlock = %d\n",
|
|
resource, decode_locking_cmd(command), decode_flags(lock_flags), memlock());
|
|
|
|
/* P_#global causes a full cache refresh */
|
|
if (!strcmp(resource, "P_" VG_GLOBAL)) {
|
|
do_refresh_cache();
|
|
return;
|
|
}
|
|
|
|
lock_cmd &= (LCK_SCOPE_MASK | LCK_TYPE_MASK | LCK_HOLD);
|
|
|
|
/*
|
|
* Check if LCK_CACHE should be set. All P_ locks except # are cache related.
|
|
*/
|
|
if (strncmp(resource, "P_#", 3) && !strncmp(resource, "P_", 2))
|
|
lock_cmd |= LCK_CACHE;
|
|
|
|
pthread_mutex_lock(&lvm_lock);
|
|
switch (lock_cmd) {
|
|
case LCK_VG_COMMIT:
|
|
DEBUGLOG("vg_commit notification for VG %s\n", vgname);
|
|
lvmcache_commit_metadata(vgname);
|
|
break;
|
|
case LCK_VG_REVERT:
|
|
DEBUGLOG("vg_revert notification for VG %s\n", vgname);
|
|
lvmcache_drop_metadata(vgname, 1);
|
|
break;
|
|
case LCK_VG_DROP_CACHE:
|
|
default:
|
|
DEBUGLOG("Invalidating cached metadata for VG %s\n", vgname);
|
|
lvmcache_drop_metadata(vgname, 0);
|
|
}
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
}
|
|
|
|
/*
|
|
* Compare the uuid with the list of exclusive locks that clvmd
|
|
* held before it was restarted, so we can get the right kind
|
|
* of lock now we are restarting.
|
|
*/
|
|
static int was_ex_lock(char *uuid, char **argv)
|
|
{
|
|
int optnum = 0;
|
|
char *opt = argv[optnum];
|
|
|
|
while (opt) {
|
|
if (strcmp(opt, "-E") == 0) {
|
|
opt = argv[++optnum];
|
|
if (opt && (strcmp(opt, uuid) == 0)) {
|
|
DEBUGLOG("Lock %s is exclusive\n", uuid);
|
|
return 1;
|
|
}
|
|
}
|
|
opt = argv[++optnum];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Ideally, clvmd should be started before any LVs are active
|
|
* but this may not be the case...
|
|
* I suppose this also comes in handy if clvmd crashes, not that it would!
|
|
*/
|
|
static void *get_initial_state(char **argv)
|
|
{
|
|
int lock_mode;
|
|
char lv[64], vg[64], flags[25], vg_flags[25];
|
|
char uuid[65];
|
|
char line[255];
|
|
FILE *lvs =
|
|
popen
|
|
("lvm lvs --config 'log{command_names=0 prefix=\"\"}' --nolocking --noheadings -o vg_uuid,lv_uuid,lv_attr,vg_attr",
|
|
"r");
|
|
|
|
if (!lvs)
|
|
return NULL;
|
|
|
|
while (fgets(line, sizeof(line), lvs)) {
|
|
if (sscanf(line, "%s %s %s %s\n", vg, lv, flags, vg_flags) == 4) {
|
|
|
|
/* States: s:suspended a:active S:dropped snapshot I:invalid snapshot */
|
|
if (strlen(vg) == 38 && /* is is a valid UUID ? */
|
|
(flags[4] == 'a' || flags[4] == 's') && /* is it active or suspended? */
|
|
vg_flags[5] == 'c') { /* is it clustered ? */
|
|
/* Convert hyphen-separated UUIDs into one */
|
|
memcpy(&uuid[0], &vg[0], 6);
|
|
memcpy(&uuid[6], &vg[7], 4);
|
|
memcpy(&uuid[10], &vg[12], 4);
|
|
memcpy(&uuid[14], &vg[17], 4);
|
|
memcpy(&uuid[18], &vg[22], 4);
|
|
memcpy(&uuid[22], &vg[27], 4);
|
|
memcpy(&uuid[26], &vg[32], 6);
|
|
memcpy(&uuid[32], &lv[0], 6);
|
|
memcpy(&uuid[38], &lv[7], 4);
|
|
memcpy(&uuid[42], &lv[12], 4);
|
|
memcpy(&uuid[46], &lv[17], 4);
|
|
memcpy(&uuid[50], &lv[22], 4);
|
|
memcpy(&uuid[54], &lv[27], 4);
|
|
memcpy(&uuid[58], &lv[32], 6);
|
|
uuid[64] = '\0';
|
|
|
|
lock_mode = LKM_CRMODE;
|
|
|
|
/* Look for this lock in the list of EX locks
|
|
we were passed on the command-line */
|
|
if (was_ex_lock(uuid, argv))
|
|
lock_mode = LKM_EXMODE;
|
|
|
|
DEBUGLOG("getting initial lock for %s\n", uuid);
|
|
hold_lock(uuid, lock_mode, LKF_NOQUEUE);
|
|
}
|
|
}
|
|
}
|
|
if (fclose(lvs))
|
|
DEBUGLOG("lvs fclose failed: %s\n", strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
static void lvm2_log_fn(int level, const char *file, int line, int dm_errno,
|
|
const char *message)
|
|
{
|
|
|
|
/* Send messages to the normal LVM2 logging system too,
|
|
so we get debug output when it's asked for.
|
|
We need to NULL the function ptr otherwise it will just call
|
|
back into here! */
|
|
init_log_fn(NULL);
|
|
print_log(level, file, line, dm_errno, "%s", message);
|
|
init_log_fn(lvm2_log_fn);
|
|
|
|
/*
|
|
* Ignore non-error messages, but store the latest one for returning
|
|
* to the user.
|
|
*/
|
|
if (level != _LOG_ERR && level != _LOG_FATAL)
|
|
return;
|
|
|
|
strncpy(last_error, message, sizeof(last_error));
|
|
last_error[sizeof(last_error)-1] = '\0';
|
|
}
|
|
|
|
/* This checks some basic cluster-LVM configuration stuff */
|
|
static void check_config()
|
|
{
|
|
int locking_type;
|
|
|
|
locking_type = find_config_tree_int(cmd, "global/locking_type", 1);
|
|
|
|
if (locking_type == 3) /* compiled-in cluster support */
|
|
return;
|
|
|
|
if (locking_type == 2) { /* External library, check name */
|
|
const char *libname;
|
|
|
|
libname = find_config_tree_str(cmd, "global/locking_library",
|
|
"");
|
|
if (strstr(libname, "liblvm2clusterlock.so"))
|
|
return;
|
|
|
|
log_error("Incorrect LVM locking library specified in lvm.conf, cluster operations may not work.");
|
|
return;
|
|
}
|
|
log_error("locking_type not set correctly in lvm.conf, cluster operations will not work.");
|
|
}
|
|
|
|
/* Backups up the LVM metadata if it's changed */
|
|
void lvm_do_backup(const char *vgname)
|
|
{
|
|
struct volume_group * vg;
|
|
int consistent = 0;
|
|
|
|
DEBUGLOG("Triggering backup of VG metadata for %s. suspended=%d\n", vgname, suspended);
|
|
|
|
pthread_mutex_lock(&lvm_lock);
|
|
|
|
vg = vg_read_internal(cmd, vgname, NULL /*vgid*/, &consistent);
|
|
|
|
if (vg && consistent)
|
|
check_current_backup(vg);
|
|
else
|
|
log_error("Error backing up metadata, can't find VG for group %s", vgname);
|
|
|
|
vg_release(vg);
|
|
dm_pool_empty(cmd->mem);
|
|
|
|
pthread_mutex_unlock(&lvm_lock);
|
|
}
|
|
|
|
struct dm_hash_node *get_next_excl_lock(struct dm_hash_node *v, char **name)
|
|
{
|
|
struct lv_info *lvi;
|
|
|
|
*name = NULL;
|
|
if (!v)
|
|
v = dm_hash_get_first(lv_hash);
|
|
|
|
do {
|
|
if (v) {
|
|
lvi = dm_hash_get_data(lv_hash, v);
|
|
DEBUGLOG("Looking for EX locks. found %x mode %d\n", lvi->lock_id, lvi->lock_mode);
|
|
|
|
if (lvi->lock_mode == LCK_EXCL) {
|
|
*name = dm_hash_get_key(lv_hash, v);
|
|
}
|
|
v = dm_hash_get_next(lv_hash, v);
|
|
}
|
|
} while (v && !*name);
|
|
DEBUGLOG("returning EXclusive UUID %s\n", *name);
|
|
return v;
|
|
}
|
|
|
|
/* Called to initialise the LVM context of the daemon */
|
|
int init_lvm(int using_gulm, char **argv)
|
|
{
|
|
if (!(cmd = create_toolcontext(1, NULL))) {
|
|
log_error("Failed to allocate command context");
|
|
return 0;
|
|
}
|
|
|
|
if (stored_errno()) {
|
|
destroy_toolcontext(cmd);
|
|
return 0;
|
|
}
|
|
|
|
/* Use LOG_DAEMON for syslog messages instead of LOG_USER */
|
|
init_syslog(LOG_DAEMON);
|
|
openlog("clvmd", LOG_PID, LOG_DAEMON);
|
|
cmd->cmd_line = "clvmd";
|
|
|
|
/* Check lvm.conf is setup for cluster-LVM */
|
|
check_config();
|
|
init_ignore_suspended_devices(1);
|
|
|
|
/* Remove any non-LV locks that may have been left around */
|
|
if (using_gulm)
|
|
drop_vg_locks();
|
|
|
|
get_initial_state(argv);
|
|
|
|
/* Trap log messages so we can pass them back to the user */
|
|
init_log_fn(lvm2_log_fn);
|
|
memlock_inc_daemon(cmd);
|
|
|
|
return 1;
|
|
}
|
|
|
|
void destroy_lvm(void)
|
|
{
|
|
if (cmd) {
|
|
memlock_dec_daemon(cmd);
|
|
destroy_toolcontext(cmd);
|
|
}
|
|
cmd = NULL;
|
|
}
|