aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/sunrpc/sched.c49
1 files changed, 21 insertions, 28 deletions
diff --git a/net/sunrpc/sched.c b/net/sunrpc/sched.c
index d95fe4e40eb4..f6eed4d4e5dd 100644
--- a/net/sunrpc/sched.c
+++ b/net/sunrpc/sched.c
@@ -58,7 +58,7 @@ static DECLARE_WAIT_QUEUE_HEAD(client_kill_wait);
58 * rpciod-related stuff 58 * rpciod-related stuff
59 */ 59 */
60static DEFINE_MUTEX(rpciod_mutex); 60static DEFINE_MUTEX(rpciod_mutex);
61static unsigned int rpciod_users; 61static atomic_t rpciod_users = ATOMIC_INIT(0);
62struct workqueue_struct *rpciod_workqueue; 62struct workqueue_struct *rpciod_workqueue;
63 63
64/* 64/*
@@ -1047,28 +1047,27 @@ rpciod_up(void)
1047 struct workqueue_struct *wq; 1047 struct workqueue_struct *wq;
1048 int error = 0; 1048 int error = 0;
1049 1049
1050 if (atomic_inc_not_zero(&rpciod_users))
1051 return 0;
1052
1050 mutex_lock(&rpciod_mutex); 1053 mutex_lock(&rpciod_mutex);
1051 dprintk("RPC: rpciod_up: users %u\n", rpciod_users); 1054
1052 rpciod_users++; 1055 /* Guard against races with rpciod_down() */
1053 if (rpciod_workqueue) 1056 if (rpciod_workqueue != NULL)
1054 goto out; 1057 goto out_ok;
1055 /*
1056 * If there's no pid, we should be the first user.
1057 */
1058 if (rpciod_users > 1)
1059 printk(KERN_WARNING "rpciod_up: no workqueue, %u users??\n", rpciod_users);
1060 /* 1058 /*
1061 * Create the rpciod thread and wait for it to start. 1059 * Create the rpciod thread and wait for it to start.
1062 */ 1060 */
1061 dprintk("RPC: creating workqueue rpciod\n");
1063 error = -ENOMEM; 1062 error = -ENOMEM;
1064 wq = create_workqueue("rpciod"); 1063 wq = create_workqueue("rpciod");
1065 if (wq == NULL) { 1064 if (wq == NULL)
1066 printk(KERN_WARNING "rpciod_up: create workqueue failed, error=%d\n", error);
1067 rpciod_users--;
1068 goto out; 1065 goto out;
1069 } 1066
1070 rpciod_workqueue = wq; 1067 rpciod_workqueue = wq;
1071 error = 0; 1068 error = 0;
1069out_ok:
1070 atomic_inc(&rpciod_users);
1072out: 1071out:
1073 mutex_unlock(&rpciod_mutex); 1072 mutex_unlock(&rpciod_mutex);
1074 return error; 1073 return error;
@@ -1077,23 +1076,17 @@ out:
1077void 1076void
1078rpciod_down(void) 1077rpciod_down(void)
1079{ 1078{
1079 if (!atomic_dec_and_test(&rpciod_users))
1080 return;
1081
1080 mutex_lock(&rpciod_mutex); 1082 mutex_lock(&rpciod_mutex);
1081 dprintk("RPC: rpciod_down sema %u\n", rpciod_users); 1083 dprintk("RPC: destroying workqueue rpciod\n");
1082 if (rpciod_users) {
1083 if (--rpciod_users)
1084 goto out;
1085 } else
1086 printk(KERN_WARNING "rpciod_down: no users??\n");
1087 1084
1088 if (!rpciod_workqueue) { 1085 if (atomic_read(&rpciod_users) == 0 && rpciod_workqueue != NULL) {
1089 dprintk("RPC: rpciod_down: Nothing to do!\n"); 1086 rpciod_killall();
1090 goto out; 1087 destroy_workqueue(rpciod_workqueue);
1088 rpciod_workqueue = NULL;
1091 } 1089 }
1092 rpciod_killall();
1093
1094 destroy_workqueue(rpciod_workqueue);
1095 rpciod_workqueue = NULL;
1096 out:
1097 mutex_unlock(&rpciod_mutex); 1090 mutex_unlock(&rpciod_mutex);
1098} 1091}
1099 1092