aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2012-08-03 13:30:45 -0400
committerTejun Heo <tj@kernel.org>2012-08-03 13:30:45 -0400
commitd8e794dfd51c368ed3f686b7f4172830b60ae47b (patch)
tree72e930ab0a14bf50fa1dc6802722483247b72806 /kernel
parent8930caba3dbdd8b86dd6934a5920bf61b53a931e (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.c7
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}
1113EXPORT_SYMBOL_GPL(queue_work); 1113EXPORT_SYMBOL_GPL(queue_work);
1114 1114
1115static void delayed_work_timer_fn(unsigned long __data) 1115void 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}
1124EXPORT_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);