aboutsummaryrefslogtreecommitdiffstats
path: root/net/sunrpc
diff options
context:
space:
mode:
Diffstat (limited to 'net/sunrpc')
-rw-r--r--net/sunrpc/sched.c60
1 files changed, 10 insertions, 50 deletions
diff --git a/net/sunrpc/sched.c b/net/sunrpc/sched.c
index 29b1c1441f4d..043eef4c15a2 100644
--- a/net/sunrpc/sched.c
+++ b/net/sunrpc/sched.c
@@ -58,13 +58,15 @@ struct workqueue_struct *rpciod_workqueue;
58 * rpc_run_timer(). 58 * rpc_run_timer().
59 */ 59 */
60static void 60static void
61__rpc_disable_timer(struct rpc_task *task) 61__rpc_disable_timer(struct rpc_wait_queue *queue, struct rpc_task *task)
62{ 62{
63 if (task->tk_timeout == 0) 63 if (task->tk_timeout == 0)
64 return; 64 return;
65 dprintk("RPC: %5u disabling timer\n", task->tk_pid); 65 dprintk("RPC: %5u disabling timer\n", task->tk_pid);
66 task->tk_timeout = 0; 66 task->tk_timeout = 0;
67 list_del(&task->u.tk_wait.timer_list); 67 list_del(&task->u.tk_wait.timer_list);
68 if (list_empty(&queue->timer_list.list))
69 del_timer(&queue->timer_list.timer);
68} 70}
69 71
70static void 72static void
@@ -78,7 +80,7 @@ rpc_set_queue_timer(struct rpc_wait_queue *queue, unsigned long expires)
78 * Set up a timer for the current task. 80 * Set up a timer for the current task.
79 */ 81 */
80static void 82static void
81__rpc_add_timer(struct rpc_task *task) 83__rpc_add_timer(struct rpc_wait_queue *queue, struct rpc_task *task)
82{ 84{
83 if (!task->tk_timeout) 85 if (!task->tk_timeout)
84 return; 86 return;
@@ -86,23 +88,10 @@ __rpc_add_timer(struct rpc_task *task)
86 dprintk("RPC: %5u setting alarm for %lu ms\n", 88 dprintk("RPC: %5u setting alarm for %lu ms\n",
87 task->tk_pid, task->tk_timeout * 1000 / HZ); 89 task->tk_pid, task->tk_timeout * 1000 / HZ);
88 90
89 set_bit(RPC_TASK_HAS_TIMER, &task->tk_runstate); 91 task->u.tk_wait.expires = jiffies + task->tk_timeout;
90 mod_timer(&task->tk_timer, jiffies + task->tk_timeout); 92 if (list_empty(&queue->timer_list.list) || time_before(task->u.tk_wait.expires, queue->timer_list.expires))
91} 93 rpc_set_queue_timer(queue, task->u.tk_wait.expires);
92 94 list_add(&task->u.tk_wait.timer_list, &queue->timer_list.list);
93/*
94 * Delete any timer for the current task. Because we use del_timer_sync(),
95 * this function should never be called while holding queue->lock.
96 */
97static void
98rpc_delete_timer(struct rpc_task *task)
99{
100 if (RPC_IS_QUEUED(task))
101 return;
102 if (test_and_clear_bit(RPC_TASK_HAS_TIMER, &task->tk_runstate)) {
103 del_singleshot_timer_sync(&task->tk_timer);
104 dprintk("RPC: %5u deleting timer\n", task->tk_pid);
105 }
106} 95}
107 96
108/* 97/*
@@ -172,7 +161,7 @@ static void __rpc_remove_wait_queue_priority(struct rpc_task *task)
172 */ 161 */
173static void __rpc_remove_wait_queue(struct rpc_wait_queue *queue, struct rpc_task *task) 162static void __rpc_remove_wait_queue(struct rpc_wait_queue *queue, struct rpc_task *task)
174{ 163{
175 __rpc_disable_timer(task); 164 __rpc_disable_timer(queue, task);
176 if (RPC_IS_PRIORITY(queue)) 165 if (RPC_IS_PRIORITY(queue))
177 __rpc_remove_wait_queue_priority(task); 166 __rpc_remove_wait_queue_priority(task);
178 list_del(&task->u.tk_wait.list); 167 list_del(&task->u.tk_wait.list);
@@ -344,7 +333,7 @@ static void __rpc_sleep_on(struct rpc_wait_queue *q, struct rpc_task *task,
344 333
345 BUG_ON(task->tk_callback != NULL); 334 BUG_ON(task->tk_callback != NULL);
346 task->tk_callback = action; 335 task->tk_callback = action;
347 __rpc_add_timer(task); 336 __rpc_add_timer(q, task);
348} 337}
349 338
350void rpc_sleep_on(struct rpc_wait_queue *q, struct rpc_task *task, 339void rpc_sleep_on(struct rpc_wait_queue *q, struct rpc_task *task,
@@ -555,26 +544,6 @@ void rpc_wake_up_status(struct rpc_wait_queue *queue, int status)
555} 544}
556EXPORT_SYMBOL_GPL(rpc_wake_up_status); 545EXPORT_SYMBOL_GPL(rpc_wake_up_status);
557 546
558/*
559 * Run a timeout function.
560 */
561static void rpc_run_timer(unsigned long ptr)
562{
563 struct rpc_task *task = (struct rpc_task *)ptr;
564 struct rpc_wait_queue *queue = task->tk_waitqueue;
565
566 spin_lock(&queue->lock);
567 if (RPC_IS_QUEUED(task) && task->tk_waitqueue == queue) {
568 dprintk("RPC: %5u timeout\n", task->tk_pid);
569 task->tk_status = -ETIMEDOUT;
570 rpc_wake_up_task_queue_locked(queue, task);
571 }
572 spin_unlock(&queue->lock);
573 smp_mb__before_clear_bit();
574 clear_bit(RPC_TASK_HAS_TIMER, &task->tk_runstate);
575 smp_mb__after_clear_bit();
576}
577
578static void __rpc_queue_timer_fn(unsigned long ptr) 547static void __rpc_queue_timer_fn(unsigned long ptr)
579{ 548{
580 struct rpc_wait_queue *queue = (struct rpc_wait_queue *)ptr; 549 struct rpc_wait_queue *queue = (struct rpc_wait_queue *)ptr;
@@ -586,7 +555,6 @@ static void __rpc_queue_timer_fn(unsigned long ptr)
586 list_for_each_entry_safe(task, n, &queue->timer_list.list, u.tk_wait.timer_list) { 555 list_for_each_entry_safe(task, n, &queue->timer_list.list, u.tk_wait.timer_list) {
587 timeo = task->u.tk_wait.expires; 556 timeo = task->u.tk_wait.expires;
588 if (time_after_eq(now, timeo)) { 557 if (time_after_eq(now, timeo)) {
589 list_del_init(&task->u.tk_wait.timer_list);
590 dprintk("RPC: %5u timeout\n", task->tk_pid); 558 dprintk("RPC: %5u timeout\n", task->tk_pid);
591 task->tk_status = -ETIMEDOUT; 559 task->tk_status = -ETIMEDOUT;
592 rpc_wake_up_task_queue_locked(queue, task); 560 rpc_wake_up_task_queue_locked(queue, task);
@@ -666,10 +634,6 @@ static void __rpc_execute(struct rpc_task *task)
666 BUG_ON(RPC_IS_QUEUED(task)); 634 BUG_ON(RPC_IS_QUEUED(task));
667 635
668 for (;;) { 636 for (;;) {
669 /*
670 * Garbage collection of pending timers...
671 */
672 rpc_delete_timer(task);
673 637
674 /* 638 /*
675 * Execute any pending callback. 639 * Execute any pending callback.
@@ -838,7 +802,6 @@ EXPORT_SYMBOL_GPL(rpc_free);
838static void rpc_init_task(struct rpc_task *task, const struct rpc_task_setup *task_setup_data) 802static void rpc_init_task(struct rpc_task *task, const struct rpc_task_setup *task_setup_data)
839{ 803{
840 memset(task, 0, sizeof(*task)); 804 memset(task, 0, sizeof(*task));
841 setup_timer(&task->tk_timer, rpc_run_timer, (unsigned long)task);
842 atomic_set(&task->tk_count, 1); 805 atomic_set(&task->tk_count, 1);
843 task->tk_flags = task_setup_data->flags; 806 task->tk_flags = task_setup_data->flags;
844 task->tk_ops = task_setup_data->callback_ops; 807 task->tk_ops = task_setup_data->callback_ops;
@@ -971,9 +934,6 @@ static void rpc_release_task(struct rpc_task *task)
971 } 934 }
972 BUG_ON (RPC_IS_QUEUED(task)); 935 BUG_ON (RPC_IS_QUEUED(task));
973 936
974 /* Synchronously delete any running timer */
975 rpc_delete_timer(task);
976
977#ifdef RPC_DEBUG 937#ifdef RPC_DEBUG
978 task->tk_magic = 0; 938 task->tk_magic = 0;
979#endif 939#endif