diff options
author | Tejun Heo <tj@kernel.org> | 2012-08-03 13:30:45 -0400 |
---|---|---|
committer | Tejun Heo <tj@kernel.org> | 2012-08-03 13:30:45 -0400 |
commit | d8e794dfd51c368ed3f686b7f4172830b60ae47b (patch) | |
tree | 72e930ab0a14bf50fa1dc6802722483247b72806 /kernel | |
parent | 8930caba3dbdd8b86dd6934a5920bf61b53a931e (diff) |
workqueue: set delayed_work->timer function on initialization
delayed_work->timer.function is currently initialized during
queue_delayed_work_on(). Export delayed_work_timer_fn() and set
delayed_work timer function during delayed_work initialization
together with other fields.
This ensures the timer function is always valid on an initialized
delayed_work. This is to help mod_delayed_work() implementation.
To detect delayed_work users which diddle with the internal timer,
trigger WARN if timer function doesn't match on queue.
Signed-off-by: Tejun Heo <tj@kernel.org>
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/workqueue.c | 7 |
1 files changed, 4 insertions, 3 deletions
diff --git a/kernel/workqueue.c b/kernel/workqueue.c index 30474c4e107c..55392385fe30 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c | |||
@@ -1112,7 +1112,7 @@ bool queue_work(struct workqueue_struct *wq, struct work_struct *work) | |||
1112 | } | 1112 | } |
1113 | EXPORT_SYMBOL_GPL(queue_work); | 1113 | EXPORT_SYMBOL_GPL(queue_work); |
1114 | 1114 | ||
1115 | static void delayed_work_timer_fn(unsigned long __data) | 1115 | void delayed_work_timer_fn(unsigned long __data) |
1116 | { | 1116 | { |
1117 | struct delayed_work *dwork = (struct delayed_work *)__data; | 1117 | struct delayed_work *dwork = (struct delayed_work *)__data; |
1118 | struct cpu_workqueue_struct *cwq = get_work_cwq(&dwork->work); | 1118 | struct cpu_workqueue_struct *cwq = get_work_cwq(&dwork->work); |
@@ -1121,6 +1121,7 @@ static void delayed_work_timer_fn(unsigned long __data) | |||
1121 | __queue_work(smp_processor_id(), cwq->wq, &dwork->work); | 1121 | __queue_work(smp_processor_id(), cwq->wq, &dwork->work); |
1122 | local_irq_enable(); | 1122 | local_irq_enable(); |
1123 | } | 1123 | } |
1124 | EXPORT_SYMBOL_GPL(delayed_work_timer_fn); | ||
1124 | 1125 | ||
1125 | /** | 1126 | /** |
1126 | * queue_delayed_work_on - queue work on specific CPU after delay | 1127 | * queue_delayed_work_on - queue work on specific CPU after delay |
@@ -1145,6 +1146,8 @@ bool queue_delayed_work_on(int cpu, struct workqueue_struct *wq, | |||
1145 | if (!test_and_set_bit(WORK_STRUCT_PENDING_BIT, work_data_bits(work))) { | 1146 | if (!test_and_set_bit(WORK_STRUCT_PENDING_BIT, work_data_bits(work))) { |
1146 | unsigned int lcpu; | 1147 | unsigned int lcpu; |
1147 | 1148 | ||
1149 | WARN_ON_ONCE(timer->function != delayed_work_timer_fn || | ||
1150 | timer->data != (unsigned long)dwork); | ||
1148 | BUG_ON(timer_pending(timer)); | 1151 | BUG_ON(timer_pending(timer)); |
1149 | BUG_ON(!list_empty(&work->entry)); | 1152 | BUG_ON(!list_empty(&work->entry)); |
1150 | 1153 | ||
@@ -1168,8 +1171,6 @@ bool queue_delayed_work_on(int cpu, struct workqueue_struct *wq, | |||
1168 | set_work_cwq(work, get_cwq(lcpu, wq), 0); | 1171 | set_work_cwq(work, get_cwq(lcpu, wq), 0); |
1169 | 1172 | ||
1170 | timer->expires = jiffies + delay; | 1173 | timer->expires = jiffies + delay; |
1171 | timer->data = (unsigned long)dwork; | ||
1172 | timer->function = delayed_work_timer_fn; | ||
1173 | 1174 | ||
1174 | if (unlikely(cpu >= 0)) | 1175 | if (unlikely(cpu >= 0)) |
1175 | add_timer_on(timer, cpu); | 1176 | add_timer_on(timer, cpu); |