[ltt-dev] [PATCH 2/7] urcu, defer_rcu: fix missing respond to a cancellation request
Mathieu Desnoyers
mathieu.desnoyers at efficios.com
Thu Sep 29 13:13:36 EDT 2011
* Lai Jiangshan (laijs at cn.fujitsu.com) wrote:
> thread1: thread2
> thr_defer()
> pthread_testcancel();
> stop_defer_thread() .
> pthread_cancel(tid_defer); .
> wake_up_defer(); .
> wait_defer();
> pthread_join(tid_defer)
>
> thread2 missing respond to the cancellation request from thread1
> and wait callback forever.
>
> thread1 wait thread2 forver by pthread_join():
>
> Signed-off-by: Lai Jiangshan <laijs at cn.fujitsu.com>
> ---
> urcu-defer-impl.h | 3 ++-
> 1 files changed, 2 insertions(+), 1 deletions(-)
>
> diff --git a/urcu-defer-impl.h b/urcu-defer-impl.h
> index d1ab046..6339747 100644
> --- a/urcu-defer-impl.h
> +++ b/urcu-defer-impl.h
> @@ -187,6 +187,7 @@ static void wait_defer(void)
> {
> uatomic_dec(&defer_thread_futex);
> cmm_smp_mb(); /* Write futex before read queue */
> + pthread_testcancel();
> if (rcu_defer_num_callbacks()) {
> cmm_smp_mb(); /* Read queue before write futex */
> /* Callbacks are queued, don't wait. */
> @@ -359,7 +360,6 @@ void _defer_rcu(void (*fct)(void *p), void *p)
> void *thr_defer(void *args)
> {
> for (;;) {
> - pthread_testcancel();
> /*
> * "Be green". Don't wake up the CPU if there is no RCU work
> * to perform whatsoever. Aims at saving laptop battery life by
Shouldn't we just modify futex_noasync and futex_async to add a
pthread_testcancel there instead ? We should then document, in futex.h,
that our futex implementations are pthread cancellation points.
> @@ -387,6 +387,7 @@ static void start_defer_thread(void)
> {
> int ret;
>
> + uatomic_set(&defer_thread_futex, 0);
this would be to handle a start/stop/start sequence, right ? If we put
the testcancel within the region where the defer thread can have the
futex value set to non-zero, I guess setting it back to zero upon
restart will be required. Instead of setting it back to 0 in the
"start", we could theoretically let stop_defer_thread() do it, because
the first "start" will already have its value set to 0. We have to
notice that this whole rcu_defer scheme does not deal well with fork()
that are not followed by exec() though. We should probably just
document this and not care about it: call_rcu() is a superset of
defer_rcu anyway, and we should encourage use of call_rcu.
Thanks,
Mathieu
> ret = pthread_create(&tid_defer, NULL, thr_defer, NULL);
> assert(!ret);
> }
> --
> 1.7.4.4
>
--
Mathieu Desnoyers
Operating System Efficiency R&D Consultant
EfficiOS Inc.
http://www.efficios.com
More information about the lttng-dev
mailing list