mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-23 21:19:09 +00:00
Add tags to stasis objects using the name. This makes it easier to track the source of certain stasis ref leaks. Review: https://reviewboard.asterisk.org/r/3821/ ........ Merged revisions 418996 from http://svn.asterisk.org/svn/asterisk/branches/12 git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@418997 65c4cc65-6c06-0410-ace0-fbb531ad65f3
202 lines
4.3 KiB
C
202 lines
4.3 KiB
C
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 2013, Digium, Inc.
|
|
*
|
|
* David M. Lee, II <dlee@digium.com>
|
|
*
|
|
* See http://www.asterisk.org for more information about
|
|
* the Asterisk project. Please do not directly contact
|
|
* any of the maintainers of this project for assistance;
|
|
* the project provides a web site, mailing lists and IRC
|
|
* channels for your use.
|
|
*
|
|
* This program is free software, distributed under the terms of
|
|
* the GNU General Public License Version 2. See the LICENSE file
|
|
* at the top of the source tree.
|
|
*/
|
|
|
|
/*! \file
|
|
*
|
|
* \brief Typical cache pattern for Stasis topics.
|
|
*
|
|
* \author David M. Lee, II <dlee@digium.com>
|
|
*/
|
|
|
|
/*** MODULEINFO
|
|
<support_level>core</support_level>
|
|
***/
|
|
|
|
#include "asterisk.h"
|
|
|
|
ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
|
|
|
|
#include "asterisk/astobj2.h"
|
|
#include "asterisk/stasis_cache_pattern.h"
|
|
|
|
struct stasis_cp_all {
|
|
struct stasis_topic *topic;
|
|
struct stasis_topic *topic_cached;
|
|
struct stasis_cache *cache;
|
|
|
|
struct stasis_forward *forward_all_to_cached;
|
|
};
|
|
|
|
struct stasis_cp_single {
|
|
struct stasis_topic *topic;
|
|
struct stasis_caching_topic *topic_cached;
|
|
|
|
struct stasis_forward *forward_topic_to_all;
|
|
struct stasis_forward *forward_cached_to_all;
|
|
};
|
|
|
|
static void all_dtor(void *obj)
|
|
{
|
|
struct stasis_cp_all *all = obj;
|
|
|
|
ao2_cleanup(all->topic);
|
|
all->topic = NULL;
|
|
ao2_cleanup(all->topic_cached);
|
|
all->topic_cached = NULL;
|
|
ao2_cleanup(all->cache);
|
|
all->cache = NULL;
|
|
stasis_forward_cancel(all->forward_all_to_cached);
|
|
all->forward_all_to_cached = NULL;
|
|
}
|
|
|
|
struct stasis_cp_all *stasis_cp_all_create(const char *name,
|
|
snapshot_get_id id_fn)
|
|
{
|
|
RAII_VAR(char *, cached_name, NULL, ast_free);
|
|
RAII_VAR(struct stasis_cp_all *, all, NULL, ao2_cleanup);
|
|
|
|
all = ao2_t_alloc(sizeof(*all), all_dtor, name);
|
|
if (!all) {
|
|
return NULL;
|
|
}
|
|
|
|
ast_asprintf(&cached_name, "%s-cached", name);
|
|
if (!cached_name) {
|
|
return NULL;
|
|
}
|
|
|
|
all->topic = stasis_topic_create(name);
|
|
all->topic_cached = stasis_topic_create(cached_name);
|
|
all->cache = stasis_cache_create(id_fn);
|
|
all->forward_all_to_cached =
|
|
stasis_forward_all(all->topic, all->topic_cached);
|
|
|
|
if (!all->topic || !all->topic_cached || !all->cache ||
|
|
!all->forward_all_to_cached) {
|
|
return NULL;
|
|
}
|
|
|
|
ao2_ref(all, +1);
|
|
return all;
|
|
}
|
|
|
|
struct stasis_topic *stasis_cp_all_topic(struct stasis_cp_all *all)
|
|
{
|
|
if (!all) {
|
|
return NULL;
|
|
}
|
|
return all->topic;
|
|
}
|
|
|
|
struct stasis_topic *stasis_cp_all_topic_cached(
|
|
struct stasis_cp_all *all)
|
|
{
|
|
if (!all) {
|
|
return NULL;
|
|
}
|
|
return all->topic_cached;
|
|
}
|
|
|
|
struct stasis_cache *stasis_cp_all_cache(struct stasis_cp_all *all)
|
|
{
|
|
if (!all) {
|
|
return NULL;
|
|
}
|
|
return all->cache;
|
|
}
|
|
|
|
static void one_dtor(void *obj)
|
|
{
|
|
struct stasis_cp_single *one = obj;
|
|
|
|
/* Should already be unsubscribed */
|
|
ast_assert(one->topic_cached == NULL);
|
|
ast_assert(one->forward_topic_to_all == NULL);
|
|
ast_assert(one->forward_cached_to_all == NULL);
|
|
|
|
ao2_cleanup(one->topic);
|
|
one->topic = NULL;
|
|
}
|
|
|
|
struct stasis_cp_single *stasis_cp_single_create(struct stasis_cp_all *all,
|
|
const char *name)
|
|
{
|
|
RAII_VAR(struct stasis_cp_single *, one, NULL, ao2_cleanup);
|
|
|
|
one = ao2_t_alloc(sizeof(*one), one_dtor, name);
|
|
if (!one) {
|
|
return NULL;
|
|
}
|
|
|
|
one->topic = stasis_topic_create(name);
|
|
if (!one->topic) {
|
|
return NULL;
|
|
}
|
|
one->topic_cached = stasis_caching_topic_create(one->topic, all->cache);
|
|
if (!one->topic_cached) {
|
|
return NULL;
|
|
}
|
|
|
|
one->forward_topic_to_all = stasis_forward_all(one->topic, all->topic);
|
|
if (!one->forward_topic_to_all) {
|
|
return NULL;
|
|
}
|
|
one->forward_cached_to_all = stasis_forward_all(
|
|
stasis_caching_get_topic(one->topic_cached), all->topic_cached);
|
|
if (!one->forward_cached_to_all) {
|
|
return NULL;
|
|
}
|
|
|
|
ao2_ref(one, +1);
|
|
return one;
|
|
}
|
|
|
|
void stasis_cp_single_unsubscribe(struct stasis_cp_single *one)
|
|
{
|
|
if (!one) {
|
|
return;
|
|
}
|
|
|
|
stasis_forward_cancel(one->forward_topic_to_all);
|
|
one->forward_topic_to_all = NULL;
|
|
stasis_forward_cancel(one->forward_cached_to_all);
|
|
one->forward_cached_to_all = NULL;
|
|
stasis_caching_unsubscribe(one->topic_cached);
|
|
one->topic_cached = NULL;
|
|
|
|
ao2_cleanup(one);
|
|
}
|
|
|
|
struct stasis_topic *stasis_cp_single_topic(struct stasis_cp_single *one)
|
|
{
|
|
if (!one) {
|
|
return NULL;
|
|
}
|
|
return one->topic;
|
|
}
|
|
|
|
struct stasis_topic *stasis_cp_single_topic_cached(
|
|
struct stasis_cp_single *one)
|
|
{
|
|
if (!one) {
|
|
return NULL;
|
|
}
|
|
return stasis_caching_get_topic(one->topic_cached);
|
|
}
|
|
|