summaryrefslogtreecommitdiff
path: root/res/res_sorcery_memory_cache.c
diff options
context:
space:
mode:
authorRichard Mudgett <rmudgett@digium.com>2015-09-30 17:28:19 -0500
committerRichard Mudgett <rmudgett@digium.com>2015-10-01 17:25:37 -0500
commit40c69e78f571781b67a240554c119b870e3cd6d4 (patch)
tree53c0bdc98119d06031b3dcf66c14ab80dca518f8 /res/res_sorcery_memory_cache.c
parentdfeb513e85d13550d81b40df5e95333c1ad5c61c (diff)
res_sorcery_memory_cache.c: Fix deadlock with scheduler.
A deadlock can happen when a sorcery object is being expired from the memory cache when at the same time another object is being placed into the memory cache. There are a couple other variations on this theme that could cause the deadlock. Basically if an object is being expired from the sorcery memory cache at the same time as another thread tries to update the next object expiration timer the deadlock can happen. * Add a deadlock avoidance loop in expire_objects_from_cache() to check if someone is trying to remove the scheduler callback from the scheduler. ASTERISK-25441 #close Change-Id: Iec7b0bdb81a72b39477727b1535b2539ad0cf4dc
Diffstat (limited to 'res/res_sorcery_memory_cache.c')
-rw-r--r--res/res_sorcery_memory_cache.c35
1 files changed, 22 insertions, 13 deletions
diff --git a/res/res_sorcery_memory_cache.c b/res/res_sorcery_memory_cache.c
index 7c7d5708e..0ce0e33a1 100644
--- a/res/res_sorcery_memory_cache.c
+++ b/res/res_sorcery_memory_cache.c
@@ -122,6 +122,8 @@ struct sorcery_memory_cache {
struct ast_heap *object_heap;
/*! \brief Scheduler item for expiring oldest object. */
int expire_id;
+ /*! TRUE if trying to stop the oldest object expiration scheduler item. */
+ unsigned int del_expire:1;
#ifdef TEST_FRAMEWORK
/*! \brief Variable used to indicate we should notify a test when we reach empty */
unsigned int cache_notify;
@@ -441,7 +443,21 @@ static int expire_objects_from_cache(const void *data)
struct sorcery_memory_cache *cache = (struct sorcery_memory_cache *)data;
struct sorcery_memory_cached_object *cached;
- ao2_wrlock(cache->objects);
+ /*
+ * We need to do deadlock avoidance between a non-scheduler thread
+ * blocking when trying to delete the scheduled entry for this
+ * callback because the scheduler thread is running this callback
+ * and this callback waiting for the cache->objects container lock
+ * that the blocked non-scheduler thread already holds.
+ */
+ while (ao2_trywrlock(cache->objects)) {
+ if (cache->del_expire) {
+ cache->expire_id = -1;
+ ao2_ref(cache, -1);
+ return 0;
+ }
+ sched_yield();
+ }
cache->expire_id = -1;
@@ -486,7 +502,9 @@ static void remove_all_from_cache(struct sorcery_memory_cache *cache)
ao2_callback(cache->objects, OBJ_UNLINK | OBJ_NOLOCK | OBJ_NODATA | OBJ_MULTIPLE,
NULL, NULL);
+ cache->del_expire = 1;
AST_SCHED_DEL_UNREF(sched, cache->expire_id, ao2_ref(cache, -1));
+ cache->del_expire = 0;
}
/*!
@@ -579,18 +597,9 @@ static int schedule_cache_expiration(struct sorcery_memory_cache *cache)
return 0;
}
- if (cache->expire_id != -1) {
- /* If we can't unschedule this expiration then it is currently attempting to run,
- * so let it run - it just means that it'll be the one scheduling instead of us.
- */
- if (ast_sched_del(sched, cache->expire_id)) {
- return 0;
- }
-
- /* Since it successfully cancelled we need to drop the ref to the cache it had */
- ao2_ref(cache, -1);
- cache->expire_id = -1;
- }
+ cache->del_expire = 1;
+ AST_SCHED_DEL_UNREF(sched, cache->expire_id, ao2_ref(cache, -1));
+ cache->del_expire = 0;
cached = ast_heap_peek(cache->object_heap, 1);
if (!cached) {