1
0
mirror of https://gitlab.xiph.org/xiph/icecast-common.git synced 2025-01-03 14:56:36 -05:00
icecast-common/include/igloo/ro.h

425 lines
16 KiB
C

/* Copyright (C) 2018 Marvin Scholz <epirat07@gmail.com>
* Copyright (C) 2018-2019 Philipp "ph3-der-loewe" Schafft <lion@lion.leolix.org>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef _LIBIGLOO__RO_H_
#define _LIBIGLOO__RO_H_
#ifdef __cplusplus
extern "C" {
#endif
#include <igloo/config.h>
#ifdef IGLOO_CTC_HAVE_STDINT_H
#include <stdint.h>
#endif
#include <stdarg.h>
#include <igloo/config.h>
#include "types.h"
#include "thread.h"
/* Type used for callback called then the object is actually freed
* That is once all references to it are gone.
*
* This function must not try to deallocate or alter self.
*/
typedef void (*igloo_ro_free_t)(igloo_ro_t self);
/* Type used for callback called then the object is created
* using the generic igloo_ro_new().
*
* Additional parameters passed to igloo_ro_new() are passed
* in the list ap. All limitations of <stdarg.h> apply.
*
* This function must return zero in case of success and
* non-zero in case of error. In case of error igloo_ro_unref()
* is called internally to clear the object.
*/
typedef int (*igloo_ro_new_t)(igloo_ro_t self, const igloo_ro_type_t *type, va_list ap);
/* Type used to store flags for clone operations.
*/
#ifdef IGLOO_CTC_HAVE_STDINT_H
typedef uint_least32_t igloo_ro_cf_t;
#else
typedef unsigned long int igloo_ro_cf_t;
#endif
/* No clone flags set. Usefull for variable initialization. */
#define igloo_RO_CF_NONE ((igloo_ro_cf_t)0x0000)
/* Make a shallow copy of the object */
#define igloo_RO_CF_SHALLOW ((igloo_ro_cf_t)0x0001)
/* Make a deep copy of the object */
#define igloo_RO_CF_DEEP ((igloo_ro_cf_t)0x0002)
/* Make a copy of the object that shares part of it's state with the original.
* This is similar to using dup() on a POSIX filehandle.
* This is useful when interacting with IO.
*/
#define igloo_RO_CF_DUP ((igloo_ro_cf_t)0x0004)
/* Defaults to use if required and allowed is igloo_RO_CF_NONE */
#define igloo_RO_CF_DEFAULT (igloo_RO_CF_SHALLOW|igloo_RO_CF_DEEP)
/* Type uses for callback called when the object should be cloned.
*
* A clone of the object must never have the same address of the object that was cloned.
*
* Parameters:
* self
* The object to clone.
* required, allowed
* See igloo_ro_clone().
* name, associated
* See igloo_ro_new().
*/
typedef igloo_ro_t (*igloo_ro_clone_t)(igloo_ro_t self, igloo_ro_cf_t required, igloo_ro_cf_t allowed, const char *name, igloo_ro_t associated, igloo_ro_t instance);
/* Type used for callback called when the object needs to be converted to another type.
*
* This callback is not used when a object of the same type is requested.
* On that case clone is requested.
*
* There are two cases that must be handled by the callback:
* 0) The object self is of the same type as this callback is set for and type is pointing to different type.
* 1) The object self is of any type and type is pointing to the type this callback was set to.
*
* If the callback can not convert it must return igloo_RO_NULL.
*
* Parameters:
* self
* The object to convert.
* type
* The type to convert to.
* required, allowed
* See igloo_ro_clone().
* name, associated
* See igloo_ro_new().
*/
typedef igloo_ro_t (*igloo_ro_convert_t)(igloo_ro_t self, const igloo_ro_type_t *type, igloo_ro_cf_t required, igloo_ro_cf_t allowed, const char *name, igloo_ro_t associated, igloo_ro_t instance);
/* Type used for callback called when a specific interface to the object is requested.
*
* There are two cases that must be handled by the callback:
* 0) The object self is of the same type as this callback is set for and type is pointing to different type that is an interface.
* 1) The object self is of any type and type is pointing to the type this callback was set to which is an interface.
*
* This must not be used for converting to non-interface types. Use igloo_ro_convert() for that.
*
* Parameters:
* self
* The object to convert.
* type
* The type of the interface requested.
* name, associated
* See igloo_ro_new().
*/
typedef igloo_ro_t (*igloo_ro_get_interface_t)(igloo_ro_t self, const igloo_ro_type_t *type, const char *name, igloo_ro_t associated, igloo_ro_t instance);
/* Type used to store flags for stringify operation.
*/
#ifdef IGLOO_CTC_HAVE_STDINT_H
typedef uint_least32_t igloo_ro_sy_t;
#else
typedef unsigned long int igloo_ro_sy_t;
#endif
/* No stringify flags set. Usefull for variable initialization. */
#define igloo_RO_SY_NONE ((igloo_ro_sy_t)0x0000)
/* Stringify using defaults. */
#define igloo_RO_SY_DEFAULT ((igloo_ro_sy_t)0x1000)
/* Stringify the object itself, do not touch the content.
* When set together with igloo_RO_SY_CONTENT the libigloo will select the mode.
*/
#define igloo_RO_SY_OBJECT ((igloo_ro_sy_t)0x0001)
/* Stringify the object's content.
* When set together with igloo_RO_SY_OBJECT the libigloo will select the mode.
*/
#define igloo_RO_SY_CONTENT ((igloo_ro_sy_t)0x0002)
/* Type used for callback called when the object needs to be converted to a string.
*
* This is used mostly for debugging or preseting the object to the user.
* The callback is not expected to return a string that can be used to reconstruct the object.
*
* igloo_RO_SY_OBJECT is always set when this is called.
*/
typedef char * (*igloo_ro_stringify_t)(igloo_ro_t self, igloo_ro_sy_t flags);
/* Type used as a result of a compare between objects.
*/
typedef enum {
/* The objects could not be compared.
* Common cases may be that when one is NULL and the other is not NULL
* or when both are of difference types that can not be compared to each other
* (e.g. a IO handle and a buffer.)
*/
igloo_RO_CR__ERROR = -1,
/* Both objects represent the same thing but are not the same object.
* See also igloo_RO_CR_SAME.
*/
igloo_RO_CR_EQUAL = 0,
/* Both objects are distinct and no other relation can be applied. */
igloo_RO_CR_NOTEQUAL,
/* Object A is less than object B. */
igloo_RO_CR_ALESSTHANB,
/* Object A is greather than object B. */
igloo_RO_CR_AGREATERTHANB,
/* Both objects are the same.
* That is then they share the same address in memory.
*/
igloo_RO_CR_SAME
} igloo_ro_cr_t;
/* Type used for callback called then two objects must be compared.
*
* Object a is always a object that belongs to the type this callback was set for.
* Objects a, and b are not igloo_RO_NULL.
* Objects a, and b may be of diffrent type.
* Objects a, and b are not the same (See igloo_RO_CR_SAME).
*/
typedef igloo_ro_cr_t (*igloo_ro_compare_t)(igloo_ro_t a, igloo_ro_t b);
/* Type used for callback called when error value is requested from an object.
*/
typedef igloo_error_t (*igloo_ro_get_error_t)(igloo_ro_t self, igloo_error_t *result);
/* Meta type used to defined types.
* DO NOT use any of the members in here directly!
*/
/* ---[ PRIVATE ]--- */
/*
* Those types are defined here as they must be known to the compiler.
* Nobody should ever try to access them directly.
*/
struct igloo_ro_type_tag {
/* Size of this control structure */
size_t control_length;
/* ABI version of this structure */
int control_version;
/* Total length of the objects to be created */
size_t type_length;
/* Name of type */
const char * type_name;
/* STILL UNUSED: Parent type */
const igloo_ro_type_t * type_parent;
/* Callback to be called on final free() */
igloo_ro_free_t type_freecb;
/* Callback to be called by igloo_ro_new() */
igloo_ro_new_t type_newcb;
/* Callback to be called by igloo_ro_clone() */
igloo_ro_clone_t type_clonecb;
/* Callback to be called by igloo_ro_convert() */
igloo_ro_convert_t type_convertcb;
/* Callback to be called by igloo_ro_get_interface() */
igloo_ro_get_interface_t type_get_interfacecb;
/* Callback to be called by igloo_ro_stringify() */
igloo_ro_stringify_t type_stringifycb;
/* Callback to be called by igloo_ro_compare() */
igloo_ro_compare_t type_comparecb;
/* Callback to be called by igloo_ro_get_error() */
igloo_ro_get_error_t type_get_errorcb;
};
struct igloo_ro_base_tag {
/* Type of the object */
const igloo_ro_type_t * type;
/* Reference counters */
size_t refc;
size_t wrefc;
/* Mutex for igloo_ro_*(). */
igloo_mutex_t lock;
/* Name of the object. */
char * name;
/* Associated objects */
igloo_ro_t associated;
/* STILL UNUSED: Instance objects */
igloo_ro_t instance;
};
int igloo_ro_new__return_zero(igloo_ro_t self, const igloo_ro_type_t *type, va_list ap);
/* ---[ END PRIVATE ]--- */
igloo_ro_t igloo_RO_TO_TYPE_raw(igloo_ro_t object, const igloo_ro_type_t *type);
#define igloo_RO_GET_TYPE_BY_SYMBOL(x) (igloo_ro__type__ ## x)
#ifdef IGLOO_CTC_HAVE_TYPE_ATTRIBUTE_TRANSPARENT_UNION
#define igloo_RO__GETBASE(x) (((igloo_ro_t)(x)).subtype__igloo_ro_base_t)
#define igloo_RO_NULL ((igloo_ro_t)(igloo_ro_base_t*)NULL)
#define igloo_RO_IS_NULL(x) (igloo_RO__GETBASE((x)) == NULL)
#define igloo_RO_TO_TYPE(x,type) (((igloo_ro_t)igloo_RO_TO_TYPE_raw((x), igloo_RO_GET_TYPE_BY_SYMBOL(type))).subtype__ ## type)
#else
#define igloo_RO__GETBASE(x) ((igloo_ro_base_t*)(x))
#define igloo_RO_NULL NULL
#define igloo_RO_IS_NULL(x) ((x) == NULL)
#define igloo_RO_TO_TYPE(x,type) ((type*)igloo_RO_TO_TYPE_raw((x), igloo_RO_GET_TYPE_BY_SYMBOL(type)))
#endif
#define igloo_RO_GET_TYPE(x) (igloo_RO__GETBASE((x)) == NULL ? NULL : igloo_RO__GETBASE((x))->type)
#define igloo_RO_GET_TYPENAME(x) (igloo_RO_GET_TYPE((x)) == NULL ? NULL : igloo_RO_GET_TYPE((x))->type_name)
int igloo_RO_IS_VALID_raw(igloo_ro_t object, const igloo_ro_type_t *type);
#define igloo_RO_IS_VALID(x,type) igloo_RO_IS_VALID_raw((x), igloo_RO_GET_TYPE_BY_SYMBOL(type))
int igloo_RO_HAS_TYPE_raw(igloo_ro_t object, const igloo_ro_type_t *type);
#define igloo_RO_HAS_TYPE(x,type) igloo_RO_HAS_TYPE_raw((x), (type))
#define igloo_RO_IS_SAME(a,b) (igloo_RO__GETBASE((a)) == igloo_RO__GETBASE((b)))
/* Create a new refobject
* The type argument gives the type for the new object,
* the name for the object is given by name, and
* the associated refobject is given by associated.
*/
igloo_ro_t igloo_ro_new__raw(const igloo_ro_type_t *type, const char *name, igloo_ro_t associated, igloo_ro_t instance);
#define igloo_ro_new_raw(type, name, associated, instance) igloo_RO_TO_TYPE(igloo_ro_new__raw(igloo_RO_GET_TYPE_BY_SYMBOL(type), (name), (associated), (instance)), type)
igloo_ro_t igloo_ro_new__simple(const igloo_ro_type_t *type, const char *name, igloo_ro_t associated, igloo_ro_t instance, ...);
#define igloo_ro_new(type, ...) igloo_RO_TO_TYPE(igloo_ro_new__simple(igloo_RO_GET_TYPE_BY_SYMBOL(type), NULL, igloo_RO_NULL, igloo_RO_NULL, ## __VA_ARGS__), type)
#define igloo_ro_new_ext(type, name, associated, instance, ...) igloo_RO_TO_TYPE(igloo_ro_new__simple(igloo_RO_GET_TYPE_BY_SYMBOL(type), (name), (associated), (instance), ## __VA_ARGS__), type)
/* This increases the reference counter of the object */
igloo_error_t igloo_ro_ref(igloo_ro_t self);
/* This decreases the reference counter of the object.
* If the object's reference counter reaches zero the object is freed.
*/
igloo_error_t igloo_ro_unref(igloo_ro_t self);
/* This is the same as igloo_ro_ref() and igloo_ro_unref() but increases/decreases the weak reference counter. */
igloo_error_t igloo_ro_weak_ref(igloo_ro_t self);
igloo_error_t igloo_ro_weak_unref(igloo_ro_t self);
/* This gets the object's name */
const char * igloo_ro_get_name(igloo_ro_t self);
/* This gets the object's associated object. */
igloo_ro_t igloo_ro_get_associated(igloo_ro_t self);
igloo_error_t igloo_ro_set_associated(igloo_ro_t self, igloo_ro_t associated);
/* This gets the object's instance object. */
igloo_ro_t igloo_ro_get_instance(igloo_ro_t self);
/* Clone the given object returning a copy of it.
*
* This creates a copy of the passed object if possible.
* The mode of copy is selected by the parameters required and allowed.
* Both list flags of modes. Modes listed as required are adhered to.
* However the application may select any additional mode from the allowed
* flags.
*
* If both required and allowed are zero a set of default flags is applied
* as allowed flags.
*
* Parameters:
* self
* The object to copy.
* required
* Flags that must be adhered to when coping.
* allowed
* Flags that are allowed to be used.
* Flags that are listed as required are automatically added the list of allowed flags.
* name, associated
* See igloo_ro_new().
*/
igloo_ro_t igloo_ro_clone(igloo_ro_t self, igloo_ro_cf_t required, igloo_ro_cf_t allowed, const char *name, igloo_ro_t associated);
/* Make a copy of a object but converts it to another type as well.
*
* This makes makes a copy of the object that is also of a diffrent type.
* If the type is the same as the type of self igloo_ro_clone() is called.
*
* Parameters:
* self
* The object to copy and convert.
* type
* The type to convert to.
* required, allowed
* See igloo_ro_clone().
* name, associated
* See igloo_ro_new().
*/
igloo_ro_t igloo_ro_convert_ext(igloo_ro_t self, const igloo_ro_type_t *type, igloo_ro_cf_t required, igloo_ro_cf_t allowed, const char *name, igloo_ro_t associated);
igloo_ro_t igloo_ro_convert_simple(igloo_ro_t self, const igloo_ro_type_t *type, igloo_ro_cf_t required, igloo_ro_cf_t allowed);
#define igloo_ro_convert(self, type) igloo_RO_TO_TYPE(igloo_ro_convert_simple((self), igloo_RO_GET_TYPE_BY_SYMBOL(type), igloo_RO_CF_NONE, igloo_RO_CF_NONE), type)
/* Request a specific interface from the object.
*
* This must not be used to convert the object. This can only be used for requesting a specific interface.
*
* The object may return the same object for the same requested interface multiple times.
* However each time the same object is returned a new reference to it is returned.
*
* Parameters:
* self
* The object to copy and convert.
* type
* The interface requested.
* name, associated
* See igloo_ro_new().
* Returns:
* An object that represents the given interface or igloo_RO_NULL.
*/
igloo_ro_t igloo_ro_get_interface_ext(igloo_ro_t self, const igloo_ro_type_t *type, const char *name, igloo_ro_t associated);
#define igloo_ro_get_interface(self, type) igloo_RO_TO_TYPE(igloo_ro_get_interface_ext((self), igloo_RO_GET_TYPE_BY_SYMBOL(type), NULL, igloo_RO_NULL), type)
/* Convert a object to a string.
* This is used for debugging and presenting to the user.
*
* The resulting string can not be used to recreate the object.
*
* Parameters:
* self
* The object to convert to a string.
* flags
* Flags used to select options to the conversion.
* Should normally be igloo_RO_SY_DEFAULT.
* Returns:
* A string as allocated using malloc(3). The caller must call free(3).
*/
char * igloo_ro_stringify(igloo_ro_t self, igloo_ro_sy_t flags);
/* Compare two objects.
*
* Parameters:
* a, b
* The objects to compare.
* Returns:
* Thre result of the compare. See igloo_ro_cr_t.
*/
igloo_ro_cr_t igloo_ro_compare(igloo_ro_t a, igloo_ro_t b);
/* Get error value from a object.
*
* Parameters:
* self
* The Object to request error value from.
* result
* Pointer to the location the error value should be stored.
* The value is only written if igloo_ERROR_NONE is returned.
* Returns:
* The result of the query.
*/
igloo_error_t igloo_ro_get_error(igloo_ro_t self, igloo_error_t *result);
#ifdef __cplusplus
}
#endif
#endif