diff options
author | Anand Avati <avati@redhat.com> | 2013-12-06 17:31:57 -0800 |
---|---|---|
committer | Pranith Kumar Karampuri <pkarampu@redhat.com> | 2016-02-16 21:00:50 -0800 |
commit | dcd1db157479bf9f9ac3c39fd3fd8cb0359c0779 (patch) | |
tree | 2c9d46677a127021f0228556f0149f9b4f4f247d /libglusterfs | |
parent | e424283c1f40386e5e3323b44df1a591ca62a7e8 (diff) |
timer: fix race between gf_timer_call_cancel() and gf_timer_proc()
>Change-Id: Ie264d3d591352e4a8ddaa90ae2174d9c552396f1
>BUG: 1243187
>Signed-off-by: Anand Avati <avati@redhat.com>
>Reviewed-on: http://review.gluster.org/6459
>Reviewed-by: Krutika Dhananjay <kdhananj@redhat.com>
>Reviewed-by: Poornima G <pgurusid@redhat.com>
>Tested-by: Gluster Build System <jenkins@build.gluster.com>
>Reviewed-by: Niels de Vos <ndevos@redhat.com>
>(cherry picked from commit ea90c92820ee0ca500345863cdfb5009d08b6ca7)
Change-Id: I2bc136e7d676826428afbf57f5afe50e2238fd33
BUG: 1246121
Signed-off-by: Anand Avati <avati@redhat.com>
Reviewed-on: http://review.gluster.org/11796
Reviewed-by: Pranith Kumar Karampuri <pkarampu@redhat.com>
Tested-by: Pranith Kumar Karampuri <pkarampu@redhat.com>
Smoke: Gluster Build System <jenkins@build.gluster.com>
CentOS-regression: Gluster Build System <jenkins@build.gluster.com>
Reviewed-by: Niels de Vos <ndevos@redhat.com>
NetBSD-regression: NetBSD Build System <jenkins@build.gluster.org>
Diffstat (limited to 'libglusterfs')
-rw-r--r-- | libglusterfs/src/timer.c | 53 | ||||
-rw-r--r-- | libglusterfs/src/timer.h | 2 |
2 files changed, 18 insertions, 37 deletions
diff --git a/libglusterfs/src/timer.c b/libglusterfs/src/timer.c index 2ffe6dff865..e55abface77 100644 --- a/libglusterfs/src/timer.c +++ b/libglusterfs/src/timer.c @@ -85,31 +85,11 @@ gf_timer_call_after (glusterfs_ctx_t *ctx, } int32_t -gf_timer_call_stale (gf_timer_registry_t *reg, - gf_timer_t *event) -{ - if (reg == NULL || event == NULL) - { - gf_msg_callingfn ("timer", GF_LOG_ERROR, EINVAL, - LG_MSG_INVALID_ARG, "invalid argument"); - return 0; - } - - event->next->prev = event->prev; - event->prev->next = event->next; - event->next = ®->stale; - event->prev = event->next->prev; - event->next->prev = event; - event->prev->next = event; - - return 0; -} - -int32_t gf_timer_call_cancel (glusterfs_ctx_t *ctx, gf_timer_t *event) { gf_timer_registry_t *reg = NULL; + gf_boolean_t fired = _gf_false; if (ctx == NULL || event == NULL) { @@ -128,13 +108,21 @@ gf_timer_call_cancel (glusterfs_ctx_t *ctx, pthread_mutex_lock (®->lock); { + fired = event->fired; + if (fired) + goto unlock; + event->next->prev = event->prev; event->prev->next = event->next; } +unlock: pthread_mutex_unlock (®->lock); - GF_FREE (event); - return 0; + if (!fired) { + GF_FREE (event); + return 0; + } + return -1; } static void __delete_entry (gf_timer_t *event) { @@ -181,7 +169,9 @@ gf_timer_proc (void *ctx) at = TS (event->at); if (event != ®->active && now >= at) { need_cbk = 1; - gf_timer_call_stale (reg, event); + event->next->prev = event->prev; + event->prev->next = event->next; + event->fired = _gf_true; } } pthread_mutex_unlock (®->lock); @@ -192,15 +182,13 @@ gf_timer_proc (void *ctx) THIS = event->xl; } event->callbk (event->data); - /*This callbk above would have freed the event - * by calling timer_cancel, don't ever touch it - * again*/ + GF_FREE (event); if (old_THIS) { THIS = old_THIS; } - } - else + } else { break; + } } nanosleep (&sleepts, NULL); } @@ -216,11 +204,6 @@ gf_timer_proc (void *ctx) * list_head*/ __delete_entry (event); } - - while (reg->stale.next != ®->stale) { - event = reg->stale.next; - __delete_entry (event); - } } pthread_mutex_unlock (®->lock); pthread_mutex_destroy (®->lock); @@ -249,8 +232,6 @@ gf_timer_registry_init (glusterfs_ctx_t *ctx) pthread_mutex_init (®->lock, NULL); reg->active.next = ®->active; reg->active.prev = ®->active; - reg->stale.next = ®->stale; - reg->stale.prev = ®->stale; ctx->timer = reg; gf_thread_create (®->th, NULL, gf_timer_proc, ctx); diff --git a/libglusterfs/src/timer.h b/libglusterfs/src/timer.h index e64b350ec58..35d99be143c 100644 --- a/libglusterfs/src/timer.h +++ b/libglusterfs/src/timer.h @@ -29,12 +29,12 @@ struct _gf_timer { gf_timer_cbk_t callbk; void *data; xlator_t *xl; + gf_boolean_t fired; }; struct _gf_timer_registry { pthread_t th; char fin; - struct _gf_timer stale; struct _gf_timer active; pthread_mutex_t lock; }; |