Re: GPF in run_workqueue()/list_del_init(cwq->worklist.next) on resume
From: Tejun Heo
Date: Wed Nov 11 2009 - 03:09:32 EST
Hello,
Linus Torvalds wrote:
> On Tue, 10 Nov 2009, Rafael J. Wysocki wrote:
> And the code really is pretty subtle. queue_delayed_work_on(), for
> example, uses raw_smp_processor_id() to basically pick a target CPU for
> the work ("whatever the CPU happens to be now"). But does that have to
> match the CPU that the timeout will trigger on? I dunno.
It will queue on the cpu the timer callback runs, so delayed work will
follow timer which can be a little bit surprising at times, I suppose.
> I've walked through the code many times, but every time I end up
> forgetting all the subtleties a few days later.
A lot of those subtleties come from the fact struct work is not around
once the work function is invoked, so after that the workqueue code
doesn't have much to work with but the pointer value itself.
> The workqueue code is very fragile in many ways. I'm adding Oleg and Tejun
> to the Cc, because Oleg is definitely one of the workqueue locking
> masters, and Tejun has worked on it to make it way more robust, so they
> may have ideas.
One thing that I can think of which might cause this early release is
self-requeueing works which assume that only one instance of the
function will be executed at any given time. While preparing to bring
down a cpu, worker threads are unbound from the cpu. After cpu is
brought down, the workqueue for that cpu is flushed. This means that
any work which was running or on queue at the time of cpu down will
run on a different cpu. So, let's assume there's a work function
which looks like the following,
void my_work_fn(struct work_struct *work)
{
struct my_struct *me = container_of(work, something...);
DO SOMETHING;
if (--me->todo)
schedule_work(work);
else
free(me);
}
Which will work perfectly as long as all cpus stay alive as the work
will be pinned on a single cpu and cwq guarantees that a single work
is never executed in parallel. However, if a cpu is brought down
while my_work_fn() was doing SOMETHING and me->todo was above 1,
schedule_work() will schedule itself to a different cpu which will
happily execute the work in parallel.
As worker threads become unbound, they may bounce among different cpus
while executing and create more than two instances of parallel
execution of the work which can lead to freeings work which is still
on workqueue list with the above code but with different code just two
parallel executions should be able to achieve it.
Another related issue is the behavior flush_work() when a work ends up
scheduled on a different cpu. flush_work() will only look at a single
cpu workqueue on each flush attempt and if the work is not on the cpu
or there but also running on other cpus, it won't do nothing about it.
So, it's not too difficult to write code where the caller incorrectly
assumes the work is done after flush_work() is finished while the work
actually ended up being scheduled on a different cpu.
One way to debug I can think of is to record work pointer -> function
mapping in a percpu ring buffer and when the bug occurs (can be
detected by matching the 6b poison pattern before dereferencing next
pointer) print out function pointer for the matching work pointer.
The async nature makes the problem very difficult to track down. :-(
Thanks.
--
tejun
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/