Skip site navigation (1)Skip section navigation (2)
Date:      Fri, 04 Jul 2014 11:03:51 -0700
From:      Alfred Perlstein <alfred@freebsd.org>
To:        arch@FreeBSD.org
Subject:   Re: callout(9) really this complicated?
Message-ID:  <53B6EC87.1070700@freebsd.org>
In-Reply-To: <20140704041521.GW45513@funkthat.com>
References:  <20140704041521.GW45513@funkthat.com>

next in thread | previous in thread | raw e-mail | index | archive | help
On 7/3/14 9:15 PM, John-Mark Gurney wrote:
> So, I was going to look at using callout(9) for some time delayed
> actions...  But upon reading the docs, a) the docs are inconsistent,
> and b) the docs only talk about requirements in other section...
>
> Is there a better interface?  If so, can we mark callout(9) deprecated?
> If not, I have some questions...
>
> If you want callout_drain to work properly, you have to add extra code
> to both your callout, and around the usage of it...
>
> callout_drain does not drain the callout:
>       However, the callout subsystem does guarantee that the callout will be
>       fully stopped before callout_drain() returns.
>
> Yet other parse of the docs say that you can depend upon the callout
> being fully stopped..  I've sent email to Ian (iedowse) about why he
> added this statement...
>
> Second, the amount of work you have to do to make sure you drain
> seems pretty crazy as documented in Avoiding Race Conditions...
>
> It seems like if I have created a callout w/ callout_init_mtx,
> that I shouldn't have to do extra work to make it work correctly...
>
> When calling _callout_stop_safe as callout_drain, there is no assert
> that the lock is held, though it is documented as requiring it by:
>       The function callout_drain() is identical to callout_stop() except that
>       it will wait for the callout to be completed if it is already in
>       progress.
>
> Maybe we need to add an additional statement here?  and assert that it
> isn't locked?
>
> Also, I have tried to follow the code, but it's complicated, so I'm
> hoping that I can get some help here.
>
> Thanks.
>
This isn't so bad, basically just think about there being another thread 
launching threads to service a routine, the logic to stop/drain is 
moderately complex but given a bit more time it should be relatively 
obvious what is needed to be done.

As far as the docs regarding callout_drain, that makes sense, I think 
the docs are a bit overly verbose (perhaps someone was drinking too much 
coffee).  In other words, with regards to callout_drain, what the docs 
mean is that when callout_drain() returns, there should be no further 
calls to the callout.  Nothing more is meant.

The important trick is NOT to hold a mutex or other lock during 
callout_drain() that the callout itself uses.  This is relatively 
straightforward.

-Alfred



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?53B6EC87.1070700>