diff options
author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2007-07-14 15:39:58 -0400 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2007-07-19 15:21:39 -0400 |
commit | e4eff1a622edd6ab7b73acd5d8763aa2fa3fee49 (patch) | |
tree | 257d6675733d4af122a77054281e1d7d5062d904 /fs/nfs/nfs3proc.c | |
parent | 4fdc17b2a7f4d9db5b08e0f963d0027f714e4104 (diff) |
SUNRPC: Clean up the sillyrename code
Fix a couple of bugs:
- Don't rely on the parent dentry still being valid when the call completes.
Fixes a race with shrink_dcache_for_umount_subtree()
- Don't remove the file if the filehandle has been labelled as stale.
Fix a couple of inefficiencies
- Remove the global list of sillyrenamed files. Instead we can cache the
sillyrename information in the dentry->d_fsdata
- Move common code from unlink_setup/unlink_done into fs/nfs/unlink.c
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs/nfs/nfs3proc.c')
-rw-r--r-- | fs/nfs/nfs3proc.c | 38 |
1 files changed, 9 insertions, 29 deletions
diff --git a/fs/nfs/nfs3proc.c b/fs/nfs/nfs3proc.c index eac07f2c99dd..c7ca5d70870b 100644 --- a/fs/nfs/nfs3proc.c +++ b/fs/nfs/nfs3proc.c | |||
@@ -370,41 +370,21 @@ nfs3_proc_remove(struct inode *dir, struct qstr *name) | |||
370 | return status; | 370 | return status; |
371 | } | 371 | } |
372 | 372 | ||
373 | static int | 373 | static void |
374 | nfs3_proc_unlink_setup(struct rpc_message *msg, struct dentry *dir, struct qstr *name) | 374 | nfs3_proc_unlink_setup(struct rpc_message *msg, struct inode *dir) |
375 | { | 375 | { |
376 | struct unlinkxdr { | ||
377 | struct nfs_removeargs arg; | ||
378 | struct nfs_removeres res; | ||
379 | } *ptr; | ||
380 | |||
381 | ptr = kmalloc(sizeof(*ptr), GFP_KERNEL); | ||
382 | if (!ptr) | ||
383 | return -ENOMEM; | ||
384 | ptr->arg.fh = NFS_FH(dir->d_inode); | ||
385 | ptr->arg.name.name = name->name; | ||
386 | ptr->arg.name.len = name->len; | ||
387 | nfs_fattr_init(&ptr->res.dir_attr); | ||
388 | msg->rpc_proc = &nfs3_procedures[NFS3PROC_REMOVE]; | 376 | msg->rpc_proc = &nfs3_procedures[NFS3PROC_REMOVE]; |
389 | msg->rpc_argp = &ptr->arg; | ||
390 | msg->rpc_resp = &ptr->res; | ||
391 | return 0; | ||
392 | } | 377 | } |
393 | 378 | ||
394 | static int | 379 | static int |
395 | nfs3_proc_unlink_done(struct dentry *dir, struct rpc_task *task) | 380 | nfs3_proc_unlink_done(struct rpc_task *task, struct inode *dir) |
396 | { | 381 | { |
397 | struct rpc_message *msg = &task->tk_msg; | 382 | struct nfs_removeres *res; |
398 | struct nfs_fattr *dir_attr; | 383 | if (nfs3_async_handle_jukebox(task, dir)) |
399 | 384 | return 0; | |
400 | if (nfs3_async_handle_jukebox(task, dir->d_inode)) | 385 | res = task->tk_msg.rpc_resp; |
401 | return 1; | 386 | nfs_post_op_update_inode(dir, &res->dir_attr); |
402 | if (msg->rpc_argp) { | 387 | return 1; |
403 | dir_attr = &((struct nfs_removeres*)msg->rpc_resp)->dir_attr; | ||
404 | nfs_post_op_update_inode(dir->d_inode, dir_attr); | ||
405 | kfree(msg->rpc_argp); | ||
406 | } | ||
407 | return 0; | ||
408 | } | 388 | } |
409 | 389 | ||
410 | static int | 390 | static int |