mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
Review locking: block signals instead of ignoring them and restore state
afterwards; avoid race condition with unlink; add LCK_HOLD to process_each_vg.
This commit is contained in:
parent
14a9cda63b
commit
568d7229bf
@ -32,11 +32,17 @@ struct lock_list {
|
|||||||
static struct list _lock_list;
|
static struct list _lock_list;
|
||||||
static char _lock_dir[NAME_LEN];
|
static char _lock_dir[NAME_LEN];
|
||||||
|
|
||||||
|
static sig_t _oldhandler;
|
||||||
|
static sigset_t _fullsigset, _intsigset;
|
||||||
|
static int _handler_installed;
|
||||||
|
|
||||||
static int _release_lock(const char *file)
|
static int _release_lock(const char *file)
|
||||||
{
|
{
|
||||||
struct lock_list *ll;
|
struct lock_list *ll;
|
||||||
struct list *llh, *llt;
|
struct list *llh, *llt;
|
||||||
|
|
||||||
|
struct stat buf1, buf2;
|
||||||
|
|
||||||
list_iterate_safe(llh, llt, &_lock_list) {
|
list_iterate_safe(llh, llt, &_lock_list) {
|
||||||
ll = list_item(llh, struct lock_list);
|
ll = list_item(llh, struct lock_list);
|
||||||
|
|
||||||
@ -44,10 +50,13 @@ static int _release_lock(const char *file)
|
|||||||
list_del(llh);
|
list_del(llh);
|
||||||
log_very_verbose("Unlocking %s", ll->res);
|
log_very_verbose("Unlocking %s", ll->res);
|
||||||
|
|
||||||
/*
|
if (flock(ll->lf, LOCK_NB | LOCK_UN))
|
||||||
* If this is the last pid using the file, remove it
|
log_sys_error("flock", ll->res);
|
||||||
*/
|
|
||||||
if (!flock(ll->lf, LOCK_NB | LOCK_EX))
|
if (!flock(ll->lf, LOCK_NB | LOCK_EX) &&
|
||||||
|
!stat(ll->res, &buf1) &&
|
||||||
|
!fstat(ll->lf, &buf2) &&
|
||||||
|
!memcmp(&buf1.st_ino, &buf2.st_ino, sizeof(ino_t)))
|
||||||
if (unlink(ll->res))
|
if (unlink(ll->res))
|
||||||
log_sys_error("unlink", ll->res);
|
log_sys_error("unlink", ll->res);
|
||||||
|
|
||||||
@ -70,22 +79,35 @@ void fin_file_locking(void)
|
|||||||
_release_lock(NULL);
|
_release_lock(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void _remove_ctrl_c_handler()
|
||||||
|
{
|
||||||
|
siginterrupt(SIGINT, 0);
|
||||||
|
if (!_handler_installed || _oldhandler == SIG_ERR)
|
||||||
|
return;
|
||||||
|
|
||||||
|
sigprocmask(SIG_SETMASK, &_fullsigset, NULL);
|
||||||
|
if (signal(SIGINT, _oldhandler) == SIG_ERR)
|
||||||
|
log_sys_error("signal", "_remove_ctrl_c_handler");
|
||||||
|
|
||||||
|
_handler_installed = 0;
|
||||||
|
}
|
||||||
|
|
||||||
void _trap_ctrl_c(int signal)
|
void _trap_ctrl_c(int signal)
|
||||||
{
|
{
|
||||||
|
_remove_ctrl_c_handler();
|
||||||
log_error("CTRL-c detected: giving up waiting for lock");
|
log_error("CTRL-c detected: giving up waiting for lock");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _install_ctrl_c_handler()
|
static void _install_ctrl_c_handler()
|
||||||
{
|
{
|
||||||
siginterrupt(SIGINT, 1);
|
if ((_oldhandler = signal(SIGINT, _trap_ctrl_c)) == SIG_ERR)
|
||||||
signal(SIGINT, _trap_ctrl_c);
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
static void _remove_ctrl_c_handler()
|
sigprocmask(SIG_SETMASK, &_intsigset, NULL);
|
||||||
{
|
siginterrupt(SIGINT, 1);
|
||||||
signal(SIGINT, SIG_IGN);
|
|
||||||
siginterrupt(SIGINT, 0);
|
_handler_installed = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _lock_file(const char *file, int flags)
|
static int _lock_file(const char *file, int flags)
|
||||||
@ -94,6 +116,7 @@ static int _lock_file(const char *file, int flags)
|
|||||||
int r = 1;
|
int r = 1;
|
||||||
|
|
||||||
struct lock_list *ll;
|
struct lock_list *ll;
|
||||||
|
struct stat buf1, buf2;
|
||||||
|
|
||||||
switch (flags & LCK_TYPE_MASK) {
|
switch (flags & LCK_TYPE_MASK) {
|
||||||
case LCK_READ:
|
case LCK_READ:
|
||||||
@ -117,12 +140,17 @@ static int _lock_file(const char *file, int flags)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ll->lf = -1;
|
||||||
|
|
||||||
log_very_verbose("Locking %s", ll->res);
|
log_very_verbose("Locking %s", ll->res);
|
||||||
if ((ll->lf = open(file, O_CREAT | O_APPEND | O_RDWR, 0777)) < 0) {
|
do {
|
||||||
|
if (ll->lf > -1)
|
||||||
|
close(ll->lf);
|
||||||
|
|
||||||
|
if ((ll->lf = open(file, O_CREAT | O_APPEND | O_RDWR, 0777))
|
||||||
|
< 0) {
|
||||||
log_sys_error("open", file);
|
log_sys_error("open", file);
|
||||||
dbg_free(ll->res);
|
goto err;
|
||||||
dbg_free(ll);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((flags & LCK_NONBLOCK))
|
if ((flags & LCK_NONBLOCK))
|
||||||
@ -130,19 +158,27 @@ static int _lock_file(const char *file, int flags)
|
|||||||
else
|
else
|
||||||
_install_ctrl_c_handler();
|
_install_ctrl_c_handler();
|
||||||
|
|
||||||
if (flock(ll->lf, operation)) {
|
r = flock(ll->lf, operation);
|
||||||
log_sys_error("flock", ll->res);
|
|
||||||
dbg_free(ll->res);
|
|
||||||
dbg_free(ll);
|
|
||||||
r = 0;
|
|
||||||
} else
|
|
||||||
list_add(&_lock_list, &ll->list);
|
|
||||||
|
|
||||||
|
|
||||||
if (!(flags & LCK_NONBLOCK))
|
if (!(flags & LCK_NONBLOCK))
|
||||||
_remove_ctrl_c_handler();
|
_remove_ctrl_c_handler();
|
||||||
|
|
||||||
return r;
|
if (r) {
|
||||||
|
log_sys_error("flock", ll->res);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!stat(ll->res, &buf1) && !fstat(ll->lf, &buf2) &&
|
||||||
|
!memcmp(&buf1.st_ino, &buf2.st_ino, sizeof(ino_t)))
|
||||||
|
break;
|
||||||
|
} while (!(flags & LCK_NONBLOCK));
|
||||||
|
|
||||||
|
list_add(&_lock_list, &ll->list);
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
err:
|
||||||
|
dbg_free(ll->res);
|
||||||
|
dbg_free(ll);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int lock_resource(struct cmd_context *cmd, const char *resource, int flags)
|
int lock_resource(struct cmd_context *cmd, const char *resource, int flags)
|
||||||
@ -191,7 +227,6 @@ int lock_resource(struct cmd_context *cmd, const char *resource, int flags)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int init_file_locking(struct locking_type *locking, struct config_file *cf)
|
int init_file_locking(struct locking_type *locking, struct config_file *cf)
|
||||||
{
|
{
|
||||||
locking->lock_resource = lock_resource;
|
locking->lock_resource = lock_resource;
|
||||||
@ -207,5 +242,15 @@ int init_file_locking(struct locking_type *locking, struct config_file *cf)
|
|||||||
|
|
||||||
list_init(&_lock_list);
|
list_init(&_lock_list);
|
||||||
|
|
||||||
|
if (sigfillset(&_intsigset) || sigfillset(&_fullsigset)) {
|
||||||
|
log_sys_error("sigfillset", "init_file_locking");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sigdelset(&_intsigset, SIGINT)) {
|
||||||
|
log_sys_error("sigdelset", "init_file_locking");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -14,37 +14,45 @@
|
|||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
|
||||||
static struct locking_type _locking;
|
static struct locking_type _locking;
|
||||||
|
static sigset_t _oldset;
|
||||||
|
|
||||||
static int _lock_count = 0; /* Number of locks held */
|
static int _lock_count = 0; /* Number of locks held */
|
||||||
static int _signals_ignored = 0;
|
static int _signals_blocked = 0;
|
||||||
|
|
||||||
static void _ignore_signals(void)
|
static void _block_signals(void)
|
||||||
{
|
{
|
||||||
int s;
|
sigset_t set;
|
||||||
|
|
||||||
if (_signals_ignored)
|
if (_signals_blocked)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
for (s = 0; s < NSIG; s++)
|
if (sigfillset(&set)) {
|
||||||
signal(s, SIG_IGN);
|
log_sys_error("sigfillset", "_block_signals");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
_signals_ignored = 1;
|
if (sigprocmask(SIG_SETMASK, &set, &_oldset)) {
|
||||||
|
log_sys_error("sigprocmask", "_block_signals");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_signals_blocked = 1;
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _enable_signals(void)
|
static void _unblock_signals(void)
|
||||||
{
|
{
|
||||||
int s;
|
/* Don't unblock signals while any locks are held */
|
||||||
|
if (!_signals_blocked || _lock_count)
|
||||||
/* Don't enable signals while any locks are held */
|
|
||||||
if (!_signals_ignored || _lock_count)
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
for (s = 0; s < NSIG; s++)
|
if (sigprocmask(SIG_SETMASK, &_oldset, NULL)) {
|
||||||
signal(s, SIG_DFL);
|
log_sys_error("sigprocmask", "_block_signals");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
_signals_ignored = 0;
|
_signals_blocked = 0;
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -114,21 +122,20 @@ void fin_locking(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* VG locking is by name
|
* VG locking is by VG name.
|
||||||
* LV locking is by VG_name/LV_uuid
|
* FIXME This should become VG uuid.
|
||||||
* FIXME This should take a VG_uuid instead of VG_name
|
|
||||||
*/
|
*/
|
||||||
int _lock_vol(struct cmd_context *cmd, const char *resource, int flags)
|
int _lock_vol(struct cmd_context *cmd, const char *resource, int flags)
|
||||||
{
|
{
|
||||||
_ignore_signals();
|
_block_signals();
|
||||||
|
|
||||||
if (!(_locking.lock_resource(cmd, resource, flags))) {
|
if (!(_locking.lock_resource(cmd, resource, flags))) {
|
||||||
_enable_signals();
|
_unblock_signals();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
_update_lock_count(flags);
|
_update_lock_count(flags);
|
||||||
_enable_signals();
|
_unblock_signals();
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -138,7 +145,7 @@ int lock_vol(struct cmd_context *cmd, const char *vol, int flags)
|
|||||||
char resource[258];
|
char resource[258];
|
||||||
|
|
||||||
switch (flags & LCK_SCOPE_MASK) {
|
switch (flags & LCK_SCOPE_MASK) {
|
||||||
case LCK_VG: /* Lock volume group before changing on-disk metadata. */
|
case LCK_VG: /* Lock VG to change on-disk metadata. */
|
||||||
case LCK_LV: /* Suspends LV if it's active. */
|
case LCK_LV: /* Suspends LV if it's active. */
|
||||||
strncpy(resource, (char *) vol, sizeof(resource));
|
strncpy(resource, (char *) vol, sizeof(resource));
|
||||||
break;
|
break;
|
||||||
@ -160,4 +167,3 @@ int lock_vol(struct cmd_context *cmd, const char *vol, int flags)
|
|||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -134,7 +134,7 @@ int process_each_vg(struct cmd_context *cmd, int argc, char **argv,
|
|||||||
log_verbose("Using volume group(s) on command line");
|
log_verbose("Using volume group(s) on command line");
|
||||||
for (; opt < argc; opt++) {
|
for (; opt < argc; opt++) {
|
||||||
vg_name = argv[opt];
|
vg_name = argv[opt];
|
||||||
if (!lock_vol(cmd, vg_name, LCK_VG | lock_type)) {
|
if (!lock_vol(cmd, vg_name, lock_type)) {
|
||||||
log_error("Can't lock %s: skipping", vg_name);
|
log_error("Can't lock %s: skipping", vg_name);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -150,7 +150,7 @@ int process_each_vg(struct cmd_context *cmd, int argc, char **argv,
|
|||||||
}
|
}
|
||||||
list_iterate(vgh, vgs) {
|
list_iterate(vgh, vgs) {
|
||||||
vg_name = list_item(vgh, struct name_list)->name;
|
vg_name = list_item(vgh, struct name_list)->name;
|
||||||
if (!lock_vol(cmd, vg_name, LCK_VG | lock_type)) {
|
if (!lock_vol(cmd, vg_name, lock_type)) {
|
||||||
log_error("Can't lock %s: skipping", vg_name);
|
log_error("Can't lock %s: skipping", vg_name);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -54,5 +54,5 @@ static int vg_backup_single(struct cmd_context *cmd, const char *vg_name)
|
|||||||
|
|
||||||
int vgcfgbackup(struct cmd_context *cmd, int argc, char **argv)
|
int vgcfgbackup(struct cmd_context *cmd, int argc, char **argv)
|
||||||
{
|
{
|
||||||
return process_each_vg(cmd, argc, argv, LCK_READ, &vg_backup_single);
|
return process_each_vg(cmd, argc, argv, LCK_VG_READ, &vg_backup_single);
|
||||||
}
|
}
|
||||||
|
@ -67,7 +67,7 @@ int vgchange(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
|
|
||||||
return process_each_vg(cmd, argc, argv,
|
return process_each_vg(cmd, argc, argv,
|
||||||
(arg_count(cmd, available_ARG)) ?
|
(arg_count(cmd, available_ARG)) ?
|
||||||
LCK_READ : LCK_WRITE, &vgchange_single);
|
LCK_VG_READ : LCK_VG_WRITE, &vgchange_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vgchange_single(struct cmd_context *cmd, const char *vg_name)
|
static int vgchange_single(struct cmd_context *cmd, const char *vg_name)
|
||||||
|
@ -24,7 +24,7 @@ static int vgck_single(struct cmd_context *cmd, const char *vg_name);
|
|||||||
|
|
||||||
int vgck(struct cmd_context *cmd, int argc, char **argv)
|
int vgck(struct cmd_context *cmd, int argc, char **argv)
|
||||||
{
|
{
|
||||||
return process_each_vg(cmd, argc, argv, LCK_READ, &vgck_single);
|
return process_each_vg(cmd, argc, argv, LCK_VG_READ, &vgck_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vgck_single(struct cmd_context *cmd, const char *vg_name)
|
static int vgck_single(struct cmd_context *cmd, const char *vg_name)
|
||||||
|
@ -45,7 +45,7 @@ int vgdisplay(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
}
|
}
|
||||||
**********/
|
**********/
|
||||||
|
|
||||||
process_each_vg(cmd, argc, argv, LCK_READ, &vgdisplay_single);
|
process_each_vg(cmd, argc, argv, LCK_VG_READ, &vgdisplay_single);
|
||||||
|
|
||||||
/******** FIXME Need to count number processed
|
/******** FIXME Need to count number processed
|
||||||
Add this to process_each_vg if arg_count(cmd,activevolumegroups_ARG) ?
|
Add this to process_each_vg if arg_count(cmd,activevolumegroups_ARG) ?
|
||||||
|
@ -34,7 +34,7 @@ int vgexport(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
return ECMD_FAILED;
|
return ECMD_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
return process_each_vg(cmd, argc, argv, LCK_READ, &vgexport_single);
|
return process_each_vg(cmd, argc, argv, LCK_VG_READ, &vgexport_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vgexport_single(struct cmd_context *cmd, const char *vg_name)
|
static int vgexport_single(struct cmd_context *cmd, const char *vg_name)
|
||||||
|
@ -34,7 +34,7 @@ int vgimport(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
return ECMD_FAILED;
|
return ECMD_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
return process_each_vg(cmd, argc, argv, LCK_WRITE, &vgimport_single);
|
return process_each_vg(cmd, argc, argv, LCK_VG_WRITE, &vgimport_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vgimport_single(struct cmd_context *cmd, const char *vg_name)
|
static int vgimport_single(struct cmd_context *cmd, const char *vg_name)
|
||||||
|
@ -31,7 +31,8 @@ int vgremove(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
return ECMD_FAILED;
|
return ECMD_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = process_each_vg(cmd, argc, argv, LCK_WRITE | LCK_NONBLOCK,
|
ret = process_each_vg(cmd, argc, argv,
|
||||||
|
LCK_VG | LCK_WRITE | LCK_NONBLOCK,
|
||||||
&vgremove_single);
|
&vgremove_single);
|
||||||
|
|
||||||
unlock_vg(cmd, "");
|
unlock_vg(cmd, "");
|
||||||
|
@ -37,7 +37,7 @@ int vgscan(struct cmd_context *cmd, int argc, char **argv)
|
|||||||
|
|
||||||
log_print("Reading all physical volumes. This may take a while...");
|
log_print("Reading all physical volumes. This may take a while...");
|
||||||
|
|
||||||
return process_each_vg(cmd, argc, argv, LCK_READ, &vgscan_single);
|
return process_each_vg(cmd, argc, argv, LCK_VG_READ, &vgscan_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vgscan_single(struct cmd_context *cmd, const char *vg_name)
|
static int vgscan_single(struct cmd_context *cmd, const char *vg_name)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user