blob: 1dc72105ab125171b4bb3356ea0a09577178ec77 [file] [log] [blame]
David Teiglande7fd4172006-01-18 09:30:29 +00001/******************************************************************************
2*******************************************************************************
3**
4** Copyright (C) Sistina Software, Inc. 1997-2003 All rights reserved.
David Teiglandef0c2bb2007-03-28 09:56:46 -05005** Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
David Teiglande7fd4172006-01-18 09:30:29 +00006**
7** This copyrighted material is made available to anyone wishing to use,
8** modify, copy, or redistribute it subject to the terms and conditions
9** of the GNU General Public License v.2.
10**
11*******************************************************************************
12******************************************************************************/
13
14#include "dlm_internal.h"
15#include "lockspace.h"
16#include "member.h"
17#include "recoverd.h"
18#include "ast.h"
19#include "dir.h"
20#include "lowcomms.h"
21#include "config.h"
22#include "memory.h"
23#include "lock.h"
David Teiglandc56b39c2006-04-28 10:51:53 -040024#include "recover.h"
David Teigland2896ee32006-11-27 11:31:22 -060025#include "requestqueue.h"
David Teiglande7fd4172006-01-18 09:30:29 +000026
27#ifdef CONFIG_DLM_DEBUG
28int dlm_create_debug_file(struct dlm_ls *ls);
29void dlm_delete_debug_file(struct dlm_ls *ls);
30#else
31static inline int dlm_create_debug_file(struct dlm_ls *ls) { return 0; }
32static inline void dlm_delete_debug_file(struct dlm_ls *ls) { }
33#endif
34
35static int ls_count;
David Teigland90135922006-01-20 08:47:07 +000036static struct mutex ls_lock;
David Teiglande7fd4172006-01-18 09:30:29 +000037static struct list_head lslist;
38static spinlock_t lslist_lock;
39static struct task_struct * scand_task;
40
41
42static ssize_t dlm_control_store(struct dlm_ls *ls, const char *buf, size_t len)
43{
44 ssize_t ret = len;
45 int n = simple_strtol(buf, NULL, 0);
46
Patrick Caulfielde2de7f52006-11-06 08:53:28 +000047 ls = dlm_find_lockspace_local(ls->ls_local_handle);
48 if (!ls)
49 return -EINVAL;
50
David Teiglande7fd4172006-01-18 09:30:29 +000051 switch (n) {
52 case 0:
53 dlm_ls_stop(ls);
54 break;
55 case 1:
56 dlm_ls_start(ls);
57 break;
58 default:
59 ret = -EINVAL;
60 }
Patrick Caulfielde2de7f52006-11-06 08:53:28 +000061 dlm_put_lockspace(ls);
David Teiglande7fd4172006-01-18 09:30:29 +000062 return ret;
63}
64
65static ssize_t dlm_event_store(struct dlm_ls *ls, const char *buf, size_t len)
66{
67 ls->ls_uevent_result = simple_strtol(buf, NULL, 0);
68 set_bit(LSFL_UEVENT_WAIT, &ls->ls_flags);
69 wake_up(&ls->ls_uevent_wait);
70 return len;
71}
72
73static ssize_t dlm_id_show(struct dlm_ls *ls, char *buf)
74{
David Teiglanda1d144c2006-09-06 17:01:40 -050075 return snprintf(buf, PAGE_SIZE, "%u\n", ls->ls_global_id);
David Teiglande7fd4172006-01-18 09:30:29 +000076}
77
78static ssize_t dlm_id_store(struct dlm_ls *ls, const char *buf, size_t len)
79{
80 ls->ls_global_id = simple_strtoul(buf, NULL, 0);
81 return len;
82}
83
David Teiglandc56b39c2006-04-28 10:51:53 -040084static ssize_t dlm_recover_status_show(struct dlm_ls *ls, char *buf)
85{
86 uint32_t status = dlm_recover_status(ls);
David Teiglanda1d144c2006-09-06 17:01:40 -050087 return snprintf(buf, PAGE_SIZE, "%x\n", status);
David Teiglandc56b39c2006-04-28 10:51:53 -040088}
89
David Teiglandfaa0f262006-08-08 17:08:42 -050090static ssize_t dlm_recover_nodeid_show(struct dlm_ls *ls, char *buf)
91{
David Teiglanda1d144c2006-09-06 17:01:40 -050092 return snprintf(buf, PAGE_SIZE, "%d\n", ls->ls_recover_nodeid);
David Teiglandfaa0f262006-08-08 17:08:42 -050093}
94
David Teiglande7fd4172006-01-18 09:30:29 +000095struct dlm_attr {
96 struct attribute attr;
97 ssize_t (*show)(struct dlm_ls *, char *);
98 ssize_t (*store)(struct dlm_ls *, const char *, size_t);
99};
100
101static struct dlm_attr dlm_attr_control = {
102 .attr = {.name = "control", .mode = S_IWUSR},
103 .store = dlm_control_store
104};
105
106static struct dlm_attr dlm_attr_event = {
107 .attr = {.name = "event_done", .mode = S_IWUSR},
108 .store = dlm_event_store
109};
110
111static struct dlm_attr dlm_attr_id = {
112 .attr = {.name = "id", .mode = S_IRUGO | S_IWUSR},
113 .show = dlm_id_show,
114 .store = dlm_id_store
115};
116
David Teiglandc56b39c2006-04-28 10:51:53 -0400117static struct dlm_attr dlm_attr_recover_status = {
118 .attr = {.name = "recover_status", .mode = S_IRUGO},
119 .show = dlm_recover_status_show
120};
121
David Teiglandfaa0f262006-08-08 17:08:42 -0500122static struct dlm_attr dlm_attr_recover_nodeid = {
123 .attr = {.name = "recover_nodeid", .mode = S_IRUGO},
124 .show = dlm_recover_nodeid_show
125};
126
David Teiglande7fd4172006-01-18 09:30:29 +0000127static struct attribute *dlm_attrs[] = {
128 &dlm_attr_control.attr,
129 &dlm_attr_event.attr,
130 &dlm_attr_id.attr,
David Teiglandc56b39c2006-04-28 10:51:53 -0400131 &dlm_attr_recover_status.attr,
David Teiglandfaa0f262006-08-08 17:08:42 -0500132 &dlm_attr_recover_nodeid.attr,
David Teiglande7fd4172006-01-18 09:30:29 +0000133 NULL,
134};
135
136static ssize_t dlm_attr_show(struct kobject *kobj, struct attribute *attr,
137 char *buf)
138{
139 struct dlm_ls *ls = container_of(kobj, struct dlm_ls, ls_kobj);
140 struct dlm_attr *a = container_of(attr, struct dlm_attr, attr);
141 return a->show ? a->show(ls, buf) : 0;
142}
143
144static ssize_t dlm_attr_store(struct kobject *kobj, struct attribute *attr,
145 const char *buf, size_t len)
146{
147 struct dlm_ls *ls = container_of(kobj, struct dlm_ls, ls_kobj);
148 struct dlm_attr *a = container_of(attr, struct dlm_attr, attr);
149 return a->store ? a->store(ls, buf, len) : len;
150}
151
Patrick Caulfieldba542e32006-11-02 14:41:23 +0000152static void lockspace_kobj_release(struct kobject *k)
153{
154 struct dlm_ls *ls = container_of(k, struct dlm_ls, ls_kobj);
155 kfree(ls);
156}
157
David Teiglande7fd4172006-01-18 09:30:29 +0000158static struct sysfs_ops dlm_attr_ops = {
159 .show = dlm_attr_show,
160 .store = dlm_attr_store,
161};
162
163static struct kobj_type dlm_ktype = {
164 .default_attrs = dlm_attrs,
165 .sysfs_ops = &dlm_attr_ops,
Patrick Caulfieldba542e32006-11-02 14:41:23 +0000166 .release = lockspace_kobj_release,
David Teiglande7fd4172006-01-18 09:30:29 +0000167};
168
169static struct kset dlm_kset = {
David Teiglande7fd4172006-01-18 09:30:29 +0000170 .kobj = {.name = "dlm",},
171 .ktype = &dlm_ktype,
172};
173
174static int kobject_setup(struct dlm_ls *ls)
175{
176 char lsname[DLM_LOCKSPACE_LEN];
177 int error;
178
179 memset(lsname, 0, DLM_LOCKSPACE_LEN);
180 snprintf(lsname, DLM_LOCKSPACE_LEN, "%s", ls->ls_name);
181
182 error = kobject_set_name(&ls->ls_kobj, "%s", lsname);
183 if (error)
184 return error;
185
186 ls->ls_kobj.kset = &dlm_kset;
187 ls->ls_kobj.ktype = &dlm_ktype;
188 return 0;
189}
190
191static int do_uevent(struct dlm_ls *ls, int in)
192{
193 int error;
194
195 if (in)
196 kobject_uevent(&ls->ls_kobj, KOBJ_ONLINE);
197 else
198 kobject_uevent(&ls->ls_kobj, KOBJ_OFFLINE);
199
David Teigland8b0e7b22007-05-18 09:03:35 -0500200 log_debug(ls, "%s the lockspace group...", in ? "joining" : "leaving");
201
202 /* dlm_controld will see the uevent, do the necessary group management
203 and then write to sysfs to wake us */
204
David Teiglande7fd4172006-01-18 09:30:29 +0000205 error = wait_event_interruptible(ls->ls_uevent_wait,
206 test_and_clear_bit(LSFL_UEVENT_WAIT, &ls->ls_flags));
David Teigland8b0e7b22007-05-18 09:03:35 -0500207
208 log_debug(ls, "group event done %d %d", error, ls->ls_uevent_result);
209
David Teiglande7fd4172006-01-18 09:30:29 +0000210 if (error)
211 goto out;
212
213 error = ls->ls_uevent_result;
214 out:
David Teigland8b0e7b22007-05-18 09:03:35 -0500215 if (error)
216 log_error(ls, "group %s failed %d %d", in ? "join" : "leave",
217 error, ls->ls_uevent_result);
David Teiglande7fd4172006-01-18 09:30:29 +0000218 return error;
219}
220
221
222int dlm_lockspace_init(void)
223{
224 int error;
225
226 ls_count = 0;
David Teigland90135922006-01-20 08:47:07 +0000227 mutex_init(&ls_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000228 INIT_LIST_HEAD(&lslist);
229 spin_lock_init(&lslist_lock);
230
Greg Kroah-Hartman823bccf2007-04-13 13:15:19 -0700231 kobj_set_kset_s(&dlm_kset, kernel_subsys);
David Teiglande7fd4172006-01-18 09:30:29 +0000232 error = kset_register(&dlm_kset);
233 if (error)
234 printk("dlm_lockspace_init: cannot register kset %d\n", error);
235 return error;
236}
237
238void dlm_lockspace_exit(void)
239{
240 kset_unregister(&dlm_kset);
241}
242
243static int dlm_scand(void *data)
244{
245 struct dlm_ls *ls;
246
247 while (!kthread_should_stop()) {
David Teigland85e86ed2007-05-18 08:58:15 -0500248 list_for_each_entry(ls, &lslist, ls_list) {
249 if (dlm_lock_recovery_try(ls)) {
250 dlm_scan_rsbs(ls);
David Teigland3ae1acf2007-05-18 08:59:31 -0500251 dlm_scan_timeout(ls);
David Teigland85e86ed2007-05-18 08:58:15 -0500252 dlm_unlock_recovery(ls);
253 }
254 }
David Teigland68c817a2007-01-09 09:41:48 -0600255 schedule_timeout_interruptible(dlm_config.ci_scan_secs * HZ);
David Teiglande7fd4172006-01-18 09:30:29 +0000256 }
257 return 0;
258}
259
260static int dlm_scand_start(void)
261{
262 struct task_struct *p;
263 int error = 0;
264
265 p = kthread_run(dlm_scand, NULL, "dlm_scand");
266 if (IS_ERR(p))
267 error = PTR_ERR(p);
268 else
269 scand_task = p;
270 return error;
271}
272
273static void dlm_scand_stop(void)
274{
275 kthread_stop(scand_task);
276}
277
278static struct dlm_ls *dlm_find_lockspace_name(char *name, int namelen)
279{
280 struct dlm_ls *ls;
281
282 spin_lock(&lslist_lock);
283
284 list_for_each_entry(ls, &lslist, ls_list) {
285 if (ls->ls_namelen == namelen &&
286 memcmp(ls->ls_name, name, namelen) == 0)
287 goto out;
288 }
289 ls = NULL;
290 out:
291 spin_unlock(&lslist_lock);
292 return ls;
293}
294
295struct dlm_ls *dlm_find_lockspace_global(uint32_t id)
296{
297 struct dlm_ls *ls;
298
299 spin_lock(&lslist_lock);
300
301 list_for_each_entry(ls, &lslist, ls_list) {
302 if (ls->ls_global_id == id) {
303 ls->ls_count++;
304 goto out;
305 }
306 }
307 ls = NULL;
308 out:
309 spin_unlock(&lslist_lock);
310 return ls;
311}
312
David Teigland597d0ca2006-07-12 16:44:04 -0500313struct dlm_ls *dlm_find_lockspace_local(dlm_lockspace_t *lockspace)
David Teiglande7fd4172006-01-18 09:30:29 +0000314{
David Teigland597d0ca2006-07-12 16:44:04 -0500315 struct dlm_ls *ls;
David Teiglande7fd4172006-01-18 09:30:29 +0000316
317 spin_lock(&lslist_lock);
David Teigland597d0ca2006-07-12 16:44:04 -0500318 list_for_each_entry(ls, &lslist, ls_list) {
319 if (ls->ls_local_handle == lockspace) {
320 ls->ls_count++;
321 goto out;
322 }
323 }
324 ls = NULL;
325 out:
326 spin_unlock(&lslist_lock);
327 return ls;
328}
329
330struct dlm_ls *dlm_find_lockspace_device(int minor)
331{
332 struct dlm_ls *ls;
333
334 spin_lock(&lslist_lock);
335 list_for_each_entry(ls, &lslist, ls_list) {
336 if (ls->ls_device.minor == minor) {
337 ls->ls_count++;
338 goto out;
339 }
340 }
341 ls = NULL;
342 out:
David Teiglande7fd4172006-01-18 09:30:29 +0000343 spin_unlock(&lslist_lock);
344 return ls;
345}
346
347void dlm_put_lockspace(struct dlm_ls *ls)
348{
349 spin_lock(&lslist_lock);
350 ls->ls_count--;
351 spin_unlock(&lslist_lock);
352}
353
354static void remove_lockspace(struct dlm_ls *ls)
355{
356 for (;;) {
357 spin_lock(&lslist_lock);
358 if (ls->ls_count == 0) {
359 list_del(&ls->ls_list);
360 spin_unlock(&lslist_lock);
361 return;
362 }
363 spin_unlock(&lslist_lock);
364 ssleep(1);
365 }
366}
367
368static int threads_start(void)
369{
370 int error;
371
372 /* Thread which process lock requests for all lockspace's */
373 error = dlm_astd_start();
374 if (error) {
375 log_print("cannot start dlm_astd thread %d", error);
376 goto fail;
377 }
378
379 error = dlm_scand_start();
380 if (error) {
381 log_print("cannot start dlm_scand thread %d", error);
382 goto astd_fail;
383 }
384
385 /* Thread for sending/receiving messages for all lockspace's */
386 error = dlm_lowcomms_start();
387 if (error) {
388 log_print("cannot start dlm lowcomms %d", error);
389 goto scand_fail;
390 }
391
392 return 0;
393
394 scand_fail:
395 dlm_scand_stop();
396 astd_fail:
397 dlm_astd_stop();
398 fail:
399 return error;
400}
401
402static void threads_stop(void)
403{
404 dlm_scand_stop();
405 dlm_lowcomms_stop();
406 dlm_astd_stop();
407}
408
409static int new_lockspace(char *name, int namelen, void **lockspace,
410 uint32_t flags, int lvblen)
411{
412 struct dlm_ls *ls;
413 int i, size, error = -ENOMEM;
David Teigland79d72b52007-05-18 09:02:20 -0500414 int do_unreg = 0;
David Teiglande7fd4172006-01-18 09:30:29 +0000415
416 if (namelen > DLM_LOCKSPACE_LEN)
417 return -EINVAL;
418
419 if (!lvblen || (lvblen % 8))
420 return -EINVAL;
421
422 if (!try_module_get(THIS_MODULE))
423 return -EINVAL;
424
425 ls = dlm_find_lockspace_name(name, namelen);
426 if (ls) {
427 *lockspace = ls;
428 module_put(THIS_MODULE);
429 return -EEXIST;
430 }
431
David Teigland90135922006-01-20 08:47:07 +0000432 ls = kzalloc(sizeof(struct dlm_ls) + namelen, GFP_KERNEL);
David Teiglande7fd4172006-01-18 09:30:29 +0000433 if (!ls)
434 goto out;
David Teiglande7fd4172006-01-18 09:30:29 +0000435 memcpy(ls->ls_name, name, namelen);
436 ls->ls_namelen = namelen;
David Teiglande7fd4172006-01-18 09:30:29 +0000437 ls->ls_lvblen = lvblen;
438 ls->ls_count = 0;
439 ls->ls_flags = 0;
440
David Teigland3ae1acf2007-05-18 08:59:31 -0500441 if (flags & DLM_LSFL_TIMEWARN)
442 set_bit(LSFL_TIMEWARN, &ls->ls_flags);
David Teigland3ae1acf2007-05-18 08:59:31 -0500443
Patrick Caulfield44f487a2007-06-06 09:21:22 -0500444 if (flags & DLM_LSFL_FS)
445 ls->ls_allocation = GFP_NOFS;
446 else
447 ls->ls_allocation = GFP_KERNEL;
448
David Teiglandfad59c12007-06-11 10:47:18 -0500449 /* ls_exflags are forced to match among nodes, and we don't
450 need to require all nodes to have TIMEWARN or FS set */
451 ls->ls_exflags = (flags & ~(DLM_LSFL_TIMEWARN | DLM_LSFL_FS));
452
David Teigland68c817a2007-01-09 09:41:48 -0600453 size = dlm_config.ci_rsbtbl_size;
David Teiglande7fd4172006-01-18 09:30:29 +0000454 ls->ls_rsbtbl_size = size;
455
456 ls->ls_rsbtbl = kmalloc(sizeof(struct dlm_rsbtable) * size, GFP_KERNEL);
457 if (!ls->ls_rsbtbl)
458 goto out_lsfree;
459 for (i = 0; i < size; i++) {
460 INIT_LIST_HEAD(&ls->ls_rsbtbl[i].list);
461 INIT_LIST_HEAD(&ls->ls_rsbtbl[i].toss);
462 rwlock_init(&ls->ls_rsbtbl[i].lock);
463 }
464
David Teigland68c817a2007-01-09 09:41:48 -0600465 size = dlm_config.ci_lkbtbl_size;
David Teiglande7fd4172006-01-18 09:30:29 +0000466 ls->ls_lkbtbl_size = size;
467
468 ls->ls_lkbtbl = kmalloc(sizeof(struct dlm_lkbtable) * size, GFP_KERNEL);
469 if (!ls->ls_lkbtbl)
470 goto out_rsbfree;
471 for (i = 0; i < size; i++) {
472 INIT_LIST_HEAD(&ls->ls_lkbtbl[i].list);
473 rwlock_init(&ls->ls_lkbtbl[i].lock);
474 ls->ls_lkbtbl[i].counter = 1;
475 }
476
David Teigland68c817a2007-01-09 09:41:48 -0600477 size = dlm_config.ci_dirtbl_size;
David Teiglande7fd4172006-01-18 09:30:29 +0000478 ls->ls_dirtbl_size = size;
479
480 ls->ls_dirtbl = kmalloc(sizeof(struct dlm_dirtable) * size, GFP_KERNEL);
481 if (!ls->ls_dirtbl)
482 goto out_lkbfree;
483 for (i = 0; i < size; i++) {
484 INIT_LIST_HEAD(&ls->ls_dirtbl[i].list);
485 rwlock_init(&ls->ls_dirtbl[i].lock);
486 }
487
488 INIT_LIST_HEAD(&ls->ls_waiters);
David Teigland90135922006-01-20 08:47:07 +0000489 mutex_init(&ls->ls_waiters_mutex);
David Teiglandef0c2bb2007-03-28 09:56:46 -0500490 INIT_LIST_HEAD(&ls->ls_orphans);
491 mutex_init(&ls->ls_orphans_mutex);
David Teigland3ae1acf2007-05-18 08:59:31 -0500492 INIT_LIST_HEAD(&ls->ls_timeout);
493 mutex_init(&ls->ls_timeout_mutex);
David Teiglande7fd4172006-01-18 09:30:29 +0000494
495 INIT_LIST_HEAD(&ls->ls_nodes);
496 INIT_LIST_HEAD(&ls->ls_nodes_gone);
497 ls->ls_num_nodes = 0;
498 ls->ls_low_nodeid = 0;
499 ls->ls_total_weight = 0;
500 ls->ls_node_array = NULL;
501
502 memset(&ls->ls_stub_rsb, 0, sizeof(struct dlm_rsb));
503 ls->ls_stub_rsb.res_ls = ls;
504
David Teigland5de63192006-07-25 13:44:31 -0500505 ls->ls_debug_rsb_dentry = NULL;
506 ls->ls_debug_waiters_dentry = NULL;
David Teiglande7fd4172006-01-18 09:30:29 +0000507
508 init_waitqueue_head(&ls->ls_uevent_wait);
509 ls->ls_uevent_result = 0;
David Teigland8b0e7b22007-05-18 09:03:35 -0500510 init_completion(&ls->ls_members_done);
511 ls->ls_members_result = -1;
David Teiglande7fd4172006-01-18 09:30:29 +0000512
513 ls->ls_recoverd_task = NULL;
David Teigland90135922006-01-20 08:47:07 +0000514 mutex_init(&ls->ls_recoverd_active);
David Teiglande7fd4172006-01-18 09:30:29 +0000515 spin_lock_init(&ls->ls_recover_lock);
David Teigland98f176f2006-11-27 13:19:28 -0600516 spin_lock_init(&ls->ls_rcom_spin);
517 get_random_bytes(&ls->ls_rcom_seq, sizeof(uint64_t));
David Teiglande7fd4172006-01-18 09:30:29 +0000518 ls->ls_recover_status = 0;
519 ls->ls_recover_seq = 0;
520 ls->ls_recover_args = NULL;
521 init_rwsem(&ls->ls_in_recovery);
522 INIT_LIST_HEAD(&ls->ls_requestqueue);
David Teigland90135922006-01-20 08:47:07 +0000523 mutex_init(&ls->ls_requestqueue_mutex);
David Teigland597d0ca2006-07-12 16:44:04 -0500524 mutex_init(&ls->ls_clear_proc_locks);
David Teiglande7fd4172006-01-18 09:30:29 +0000525
David Teigland68c817a2007-01-09 09:41:48 -0600526 ls->ls_recover_buf = kmalloc(dlm_config.ci_buffer_size, GFP_KERNEL);
David Teiglande7fd4172006-01-18 09:30:29 +0000527 if (!ls->ls_recover_buf)
528 goto out_dirfree;
529
530 INIT_LIST_HEAD(&ls->ls_recover_list);
531 spin_lock_init(&ls->ls_recover_list_lock);
532 ls->ls_recover_list_count = 0;
David Teigland597d0ca2006-07-12 16:44:04 -0500533 ls->ls_local_handle = ls;
David Teiglande7fd4172006-01-18 09:30:29 +0000534 init_waitqueue_head(&ls->ls_wait_general);
535 INIT_LIST_HEAD(&ls->ls_root_list);
536 init_rwsem(&ls->ls_root_sem);
537
538 down_write(&ls->ls_in_recovery);
539
David Teigland5f88f1e2006-08-24 14:47:20 -0500540 spin_lock(&lslist_lock);
541 list_add(&ls->ls_list, &lslist);
542 spin_unlock(&lslist_lock);
543
544 /* needs to find ls in lslist */
David Teiglande7fd4172006-01-18 09:30:29 +0000545 error = dlm_recoverd_start(ls);
546 if (error) {
547 log_error(ls, "can't start dlm_recoverd %d", error);
David Teigland79d72b52007-05-18 09:02:20 -0500548 goto out_delist;
David Teiglande7fd4172006-01-18 09:30:29 +0000549 }
550
David Teiglande7fd4172006-01-18 09:30:29 +0000551 error = kobject_setup(ls);
552 if (error)
David Teigland79d72b52007-05-18 09:02:20 -0500553 goto out_stop;
David Teiglande7fd4172006-01-18 09:30:29 +0000554
555 error = kobject_register(&ls->ls_kobj);
556 if (error)
David Teigland79d72b52007-05-18 09:02:20 -0500557 goto out_stop;
558
559 /* let kobject handle freeing of ls if there's an error */
560 do_unreg = 1;
David Teiglande7fd4172006-01-18 09:30:29 +0000561
David Teigland8b0e7b22007-05-18 09:03:35 -0500562 /* This uevent triggers dlm_controld in userspace to add us to the
563 group of nodes that are members of this lockspace (managed by the
564 cluster infrastructure.) Once it's done that, it tells us who the
565 current lockspace members are (via configfs) and then tells the
566 lockspace to start running (via sysfs) in dlm_ls_start(). */
567
David Teiglande7fd4172006-01-18 09:30:29 +0000568 error = do_uevent(ls, 1);
569 if (error)
David Teigland79d72b52007-05-18 09:02:20 -0500570 goto out_stop;
571
David Teigland8b0e7b22007-05-18 09:03:35 -0500572 wait_for_completion(&ls->ls_members_done);
573 error = ls->ls_members_result;
574 if (error)
575 goto out_members;
576
David Teigland79d72b52007-05-18 09:02:20 -0500577 dlm_create_debug_file(ls);
578
579 log_debug(ls, "join complete");
David Teiglande7fd4172006-01-18 09:30:29 +0000580
581 *lockspace = ls;
582 return 0;
583
David Teigland8b0e7b22007-05-18 09:03:35 -0500584 out_members:
585 do_uevent(ls, 0);
586 dlm_clear_members(ls);
587 kfree(ls->ls_node_array);
David Teigland79d72b52007-05-18 09:02:20 -0500588 out_stop:
David Teigland5f88f1e2006-08-24 14:47:20 -0500589 dlm_recoverd_stop(ls);
David Teigland79d72b52007-05-18 09:02:20 -0500590 out_delist:
David Teiglande7fd4172006-01-18 09:30:29 +0000591 spin_lock(&lslist_lock);
592 list_del(&ls->ls_list);
593 spin_unlock(&lslist_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000594 kfree(ls->ls_recover_buf);
595 out_dirfree:
596 kfree(ls->ls_dirtbl);
597 out_lkbfree:
598 kfree(ls->ls_lkbtbl);
599 out_rsbfree:
600 kfree(ls->ls_rsbtbl);
601 out_lsfree:
David Teigland79d72b52007-05-18 09:02:20 -0500602 if (do_unreg)
603 kobject_unregister(&ls->ls_kobj);
604 else
605 kfree(ls);
David Teiglande7fd4172006-01-18 09:30:29 +0000606 out:
607 module_put(THIS_MODULE);
608 return error;
609}
610
611int dlm_new_lockspace(char *name, int namelen, void **lockspace,
612 uint32_t flags, int lvblen)
613{
614 int error = 0;
615
David Teigland90135922006-01-20 08:47:07 +0000616 mutex_lock(&ls_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000617 if (!ls_count)
618 error = threads_start();
619 if (error)
620 goto out;
621
622 error = new_lockspace(name, namelen, lockspace, flags, lvblen);
623 if (!error)
624 ls_count++;
David Teigland8b0e7b22007-05-18 09:03:35 -0500625 else if (!ls_count)
626 threads_stop();
David Teiglande7fd4172006-01-18 09:30:29 +0000627 out:
David Teigland90135922006-01-20 08:47:07 +0000628 mutex_unlock(&ls_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000629 return error;
630}
631
632/* Return 1 if the lockspace still has active remote locks,
633 * 2 if the lockspace still has active local locks.
634 */
635static int lockspace_busy(struct dlm_ls *ls)
636{
637 int i, lkb_found = 0;
638 struct dlm_lkb *lkb;
639
640 /* NOTE: We check the lockidtbl here rather than the resource table.
641 This is because there may be LKBs queued as ASTs that have been
642 unlinked from their RSBs and are pending deletion once the AST has
643 been delivered */
644
645 for (i = 0; i < ls->ls_lkbtbl_size; i++) {
646 read_lock(&ls->ls_lkbtbl[i].lock);
647 if (!list_empty(&ls->ls_lkbtbl[i].list)) {
648 lkb_found = 1;
649 list_for_each_entry(lkb, &ls->ls_lkbtbl[i].list,
650 lkb_idtbl_list) {
651 if (!lkb->lkb_nodeid) {
652 read_unlock(&ls->ls_lkbtbl[i].lock);
653 return 2;
654 }
655 }
656 }
657 read_unlock(&ls->ls_lkbtbl[i].lock);
658 }
659 return lkb_found;
660}
661
662static int release_lockspace(struct dlm_ls *ls, int force)
663{
664 struct dlm_lkb *lkb;
665 struct dlm_rsb *rsb;
666 struct list_head *head;
667 int i;
668 int busy = lockspace_busy(ls);
669
670 if (busy > force)
671 return -EBUSY;
672
673 if (force < 3)
674 do_uevent(ls, 0);
675
676 dlm_recoverd_stop(ls);
677
678 remove_lockspace(ls);
679
680 dlm_delete_debug_file(ls);
681
682 dlm_astd_suspend();
683
684 kfree(ls->ls_recover_buf);
685
686 /*
687 * Free direntry structs.
688 */
689
690 dlm_dir_clear(ls);
691 kfree(ls->ls_dirtbl);
692
693 /*
694 * Free all lkb's on lkbtbl[] lists.
695 */
696
697 for (i = 0; i < ls->ls_lkbtbl_size; i++) {
698 head = &ls->ls_lkbtbl[i].list;
699 while (!list_empty(head)) {
700 lkb = list_entry(head->next, struct dlm_lkb,
701 lkb_idtbl_list);
702
703 list_del(&lkb->lkb_idtbl_list);
704
705 dlm_del_ast(lkb);
706
707 if (lkb->lkb_lvbptr && lkb->lkb_flags & DLM_IFL_MSTCPY)
708 free_lvb(lkb->lkb_lvbptr);
709
710 free_lkb(lkb);
711 }
712 }
713 dlm_astd_resume();
714
715 kfree(ls->ls_lkbtbl);
716
717 /*
718 * Free all rsb's on rsbtbl[] lists
719 */
720
721 for (i = 0; i < ls->ls_rsbtbl_size; i++) {
722 head = &ls->ls_rsbtbl[i].list;
723 while (!list_empty(head)) {
724 rsb = list_entry(head->next, struct dlm_rsb,
725 res_hashchain);
726
727 list_del(&rsb->res_hashchain);
728 free_rsb(rsb);
729 }
730
731 head = &ls->ls_rsbtbl[i].toss;
732 while (!list_empty(head)) {
733 rsb = list_entry(head->next, struct dlm_rsb,
734 res_hashchain);
735 list_del(&rsb->res_hashchain);
736 free_rsb(rsb);
737 }
738 }
739
740 kfree(ls->ls_rsbtbl);
741
742 /*
743 * Free structures on any other lists
744 */
745
David Teigland2896ee32006-11-27 11:31:22 -0600746 dlm_purge_requestqueue(ls);
David Teiglande7fd4172006-01-18 09:30:29 +0000747 kfree(ls->ls_recover_args);
748 dlm_clear_free_entries(ls);
749 dlm_clear_members(ls);
750 dlm_clear_members_gone(ls);
751 kfree(ls->ls_node_array);
752 kobject_unregister(&ls->ls_kobj);
David Teigland79d72b52007-05-18 09:02:20 -0500753 /* The ls structure will be freed when the kobject is done with */
David Teiglande7fd4172006-01-18 09:30:29 +0000754
David Teigland90135922006-01-20 08:47:07 +0000755 mutex_lock(&ls_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000756 ls_count--;
757 if (!ls_count)
758 threads_stop();
David Teigland90135922006-01-20 08:47:07 +0000759 mutex_unlock(&ls_lock);
David Teiglande7fd4172006-01-18 09:30:29 +0000760
761 module_put(THIS_MODULE);
762 return 0;
763}
764
765/*
766 * Called when a system has released all its locks and is not going to use the
767 * lockspace any longer. We free everything we're managing for this lockspace.
768 * Remaining nodes will go through the recovery process as if we'd died. The
769 * lockspace must continue to function as usual, participating in recoveries,
770 * until this returns.
771 *
772 * Force has 4 possible values:
773 * 0 - don't destroy locksapce if it has any LKBs
774 * 1 - destroy lockspace if it has remote LKBs but not if it has local LKBs
775 * 2 - destroy lockspace regardless of LKBs
776 * 3 - destroy lockspace as part of a forced shutdown
777 */
778
779int dlm_release_lockspace(void *lockspace, int force)
780{
781 struct dlm_ls *ls;
782
783 ls = dlm_find_lockspace_local(lockspace);
784 if (!ls)
785 return -EINVAL;
786 dlm_put_lockspace(ls);
787 return release_lockspace(ls, force);
788}
789