cvs commit: src/sys/kern subr_taskqueue.c

John Baldwin jhb at freebsd.org
Wed Jan 11 06:20:35 PST 2006


On Wednesday 11 January 2006 08:56 am, Scott Long wrote:
> John Baldwin wrote:
> > On Tuesday 10 January 2006 07:37 pm, Scott Long wrote:
> >>scottl      2006-01-11 00:37:13 UTC
> >>
> >>  FreeBSD src repository
> >>
> >>  Modified files:
> >>    sys/kern             subr_taskqueue.c
> >>  Log:
> >>  The interlock in taskqueue_terminate() is completely wrong for
> >> taskqueues that use spinlocks.  Remove it for now.
> >
> > Eh?  It's waiting for the wakeup that comes from kthread_exit() after the
> > thread has exited which is locked via the proc lock.  Sleeping on the
> > taskqueue itself doesn't buy you anything.  (In fact, it might sleep
> > forever.)   The simplest solution might be to acquire the proc lock a lot
> > earlier before the taskqueue lock in this function so that you don't have
> > to acquire it while holding the taskqueue lock since that is what gives
> > you problems.
>
> With the code the way it was, kthread_exit() in taskqueue_thread_loop
> can wind up blocking on the proc lock while the lock is still held in
> taskqueue_terminate.  I don't know why this is actually a problem, but
> turning on WITNESS to investigate revealed the immediate problem of
> trying to grab the proc lock with a spinlock already held.  The
> interlock is really just a protection against drivers that don't
> adequately quiesce themselves, so I removed it for now until we can
> figure out something better.

The interlock to make sure the thread has terminated before the function 
returns.  This would be important if the taskqueue routine for this thread 
was in a kernel module that was being unloaded to avoid having a kernel page 
fault.  The solution for the LOR you saw is probably to just lock the proc 
lock earlier.  taskqueue_terminate() is not really a critical path operation 
(if it is that's a bug) so holding the proc lock a bit longer there won't 
kill.  As it is, there isn't a matching wakeup to ever resume the sleeper in 
taskqueue_terminate(), so the calling thread will block indefinitely (unless 
it is using a timeout).

-- 
John Baldwin <jhb at FreeBSD.org>  <><  http://www.FreeBSD.org/~jhb/
"Power Users Use the Power to Serve"  =  http://www.FreeBSD.org


More information about the cvs-all mailing list