forked from meetecho/janus-gateway
-
Notifications
You must be signed in to change notification settings - Fork 1
/
refcount.h
225 lines (210 loc) · 8.2 KB
/
refcount.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
/*! \file refcount.h
* \author Lorenzo Miniero <[email protected]>
* \copyright GNU General Public License v3
* \brief Reference counter mechanism
* \details Implementation of a simple reference counter that can be
* used to keep track of memory management in Janus, in order to avoid
* the need for timed garbage collectord and the like which have proven
* ineffective in the past (e.g., crashes whenever race conditions
* occurred). This implementation is heavily based on an excellent
* <a href="http://nullprogram.com/blog/2015/02/17/">blog post</a>
* written by Chris Wellons.
*
* Objects interested in leveraging this reference counter mechanism
* must add a janus_refcount instance as one of the members of the object
* itself, and then call janus_refcall_init() to set it up. Initializing
* the reference counter just needs a pointer to the function to invoke
* when the object needs to be destroyed (counter reaches 0), while it
* will automatically set the counter to 1. To increase and decrease the
* counter just call janus_refcount_increase() and janus_refcount_decrease().
* When the counter reaches 0, the function passed when initializing it will
* be invoked: this means it's up to you to then free all the resources
* the object may have allocated. Notice that if this involves other
* objects that are reference counted, freeing the resource will just
* mean decreasing the related counter, and not destroying it right away.
*
* The free function must be defined like this:
*
\verbatim
void my_free_function(janus_refcount *counter);
\endverbatim
*
* Since the reference counter cannot know the size of the object to be
* freed, or where in the list of members the counter has been placed,
* retrieving the pointer to the object to free is up to you, using the
* janus_refcount_containerof macro. This is an example of how the
* free function we have defined above may be implemented:
*
\verbatim
typedef my_struct {
int number;
char *string;
janus_refcount myref;
}
void my_free_function(janus_refcount *counter) {
struct my_struct *my_object = janus_refcount_containerof(counter, struct my_struct, myref);
if(my_object->string)
free(my_object->string);
free(my_object);
}
\endverbatim
*
* \ingroup core
* \ref core
*/
#ifndef JANUS_REFCOUNT_H
#define JANUS_REFCOUNT_H
#include <glib.h>
#include "mutex.h"
//~ #define REFCOUNT_DEBUG
extern int refcount_debug;
/*! \brief Macro to programmatically address the object itself from its counter
* \details \c refptr is the pointer to the janus_refcount instance, \c type
* is the type of the object itself (e.g., <code>struct mystruct</code>),
* while \c member is how the janus_refcount instance is called in the
* object that contains it. */
#define janus_refcount_containerof(refptr, type, member) \
((type *)((char *)(refptr) - offsetof(type, member)))
/*! \brief Janus reference counter structure */
typedef struct janus_refcount janus_refcount;
struct janus_refcount {
/*! \brief The reference counter itself */
gint count;
/*! \brief Pointer to the function that will be used to free the object */
void (*free)(const janus_refcount *);
};
#ifdef REFCOUNT_DEBUG
/* Reference counters debugging */
extern GHashTable *counters;
extern janus_mutex counters_mutex;
#define janus_refcount_track(refp) { \
janus_mutex_lock(&counters_mutex); \
if(counters == NULL) \
counters = g_hash_table_new(NULL, NULL); \
g_hash_table_insert(counters, refp, refp); \
janus_mutex_unlock(&counters_mutex); \
}
#define janus_refcount_untrack(refp) { \
janus_mutex_lock(&counters_mutex); \
g_hash_table_remove(counters, refp); \
janus_mutex_unlock(&counters_mutex); \
}
#endif
/*! \brief Janus reference counter initialization (debug according to settings)
* \note Also sets the counter to 1 automatically, so no need to increase
* it again manually via janus_refcount_increase() after the initialization
* @param refp Pointer to the Janus reference counter instance
* @param free_fn Pointer to the function to invoke when the object the counter
* refers to needs to be destroyed */
#define janus_refcount_init(refp, free_fn) { \
if(!refcount_debug) { \
janus_refcount_init_nodebug(refp, free_fn); \
} else { \
janus_refcount_init_debug(refp, free_fn); \
} \
}
/*! \brief Janus reference counter initialization (no debug)
* \note Also sets the counter to 1 automatically, so no need to increase
* it again manually via janus_refcount_increase() after the initialization
* @param refp Pointer to the Janus reference counter instance
* @param free_fn Pointer to the function to invoke when the object the counter
* refers to needs to be destroyed */
#ifdef REFCOUNT_DEBUG
#define janus_refcount_init_nodebug(refp, free_fn) { \
(refp)->count = 1; \
(refp)->free = free_fn; \
janus_refcount_track((refp)); \
}
#else
#define janus_refcount_init_nodebug(refp, free_fn) { \
(refp)->count = 1; \
(refp)->free = free_fn; \
}
#endif
/*! \brief Janus reference counter initialization (debug)
* \note Also sets the counter to 1 automatically, so no need to increase
* it again manually via janus_refcount_increase() after the initialization
* @param refp Pointer to the Janus reference counter instance
* @param free_fn Pointer to the function to invoke when the object the counter
* refers to needs to be destroyed */
#ifdef REFCOUNT_DEBUG
#define janus_refcount_init_debug(refp, free_fn) { \
(refp)->count = 1; \
JANUS_PRINT("[%s:%s:%d:init] %p (%d)\n", __FILE__, __FUNCTION__, __LINE__, refp, (refp)->count); \
(refp)->free = free_fn; \
janus_refcount_track((refp)); \
}
#else
#define janus_refcount_init_debug(refp, free_fn) { \
(refp)->count = 1; \
JANUS_PRINT("[%s:%s:%d:init] %p (%d)\n", __FILE__, __FUNCTION__, __LINE__, refp, (refp)->count); \
(refp)->free = free_fn; \
}
#endif
/*! \brief Increase the Janus reference counter (debug according to settings)
* @param refp Pointer to the Janus reference counter instance */
#define janus_refcount_increase(refp) { \
if(!refcount_debug) { \
janus_refcount_increase_nodebug(refp); \
} else { \
janus_refcount_increase_debug(refp); \
} \
}
/*! \brief Increase the Janus reference counter (no debug)
* @param refp Pointer to the Janus reference counter instance */
#define janus_refcount_increase_nodebug(refp) { \
g_atomic_int_inc((gint *)&(refp)->count); \
}
/*! \brief Increase the Janus reference counter (debug)
* @param refp Pointer to the Janus reference counter instance */
#define janus_refcount_increase_debug(refp) { \
JANUS_PRINT("[%s:%s:%d:increase] %p (%d)\n", __FILE__, __FUNCTION__, __LINE__, refp, (refp)->count+1); \
g_atomic_int_inc((gint *)&(refp)->count); \
}
/*! \brief Decrease the Janus reference counter (debug according to settings)
* \note Will invoke the \c free function if the counter reaches 0
* @param refp Pointer to the Janus reference counter instance */
#define janus_refcount_decrease(refp) { \
if(!refcount_debug) { \
janus_refcount_decrease_nodebug(refp); \
} else { \
janus_refcount_decrease_debug(refp); \
} \
}
/*! \brief Decrease the Janus reference counter (debug)
* \note Will invoke the \c free function if the counter reaches 0
* @param refp Pointer to the Janus reference counter instance */
#ifdef REFCOUNT_DEBUG
#define janus_refcount_decrease_debug(refp) { \
JANUS_PRINT("[%s:%s:%d:decrease] %p (%d)\n", __FILE__, __FUNCTION__, __LINE__, refp, (refp)->count-1); \
if(g_atomic_int_dec_and_test((gint *)&(refp)->count)) { \
(refp)->free(refp); \
janus_refcount_untrack((refp)); \
} \
}
#else
#define janus_refcount_decrease_debug(refp) { \
JANUS_PRINT("[%s:%s:%d:decrease] %p (%d)\n", __FILE__, __FUNCTION__, __LINE__, refp, (refp)->count-1); \
if(g_atomic_int_dec_and_test((gint *)&(refp)->count)) { \
(refp)->free(refp); \
} \
}
#endif
/*! \brief Decrease the Janus reference counter (no debug)
* \note Will invoke the \c free function if the counter reaches 0
* @param refp Pointer to the Janus reference counter instance */
#ifdef REFCOUNT_DEBUG
#define janus_refcount_decrease_nodebug(refp) { \
if(g_atomic_int_dec_and_test((gint *)&(refp)->count)) { \
(refp)->free(refp); \
janus_refcount_untrack((refp)); \
} \
}
#else
#define janus_refcount_decrease_nodebug(refp) { \
if(g_atomic_int_dec_and_test((gint *)&(refp)->count)) { \
(refp)->free(refp); \
} \
}
#endif
#endif