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.c49
1 files changed, 28 insertions, 21 deletions
diff --git a/fs/nfs/nfs4state.c b/fs/nfs/nfs4state.c
index 2894502b94dc..83f3dd39ac55 100644
--- a/fs/nfs/nfs4state.c
+++ b/fs/nfs/nfs4state.c
@@ -767,32 +767,34 @@ unlock:
767 return status; 767 return status;
768} 768}
769 769
770static int reclaimer(void *); 770static int nfs4_run_state_manager(void *);
771 771
772static inline void nfs4_clear_recover_bit(struct nfs_client *clp) 772static void nfs4_clear_state_manager_bit(struct nfs_client *clp)
773{ 773{
774 smp_mb__before_clear_bit(); 774 smp_mb__before_clear_bit();
775 clear_bit(NFS4CLNT_STATE_RECOVER, &clp->cl_state); 775 clear_bit(NFS4CLNT_MANAGER_RUNNING, &clp->cl_state);
776 smp_mb__after_clear_bit(); 776 smp_mb__after_clear_bit();
777 wake_up_bit(&clp->cl_state, NFS4CLNT_STATE_RECOVER); 777 wake_up_bit(&clp->cl_state, NFS4CLNT_MANAGER_RUNNING);
778 rpc_wake_up(&clp->cl_rpcwaitq); 778 rpc_wake_up(&clp->cl_rpcwaitq);
779} 779}
780 780
781/* 781/*
782 * State recovery routine 782 * Schedule the nfs_client asynchronous state management routine
783 */ 783 */
784static void nfs4_recover_state(struct nfs_client *clp) 784static void nfs4_schedule_state_manager(struct nfs_client *clp)
785{ 785{
786 struct task_struct *task; 786 struct task_struct *task;
787 787
788 if (test_and_set_bit(NFS4CLNT_MANAGER_RUNNING, &clp->cl_state) != 0)
789 return;
788 __module_get(THIS_MODULE); 790 __module_get(THIS_MODULE);
789 atomic_inc(&clp->cl_count); 791 atomic_inc(&clp->cl_count);
790 task = kthread_run(reclaimer, clp, "%s-reclaim", 792 task = kthread_run(nfs4_run_state_manager, clp, "%s-manager",
791 rpc_peeraddr2str(clp->cl_rpcclient, 793 rpc_peeraddr2str(clp->cl_rpcclient,
792 RPC_DISPLAY_ADDR)); 794 RPC_DISPLAY_ADDR));
793 if (!IS_ERR(task)) 795 if (!IS_ERR(task))
794 return; 796 return;
795 nfs4_clear_recover_bit(clp); 797 nfs4_clear_state_manager_bit(clp);
796 nfs_put_client(clp); 798 nfs_put_client(clp);
797 module_put(THIS_MODULE); 799 module_put(THIS_MODULE);
798} 800}
@@ -806,8 +808,7 @@ void nfs4_schedule_state_recovery(struct nfs_client *clp)
806 return; 808 return;
807 if (!test_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state)) 809 if (!test_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state))
808 set_bit(NFS4CLNT_CHECK_LEASE, &clp->cl_state); 810 set_bit(NFS4CLNT_CHECK_LEASE, &clp->cl_state);
809 if (test_and_set_bit(NFS4CLNT_STATE_RECOVER, &clp->cl_state) == 0) 811 nfs4_schedule_state_manager(clp);
810 nfs4_recover_state(clp);
811} 812}
812 813
813static int nfs4_state_mark_reclaim_reboot(struct nfs_client *clp, struct nfs4_state *state) 814static int nfs4_state_mark_reclaim_reboot(struct nfs_client *clp, struct nfs4_state *state)
@@ -1106,13 +1107,10 @@ static int nfs4_reclaim_lease(struct nfs_client *clp)
1106 return status; 1107 return status;
1107} 1108}
1108 1109
1109static int reclaimer(void *ptr) 1110static void nfs4_state_manager(struct nfs_client *clp)
1110{ 1111{
1111 struct nfs_client *clp = ptr;
1112 int status = 0; 1112 int status = 0;
1113 1113
1114 allow_signal(SIGKILL);
1115
1116 /* Ensure exclusive access to NFSv4 state */ 1114 /* Ensure exclusive access to NFSv4 state */
1117 while (!list_empty(&clp->cl_superblocks)) { 1115 while (!list_empty(&clp->cl_superblocks)) {
1118 if (test_and_clear_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state)) { 1116 if (test_and_clear_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state)) {
@@ -1161,19 +1159,28 @@ static int reclaimer(void *ptr)
1161 nfs_client_return_marked_delegations(clp); 1159 nfs_client_return_marked_delegations(clp);
1162 continue; 1160 continue;
1163 } 1161 }
1162
1163 nfs4_clear_state_manager_bit(clp);
1164 break; 1164 break;
1165 } 1165 }
1166out: 1166 return;
1167 nfs4_clear_recover_bit(clp);
1168 nfs_put_client(clp);
1169 module_put_and_exit(0);
1170 return 0;
1171out_error: 1167out_error:
1172 printk(KERN_WARNING "Error: state recovery failed on NFSv4 server %s" 1168 printk(KERN_WARNING "Error: state manager failed on NFSv4 server %s"
1173 " with error %d\n", clp->cl_hostname, -status); 1169 " with error %d\n", clp->cl_hostname, -status);
1174 if (test_bit(NFS4CLNT_RECLAIM_REBOOT, &clp->cl_state)) 1170 if (test_bit(NFS4CLNT_RECLAIM_REBOOT, &clp->cl_state))
1175 nfs4_state_end_reclaim_reboot(clp); 1171 nfs4_state_end_reclaim_reboot(clp);
1176 goto out; 1172 nfs4_clear_state_manager_bit(clp);
1173}
1174
1175static int nfs4_run_state_manager(void *ptr)
1176{
1177 struct nfs_client *clp = ptr;
1178
1179 allow_signal(SIGKILL);
1180 nfs4_state_manager(clp);
1181 nfs_put_client(clp);
1182 module_put_and_exit(0);
1183 return 0;
1177} 1184}
1178 1185
1179/* 1186/*