Skip to content

Commit

Permalink
dlm: detect available userspace daemon
Browse files Browse the repository at this point in the history
If dlm_controld (the userspace daemon that controls the setup and
recovery of the dlm) fails, the kernel should shut down the lockspaces
in the kernel rather than leaving them running.  This is detected by
having dlm_controld hold a misc device open while running, and if
the kernel detects a close while the daemon is still needed, it stops
the lockspaces in the kernel.

Knowing that the userspace daemon isn't running also allows the
lockspace create/remove routines to avoid waiting on the daemon
for join/leave operations.

Signed-off-by: David Teigland <teigland@redhat.com>
  • Loading branch information
David Teigland committed Aug 28, 2008
1 parent 0f8e0d9 commit dc68c7e
Show file tree
Hide file tree
Showing 4 changed files with 85 additions and 2 deletions.
24 changes: 23 additions & 1 deletion fs/dlm/lockspace.c
Original file line number Diff line number Diff line change
Expand Up @@ -378,6 +378,11 @@ static int new_lockspace(char *name, int namelen, void **lockspace,
if (!try_module_get(THIS_MODULE))
return -EINVAL;

if (!dlm_user_daemon_available()) {
module_put(THIS_MODULE);
return -EUNATCH;
}

error = 0;

spin_lock(&lslist_lock);
Expand Down Expand Up @@ -669,7 +674,7 @@ static int release_lockspace(struct dlm_ls *ls, int force)

dlm_device_deregister(ls);

if (force < 3)
if (force < 3 && dlm_user_daemon_available())
do_uevent(ls, 0);

dlm_recoverd_stop(ls);
Expand Down Expand Up @@ -791,3 +796,20 @@ int dlm_release_lockspace(void *lockspace, int force)
return error;
}

void dlm_stop_lockspaces(void)
{
struct dlm_ls *ls;

restart:
spin_lock(&lslist_lock);
list_for_each_entry(ls, &lslist, ls_list) {
if (!test_bit(LSFL_RUNNING, &ls->ls_flags))
continue;
spin_unlock(&lslist_lock);
log_error(ls, "no userland control daemon, stopping lockspace");
dlm_ls_stop(ls);
goto restart;
}
spin_unlock(&lslist_lock);
}

1 change: 1 addition & 0 deletions fs/dlm/lockspace.h
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ struct dlm_ls *dlm_find_lockspace_global(uint32_t id);
struct dlm_ls *dlm_find_lockspace_local(void *id);
struct dlm_ls *dlm_find_lockspace_device(int minor);
void dlm_put_lockspace(struct dlm_ls *ls);
void dlm_stop_lockspaces(void);

#endif /* __LOCKSPACE_DOT_H__ */

61 changes: 60 additions & 1 deletion fs/dlm/user.c
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@

static const char name_prefix[] = "dlm";
static const struct file_operations device_fops;
static atomic_t dlm_monitor_opened;
static int dlm_monitor_unused = 1;

#ifdef CONFIG_COMPAT

Expand Down Expand Up @@ -890,6 +892,26 @@ static unsigned int device_poll(struct file *file, poll_table *wait)
return 0;
}

int dlm_user_daemon_available(void)
{
/* dlm_controld hasn't started (or, has started, but not
properly populated configfs) */

if (!dlm_our_nodeid())
return 0;

/* This is to deal with versions of dlm_controld that don't
know about the monitor device. We assume that if the
dlm_controld was started (above), but the monitor device
was never opened, that it's an old version. dlm_controld
should open the monitor device before populating configfs. */

if (dlm_monitor_unused)
return 1;

return atomic_read(&dlm_monitor_opened) ? 1 : 0;
}

static int ctl_device_open(struct inode *inode, struct file *file)
{
cycle_kernel_lock();
Expand All @@ -902,6 +924,20 @@ static int ctl_device_close(struct inode *inode, struct file *file)
return 0;
}

static int monitor_device_open(struct inode *inode, struct file *file)
{
atomic_inc(&dlm_monitor_opened);
dlm_monitor_unused = 0;
return 0;
}

static int monitor_device_close(struct inode *inode, struct file *file)
{
if (atomic_dec_and_test(&dlm_monitor_opened))
dlm_stop_lockspaces();
return 0;
}

static const struct file_operations device_fops = {
.open = device_open,
.release = device_close,
Expand All @@ -925,19 +961,42 @@ static struct miscdevice ctl_device = {
.minor = MISC_DYNAMIC_MINOR,
};

static const struct file_operations monitor_device_fops = {
.open = monitor_device_open,
.release = monitor_device_close,
.owner = THIS_MODULE,
};

static struct miscdevice monitor_device = {
.name = "dlm-monitor",
.fops = &monitor_device_fops,
.minor = MISC_DYNAMIC_MINOR,
};

int __init dlm_user_init(void)
{
int error;

atomic_set(&dlm_monitor_opened, 0);

error = misc_register(&ctl_device);
if (error)
if (error) {
log_print("misc_register failed for control device");
goto out;
}

error = misc_register(&monitor_device);
if (error) {
log_print("misc_register failed for monitor device");
misc_deregister(&ctl_device);
}
out:
return error;
}

void dlm_user_exit(void)
{
misc_deregister(&ctl_device);
misc_deregister(&monitor_device);
}

1 change: 1 addition & 0 deletions fs/dlm/user.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,5 +13,6 @@ void dlm_user_add_ast(struct dlm_lkb *lkb, int type);
int dlm_user_init(void);
void dlm_user_exit(void);
int dlm_device_deregister(struct dlm_ls *ls);
int dlm_user_daemon_available(void);

#endif

0 comments on commit dc68c7e

Please sign in to comment.