aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs/nfs4state.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/nfs/nfs4state.c')
-rw-r--r--fs/nfs/nfs4state.c46
1 files changed, 16 insertions, 30 deletions
diff --git a/fs/nfs/nfs4state.c b/fs/nfs/nfs4state.c
index 4ebf4df7f191..f2148376f1a7 100644
--- a/fs/nfs/nfs4state.c
+++ b/fs/nfs/nfs4state.c
@@ -43,6 +43,8 @@
43#include <linux/smp_lock.h> 43#include <linux/smp_lock.h>
44#include <linux/nfs_fs.h> 44#include <linux/nfs_fs.h>
45#include <linux/nfs_idmap.h> 45#include <linux/nfs_idmap.h>
46#include <linux/kthread.h>
47#include <linux/module.h>
46#include <linux/workqueue.h> 48#include <linux/workqueue.h>
47#include <linux/bitops.h> 49#include <linux/bitops.h>
48 50
@@ -57,8 +59,6 @@ const nfs4_stateid zero_stateid;
57static DEFINE_SPINLOCK(state_spinlock); 59static DEFINE_SPINLOCK(state_spinlock);
58static LIST_HEAD(nfs4_clientid_list); 60static LIST_HEAD(nfs4_clientid_list);
59 61
60static void nfs4_recover_state(void *);
61
62void 62void
63init_nfsv4_state(struct nfs_server *server) 63init_nfsv4_state(struct nfs_server *server)
64{ 64{
@@ -103,7 +103,6 @@ nfs4_alloc_client(struct in_addr *addr)
103 INIT_LIST_HEAD(&clp->cl_unused); 103 INIT_LIST_HEAD(&clp->cl_unused);
104 spin_lock_init(&clp->cl_lock); 104 spin_lock_init(&clp->cl_lock);
105 atomic_set(&clp->cl_count, 1); 105 atomic_set(&clp->cl_count, 1);
106 INIT_WORK(&clp->cl_recoverd, nfs4_recover_state, clp);
107 INIT_WORK(&clp->cl_renewd, nfs4_renew_state, clp); 106 INIT_WORK(&clp->cl_renewd, nfs4_renew_state, clp);
108 INIT_LIST_HEAD(&clp->cl_superblocks); 107 INIT_LIST_HEAD(&clp->cl_superblocks);
109 rpc_init_wait_queue(&clp->cl_rpcwaitq, "NFS4 client"); 108 rpc_init_wait_queue(&clp->cl_rpcwaitq, "NFS4 client");
@@ -734,10 +733,6 @@ out:
734} 733}
735 734
736static int reclaimer(void *); 735static int reclaimer(void *);
737struct reclaimer_args {
738 struct nfs4_client *clp;
739 struct completion complete;
740};
741 736
742static inline void nfs4_clear_recover_bit(struct nfs4_client *clp) 737static inline void nfs4_clear_recover_bit(struct nfs4_client *clp)
743{ 738{
@@ -751,35 +746,30 @@ static inline void nfs4_clear_recover_bit(struct nfs4_client *clp)
751/* 746/*
752 * State recovery routine 747 * State recovery routine
753 */ 748 */
754void 749static void nfs4_recover_state(struct nfs4_client *clp)
755nfs4_recover_state(void *data)
756{ 750{
757 struct nfs4_client *clp = (struct nfs4_client *)data; 751 struct task_struct *task;
758 struct reclaimer_args args = {
759 .clp = clp,
760 };
761 might_sleep();
762
763 init_completion(&args.complete);
764 752
765 if (kernel_thread(reclaimer, &args, CLONE_KERNEL) < 0) 753 __module_get(THIS_MODULE);
766 goto out_failed_clear; 754 atomic_inc(&clp->cl_count);
767 wait_for_completion(&args.complete); 755 task = kthread_run(reclaimer, clp, "%u.%u.%u.%u-reclaim",
768 return; 756 NIPQUAD(clp->cl_addr));
769out_failed_clear: 757 if (!IS_ERR(task))
758 return;
770 nfs4_clear_recover_bit(clp); 759 nfs4_clear_recover_bit(clp);
760 nfs4_put_client(clp);
761 module_put(THIS_MODULE);
771} 762}
772 763
773/* 764/*
774 * Schedule a state recovery attempt 765 * Schedule a state recovery attempt
775 */ 766 */
776void 767void nfs4_schedule_state_recovery(struct nfs4_client *clp)
777nfs4_schedule_state_recovery(struct nfs4_client *clp)
778{ 768{
779 if (!clp) 769 if (!clp)
780 return; 770 return;
781 if (test_and_set_bit(NFS4CLNT_STATE_RECOVER, &clp->cl_state) == 0) 771 if (test_and_set_bit(NFS4CLNT_STATE_RECOVER, &clp->cl_state) == 0)
782 schedule_work(&clp->cl_recoverd); 772 nfs4_recover_state(clp);
783} 773}
784 774
785static int nfs4_reclaim_locks(struct nfs4_state_recovery_ops *ops, struct nfs4_state *state) 775static int nfs4_reclaim_locks(struct nfs4_state_recovery_ops *ops, struct nfs4_state *state)
@@ -895,18 +885,13 @@ static void nfs4_state_mark_reclaim(struct nfs4_client *clp)
895 885
896static int reclaimer(void *ptr) 886static int reclaimer(void *ptr)
897{ 887{
898 struct reclaimer_args *args = (struct reclaimer_args *)ptr; 888 struct nfs4_client *clp = ptr;
899 struct nfs4_client *clp = args->clp;
900 struct nfs4_state_owner *sp; 889 struct nfs4_state_owner *sp;
901 struct nfs4_state_recovery_ops *ops; 890 struct nfs4_state_recovery_ops *ops;
902 int status = 0; 891 int status = 0;
903 892
904 daemonize("%u.%u.%u.%u-reclaim", NIPQUAD(clp->cl_addr));
905 allow_signal(SIGKILL); 893 allow_signal(SIGKILL);
906 894
907 atomic_inc(&clp->cl_count);
908 complete(&args->complete);
909
910 /* Ensure exclusive access to NFSv4 state */ 895 /* Ensure exclusive access to NFSv4 state */
911 lock_kernel(); 896 lock_kernel();
912 down_write(&clp->cl_sem); 897 down_write(&clp->cl_sem);
@@ -954,6 +939,7 @@ out:
954 nfs_handle_cb_pathdown(clp); 939 nfs_handle_cb_pathdown(clp);
955 nfs4_clear_recover_bit(clp); 940 nfs4_clear_recover_bit(clp);
956 nfs4_put_client(clp); 941 nfs4_put_client(clp);
942 module_put_and_exit(0);
957 return 0; 943 return 0;
958out_error: 944out_error:
959 printk(KERN_WARNING "Error: state recovery failed on NFSv4 server %u.%u.%u.%u with error %d\n", 945 printk(KERN_WARNING "Error: state recovery failed on NFSv4 server %u.%u.%u.%u with error %d\n",