On Tue, 5 Dec 2006 14:59:31 -0600
Andy Fleming <[email protected]> wrote:
> Ok, I think this is the summary:
>
> - phy_change() is the work queue callback function (scheduled when a
> PHY interrupt occurs)
>
> - dev_close() invokes the controller's stop/close/whatever function,
> and it calls phy_disconnect()
>
> - phy_disconnect() calls phy_stop_interrupts(). To prevent any
> pending phy_change() calls from getting confused, phy_stop_interrupts
> () needs to flush the queue. Otherwise, subsequent memory freeings
> will leave phy_change() hanging.
>
> - If phy_stop_interrupts() calls flush_scheduled_work(), keventd will
> execute its queues while rtnl_lock is held, providing opportunity for
> other callbacks to deadlock.
>
> - innocent puppies are slaughtered, and the world mourns.
>
ah, OK. So it's some other queued-up callback which takes rtnl_lock.
But I still don't see what's special about keventd. If, say, /sbin/ip is
running flush_scheduled_work() under rtnl_lock then it too should deadlock
in this scenario.
>
> Maciej's solution is to schedule phy_disconnect() to be called from a
> work queue. That solution should work, but it sounds like it doesn't
> require the check for if keventd is running.
>
> Of course, my objection to it is that it now requires the ethernet
> controller to be excessively aware of the details of how the PHY Lib
> is handling the PHY interrupts (by scheduling them on a work queue).
So what's a good fix?
a) Ban the calling of flush_scheduled_work() from under rtnl_lock().
Sounds hard.
b) Ban the queueing of callback functions which take rtnl_lock(). This
sounds like a plain bad idea - callbacks are low-level things which
ought to be able to take locks.
c) Cancel the phy_change() callback within phy_stop_interrupts() so we
don't need to run flush_scheduled_work() at all.
This will almost work, as long as it's done in workqueue.c with
appropriate locking. The bug occurs when some other CPU is running
phy_change() right now - we'll end up freeing data which that CPU is
presently playing with.
But perhaps we can take care of this within workqueue.c. We need a
cancel function which will cancel the work and, if its callback is
presently executing it will block until that execution has completed.
If we require of the calling subsystem a) that the work will not get
rescheduled (that means phy_interrupt()) and b) that the callback does
not rearm the work then things get simpler.
But still not very simple. It gets ugly with per-CPU qorkqueues.
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [email protected]
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
[Index of Archives]
[Kernel Newbies]
[Netfilter]
[Bugtraq]
[Photo]
[Stuff]
[Gimp]
[Yosemite News]
[MIPS Linux]
[ARM Linux]
[Linux Security]
[Linux RAID]
[Video 4 Linux]
[Linux for the blind]
[Linux Resources]