diff options
Diffstat (limited to 'fs/lockd')
| -rw-r--r-- | fs/lockd/svc.c | 33 | ||||
| -rw-r--r-- | fs/lockd/svc4proc.c | 7 | ||||
| -rw-r--r-- | fs/lockd/svclock.c | 33 | ||||
| -rw-r--r-- | fs/lockd/svcproc.c | 7 | ||||
| -rw-r--r-- | fs/lockd/svcsubs.c | 32 |
5 files changed, 52 insertions, 60 deletions
diff --git a/fs/lockd/svc.c b/fs/lockd/svc.c index 2169af4d545..5bd9bf0fa9d 100644 --- a/fs/lockd/svc.c +++ b/fs/lockd/svc.c | |||
| @@ -50,7 +50,7 @@ EXPORT_SYMBOL(nlmsvc_ops); | |||
| 50 | static DEFINE_MUTEX(nlmsvc_mutex); | 50 | static DEFINE_MUTEX(nlmsvc_mutex); |
| 51 | static unsigned int nlmsvc_users; | 51 | static unsigned int nlmsvc_users; |
| 52 | static struct task_struct *nlmsvc_task; | 52 | static struct task_struct *nlmsvc_task; |
| 53 | static struct svc_serv *nlmsvc_serv; | 53 | static struct svc_rqst *nlmsvc_rqst; |
| 54 | int nlmsvc_grace_period; | 54 | int nlmsvc_grace_period; |
| 55 | unsigned long nlmsvc_timeout; | 55 | unsigned long nlmsvc_timeout; |
| 56 | 56 | ||
| @@ -194,20 +194,11 @@ lockd(void *vrqstp) | |||
| 194 | 194 | ||
| 195 | svc_process(rqstp); | 195 | svc_process(rqstp); |
| 196 | } | 196 | } |
| 197 | |||
| 198 | flush_signals(current); | 197 | flush_signals(current); |
| 199 | if (nlmsvc_ops) | 198 | if (nlmsvc_ops) |
| 200 | nlmsvc_invalidate_all(); | 199 | nlmsvc_invalidate_all(); |
| 201 | nlm_shutdown_hosts(); | 200 | nlm_shutdown_hosts(); |
| 202 | |||
| 203 | unlock_kernel(); | 201 | unlock_kernel(); |
| 204 | |||
| 205 | nlmsvc_task = NULL; | ||
| 206 | nlmsvc_serv = NULL; | ||
| 207 | |||
| 208 | /* Exit the RPC thread */ | ||
| 209 | svc_exit_thread(rqstp); | ||
| 210 | |||
| 211 | return 0; | 202 | return 0; |
| 212 | } | 203 | } |
| 213 | 204 | ||
| @@ -254,16 +245,15 @@ int | |||
| 254 | lockd_up(int proto) /* Maybe add a 'family' option when IPv6 is supported ?? */ | 245 | lockd_up(int proto) /* Maybe add a 'family' option when IPv6 is supported ?? */ |
| 255 | { | 246 | { |
| 256 | struct svc_serv *serv; | 247 | struct svc_serv *serv; |
| 257 | struct svc_rqst *rqstp; | ||
| 258 | int error = 0; | 248 | int error = 0; |
| 259 | 249 | ||
| 260 | mutex_lock(&nlmsvc_mutex); | 250 | mutex_lock(&nlmsvc_mutex); |
| 261 | /* | 251 | /* |
| 262 | * Check whether we're already up and running. | 252 | * Check whether we're already up and running. |
| 263 | */ | 253 | */ |
| 264 | if (nlmsvc_serv) { | 254 | if (nlmsvc_rqst) { |
| 265 | if (proto) | 255 | if (proto) |
| 266 | error = make_socks(nlmsvc_serv, proto); | 256 | error = make_socks(nlmsvc_rqst->rq_server, proto); |
| 267 | goto out; | 257 | goto out; |
| 268 | } | 258 | } |
| 269 | 259 | ||
| @@ -288,9 +278,10 @@ lockd_up(int proto) /* Maybe add a 'family' option when IPv6 is supported ?? */ | |||
| 288 | /* | 278 | /* |
| 289 | * Create the kernel thread and wait for it to start. | 279 | * Create the kernel thread and wait for it to start. |
| 290 | */ | 280 | */ |
| 291 | rqstp = svc_prepare_thread(serv, &serv->sv_pools[0]); | 281 | nlmsvc_rqst = svc_prepare_thread(serv, &serv->sv_pools[0]); |
| 292 | if (IS_ERR(rqstp)) { | 282 | if (IS_ERR(nlmsvc_rqst)) { |
| 293 | error = PTR_ERR(rqstp); | 283 | error = PTR_ERR(nlmsvc_rqst); |
| 284 | nlmsvc_rqst = NULL; | ||
| 294 | printk(KERN_WARNING | 285 | printk(KERN_WARNING |
| 295 | "lockd_up: svc_rqst allocation failed, error=%d\n", | 286 | "lockd_up: svc_rqst allocation failed, error=%d\n", |
| 296 | error); | 287 | error); |
| @@ -298,16 +289,15 @@ lockd_up(int proto) /* Maybe add a 'family' option when IPv6 is supported ?? */ | |||
| 298 | } | 289 | } |
| 299 | 290 | ||
| 300 | svc_sock_update_bufs(serv); | 291 | svc_sock_update_bufs(serv); |
| 301 | nlmsvc_serv = rqstp->rq_server; | ||
| 302 | 292 | ||
| 303 | nlmsvc_task = kthread_run(lockd, rqstp, serv->sv_name); | 293 | nlmsvc_task = kthread_run(lockd, nlmsvc_rqst, serv->sv_name); |
| 304 | if (IS_ERR(nlmsvc_task)) { | 294 | if (IS_ERR(nlmsvc_task)) { |
| 305 | error = PTR_ERR(nlmsvc_task); | 295 | error = PTR_ERR(nlmsvc_task); |
| 296 | svc_exit_thread(nlmsvc_rqst); | ||
| 306 | nlmsvc_task = NULL; | 297 | nlmsvc_task = NULL; |
| 307 | nlmsvc_serv = NULL; | 298 | nlmsvc_rqst = NULL; |
| 308 | printk(KERN_WARNING | 299 | printk(KERN_WARNING |
| 309 | "lockd_up: kthread_run failed, error=%d\n", error); | 300 | "lockd_up: kthread_run failed, error=%d\n", error); |
| 310 | svc_exit_thread(rqstp); | ||
| 311 | goto destroy_and_out; | 301 | goto destroy_and_out; |
| 312 | } | 302 | } |
| 313 | 303 | ||
| @@ -346,6 +336,9 @@ lockd_down(void) | |||
| 346 | BUG(); | 336 | BUG(); |
| 347 | } | 337 | } |
| 348 | kthread_stop(nlmsvc_task); | 338 | kthread_stop(nlmsvc_task); |
| 339 | svc_exit_thread(nlmsvc_rqst); | ||
| 340 | nlmsvc_task = NULL; | ||
| 341 | nlmsvc_rqst = NULL; | ||
| 349 | out: | 342 | out: |
| 350 | mutex_unlock(&nlmsvc_mutex); | 343 | mutex_unlock(&nlmsvc_mutex); |
| 351 | } | 344 | } |
diff --git a/fs/lockd/svc4proc.c b/fs/lockd/svc4proc.c index 2e27176ff42..39944463933 100644 --- a/fs/lockd/svc4proc.c +++ b/fs/lockd/svc4proc.c | |||
| @@ -58,8 +58,7 @@ nlm4svc_retrieve_args(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 58 | return 0; | 58 | return 0; |
| 59 | 59 | ||
| 60 | no_locks: | 60 | no_locks: |
| 61 | if (host) | 61 | nlm_release_host(host); |
| 62 | nlm_release_host(host); | ||
| 63 | if (error) | 62 | if (error) |
| 64 | return error; | 63 | return error; |
| 65 | return nlm_lck_denied_nolocks; | 64 | return nlm_lck_denied_nolocks; |
| @@ -100,7 +99,7 @@ nlm4svc_proc_test(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 100 | return resp->status == nlm_drop_reply ? rpc_drop_reply :rpc_success; | 99 | return resp->status == nlm_drop_reply ? rpc_drop_reply :rpc_success; |
| 101 | 100 | ||
| 102 | /* Now check for conflicting locks */ | 101 | /* Now check for conflicting locks */ |
| 103 | resp->status = nlmsvc_testlock(rqstp, file, &argp->lock, &resp->lock, &resp->cookie); | 102 | resp->status = nlmsvc_testlock(rqstp, file, host, &argp->lock, &resp->lock, &resp->cookie); |
| 104 | if (resp->status == nlm_drop_reply) | 103 | if (resp->status == nlm_drop_reply) |
| 105 | rc = rpc_drop_reply; | 104 | rc = rpc_drop_reply; |
| 106 | else | 105 | else |
| @@ -146,7 +145,7 @@ nlm4svc_proc_lock(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 146 | #endif | 145 | #endif |
| 147 | 146 | ||
| 148 | /* Now try to lock the file */ | 147 | /* Now try to lock the file */ |
| 149 | resp->status = nlmsvc_lock(rqstp, file, &argp->lock, | 148 | resp->status = nlmsvc_lock(rqstp, file, host, &argp->lock, |
| 150 | argp->block, &argp->cookie); | 149 | argp->block, &argp->cookie); |
| 151 | if (resp->status == nlm_drop_reply) | 150 | if (resp->status == nlm_drop_reply) |
| 152 | rc = rpc_drop_reply; | 151 | rc = rpc_drop_reply; |
diff --git a/fs/lockd/svclock.c b/fs/lockd/svclock.c index 56a08ab9a4c..821b9acdfb6 100644 --- a/fs/lockd/svclock.c +++ b/fs/lockd/svclock.c | |||
| @@ -129,9 +129,9 @@ nlmsvc_lookup_block(struct nlm_file *file, struct nlm_lock *lock) | |||
| 129 | 129 | ||
| 130 | static inline int nlm_cookie_match(struct nlm_cookie *a, struct nlm_cookie *b) | 130 | static inline int nlm_cookie_match(struct nlm_cookie *a, struct nlm_cookie *b) |
| 131 | { | 131 | { |
| 132 | if(a->len != b->len) | 132 | if (a->len != b->len) |
| 133 | return 0; | 133 | return 0; |
| 134 | if(memcmp(a->data,b->data,a->len)) | 134 | if (memcmp(a->data, b->data, a->len)) |
| 135 | return 0; | 135 | return 0; |
| 136 | return 1; | 136 | return 1; |
| 137 | } | 137 | } |
| @@ -180,6 +180,7 @@ nlmsvc_create_block(struct svc_rqst *rqstp, struct nlm_host *host, | |||
| 180 | struct nlm_block *block; | 180 | struct nlm_block *block; |
| 181 | struct nlm_rqst *call = NULL; | 181 | struct nlm_rqst *call = NULL; |
| 182 | 182 | ||
| 183 | nlm_get_host(host); | ||
| 183 | call = nlm_alloc_call(host); | 184 | call = nlm_alloc_call(host); |
| 184 | if (call == NULL) | 185 | if (call == NULL) |
| 185 | return NULL; | 186 | return NULL; |
| @@ -358,10 +359,10 @@ nlmsvc_defer_lock_rqst(struct svc_rqst *rqstp, struct nlm_block *block) | |||
| 358 | */ | 359 | */ |
| 359 | __be32 | 360 | __be32 |
| 360 | nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, | 361 | nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, |
| 361 | struct nlm_lock *lock, int wait, struct nlm_cookie *cookie) | 362 | struct nlm_host *host, struct nlm_lock *lock, int wait, |
| 363 | struct nlm_cookie *cookie) | ||
| 362 | { | 364 | { |
| 363 | struct nlm_block *block = NULL; | 365 | struct nlm_block *block = NULL; |
| 364 | struct nlm_host *host; | ||
| 365 | int error; | 366 | int error; |
| 366 | __be32 ret; | 367 | __be32 ret; |
| 367 | 368 | ||
| @@ -373,11 +374,6 @@ nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, | |||
| 373 | (long long)lock->fl.fl_end, | 374 | (long long)lock->fl.fl_end, |
| 374 | wait); | 375 | wait); |
| 375 | 376 | ||
| 376 | /* Create host handle for callback */ | ||
| 377 | host = nlmsvc_lookup_host(rqstp, lock->caller, lock->len); | ||
| 378 | if (host == NULL) | ||
| 379 | return nlm_lck_denied_nolocks; | ||
| 380 | |||
| 381 | /* Lock file against concurrent access */ | 377 | /* Lock file against concurrent access */ |
| 382 | mutex_lock(&file->f_mutex); | 378 | mutex_lock(&file->f_mutex); |
| 383 | /* Get existing block (in case client is busy-waiting) | 379 | /* Get existing block (in case client is busy-waiting) |
| @@ -385,8 +381,7 @@ nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, | |||
| 385 | */ | 381 | */ |
| 386 | block = nlmsvc_lookup_block(file, lock); | 382 | block = nlmsvc_lookup_block(file, lock); |
| 387 | if (block == NULL) { | 383 | if (block == NULL) { |
| 388 | block = nlmsvc_create_block(rqstp, nlm_get_host(host), file, | 384 | block = nlmsvc_create_block(rqstp, host, file, lock, cookie); |
| 389 | lock, cookie); | ||
| 390 | ret = nlm_lck_denied_nolocks; | 385 | ret = nlm_lck_denied_nolocks; |
| 391 | if (block == NULL) | 386 | if (block == NULL) |
| 392 | goto out; | 387 | goto out; |
| @@ -417,7 +412,7 @@ nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, | |||
| 417 | lock->fl.fl_flags &= ~FL_SLEEP; | 412 | lock->fl.fl_flags &= ~FL_SLEEP; |
| 418 | 413 | ||
| 419 | dprintk("lockd: vfs_lock_file returned %d\n", error); | 414 | dprintk("lockd: vfs_lock_file returned %d\n", error); |
| 420 | switch(error) { | 415 | switch (error) { |
| 421 | case 0: | 416 | case 0: |
| 422 | ret = nlm_granted; | 417 | ret = nlm_granted; |
| 423 | goto out; | 418 | goto out; |
| @@ -450,7 +445,6 @@ nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file, | |||
| 450 | out: | 445 | out: |
| 451 | mutex_unlock(&file->f_mutex); | 446 | mutex_unlock(&file->f_mutex); |
| 452 | nlmsvc_release_block(block); | 447 | nlmsvc_release_block(block); |
| 453 | nlm_release_host(host); | ||
| 454 | dprintk("lockd: nlmsvc_lock returned %u\n", ret); | 448 | dprintk("lockd: nlmsvc_lock returned %u\n", ret); |
| 455 | return ret; | 449 | return ret; |
| 456 | } | 450 | } |
| @@ -460,8 +454,8 @@ out: | |||
| 460 | */ | 454 | */ |
| 461 | __be32 | 455 | __be32 |
| 462 | nlmsvc_testlock(struct svc_rqst *rqstp, struct nlm_file *file, | 456 | nlmsvc_testlock(struct svc_rqst *rqstp, struct nlm_file *file, |
| 463 | struct nlm_lock *lock, struct nlm_lock *conflock, | 457 | struct nlm_host *host, struct nlm_lock *lock, |
| 464 | struct nlm_cookie *cookie) | 458 | struct nlm_lock *conflock, struct nlm_cookie *cookie) |
| 465 | { | 459 | { |
| 466 | struct nlm_block *block = NULL; | 460 | struct nlm_block *block = NULL; |
| 467 | int error; | 461 | int error; |
| @@ -479,16 +473,9 @@ nlmsvc_testlock(struct svc_rqst *rqstp, struct nlm_file *file, | |||
| 479 | 473 | ||
| 480 | if (block == NULL) { | 474 | if (block == NULL) { |
| 481 | struct file_lock *conf = kzalloc(sizeof(*conf), GFP_KERNEL); | 475 | struct file_lock *conf = kzalloc(sizeof(*conf), GFP_KERNEL); |
| 482 | struct nlm_host *host; | ||
| 483 | 476 | ||
| 484 | if (conf == NULL) | 477 | if (conf == NULL) |
| 485 | return nlm_granted; | 478 | return nlm_granted; |
| 486 | /* Create host handle for callback */ | ||
| 487 | host = nlmsvc_lookup_host(rqstp, lock->caller, lock->len); | ||
| 488 | if (host == NULL) { | ||
| 489 | kfree(conf); | ||
| 490 | return nlm_lck_denied_nolocks; | ||
| 491 | } | ||
| 492 | block = nlmsvc_create_block(rqstp, host, file, lock, cookie); | 479 | block = nlmsvc_create_block(rqstp, host, file, lock, cookie); |
| 493 | if (block == NULL) { | 480 | if (block == NULL) { |
| 494 | kfree(conf); | 481 | kfree(conf); |
| @@ -897,7 +884,7 @@ nlmsvc_retry_blocked(void) | |||
| 897 | 884 | ||
| 898 | if (block->b_when == NLM_NEVER) | 885 | if (block->b_when == NLM_NEVER) |
| 899 | break; | 886 | break; |
| 900 | if (time_after(block->b_when,jiffies)) { | 887 | if (time_after(block->b_when, jiffies)) { |
| 901 | timeout = block->b_when - jiffies; | 888 | timeout = block->b_when - jiffies; |
| 902 | break; | 889 | break; |
| 903 | } | 890 | } |
diff --git a/fs/lockd/svcproc.c b/fs/lockd/svcproc.c index ce6952b50a7..76019d2ff72 100644 --- a/fs/lockd/svcproc.c +++ b/fs/lockd/svcproc.c | |||
| @@ -87,8 +87,7 @@ nlmsvc_retrieve_args(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 87 | return 0; | 87 | return 0; |
| 88 | 88 | ||
| 89 | no_locks: | 89 | no_locks: |
| 90 | if (host) | 90 | nlm_release_host(host); |
| 91 | nlm_release_host(host); | ||
| 92 | if (error) | 91 | if (error) |
| 93 | return error; | 92 | return error; |
| 94 | return nlm_lck_denied_nolocks; | 93 | return nlm_lck_denied_nolocks; |
| @@ -129,7 +128,7 @@ nlmsvc_proc_test(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 129 | return resp->status == nlm_drop_reply ? rpc_drop_reply :rpc_success; | 128 | return resp->status == nlm_drop_reply ? rpc_drop_reply :rpc_success; |
| 130 | 129 | ||
| 131 | /* Now check for conflicting locks */ | 130 | /* Now check for conflicting locks */ |
| 132 | resp->status = cast_status(nlmsvc_testlock(rqstp, file, &argp->lock, &resp->lock, &resp->cookie)); | 131 | resp->status = cast_status(nlmsvc_testlock(rqstp, file, host, &argp->lock, &resp->lock, &resp->cookie)); |
| 133 | if (resp->status == nlm_drop_reply) | 132 | if (resp->status == nlm_drop_reply) |
| 134 | rc = rpc_drop_reply; | 133 | rc = rpc_drop_reply; |
| 135 | else | 134 | else |
| @@ -176,7 +175,7 @@ nlmsvc_proc_lock(struct svc_rqst *rqstp, struct nlm_args *argp, | |||
| 176 | #endif | 175 | #endif |
| 177 | 176 | ||
| 178 | /* Now try to lock the file */ | 177 | /* Now try to lock the file */ |
| 179 | resp->status = cast_status(nlmsvc_lock(rqstp, file, &argp->lock, | 178 | resp->status = cast_status(nlmsvc_lock(rqstp, file, host, &argp->lock, |
| 180 | argp->block, &argp->cookie)); | 179 | argp->block, &argp->cookie)); |
| 181 | if (resp->status == nlm_drop_reply) | 180 | if (resp->status == nlm_drop_reply) |
| 182 | rc = rpc_drop_reply; | 181 | rc = rpc_drop_reply; |
diff --git a/fs/lockd/svcsubs.c b/fs/lockd/svcsubs.c index d1c48b539df..198b4e55b37 100644 --- a/fs/lockd/svcsubs.c +++ b/fs/lockd/svcsubs.c | |||
| @@ -373,13 +373,16 @@ nlmsvc_free_host_resources(struct nlm_host *host) | |||
| 373 | } | 373 | } |
| 374 | } | 374 | } |
| 375 | 375 | ||
| 376 | /* | 376 | /** |
| 377 | * Remove all locks held for clients | 377 | * nlmsvc_invalidate_all - remove all locks held for clients |
| 378 | * | ||
| 379 | * Release all locks held by NFS clients. | ||
| 380 | * | ||
| 378 | */ | 381 | */ |
| 379 | void | 382 | void |
| 380 | nlmsvc_invalidate_all(void) | 383 | nlmsvc_invalidate_all(void) |
| 381 | { | 384 | { |
| 382 | /* Release all locks held by NFS clients. | 385 | /* |
| 383 | * Previously, the code would call | 386 | * Previously, the code would call |
| 384 | * nlmsvc_free_host_resources for each client in | 387 | * nlmsvc_free_host_resources for each client in |
| 385 | * turn, which is about as inefficient as it gets. | 388 | * turn, which is about as inefficient as it gets. |
| @@ -396,6 +399,12 @@ nlmsvc_match_sb(void *datap, struct nlm_file *file) | |||
| 396 | return sb == file->f_file->f_path.mnt->mnt_sb; | 399 | return sb == file->f_file->f_path.mnt->mnt_sb; |
| 397 | } | 400 | } |
| 398 | 401 | ||
| 402 | /** | ||
| 403 | * nlmsvc_unlock_all_by_sb - release locks held on this file system | ||
| 404 | * @sb: super block | ||
| 405 | * | ||
| 406 | * Release all locks held by clients accessing this file system. | ||
| 407 | */ | ||
| 399 | int | 408 | int |
| 400 | nlmsvc_unlock_all_by_sb(struct super_block *sb) | 409 | nlmsvc_unlock_all_by_sb(struct super_block *sb) |
| 401 | { | 410 | { |
| @@ -409,17 +418,22 @@ EXPORT_SYMBOL_GPL(nlmsvc_unlock_all_by_sb); | |||
| 409 | static int | 418 | static int |
| 410 | nlmsvc_match_ip(void *datap, struct nlm_host *host) | 419 | nlmsvc_match_ip(void *datap, struct nlm_host *host) |
| 411 | { | 420 | { |
| 412 | __be32 *server_addr = datap; | 421 | return nlm_cmp_addr(&host->h_saddr, datap); |
| 413 | |||
| 414 | return host->h_saddr.sin_addr.s_addr == *server_addr; | ||
| 415 | } | 422 | } |
| 416 | 423 | ||
| 424 | /** | ||
| 425 | * nlmsvc_unlock_all_by_ip - release local locks by IP address | ||
| 426 | * @server_addr: server's IP address as seen by clients | ||
| 427 | * | ||
| 428 | * Release all locks held by clients accessing this host | ||
| 429 | * via the passed in IP address. | ||
| 430 | */ | ||
| 417 | int | 431 | int |
| 418 | nlmsvc_unlock_all_by_ip(__be32 server_addr) | 432 | nlmsvc_unlock_all_by_ip(struct sockaddr *server_addr) |
| 419 | { | 433 | { |
| 420 | int ret; | 434 | int ret; |
| 421 | ret = nlm_traverse_files(&server_addr, nlmsvc_match_ip, NULL); | ||
| 422 | return ret ? -EIO : 0; | ||
| 423 | 435 | ||
| 436 | ret = nlm_traverse_files(server_addr, nlmsvc_match_ip, NULL); | ||
| 437 | return ret ? -EIO : 0; | ||
| 424 | } | 438 | } |
| 425 | EXPORT_SYMBOL_GPL(nlmsvc_unlock_all_by_ip); | 439 | EXPORT_SYMBOL_GPL(nlmsvc_unlock_all_by_ip); |
