18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/******************************************************************************
38c2ecf20Sopenharmony_ci*******************************************************************************
48c2ecf20Sopenharmony_ci**
58c2ecf20Sopenharmony_ci**  Copyright (C) Sistina Software, Inc.  1997-2003  All rights reserved.
68c2ecf20Sopenharmony_ci**  Copyright (C) 2004-2011 Red Hat, Inc.  All rights reserved.
78c2ecf20Sopenharmony_ci**
88c2ecf20Sopenharmony_ci**
98c2ecf20Sopenharmony_ci*******************************************************************************
108c2ecf20Sopenharmony_ci******************************************************************************/
118c2ecf20Sopenharmony_ci
128c2ecf20Sopenharmony_ci#include <linux/module.h>
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_ci#include "dlm_internal.h"
158c2ecf20Sopenharmony_ci#include "lockspace.h"
168c2ecf20Sopenharmony_ci#include "member.h"
178c2ecf20Sopenharmony_ci#include "recoverd.h"
188c2ecf20Sopenharmony_ci#include "dir.h"
198c2ecf20Sopenharmony_ci#include "lowcomms.h"
208c2ecf20Sopenharmony_ci#include "config.h"
218c2ecf20Sopenharmony_ci#include "memory.h"
228c2ecf20Sopenharmony_ci#include "lock.h"
238c2ecf20Sopenharmony_ci#include "recover.h"
248c2ecf20Sopenharmony_ci#include "requestqueue.h"
258c2ecf20Sopenharmony_ci#include "user.h"
268c2ecf20Sopenharmony_ci#include "ast.h"
278c2ecf20Sopenharmony_ci
288c2ecf20Sopenharmony_cistatic int			ls_count;
298c2ecf20Sopenharmony_cistatic struct mutex		ls_lock;
308c2ecf20Sopenharmony_cistatic struct list_head		lslist;
318c2ecf20Sopenharmony_cistatic spinlock_t		lslist_lock;
328c2ecf20Sopenharmony_cistatic struct task_struct *	scand_task;
338c2ecf20Sopenharmony_ci
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_cistatic ssize_t dlm_control_store(struct dlm_ls *ls, const char *buf, size_t len)
368c2ecf20Sopenharmony_ci{
378c2ecf20Sopenharmony_ci	ssize_t ret = len;
388c2ecf20Sopenharmony_ci	int n;
398c2ecf20Sopenharmony_ci	int rc = kstrtoint(buf, 0, &n);
408c2ecf20Sopenharmony_ci
418c2ecf20Sopenharmony_ci	if (rc)
428c2ecf20Sopenharmony_ci		return rc;
438c2ecf20Sopenharmony_ci	ls = dlm_find_lockspace_local(ls->ls_local_handle);
448c2ecf20Sopenharmony_ci	if (!ls)
458c2ecf20Sopenharmony_ci		return -EINVAL;
468c2ecf20Sopenharmony_ci
478c2ecf20Sopenharmony_ci	switch (n) {
488c2ecf20Sopenharmony_ci	case 0:
498c2ecf20Sopenharmony_ci		dlm_ls_stop(ls);
508c2ecf20Sopenharmony_ci		break;
518c2ecf20Sopenharmony_ci	case 1:
528c2ecf20Sopenharmony_ci		dlm_ls_start(ls);
538c2ecf20Sopenharmony_ci		break;
548c2ecf20Sopenharmony_ci	default:
558c2ecf20Sopenharmony_ci		ret = -EINVAL;
568c2ecf20Sopenharmony_ci	}
578c2ecf20Sopenharmony_ci	dlm_put_lockspace(ls);
588c2ecf20Sopenharmony_ci	return ret;
598c2ecf20Sopenharmony_ci}
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_cistatic ssize_t dlm_event_store(struct dlm_ls *ls, const char *buf, size_t len)
628c2ecf20Sopenharmony_ci{
638c2ecf20Sopenharmony_ci	int rc = kstrtoint(buf, 0, &ls->ls_uevent_result);
648c2ecf20Sopenharmony_ci
658c2ecf20Sopenharmony_ci	if (rc)
668c2ecf20Sopenharmony_ci		return rc;
678c2ecf20Sopenharmony_ci	set_bit(LSFL_UEVENT_WAIT, &ls->ls_flags);
688c2ecf20Sopenharmony_ci	wake_up(&ls->ls_uevent_wait);
698c2ecf20Sopenharmony_ci	return len;
708c2ecf20Sopenharmony_ci}
718c2ecf20Sopenharmony_ci
728c2ecf20Sopenharmony_cistatic ssize_t dlm_id_show(struct dlm_ls *ls, char *buf)
738c2ecf20Sopenharmony_ci{
748c2ecf20Sopenharmony_ci	return snprintf(buf, PAGE_SIZE, "%u\n", ls->ls_global_id);
758c2ecf20Sopenharmony_ci}
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_cistatic ssize_t dlm_id_store(struct dlm_ls *ls, const char *buf, size_t len)
788c2ecf20Sopenharmony_ci{
798c2ecf20Sopenharmony_ci	int rc = kstrtouint(buf, 0, &ls->ls_global_id);
808c2ecf20Sopenharmony_ci
818c2ecf20Sopenharmony_ci	if (rc)
828c2ecf20Sopenharmony_ci		return rc;
838c2ecf20Sopenharmony_ci	return len;
848c2ecf20Sopenharmony_ci}
858c2ecf20Sopenharmony_ci
868c2ecf20Sopenharmony_cistatic ssize_t dlm_nodir_show(struct dlm_ls *ls, char *buf)
878c2ecf20Sopenharmony_ci{
888c2ecf20Sopenharmony_ci	return snprintf(buf, PAGE_SIZE, "%u\n", dlm_no_directory(ls));
898c2ecf20Sopenharmony_ci}
908c2ecf20Sopenharmony_ci
918c2ecf20Sopenharmony_cistatic ssize_t dlm_nodir_store(struct dlm_ls *ls, const char *buf, size_t len)
928c2ecf20Sopenharmony_ci{
938c2ecf20Sopenharmony_ci	int val;
948c2ecf20Sopenharmony_ci	int rc = kstrtoint(buf, 0, &val);
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_ci	if (rc)
978c2ecf20Sopenharmony_ci		return rc;
988c2ecf20Sopenharmony_ci	if (val == 1)
998c2ecf20Sopenharmony_ci		set_bit(LSFL_NODIR, &ls->ls_flags);
1008c2ecf20Sopenharmony_ci	return len;
1018c2ecf20Sopenharmony_ci}
1028c2ecf20Sopenharmony_ci
1038c2ecf20Sopenharmony_cistatic ssize_t dlm_recover_status_show(struct dlm_ls *ls, char *buf)
1048c2ecf20Sopenharmony_ci{
1058c2ecf20Sopenharmony_ci	uint32_t status = dlm_recover_status(ls);
1068c2ecf20Sopenharmony_ci	return snprintf(buf, PAGE_SIZE, "%x\n", status);
1078c2ecf20Sopenharmony_ci}
1088c2ecf20Sopenharmony_ci
1098c2ecf20Sopenharmony_cistatic ssize_t dlm_recover_nodeid_show(struct dlm_ls *ls, char *buf)
1108c2ecf20Sopenharmony_ci{
1118c2ecf20Sopenharmony_ci	return snprintf(buf, PAGE_SIZE, "%d\n", ls->ls_recover_nodeid);
1128c2ecf20Sopenharmony_ci}
1138c2ecf20Sopenharmony_ci
1148c2ecf20Sopenharmony_cistruct dlm_attr {
1158c2ecf20Sopenharmony_ci	struct attribute attr;
1168c2ecf20Sopenharmony_ci	ssize_t (*show)(struct dlm_ls *, char *);
1178c2ecf20Sopenharmony_ci	ssize_t (*store)(struct dlm_ls *, const char *, size_t);
1188c2ecf20Sopenharmony_ci};
1198c2ecf20Sopenharmony_ci
1208c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_control = {
1218c2ecf20Sopenharmony_ci	.attr  = {.name = "control", .mode = S_IWUSR},
1228c2ecf20Sopenharmony_ci	.store = dlm_control_store
1238c2ecf20Sopenharmony_ci};
1248c2ecf20Sopenharmony_ci
1258c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_event = {
1268c2ecf20Sopenharmony_ci	.attr  = {.name = "event_done", .mode = S_IWUSR},
1278c2ecf20Sopenharmony_ci	.store = dlm_event_store
1288c2ecf20Sopenharmony_ci};
1298c2ecf20Sopenharmony_ci
1308c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_id = {
1318c2ecf20Sopenharmony_ci	.attr  = {.name = "id", .mode = S_IRUGO | S_IWUSR},
1328c2ecf20Sopenharmony_ci	.show  = dlm_id_show,
1338c2ecf20Sopenharmony_ci	.store = dlm_id_store
1348c2ecf20Sopenharmony_ci};
1358c2ecf20Sopenharmony_ci
1368c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_nodir = {
1378c2ecf20Sopenharmony_ci	.attr  = {.name = "nodir", .mode = S_IRUGO | S_IWUSR},
1388c2ecf20Sopenharmony_ci	.show  = dlm_nodir_show,
1398c2ecf20Sopenharmony_ci	.store = dlm_nodir_store
1408c2ecf20Sopenharmony_ci};
1418c2ecf20Sopenharmony_ci
1428c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_recover_status = {
1438c2ecf20Sopenharmony_ci	.attr  = {.name = "recover_status", .mode = S_IRUGO},
1448c2ecf20Sopenharmony_ci	.show  = dlm_recover_status_show
1458c2ecf20Sopenharmony_ci};
1468c2ecf20Sopenharmony_ci
1478c2ecf20Sopenharmony_cistatic struct dlm_attr dlm_attr_recover_nodeid = {
1488c2ecf20Sopenharmony_ci	.attr  = {.name = "recover_nodeid", .mode = S_IRUGO},
1498c2ecf20Sopenharmony_ci	.show  = dlm_recover_nodeid_show
1508c2ecf20Sopenharmony_ci};
1518c2ecf20Sopenharmony_ci
1528c2ecf20Sopenharmony_cistatic struct attribute *dlm_attrs[] = {
1538c2ecf20Sopenharmony_ci	&dlm_attr_control.attr,
1548c2ecf20Sopenharmony_ci	&dlm_attr_event.attr,
1558c2ecf20Sopenharmony_ci	&dlm_attr_id.attr,
1568c2ecf20Sopenharmony_ci	&dlm_attr_nodir.attr,
1578c2ecf20Sopenharmony_ci	&dlm_attr_recover_status.attr,
1588c2ecf20Sopenharmony_ci	&dlm_attr_recover_nodeid.attr,
1598c2ecf20Sopenharmony_ci	NULL,
1608c2ecf20Sopenharmony_ci};
1618c2ecf20Sopenharmony_ciATTRIBUTE_GROUPS(dlm);
1628c2ecf20Sopenharmony_ci
1638c2ecf20Sopenharmony_cistatic ssize_t dlm_attr_show(struct kobject *kobj, struct attribute *attr,
1648c2ecf20Sopenharmony_ci			     char *buf)
1658c2ecf20Sopenharmony_ci{
1668c2ecf20Sopenharmony_ci	struct dlm_ls *ls  = container_of(kobj, struct dlm_ls, ls_kobj);
1678c2ecf20Sopenharmony_ci	struct dlm_attr *a = container_of(attr, struct dlm_attr, attr);
1688c2ecf20Sopenharmony_ci	return a->show ? a->show(ls, buf) : 0;
1698c2ecf20Sopenharmony_ci}
1708c2ecf20Sopenharmony_ci
1718c2ecf20Sopenharmony_cistatic ssize_t dlm_attr_store(struct kobject *kobj, struct attribute *attr,
1728c2ecf20Sopenharmony_ci			      const char *buf, size_t len)
1738c2ecf20Sopenharmony_ci{
1748c2ecf20Sopenharmony_ci	struct dlm_ls *ls  = container_of(kobj, struct dlm_ls, ls_kobj);
1758c2ecf20Sopenharmony_ci	struct dlm_attr *a = container_of(attr, struct dlm_attr, attr);
1768c2ecf20Sopenharmony_ci	return a->store ? a->store(ls, buf, len) : len;
1778c2ecf20Sopenharmony_ci}
1788c2ecf20Sopenharmony_ci
1798c2ecf20Sopenharmony_cistatic void lockspace_kobj_release(struct kobject *k)
1808c2ecf20Sopenharmony_ci{
1818c2ecf20Sopenharmony_ci	struct dlm_ls *ls  = container_of(k, struct dlm_ls, ls_kobj);
1828c2ecf20Sopenharmony_ci	kfree(ls);
1838c2ecf20Sopenharmony_ci}
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_cistatic const struct sysfs_ops dlm_attr_ops = {
1868c2ecf20Sopenharmony_ci	.show  = dlm_attr_show,
1878c2ecf20Sopenharmony_ci	.store = dlm_attr_store,
1888c2ecf20Sopenharmony_ci};
1898c2ecf20Sopenharmony_ci
1908c2ecf20Sopenharmony_cistatic struct kobj_type dlm_ktype = {
1918c2ecf20Sopenharmony_ci	.default_groups = dlm_groups,
1928c2ecf20Sopenharmony_ci	.sysfs_ops     = &dlm_attr_ops,
1938c2ecf20Sopenharmony_ci	.release       = lockspace_kobj_release,
1948c2ecf20Sopenharmony_ci};
1958c2ecf20Sopenharmony_ci
1968c2ecf20Sopenharmony_cistatic struct kset *dlm_kset;
1978c2ecf20Sopenharmony_ci
1988c2ecf20Sopenharmony_cistatic int do_uevent(struct dlm_ls *ls, int in)
1998c2ecf20Sopenharmony_ci{
2008c2ecf20Sopenharmony_ci	if (in)
2018c2ecf20Sopenharmony_ci		kobject_uevent(&ls->ls_kobj, KOBJ_ONLINE);
2028c2ecf20Sopenharmony_ci	else
2038c2ecf20Sopenharmony_ci		kobject_uevent(&ls->ls_kobj, KOBJ_OFFLINE);
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci	log_rinfo(ls, "%s the lockspace group...", in ? "joining" : "leaving");
2068c2ecf20Sopenharmony_ci
2078c2ecf20Sopenharmony_ci	/* dlm_controld will see the uevent, do the necessary group management
2088c2ecf20Sopenharmony_ci	   and then write to sysfs to wake us */
2098c2ecf20Sopenharmony_ci
2108c2ecf20Sopenharmony_ci	wait_event(ls->ls_uevent_wait,
2118c2ecf20Sopenharmony_ci		   test_and_clear_bit(LSFL_UEVENT_WAIT, &ls->ls_flags));
2128c2ecf20Sopenharmony_ci
2138c2ecf20Sopenharmony_ci	log_rinfo(ls, "group event done %d", ls->ls_uevent_result);
2148c2ecf20Sopenharmony_ci
2158c2ecf20Sopenharmony_ci	return ls->ls_uevent_result;
2168c2ecf20Sopenharmony_ci}
2178c2ecf20Sopenharmony_ci
2188c2ecf20Sopenharmony_cistatic int dlm_uevent(struct kset *kset, struct kobject *kobj,
2198c2ecf20Sopenharmony_ci		      struct kobj_uevent_env *env)
2208c2ecf20Sopenharmony_ci{
2218c2ecf20Sopenharmony_ci	struct dlm_ls *ls = container_of(kobj, struct dlm_ls, ls_kobj);
2228c2ecf20Sopenharmony_ci
2238c2ecf20Sopenharmony_ci	add_uevent_var(env, "LOCKSPACE=%s", ls->ls_name);
2248c2ecf20Sopenharmony_ci	return 0;
2258c2ecf20Sopenharmony_ci}
2268c2ecf20Sopenharmony_ci
2278c2ecf20Sopenharmony_cistatic const struct kset_uevent_ops dlm_uevent_ops = {
2288c2ecf20Sopenharmony_ci	.uevent = dlm_uevent,
2298c2ecf20Sopenharmony_ci};
2308c2ecf20Sopenharmony_ci
2318c2ecf20Sopenharmony_ciint __init dlm_lockspace_init(void)
2328c2ecf20Sopenharmony_ci{
2338c2ecf20Sopenharmony_ci	ls_count = 0;
2348c2ecf20Sopenharmony_ci	mutex_init(&ls_lock);
2358c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&lslist);
2368c2ecf20Sopenharmony_ci	spin_lock_init(&lslist_lock);
2378c2ecf20Sopenharmony_ci
2388c2ecf20Sopenharmony_ci	dlm_kset = kset_create_and_add("dlm", &dlm_uevent_ops, kernel_kobj);
2398c2ecf20Sopenharmony_ci	if (!dlm_kset) {
2408c2ecf20Sopenharmony_ci		printk(KERN_WARNING "%s: can not create kset\n", __func__);
2418c2ecf20Sopenharmony_ci		return -ENOMEM;
2428c2ecf20Sopenharmony_ci	}
2438c2ecf20Sopenharmony_ci	return 0;
2448c2ecf20Sopenharmony_ci}
2458c2ecf20Sopenharmony_ci
2468c2ecf20Sopenharmony_civoid dlm_lockspace_exit(void)
2478c2ecf20Sopenharmony_ci{
2488c2ecf20Sopenharmony_ci	kset_unregister(dlm_kset);
2498c2ecf20Sopenharmony_ci}
2508c2ecf20Sopenharmony_ci
2518c2ecf20Sopenharmony_cistatic struct dlm_ls *find_ls_to_scan(void)
2528c2ecf20Sopenharmony_ci{
2538c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
2548c2ecf20Sopenharmony_ci
2558c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
2568c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
2578c2ecf20Sopenharmony_ci		if (time_after_eq(jiffies, ls->ls_scan_time +
2588c2ecf20Sopenharmony_ci					    dlm_config.ci_scan_secs * HZ)) {
2598c2ecf20Sopenharmony_ci			spin_unlock(&lslist_lock);
2608c2ecf20Sopenharmony_ci			return ls;
2618c2ecf20Sopenharmony_ci		}
2628c2ecf20Sopenharmony_ci	}
2638c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
2648c2ecf20Sopenharmony_ci	return NULL;
2658c2ecf20Sopenharmony_ci}
2668c2ecf20Sopenharmony_ci
2678c2ecf20Sopenharmony_cistatic int dlm_scand(void *data)
2688c2ecf20Sopenharmony_ci{
2698c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
2708c2ecf20Sopenharmony_ci
2718c2ecf20Sopenharmony_ci	while (!kthread_should_stop()) {
2728c2ecf20Sopenharmony_ci		ls = find_ls_to_scan();
2738c2ecf20Sopenharmony_ci		if (ls) {
2748c2ecf20Sopenharmony_ci			if (dlm_lock_recovery_try(ls)) {
2758c2ecf20Sopenharmony_ci				ls->ls_scan_time = jiffies;
2768c2ecf20Sopenharmony_ci				dlm_scan_rsbs(ls);
2778c2ecf20Sopenharmony_ci				dlm_scan_timeout(ls);
2788c2ecf20Sopenharmony_ci				dlm_scan_waiters(ls);
2798c2ecf20Sopenharmony_ci				dlm_unlock_recovery(ls);
2808c2ecf20Sopenharmony_ci			} else {
2818c2ecf20Sopenharmony_ci				ls->ls_scan_time += HZ;
2828c2ecf20Sopenharmony_ci			}
2838c2ecf20Sopenharmony_ci			continue;
2848c2ecf20Sopenharmony_ci		}
2858c2ecf20Sopenharmony_ci		schedule_timeout_interruptible(dlm_config.ci_scan_secs * HZ);
2868c2ecf20Sopenharmony_ci	}
2878c2ecf20Sopenharmony_ci	return 0;
2888c2ecf20Sopenharmony_ci}
2898c2ecf20Sopenharmony_ci
2908c2ecf20Sopenharmony_cistatic int dlm_scand_start(void)
2918c2ecf20Sopenharmony_ci{
2928c2ecf20Sopenharmony_ci	struct task_struct *p;
2938c2ecf20Sopenharmony_ci	int error = 0;
2948c2ecf20Sopenharmony_ci
2958c2ecf20Sopenharmony_ci	p = kthread_run(dlm_scand, NULL, "dlm_scand");
2968c2ecf20Sopenharmony_ci	if (IS_ERR(p))
2978c2ecf20Sopenharmony_ci		error = PTR_ERR(p);
2988c2ecf20Sopenharmony_ci	else
2998c2ecf20Sopenharmony_ci		scand_task = p;
3008c2ecf20Sopenharmony_ci	return error;
3018c2ecf20Sopenharmony_ci}
3028c2ecf20Sopenharmony_ci
3038c2ecf20Sopenharmony_cistatic void dlm_scand_stop(void)
3048c2ecf20Sopenharmony_ci{
3058c2ecf20Sopenharmony_ci	kthread_stop(scand_task);
3068c2ecf20Sopenharmony_ci}
3078c2ecf20Sopenharmony_ci
3088c2ecf20Sopenharmony_cistruct dlm_ls *dlm_find_lockspace_global(uint32_t id)
3098c2ecf20Sopenharmony_ci{
3108c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
3118c2ecf20Sopenharmony_ci
3128c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
3138c2ecf20Sopenharmony_ci
3148c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
3158c2ecf20Sopenharmony_ci		if (ls->ls_global_id == id) {
3168c2ecf20Sopenharmony_ci			ls->ls_count++;
3178c2ecf20Sopenharmony_ci			goto out;
3188c2ecf20Sopenharmony_ci		}
3198c2ecf20Sopenharmony_ci	}
3208c2ecf20Sopenharmony_ci	ls = NULL;
3218c2ecf20Sopenharmony_ci out:
3228c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
3238c2ecf20Sopenharmony_ci	return ls;
3248c2ecf20Sopenharmony_ci}
3258c2ecf20Sopenharmony_ci
3268c2ecf20Sopenharmony_cistruct dlm_ls *dlm_find_lockspace_local(dlm_lockspace_t *lockspace)
3278c2ecf20Sopenharmony_ci{
3288c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
3298c2ecf20Sopenharmony_ci
3308c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
3318c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
3328c2ecf20Sopenharmony_ci		if (ls->ls_local_handle == lockspace) {
3338c2ecf20Sopenharmony_ci			ls->ls_count++;
3348c2ecf20Sopenharmony_ci			goto out;
3358c2ecf20Sopenharmony_ci		}
3368c2ecf20Sopenharmony_ci	}
3378c2ecf20Sopenharmony_ci	ls = NULL;
3388c2ecf20Sopenharmony_ci out:
3398c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
3408c2ecf20Sopenharmony_ci	return ls;
3418c2ecf20Sopenharmony_ci}
3428c2ecf20Sopenharmony_ci
3438c2ecf20Sopenharmony_cistruct dlm_ls *dlm_find_lockspace_device(int minor)
3448c2ecf20Sopenharmony_ci{
3458c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
3468c2ecf20Sopenharmony_ci
3478c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
3488c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
3498c2ecf20Sopenharmony_ci		if (ls->ls_device.minor == minor) {
3508c2ecf20Sopenharmony_ci			ls->ls_count++;
3518c2ecf20Sopenharmony_ci			goto out;
3528c2ecf20Sopenharmony_ci		}
3538c2ecf20Sopenharmony_ci	}
3548c2ecf20Sopenharmony_ci	ls = NULL;
3558c2ecf20Sopenharmony_ci out:
3568c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
3578c2ecf20Sopenharmony_ci	return ls;
3588c2ecf20Sopenharmony_ci}
3598c2ecf20Sopenharmony_ci
3608c2ecf20Sopenharmony_civoid dlm_put_lockspace(struct dlm_ls *ls)
3618c2ecf20Sopenharmony_ci{
3628c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
3638c2ecf20Sopenharmony_ci	ls->ls_count--;
3648c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
3658c2ecf20Sopenharmony_ci}
3668c2ecf20Sopenharmony_ci
3678c2ecf20Sopenharmony_cistatic void remove_lockspace(struct dlm_ls *ls)
3688c2ecf20Sopenharmony_ci{
3698c2ecf20Sopenharmony_ci	for (;;) {
3708c2ecf20Sopenharmony_ci		spin_lock(&lslist_lock);
3718c2ecf20Sopenharmony_ci		if (ls->ls_count == 0) {
3728c2ecf20Sopenharmony_ci			WARN_ON(ls->ls_create_count != 0);
3738c2ecf20Sopenharmony_ci			list_del(&ls->ls_list);
3748c2ecf20Sopenharmony_ci			spin_unlock(&lslist_lock);
3758c2ecf20Sopenharmony_ci			return;
3768c2ecf20Sopenharmony_ci		}
3778c2ecf20Sopenharmony_ci		spin_unlock(&lslist_lock);
3788c2ecf20Sopenharmony_ci		ssleep(1);
3798c2ecf20Sopenharmony_ci	}
3808c2ecf20Sopenharmony_ci}
3818c2ecf20Sopenharmony_ci
3828c2ecf20Sopenharmony_cistatic int threads_start(void)
3838c2ecf20Sopenharmony_ci{
3848c2ecf20Sopenharmony_ci	int error;
3858c2ecf20Sopenharmony_ci
3868c2ecf20Sopenharmony_ci	error = dlm_scand_start();
3878c2ecf20Sopenharmony_ci	if (error) {
3888c2ecf20Sopenharmony_ci		log_print("cannot start dlm_scand thread %d", error);
3898c2ecf20Sopenharmony_ci		goto fail;
3908c2ecf20Sopenharmony_ci	}
3918c2ecf20Sopenharmony_ci
3928c2ecf20Sopenharmony_ci	/* Thread for sending/receiving messages for all lockspace's */
3938c2ecf20Sopenharmony_ci	error = dlm_lowcomms_start();
3948c2ecf20Sopenharmony_ci	if (error) {
3958c2ecf20Sopenharmony_ci		log_print("cannot start dlm lowcomms %d", error);
3968c2ecf20Sopenharmony_ci		goto scand_fail;
3978c2ecf20Sopenharmony_ci	}
3988c2ecf20Sopenharmony_ci
3998c2ecf20Sopenharmony_ci	return 0;
4008c2ecf20Sopenharmony_ci
4018c2ecf20Sopenharmony_ci scand_fail:
4028c2ecf20Sopenharmony_ci	dlm_scand_stop();
4038c2ecf20Sopenharmony_ci fail:
4048c2ecf20Sopenharmony_ci	return error;
4058c2ecf20Sopenharmony_ci}
4068c2ecf20Sopenharmony_ci
4078c2ecf20Sopenharmony_cistatic void threads_stop(void)
4088c2ecf20Sopenharmony_ci{
4098c2ecf20Sopenharmony_ci	dlm_scand_stop();
4108c2ecf20Sopenharmony_ci	dlm_lowcomms_stop();
4118c2ecf20Sopenharmony_ci}
4128c2ecf20Sopenharmony_ci
4138c2ecf20Sopenharmony_cistatic int new_lockspace(const char *name, const char *cluster,
4148c2ecf20Sopenharmony_ci			 uint32_t flags, int lvblen,
4158c2ecf20Sopenharmony_ci			 const struct dlm_lockspace_ops *ops, void *ops_arg,
4168c2ecf20Sopenharmony_ci			 int *ops_result, dlm_lockspace_t **lockspace)
4178c2ecf20Sopenharmony_ci{
4188c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
4198c2ecf20Sopenharmony_ci	int i, size, error;
4208c2ecf20Sopenharmony_ci	int do_unreg = 0;
4218c2ecf20Sopenharmony_ci	int namelen = strlen(name);
4228c2ecf20Sopenharmony_ci
4238c2ecf20Sopenharmony_ci	if (namelen > DLM_LOCKSPACE_LEN || namelen == 0)
4248c2ecf20Sopenharmony_ci		return -EINVAL;
4258c2ecf20Sopenharmony_ci
4268c2ecf20Sopenharmony_ci	if (!lvblen || (lvblen % 8))
4278c2ecf20Sopenharmony_ci		return -EINVAL;
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_ci	if (!try_module_get(THIS_MODULE))
4308c2ecf20Sopenharmony_ci		return -EINVAL;
4318c2ecf20Sopenharmony_ci
4328c2ecf20Sopenharmony_ci	if (!dlm_user_daemon_available()) {
4338c2ecf20Sopenharmony_ci		log_print("dlm user daemon not available");
4348c2ecf20Sopenharmony_ci		error = -EUNATCH;
4358c2ecf20Sopenharmony_ci		goto out;
4368c2ecf20Sopenharmony_ci	}
4378c2ecf20Sopenharmony_ci
4388c2ecf20Sopenharmony_ci	if (ops && ops_result) {
4398c2ecf20Sopenharmony_ci	       	if (!dlm_config.ci_recover_callbacks)
4408c2ecf20Sopenharmony_ci			*ops_result = -EOPNOTSUPP;
4418c2ecf20Sopenharmony_ci		else
4428c2ecf20Sopenharmony_ci			*ops_result = 0;
4438c2ecf20Sopenharmony_ci	}
4448c2ecf20Sopenharmony_ci
4458c2ecf20Sopenharmony_ci	if (!cluster)
4468c2ecf20Sopenharmony_ci		log_print("dlm cluster name '%s' is being used without an application provided cluster name",
4478c2ecf20Sopenharmony_ci			  dlm_config.ci_cluster_name);
4488c2ecf20Sopenharmony_ci
4498c2ecf20Sopenharmony_ci	if (dlm_config.ci_recover_callbacks && cluster &&
4508c2ecf20Sopenharmony_ci	    strncmp(cluster, dlm_config.ci_cluster_name, DLM_LOCKSPACE_LEN)) {
4518c2ecf20Sopenharmony_ci		log_print("dlm cluster name '%s' does not match "
4528c2ecf20Sopenharmony_ci			  "the application cluster name '%s'",
4538c2ecf20Sopenharmony_ci			  dlm_config.ci_cluster_name, cluster);
4548c2ecf20Sopenharmony_ci		error = -EBADR;
4558c2ecf20Sopenharmony_ci		goto out;
4568c2ecf20Sopenharmony_ci	}
4578c2ecf20Sopenharmony_ci
4588c2ecf20Sopenharmony_ci	error = 0;
4598c2ecf20Sopenharmony_ci
4608c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
4618c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
4628c2ecf20Sopenharmony_ci		WARN_ON(ls->ls_create_count <= 0);
4638c2ecf20Sopenharmony_ci		if (ls->ls_namelen != namelen)
4648c2ecf20Sopenharmony_ci			continue;
4658c2ecf20Sopenharmony_ci		if (memcmp(ls->ls_name, name, namelen))
4668c2ecf20Sopenharmony_ci			continue;
4678c2ecf20Sopenharmony_ci		if (flags & DLM_LSFL_NEWEXCL) {
4688c2ecf20Sopenharmony_ci			error = -EEXIST;
4698c2ecf20Sopenharmony_ci			break;
4708c2ecf20Sopenharmony_ci		}
4718c2ecf20Sopenharmony_ci		ls->ls_create_count++;
4728c2ecf20Sopenharmony_ci		*lockspace = ls;
4738c2ecf20Sopenharmony_ci		error = 1;
4748c2ecf20Sopenharmony_ci		break;
4758c2ecf20Sopenharmony_ci	}
4768c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
4778c2ecf20Sopenharmony_ci
4788c2ecf20Sopenharmony_ci	if (error)
4798c2ecf20Sopenharmony_ci		goto out;
4808c2ecf20Sopenharmony_ci
4818c2ecf20Sopenharmony_ci	error = -ENOMEM;
4828c2ecf20Sopenharmony_ci
4838c2ecf20Sopenharmony_ci	ls = kzalloc(sizeof(struct dlm_ls) + namelen, GFP_NOFS);
4848c2ecf20Sopenharmony_ci	if (!ls)
4858c2ecf20Sopenharmony_ci		goto out;
4868c2ecf20Sopenharmony_ci	memcpy(ls->ls_name, name, namelen);
4878c2ecf20Sopenharmony_ci	ls->ls_namelen = namelen;
4888c2ecf20Sopenharmony_ci	ls->ls_lvblen = lvblen;
4898c2ecf20Sopenharmony_ci	ls->ls_count = 0;
4908c2ecf20Sopenharmony_ci	ls->ls_flags = 0;
4918c2ecf20Sopenharmony_ci	ls->ls_scan_time = jiffies;
4928c2ecf20Sopenharmony_ci
4938c2ecf20Sopenharmony_ci	if (ops && dlm_config.ci_recover_callbacks) {
4948c2ecf20Sopenharmony_ci		ls->ls_ops = ops;
4958c2ecf20Sopenharmony_ci		ls->ls_ops_arg = ops_arg;
4968c2ecf20Sopenharmony_ci	}
4978c2ecf20Sopenharmony_ci
4988c2ecf20Sopenharmony_ci	if (flags & DLM_LSFL_TIMEWARN)
4998c2ecf20Sopenharmony_ci		set_bit(LSFL_TIMEWARN, &ls->ls_flags);
5008c2ecf20Sopenharmony_ci
5018c2ecf20Sopenharmony_ci	/* ls_exflags are forced to match among nodes, and we don't
5028c2ecf20Sopenharmony_ci	   need to require all nodes to have some flags set */
5038c2ecf20Sopenharmony_ci	ls->ls_exflags = (flags & ~(DLM_LSFL_TIMEWARN | DLM_LSFL_FS |
5048c2ecf20Sopenharmony_ci				    DLM_LSFL_NEWEXCL));
5058c2ecf20Sopenharmony_ci
5068c2ecf20Sopenharmony_ci	size = dlm_config.ci_rsbtbl_size;
5078c2ecf20Sopenharmony_ci	ls->ls_rsbtbl_size = size;
5088c2ecf20Sopenharmony_ci
5098c2ecf20Sopenharmony_ci	ls->ls_rsbtbl = vmalloc(array_size(size, sizeof(struct dlm_rsbtable)));
5108c2ecf20Sopenharmony_ci	if (!ls->ls_rsbtbl)
5118c2ecf20Sopenharmony_ci		goto out_lsfree;
5128c2ecf20Sopenharmony_ci	for (i = 0; i < size; i++) {
5138c2ecf20Sopenharmony_ci		ls->ls_rsbtbl[i].keep.rb_node = NULL;
5148c2ecf20Sopenharmony_ci		ls->ls_rsbtbl[i].toss.rb_node = NULL;
5158c2ecf20Sopenharmony_ci		spin_lock_init(&ls->ls_rsbtbl[i].lock);
5168c2ecf20Sopenharmony_ci	}
5178c2ecf20Sopenharmony_ci
5188c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_remove_spin);
5198c2ecf20Sopenharmony_ci
5208c2ecf20Sopenharmony_ci	for (i = 0; i < DLM_REMOVE_NAMES_MAX; i++) {
5218c2ecf20Sopenharmony_ci		ls->ls_remove_names[i] = kzalloc(DLM_RESNAME_MAXLEN+1,
5228c2ecf20Sopenharmony_ci						 GFP_KERNEL);
5238c2ecf20Sopenharmony_ci		if (!ls->ls_remove_names[i])
5248c2ecf20Sopenharmony_ci			goto out_rsbtbl;
5258c2ecf20Sopenharmony_ci	}
5268c2ecf20Sopenharmony_ci
5278c2ecf20Sopenharmony_ci	idr_init(&ls->ls_lkbidr);
5288c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_lkbidr_spin);
5298c2ecf20Sopenharmony_ci
5308c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_waiters);
5318c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_waiters_mutex);
5328c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_orphans);
5338c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_orphans_mutex);
5348c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_timeout);
5358c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_timeout_mutex);
5368c2ecf20Sopenharmony_ci
5378c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_new_rsb);
5388c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_new_rsb_spin);
5398c2ecf20Sopenharmony_ci
5408c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_nodes);
5418c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_nodes_gone);
5428c2ecf20Sopenharmony_ci	ls->ls_num_nodes = 0;
5438c2ecf20Sopenharmony_ci	ls->ls_low_nodeid = 0;
5448c2ecf20Sopenharmony_ci	ls->ls_total_weight = 0;
5458c2ecf20Sopenharmony_ci	ls->ls_node_array = NULL;
5468c2ecf20Sopenharmony_ci
5478c2ecf20Sopenharmony_ci	memset(&ls->ls_stub_rsb, 0, sizeof(struct dlm_rsb));
5488c2ecf20Sopenharmony_ci	ls->ls_stub_rsb.res_ls = ls;
5498c2ecf20Sopenharmony_ci
5508c2ecf20Sopenharmony_ci	ls->ls_debug_rsb_dentry = NULL;
5518c2ecf20Sopenharmony_ci	ls->ls_debug_waiters_dentry = NULL;
5528c2ecf20Sopenharmony_ci
5538c2ecf20Sopenharmony_ci	init_waitqueue_head(&ls->ls_uevent_wait);
5548c2ecf20Sopenharmony_ci	ls->ls_uevent_result = 0;
5558c2ecf20Sopenharmony_ci	init_completion(&ls->ls_members_done);
5568c2ecf20Sopenharmony_ci	ls->ls_members_result = -1;
5578c2ecf20Sopenharmony_ci
5588c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_cb_mutex);
5598c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_cb_delay);
5608c2ecf20Sopenharmony_ci
5618c2ecf20Sopenharmony_ci	ls->ls_recoverd_task = NULL;
5628c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_recoverd_active);
5638c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_recover_lock);
5648c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_rcom_spin);
5658c2ecf20Sopenharmony_ci	get_random_bytes(&ls->ls_rcom_seq, sizeof(uint64_t));
5668c2ecf20Sopenharmony_ci	ls->ls_recover_status = 0;
5678c2ecf20Sopenharmony_ci	ls->ls_recover_seq = 0;
5688c2ecf20Sopenharmony_ci	ls->ls_recover_args = NULL;
5698c2ecf20Sopenharmony_ci	init_rwsem(&ls->ls_in_recovery);
5708c2ecf20Sopenharmony_ci	init_rwsem(&ls->ls_recv_active);
5718c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_requestqueue);
5728c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_requestqueue_mutex);
5738c2ecf20Sopenharmony_ci	mutex_init(&ls->ls_clear_proc_locks);
5748c2ecf20Sopenharmony_ci
5758c2ecf20Sopenharmony_ci	ls->ls_recover_buf = kmalloc(dlm_config.ci_buffer_size, GFP_NOFS);
5768c2ecf20Sopenharmony_ci	if (!ls->ls_recover_buf)
5778c2ecf20Sopenharmony_ci		goto out_lkbidr;
5788c2ecf20Sopenharmony_ci
5798c2ecf20Sopenharmony_ci	ls->ls_slot = 0;
5808c2ecf20Sopenharmony_ci	ls->ls_num_slots = 0;
5818c2ecf20Sopenharmony_ci	ls->ls_slots_size = 0;
5828c2ecf20Sopenharmony_ci	ls->ls_slots = NULL;
5838c2ecf20Sopenharmony_ci
5848c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_recover_list);
5858c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_recover_list_lock);
5868c2ecf20Sopenharmony_ci	idr_init(&ls->ls_recover_idr);
5878c2ecf20Sopenharmony_ci	spin_lock_init(&ls->ls_recover_idr_lock);
5888c2ecf20Sopenharmony_ci	ls->ls_recover_list_count = 0;
5898c2ecf20Sopenharmony_ci	ls->ls_local_handle = ls;
5908c2ecf20Sopenharmony_ci	init_waitqueue_head(&ls->ls_wait_general);
5918c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&ls->ls_root_list);
5928c2ecf20Sopenharmony_ci	init_rwsem(&ls->ls_root_sem);
5938c2ecf20Sopenharmony_ci
5948c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
5958c2ecf20Sopenharmony_ci	ls->ls_create_count = 1;
5968c2ecf20Sopenharmony_ci	list_add(&ls->ls_list, &lslist);
5978c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
5988c2ecf20Sopenharmony_ci
5998c2ecf20Sopenharmony_ci	if (flags & DLM_LSFL_FS) {
6008c2ecf20Sopenharmony_ci		error = dlm_callback_start(ls);
6018c2ecf20Sopenharmony_ci		if (error) {
6028c2ecf20Sopenharmony_ci			log_error(ls, "can't start dlm_callback %d", error);
6038c2ecf20Sopenharmony_ci			goto out_delist;
6048c2ecf20Sopenharmony_ci		}
6058c2ecf20Sopenharmony_ci	}
6068c2ecf20Sopenharmony_ci
6078c2ecf20Sopenharmony_ci	init_waitqueue_head(&ls->ls_recover_lock_wait);
6088c2ecf20Sopenharmony_ci
6098c2ecf20Sopenharmony_ci	/*
6108c2ecf20Sopenharmony_ci	 * Once started, dlm_recoverd first looks for ls in lslist, then
6118c2ecf20Sopenharmony_ci	 * initializes ls_in_recovery as locked in "down" mode.  We need
6128c2ecf20Sopenharmony_ci	 * to wait for the wakeup from dlm_recoverd because in_recovery
6138c2ecf20Sopenharmony_ci	 * has to start out in down mode.
6148c2ecf20Sopenharmony_ci	 */
6158c2ecf20Sopenharmony_ci
6168c2ecf20Sopenharmony_ci	error = dlm_recoverd_start(ls);
6178c2ecf20Sopenharmony_ci	if (error) {
6188c2ecf20Sopenharmony_ci		log_error(ls, "can't start dlm_recoverd %d", error);
6198c2ecf20Sopenharmony_ci		goto out_callback;
6208c2ecf20Sopenharmony_ci	}
6218c2ecf20Sopenharmony_ci
6228c2ecf20Sopenharmony_ci	wait_event(ls->ls_recover_lock_wait,
6238c2ecf20Sopenharmony_ci		   test_bit(LSFL_RECOVER_LOCK, &ls->ls_flags));
6248c2ecf20Sopenharmony_ci
6258c2ecf20Sopenharmony_ci	/* let kobject handle freeing of ls if there's an error */
6268c2ecf20Sopenharmony_ci	do_unreg = 1;
6278c2ecf20Sopenharmony_ci
6288c2ecf20Sopenharmony_ci	ls->ls_kobj.kset = dlm_kset;
6298c2ecf20Sopenharmony_ci	error = kobject_init_and_add(&ls->ls_kobj, &dlm_ktype, NULL,
6308c2ecf20Sopenharmony_ci				     "%s", ls->ls_name);
6318c2ecf20Sopenharmony_ci	if (error)
6328c2ecf20Sopenharmony_ci		goto out_recoverd;
6338c2ecf20Sopenharmony_ci	kobject_uevent(&ls->ls_kobj, KOBJ_ADD);
6348c2ecf20Sopenharmony_ci
6358c2ecf20Sopenharmony_ci	/* This uevent triggers dlm_controld in userspace to add us to the
6368c2ecf20Sopenharmony_ci	   group of nodes that are members of this lockspace (managed by the
6378c2ecf20Sopenharmony_ci	   cluster infrastructure.)  Once it's done that, it tells us who the
6388c2ecf20Sopenharmony_ci	   current lockspace members are (via configfs) and then tells the
6398c2ecf20Sopenharmony_ci	   lockspace to start running (via sysfs) in dlm_ls_start(). */
6408c2ecf20Sopenharmony_ci
6418c2ecf20Sopenharmony_ci	error = do_uevent(ls, 1);
6428c2ecf20Sopenharmony_ci	if (error)
6438c2ecf20Sopenharmony_ci		goto out_recoverd;
6448c2ecf20Sopenharmony_ci
6458c2ecf20Sopenharmony_ci	wait_for_completion(&ls->ls_members_done);
6468c2ecf20Sopenharmony_ci	error = ls->ls_members_result;
6478c2ecf20Sopenharmony_ci	if (error)
6488c2ecf20Sopenharmony_ci		goto out_members;
6498c2ecf20Sopenharmony_ci
6508c2ecf20Sopenharmony_ci	dlm_create_debug_file(ls);
6518c2ecf20Sopenharmony_ci
6528c2ecf20Sopenharmony_ci	log_rinfo(ls, "join complete");
6538c2ecf20Sopenharmony_ci	*lockspace = ls;
6548c2ecf20Sopenharmony_ci	return 0;
6558c2ecf20Sopenharmony_ci
6568c2ecf20Sopenharmony_ci out_members:
6578c2ecf20Sopenharmony_ci	do_uevent(ls, 0);
6588c2ecf20Sopenharmony_ci	dlm_clear_members(ls);
6598c2ecf20Sopenharmony_ci	kfree(ls->ls_node_array);
6608c2ecf20Sopenharmony_ci out_recoverd:
6618c2ecf20Sopenharmony_ci	dlm_recoverd_stop(ls);
6628c2ecf20Sopenharmony_ci out_callback:
6638c2ecf20Sopenharmony_ci	dlm_callback_stop(ls);
6648c2ecf20Sopenharmony_ci out_delist:
6658c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
6668c2ecf20Sopenharmony_ci	list_del(&ls->ls_list);
6678c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
6688c2ecf20Sopenharmony_ci	idr_destroy(&ls->ls_recover_idr);
6698c2ecf20Sopenharmony_ci	kfree(ls->ls_recover_buf);
6708c2ecf20Sopenharmony_ci out_lkbidr:
6718c2ecf20Sopenharmony_ci	idr_destroy(&ls->ls_lkbidr);
6728c2ecf20Sopenharmony_ci out_rsbtbl:
6738c2ecf20Sopenharmony_ci	for (i = 0; i < DLM_REMOVE_NAMES_MAX; i++)
6748c2ecf20Sopenharmony_ci		kfree(ls->ls_remove_names[i]);
6758c2ecf20Sopenharmony_ci	vfree(ls->ls_rsbtbl);
6768c2ecf20Sopenharmony_ci out_lsfree:
6778c2ecf20Sopenharmony_ci	if (do_unreg)
6788c2ecf20Sopenharmony_ci		kobject_put(&ls->ls_kobj);
6798c2ecf20Sopenharmony_ci	else
6808c2ecf20Sopenharmony_ci		kfree(ls);
6818c2ecf20Sopenharmony_ci out:
6828c2ecf20Sopenharmony_ci	module_put(THIS_MODULE);
6838c2ecf20Sopenharmony_ci	return error;
6848c2ecf20Sopenharmony_ci}
6858c2ecf20Sopenharmony_ci
6868c2ecf20Sopenharmony_ciint dlm_new_lockspace(const char *name, const char *cluster,
6878c2ecf20Sopenharmony_ci		      uint32_t flags, int lvblen,
6888c2ecf20Sopenharmony_ci		      const struct dlm_lockspace_ops *ops, void *ops_arg,
6898c2ecf20Sopenharmony_ci		      int *ops_result, dlm_lockspace_t **lockspace)
6908c2ecf20Sopenharmony_ci{
6918c2ecf20Sopenharmony_ci	int error = 0;
6928c2ecf20Sopenharmony_ci
6938c2ecf20Sopenharmony_ci	mutex_lock(&ls_lock);
6948c2ecf20Sopenharmony_ci	if (!ls_count)
6958c2ecf20Sopenharmony_ci		error = threads_start();
6968c2ecf20Sopenharmony_ci	if (error)
6978c2ecf20Sopenharmony_ci		goto out;
6988c2ecf20Sopenharmony_ci
6998c2ecf20Sopenharmony_ci	error = new_lockspace(name, cluster, flags, lvblen, ops, ops_arg,
7008c2ecf20Sopenharmony_ci			      ops_result, lockspace);
7018c2ecf20Sopenharmony_ci	if (!error)
7028c2ecf20Sopenharmony_ci		ls_count++;
7038c2ecf20Sopenharmony_ci	if (error > 0)
7048c2ecf20Sopenharmony_ci		error = 0;
7058c2ecf20Sopenharmony_ci	if (!ls_count)
7068c2ecf20Sopenharmony_ci		threads_stop();
7078c2ecf20Sopenharmony_ci out:
7088c2ecf20Sopenharmony_ci	mutex_unlock(&ls_lock);
7098c2ecf20Sopenharmony_ci	return error;
7108c2ecf20Sopenharmony_ci}
7118c2ecf20Sopenharmony_ci
7128c2ecf20Sopenharmony_cistatic int lkb_idr_is_local(int id, void *p, void *data)
7138c2ecf20Sopenharmony_ci{
7148c2ecf20Sopenharmony_ci	struct dlm_lkb *lkb = p;
7158c2ecf20Sopenharmony_ci
7168c2ecf20Sopenharmony_ci	return lkb->lkb_nodeid == 0 && lkb->lkb_grmode != DLM_LOCK_IV;
7178c2ecf20Sopenharmony_ci}
7188c2ecf20Sopenharmony_ci
7198c2ecf20Sopenharmony_cistatic int lkb_idr_is_any(int id, void *p, void *data)
7208c2ecf20Sopenharmony_ci{
7218c2ecf20Sopenharmony_ci	return 1;
7228c2ecf20Sopenharmony_ci}
7238c2ecf20Sopenharmony_ci
7248c2ecf20Sopenharmony_cistatic int lkb_idr_free(int id, void *p, void *data)
7258c2ecf20Sopenharmony_ci{
7268c2ecf20Sopenharmony_ci	struct dlm_lkb *lkb = p;
7278c2ecf20Sopenharmony_ci
7288c2ecf20Sopenharmony_ci	if (lkb->lkb_lvbptr && lkb->lkb_flags & DLM_IFL_MSTCPY)
7298c2ecf20Sopenharmony_ci		dlm_free_lvb(lkb->lkb_lvbptr);
7308c2ecf20Sopenharmony_ci
7318c2ecf20Sopenharmony_ci	dlm_free_lkb(lkb);
7328c2ecf20Sopenharmony_ci	return 0;
7338c2ecf20Sopenharmony_ci}
7348c2ecf20Sopenharmony_ci
7358c2ecf20Sopenharmony_ci/* NOTE: We check the lkbidr here rather than the resource table.
7368c2ecf20Sopenharmony_ci   This is because there may be LKBs queued as ASTs that have been unlinked
7378c2ecf20Sopenharmony_ci   from their RSBs and are pending deletion once the AST has been delivered */
7388c2ecf20Sopenharmony_ci
7398c2ecf20Sopenharmony_cistatic int lockspace_busy(struct dlm_ls *ls, int force)
7408c2ecf20Sopenharmony_ci{
7418c2ecf20Sopenharmony_ci	int rv;
7428c2ecf20Sopenharmony_ci
7438c2ecf20Sopenharmony_ci	spin_lock(&ls->ls_lkbidr_spin);
7448c2ecf20Sopenharmony_ci	if (force == 0) {
7458c2ecf20Sopenharmony_ci		rv = idr_for_each(&ls->ls_lkbidr, lkb_idr_is_any, ls);
7468c2ecf20Sopenharmony_ci	} else if (force == 1) {
7478c2ecf20Sopenharmony_ci		rv = idr_for_each(&ls->ls_lkbidr, lkb_idr_is_local, ls);
7488c2ecf20Sopenharmony_ci	} else {
7498c2ecf20Sopenharmony_ci		rv = 0;
7508c2ecf20Sopenharmony_ci	}
7518c2ecf20Sopenharmony_ci	spin_unlock(&ls->ls_lkbidr_spin);
7528c2ecf20Sopenharmony_ci	return rv;
7538c2ecf20Sopenharmony_ci}
7548c2ecf20Sopenharmony_ci
7558c2ecf20Sopenharmony_cistatic int release_lockspace(struct dlm_ls *ls, int force)
7568c2ecf20Sopenharmony_ci{
7578c2ecf20Sopenharmony_ci	struct dlm_rsb *rsb;
7588c2ecf20Sopenharmony_ci	struct rb_node *n;
7598c2ecf20Sopenharmony_ci	int i, busy, rv;
7608c2ecf20Sopenharmony_ci
7618c2ecf20Sopenharmony_ci	busy = lockspace_busy(ls, force);
7628c2ecf20Sopenharmony_ci
7638c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
7648c2ecf20Sopenharmony_ci	if (ls->ls_create_count == 1) {
7658c2ecf20Sopenharmony_ci		if (busy) {
7668c2ecf20Sopenharmony_ci			rv = -EBUSY;
7678c2ecf20Sopenharmony_ci		} else {
7688c2ecf20Sopenharmony_ci			/* remove_lockspace takes ls off lslist */
7698c2ecf20Sopenharmony_ci			ls->ls_create_count = 0;
7708c2ecf20Sopenharmony_ci			rv = 0;
7718c2ecf20Sopenharmony_ci		}
7728c2ecf20Sopenharmony_ci	} else if (ls->ls_create_count > 1) {
7738c2ecf20Sopenharmony_ci		rv = --ls->ls_create_count;
7748c2ecf20Sopenharmony_ci	} else {
7758c2ecf20Sopenharmony_ci		rv = -EINVAL;
7768c2ecf20Sopenharmony_ci	}
7778c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
7788c2ecf20Sopenharmony_ci
7798c2ecf20Sopenharmony_ci	if (rv) {
7808c2ecf20Sopenharmony_ci		log_debug(ls, "release_lockspace no remove %d", rv);
7818c2ecf20Sopenharmony_ci		return rv;
7828c2ecf20Sopenharmony_ci	}
7838c2ecf20Sopenharmony_ci
7848c2ecf20Sopenharmony_ci	dlm_device_deregister(ls);
7858c2ecf20Sopenharmony_ci
7868c2ecf20Sopenharmony_ci	if (force < 3 && dlm_user_daemon_available())
7878c2ecf20Sopenharmony_ci		do_uevent(ls, 0);
7888c2ecf20Sopenharmony_ci
7898c2ecf20Sopenharmony_ci	dlm_recoverd_stop(ls);
7908c2ecf20Sopenharmony_ci
7918c2ecf20Sopenharmony_ci	dlm_callback_stop(ls);
7928c2ecf20Sopenharmony_ci
7938c2ecf20Sopenharmony_ci	remove_lockspace(ls);
7948c2ecf20Sopenharmony_ci
7958c2ecf20Sopenharmony_ci	dlm_delete_debug_file(ls);
7968c2ecf20Sopenharmony_ci
7978c2ecf20Sopenharmony_ci	idr_destroy(&ls->ls_recover_idr);
7988c2ecf20Sopenharmony_ci	kfree(ls->ls_recover_buf);
7998c2ecf20Sopenharmony_ci
8008c2ecf20Sopenharmony_ci	/*
8018c2ecf20Sopenharmony_ci	 * Free all lkb's in idr
8028c2ecf20Sopenharmony_ci	 */
8038c2ecf20Sopenharmony_ci
8048c2ecf20Sopenharmony_ci	idr_for_each(&ls->ls_lkbidr, lkb_idr_free, ls);
8058c2ecf20Sopenharmony_ci	idr_destroy(&ls->ls_lkbidr);
8068c2ecf20Sopenharmony_ci
8078c2ecf20Sopenharmony_ci	/*
8088c2ecf20Sopenharmony_ci	 * Free all rsb's on rsbtbl[] lists
8098c2ecf20Sopenharmony_ci	 */
8108c2ecf20Sopenharmony_ci
8118c2ecf20Sopenharmony_ci	for (i = 0; i < ls->ls_rsbtbl_size; i++) {
8128c2ecf20Sopenharmony_ci		while ((n = rb_first(&ls->ls_rsbtbl[i].keep))) {
8138c2ecf20Sopenharmony_ci			rsb = rb_entry(n, struct dlm_rsb, res_hashnode);
8148c2ecf20Sopenharmony_ci			rb_erase(n, &ls->ls_rsbtbl[i].keep);
8158c2ecf20Sopenharmony_ci			dlm_free_rsb(rsb);
8168c2ecf20Sopenharmony_ci		}
8178c2ecf20Sopenharmony_ci
8188c2ecf20Sopenharmony_ci		while ((n = rb_first(&ls->ls_rsbtbl[i].toss))) {
8198c2ecf20Sopenharmony_ci			rsb = rb_entry(n, struct dlm_rsb, res_hashnode);
8208c2ecf20Sopenharmony_ci			rb_erase(n, &ls->ls_rsbtbl[i].toss);
8218c2ecf20Sopenharmony_ci			dlm_free_rsb(rsb);
8228c2ecf20Sopenharmony_ci		}
8238c2ecf20Sopenharmony_ci	}
8248c2ecf20Sopenharmony_ci
8258c2ecf20Sopenharmony_ci	vfree(ls->ls_rsbtbl);
8268c2ecf20Sopenharmony_ci
8278c2ecf20Sopenharmony_ci	for (i = 0; i < DLM_REMOVE_NAMES_MAX; i++)
8288c2ecf20Sopenharmony_ci		kfree(ls->ls_remove_names[i]);
8298c2ecf20Sopenharmony_ci
8308c2ecf20Sopenharmony_ci	while (!list_empty(&ls->ls_new_rsb)) {
8318c2ecf20Sopenharmony_ci		rsb = list_first_entry(&ls->ls_new_rsb, struct dlm_rsb,
8328c2ecf20Sopenharmony_ci				       res_hashchain);
8338c2ecf20Sopenharmony_ci		list_del(&rsb->res_hashchain);
8348c2ecf20Sopenharmony_ci		dlm_free_rsb(rsb);
8358c2ecf20Sopenharmony_ci	}
8368c2ecf20Sopenharmony_ci
8378c2ecf20Sopenharmony_ci	/*
8388c2ecf20Sopenharmony_ci	 * Free structures on any other lists
8398c2ecf20Sopenharmony_ci	 */
8408c2ecf20Sopenharmony_ci
8418c2ecf20Sopenharmony_ci	dlm_purge_requestqueue(ls);
8428c2ecf20Sopenharmony_ci	kfree(ls->ls_recover_args);
8438c2ecf20Sopenharmony_ci	dlm_clear_members(ls);
8448c2ecf20Sopenharmony_ci	dlm_clear_members_gone(ls);
8458c2ecf20Sopenharmony_ci	kfree(ls->ls_node_array);
8468c2ecf20Sopenharmony_ci	log_rinfo(ls, "release_lockspace final free");
8478c2ecf20Sopenharmony_ci	kobject_put(&ls->ls_kobj);
8488c2ecf20Sopenharmony_ci	/* The ls structure will be freed when the kobject is done with */
8498c2ecf20Sopenharmony_ci
8508c2ecf20Sopenharmony_ci	module_put(THIS_MODULE);
8518c2ecf20Sopenharmony_ci	return 0;
8528c2ecf20Sopenharmony_ci}
8538c2ecf20Sopenharmony_ci
8548c2ecf20Sopenharmony_ci/*
8558c2ecf20Sopenharmony_ci * Called when a system has released all its locks and is not going to use the
8568c2ecf20Sopenharmony_ci * lockspace any longer.  We free everything we're managing for this lockspace.
8578c2ecf20Sopenharmony_ci * Remaining nodes will go through the recovery process as if we'd died.  The
8588c2ecf20Sopenharmony_ci * lockspace must continue to function as usual, participating in recoveries,
8598c2ecf20Sopenharmony_ci * until this returns.
8608c2ecf20Sopenharmony_ci *
8618c2ecf20Sopenharmony_ci * Force has 4 possible values:
8628c2ecf20Sopenharmony_ci * 0 - don't destroy locksapce if it has any LKBs
8638c2ecf20Sopenharmony_ci * 1 - destroy lockspace if it has remote LKBs but not if it has local LKBs
8648c2ecf20Sopenharmony_ci * 2 - destroy lockspace regardless of LKBs
8658c2ecf20Sopenharmony_ci * 3 - destroy lockspace as part of a forced shutdown
8668c2ecf20Sopenharmony_ci */
8678c2ecf20Sopenharmony_ci
8688c2ecf20Sopenharmony_ciint dlm_release_lockspace(void *lockspace, int force)
8698c2ecf20Sopenharmony_ci{
8708c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
8718c2ecf20Sopenharmony_ci	int error;
8728c2ecf20Sopenharmony_ci
8738c2ecf20Sopenharmony_ci	ls = dlm_find_lockspace_local(lockspace);
8748c2ecf20Sopenharmony_ci	if (!ls)
8758c2ecf20Sopenharmony_ci		return -EINVAL;
8768c2ecf20Sopenharmony_ci	dlm_put_lockspace(ls);
8778c2ecf20Sopenharmony_ci
8788c2ecf20Sopenharmony_ci	mutex_lock(&ls_lock);
8798c2ecf20Sopenharmony_ci	error = release_lockspace(ls, force);
8808c2ecf20Sopenharmony_ci	if (!error)
8818c2ecf20Sopenharmony_ci		ls_count--;
8828c2ecf20Sopenharmony_ci	if (!ls_count)
8838c2ecf20Sopenharmony_ci		threads_stop();
8848c2ecf20Sopenharmony_ci	mutex_unlock(&ls_lock);
8858c2ecf20Sopenharmony_ci
8868c2ecf20Sopenharmony_ci	return error;
8878c2ecf20Sopenharmony_ci}
8888c2ecf20Sopenharmony_ci
8898c2ecf20Sopenharmony_civoid dlm_stop_lockspaces(void)
8908c2ecf20Sopenharmony_ci{
8918c2ecf20Sopenharmony_ci	struct dlm_ls *ls;
8928c2ecf20Sopenharmony_ci	int count;
8938c2ecf20Sopenharmony_ci
8948c2ecf20Sopenharmony_ci restart:
8958c2ecf20Sopenharmony_ci	count = 0;
8968c2ecf20Sopenharmony_ci	spin_lock(&lslist_lock);
8978c2ecf20Sopenharmony_ci	list_for_each_entry(ls, &lslist, ls_list) {
8988c2ecf20Sopenharmony_ci		if (!test_bit(LSFL_RUNNING, &ls->ls_flags)) {
8998c2ecf20Sopenharmony_ci			count++;
9008c2ecf20Sopenharmony_ci			continue;
9018c2ecf20Sopenharmony_ci		}
9028c2ecf20Sopenharmony_ci		spin_unlock(&lslist_lock);
9038c2ecf20Sopenharmony_ci		log_error(ls, "no userland control daemon, stopping lockspace");
9048c2ecf20Sopenharmony_ci		dlm_ls_stop(ls);
9058c2ecf20Sopenharmony_ci		goto restart;
9068c2ecf20Sopenharmony_ci	}
9078c2ecf20Sopenharmony_ci	spin_unlock(&lslist_lock);
9088c2ecf20Sopenharmony_ci
9098c2ecf20Sopenharmony_ci	if (count)
9108c2ecf20Sopenharmony_ci		log_print("dlm user daemon left %d lockspaces", count);
9118c2ecf20Sopenharmony_ci}
9128c2ecf20Sopenharmony_ci
913