mirror of
https://gitlab.xiph.org/xiph/icecast-common.git
synced 2025-01-03 14:56:36 -05:00
Feature: Implemented refobject API
This commit is contained in:
parent
904ddf1c39
commit
2bee2ee3a4
@ -61,6 +61,7 @@ struct igloo_ro_type_tag {
|
|||||||
/* Size of this control structure */
|
/* Size of this control structure */
|
||||||
size_t control_length;
|
size_t control_length;
|
||||||
/* ABI version of this structure */
|
/* ABI version of this structure */
|
||||||
|
#define igloo_RO__CONTROL_VERSION 1
|
||||||
int control_version;
|
int control_version;
|
||||||
|
|
||||||
/* Total length of the objects to be created */
|
/* Total length of the objects to be created */
|
||||||
|
185
src/ro.c
185
src/ro.c
@ -18,7 +18,190 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
#include <igloo/ro.h>
|
#include <igloo/ro.h>
|
||||||
|
|
||||||
|
static inline int check_type(const igloo_ro_type_t *type)
|
||||||
|
{
|
||||||
|
return type->control_length == sizeof(igloo_ro_type_t) && type->control_version == igloo_RO__CONTROL_VERSION &&
|
||||||
|
type->type_length >= sizeof(igloo_ro_base_t);
|
||||||
|
}
|
||||||
|
|
||||||
|
igloo_ro_t igloo_ro_new__raw(const igloo_ro_type_t *type, const char *name, igloo_ro_t associated)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base;
|
||||||
|
|
||||||
|
if (!check_type(type))
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
base = calloc(1, type->type_length);
|
||||||
|
if (!base)
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
base->type = type;
|
||||||
|
base->refc = 1;
|
||||||
|
|
||||||
|
igloo_thread_mutex_create(&(base->lock));
|
||||||
|
|
||||||
|
if (name) {
|
||||||
|
base->name = strdup(name);
|
||||||
|
if (!base->name) {
|
||||||
|
igloo_ro_unref(base);
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!igloo_RO_IS_NULL(associated)) {
|
||||||
|
if (igloo_ro_ref(associated) != 0) {
|
||||||
|
igloo_ro_unref(base);
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
base->associated = associated;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (igloo_ro_t)base;
|
||||||
|
}
|
||||||
|
|
||||||
|
igloo_ro_t igloo_ro_new__simple(const igloo_ro_type_t *type, const char *name, igloo_ro_t associated, ...)
|
||||||
|
{
|
||||||
|
igloo_ro_t ret;
|
||||||
|
int res;
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
if (!check_type(type))
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
if (!type->type_newcb)
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
ret = igloo_ro_new__raw(type, name, associated);
|
||||||
|
if (igloo_RO_IS_NULL(ret))
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
va_start(ap, associated);
|
||||||
|
res = type->type_newcb(ret, type, ap);
|
||||||
|
va_end(ap);
|
||||||
|
|
||||||
|
if (res != 0) {
|
||||||
|
igloo_ro_unref(ret);
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int igloo_ro_ref(igloo_ro_t self)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base = igloo_RO__GETBASE(self);
|
||||||
|
|
||||||
|
if (!base)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
igloo_thread_mutex_lock(&(base->lock));
|
||||||
|
base->refc++;
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int igloo_ro_unref(igloo_ro_t self)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base = igloo_RO__GETBASE(self);
|
||||||
|
|
||||||
|
if (!base)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
igloo_thread_mutex_lock(&(base->lock));
|
||||||
|
base->refc--;
|
||||||
|
|
||||||
|
if (base->refc) {
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (base->type->type_freecb)
|
||||||
|
base->type->type_freecb(self);
|
||||||
|
|
||||||
|
igloo_ro_unref(base->associated);
|
||||||
|
|
||||||
|
if (base->name)
|
||||||
|
free(base->name);
|
||||||
|
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
igloo_thread_mutex_destroy(&(base->lock));
|
||||||
|
|
||||||
|
free(base);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char * igloo_ro_get_name(igloo_ro_t self)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base = igloo_RO__GETBASE(self);
|
||||||
|
const char *ret;
|
||||||
|
|
||||||
|
if (!base)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
igloo_thread_mutex_lock(&(base->lock));
|
||||||
|
ret = base->name;
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
igloo_ro_t igloo_ro_get_associated(igloo_ro_t self)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base = igloo_RO__GETBASE(self);
|
||||||
|
igloo_ro_t ret;
|
||||||
|
|
||||||
|
if (!base)
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
|
||||||
|
igloo_thread_mutex_lock(&(base->lock));
|
||||||
|
ret = base->associated;
|
||||||
|
if (!igloo_RO_IS_NULL(ret)) {
|
||||||
|
if (igloo_ro_ref(ret) != 0) {
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
return igloo_RO_NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int igloo_ro_set_associated(igloo_ro_t self, igloo_ro_t associated)
|
||||||
|
{
|
||||||
|
igloo_ro_base_t *base = igloo_RO__GETBASE(self);
|
||||||
|
igloo_ro_t old;
|
||||||
|
|
||||||
|
if (!base)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* We can not set ourself to be our associated. */
|
||||||
|
if (base == igloo_RO__GETBASE(associated))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (!igloo_RO_IS_NULL(associated)) {
|
||||||
|
if (igloo_ro_ref(associated) != 0) {
|
||||||
|
/* Could not get a reference on the new associated object. */
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
igloo_thread_mutex_lock(&(base->lock));
|
||||||
|
old = base->associated;
|
||||||
|
base->associated = associated;
|
||||||
|
igloo_thread_mutex_unlock(&(base->lock));
|
||||||
|
|
||||||
|
igloo_ro_unref(old);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user