2 * linux/fs/lockd/clntlock.c
4 * Lock handling for the client side NLM implementation
6 * Copyright (C) 1996, Olaf Kirch <okir@monad.swb.de>
9 #define __KERNEL_SYSCALLS__
11 #include <linux/types.h>
12 #include <linux/sched.h>
13 #include <linux/nfs_fs.h>
14 #include <linux/unistd.h>
15 #include <linux/sunrpc/clnt.h>
16 #include <linux/sunrpc/svc.h>
17 #include <linux/lockd/lockd.h>
18 #include <linux/smp_lock.h>
20 #define NLMDBG_FACILITY NLMDBG_CIENT
23 * Local function prototypes
25 static int reclaimer(void *ptr
);
28 * The following functions handle blocking and granting from the
33 * This is the representation of a blocked client lock.
36 struct nlm_wait
* b_next
; /* linked list */
37 wait_queue_head_t b_wait
; /* where to wait on */
38 struct nlm_host
* b_host
;
39 struct file_lock
* b_lock
; /* local file lock */
40 unsigned short b_reclaim
; /* got to reclaim lock */
41 u32 b_status
; /* grant callback status */
44 static struct nlm_wait
* nlm_blocked
;
50 nlmclnt_block(struct nlm_host
*host
, struct file_lock
*fl
, u32
*statp
)
52 struct nlm_wait block
, **head
;
58 init_waitqueue_head(&block
.b_wait
);
59 block
.b_status
= NLM_LCK_BLOCKED
;
60 block
.b_next
= nlm_blocked
;
63 /* Remember pseudo nsm state */
64 pstate
= host
->h_state
;
66 /* Go to sleep waiting for GRANT callback. Some servers seem
67 * to lose callbacks, however, so we're going to poll from
68 * time to time just to make sure.
70 * For now, the retry frequency is pretty high; normally
71 * a 1 minute timeout would do. See the comment before
72 * nlmclnt_lock for an explanation.
74 sleep_on_timeout(&block
.b_wait
, 30*HZ
);
76 for (head
= &nlm_blocked
; *head
; head
= &(*head
)->b_next
) {
77 if (*head
== &block
) {
84 *statp
= block
.b_status
;
88 /* Okay, we were interrupted. Cancel the pending request
89 * unless the server has rebooted.
91 if (pstate
== host
->h_state
&& (err
= nlmclnt_cancel(host
, fl
)) < 0)
93 "lockd: CANCEL call failed (errno %d)\n", -err
);
99 * The server lockd has called us back to tell us the lock was granted
102 nlmclnt_grant(struct nlm_lock
*lock
)
104 struct nlm_wait
*block
;
107 * Look up blocked request based on arguments.
108 * Warning: must not use cookie to match it!
110 for (block
= nlm_blocked
; block
; block
= block
->b_next
) {
111 if (nlm_compare_locks(block
->b_lock
, &lock
->fl
))
115 /* Ooops, no blocked request found. */
117 return nlm_lck_denied
;
119 /* Alright, we found the lock. Set the return status and
120 * wake up the caller.
122 block
->b_status
= NLM_LCK_GRANTED
;
123 wake_up(&block
->b_wait
);
129 * The following procedures deal with the recovery of locks after a
134 * Reclaim all locks on server host. We do this by spawning a separate
136 * FIXME: should bump MOD_USE_COUNT while reclaiming
139 nlmclnt_recovery(struct nlm_host
*host
, u32 newstate
)
141 if (!host
->h_reclaiming
++) {
142 if (host
->h_nsmstate
== newstate
)
145 "lockd: Uh-oh! Interfering reclaims for host %s",
147 host
->h_monitored
= 0;
148 host
->h_nsmstate
= newstate
;
150 nlm_release_host(host
);
152 host
->h_monitored
= 0;
153 host
->h_nsmstate
= newstate
;
156 kernel_thread(reclaimer
, host
, 0);
163 struct nlm_host
*host
= (struct nlm_host
*) ptr
;
164 struct nlm_wait
*block
;
165 struct list_head
*tmp
;
167 /* This one ensures that our parent doesn't terminate while the
168 * reclaim is in progress */
172 /* First, reclaim all locks that have been granted previously. */
174 tmp
= file_lock_list
.next
;
175 while (tmp
!= &file_lock_list
) {
176 struct file_lock
*fl
= list_entry(tmp
, struct file_lock
, fl_link
);
177 struct inode
*inode
= fl
->fl_file
->f_dentry
->d_inode
;
178 if (inode
->i_sb
->s_magic
== NFS_SUPER_MAGIC
&&
179 nlm_cmp_addr(NFS_ADDR(inode
), &host
->h_addr
) &&
180 fl
->fl_u
.nfs_fl
.state
!= host
->h_state
&&
181 (fl
->fl_u
.nfs_fl
.flags
& NFS_LCK_GRANTED
)) {
182 fl
->fl_u
.nfs_fl
.flags
&= ~ NFS_LCK_GRANTED
;
183 nlmclnt_reclaim(host
, fl
); /* This sleeps */
189 host
->h_reclaiming
= 0;
190 wake_up(&host
->h_gracewait
);
192 /* Now, wake up all processes that sleep on a blocked lock */
193 for (block
= nlm_blocked
; block
; block
= block
->b_next
) {
194 if (block
->b_host
== host
) {
195 block
->b_status
= NLM_LCK_DENIED_GRACE_PERIOD
;
196 wake_up(&block
->b_wait
);
200 /* Release host handle after use */
201 nlm_release_host(host
);