/***************************************************************************** * vlc_objects.h: vlc_object_t definition and manipulation methods ***************************************************************************** * Copyright (C) 2002-2008 VLC authors and VideoLAN * * Authors: Samuel Hocevar * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2.1 of the License, or * (at your option) any later version. * * This program 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/ #ifndef VLC_OBJECTS_H #define VLC_OBJECTS_H 1 #ifdef __cplusplus #include extern "C" { #endif /** * \defgroup vlc_object VLC objects * \ingroup vlc * @{ * \file * Common VLC object definitions */ struct vlc_logger; struct vlc_tracer; struct vlc_object_internals; struct vlc_object_marker; /** * VLC object common members * * Common public properties for all VLC objects. * Object also have private properties maintained by the core, see * \ref vlc_object_internals_t */ struct vlc_object_t { struct vlc_logger *logger; union { struct vlc_object_internals *priv; struct vlc_object_marker *obj; }; bool no_interact; /** Module probe flag * * A boolean during module probing when the probe is "forced". * See \ref module_need(). */ bool force; }; #ifdef __cplusplus } #endif /** * Type-safe vlc_object_t cast * * This macro attempts to cast a pointer to a compound type to a * \ref vlc_object_t pointer in a type-safe manner. * It checks if the compound type actually starts with an embedded * \ref vlc_object_t structure. */ #if !defined(__cplusplus) # define VLC_OBJECT(x) \ _Generic((x)->obj, \ struct vlc_object_marker *: (x), \ struct vlc_object_t: (&((x)->obj)) \ ) #else static inline vlc_object_t *VLC_OBJECT(vlc_object_t *o) { return o; } template static inline vlc_object_t *VLC_OBJECT(T *d) { return &d->obj; } #endif #ifdef __cplusplus extern "C" { #endif /* The root object */ struct libvlc_int_t { struct vlc_object_t obj; }; /** * Allocates and initializes a vlc object. * * The object will need to be released with ::vlc_object_release() * before \p parent is released. * * @param parent A parent object to create the new object from * @param i_size object byte size * * @return the new object, or NULL on error. */ VLC_API void *vlc_object_create(vlc_object_t *parent, size_t i_size) VLC_MALLOC VLC_USED; /** * Drops the strong reference to an object. * * This removes the initial strong reference to a given object. This must be * called exactly once per allocated object after it is no longer needed, * matching vlc_object_create() or vlc_custom_create(). */ VLC_API void vlc_object_delete(vlc_object_t *obj); #define vlc_object_delete(obj) vlc_object_delete(VLC_OBJECT(obj)) VLC_API size_t vlc_list_children(vlc_object_t *, vlc_object_t **, size_t) VLC_USED; /** * Returns the object type name. * * This returns a nul-terminated string identifying the object type. * The string is valid for at least as long as the object reference. * * \param obj object whose type name to get */ VLC_API const char *vlc_object_typename(const vlc_object_t *obj) VLC_USED; /** * Gets the parent of an object. * * \return the parent object (NULL if none) * * \note The returned parent object pointer is valid as long as the child is. */ VLC_API vlc_object_t *vlc_object_parent(vlc_object_t *obj) VLC_USED; #define vlc_object_parent(o) vlc_object_parent(VLC_OBJECT(o)) static inline struct vlc_logger *vlc_object_logger(vlc_object_t *obj) { return obj->logger; } #define vlc_object_logger(o) vlc_object_logger(VLC_OBJECT(o)) /** * Get tracer of a vlc instance from an object. * * \return the tracer of a vlc instance from an object (NULL if none). */ VLC_API struct vlc_tracer *vlc_object_get_tracer(vlc_object_t *obj); /** * Tries to get the name of module bound to an object. * * \warning This function is intrinsically race-prone, as a module may be * bound or unbound asynchronously by another thread. * Do not trust the result for any purpose other than debugging/tracing. * * \return Normally, this returns a heap-allocated nul-terminated string * which is the name of the module. If no module are bound to the object, it * returns NULL. It also returns NULL on error. */ #define vlc_object_get_name(obj) var_GetString(obj, "module-name") #define vlc_object_create(a,b) vlc_object_create( VLC_OBJECT(a), b ) #define vlc_object_find_name(a,b) \ vlc_object_find_name( VLC_OBJECT(a),b) VLC_USED static inline libvlc_int_t *vlc_object_instance(vlc_object_t *obj) { vlc_object_t *parent; do parent = obj; while ((obj = vlc_object_parent(obj)) != NULL); return (libvlc_int_t *)parent; } #define vlc_object_instance(o) vlc_object_instance(VLC_OBJECT(o)) /* Here for backward compatibility. TODO: Move to ! */ VLC_API vout_thread_t *vout_Hold(vout_thread_t *vout); VLC_API void vout_Release(vout_thread_t *vout); /* Here for backward compatibility. TODO: Move to ! */ VLC_API audio_output_t *aout_Hold(audio_output_t *aout); VLC_API void aout_Release(audio_output_t *aout); /* TODO: remove vlc_object_hold/_release() for GUIs, remove this */ VLC_DEPRECATED static inline void *vlc_object_hold(vlc_object_t *o) { const char *tn = vlc_object_typename(o); if (!strcmp(tn, "audio output")) aout_Hold((audio_output_t *)o); if (!strcmp(tn, "video output")) vout_Hold((vout_thread_t *)o); return o; } static inline void vlc_object_release(vlc_object_t *o) { const char *tn = vlc_object_typename(o); if (!strcmp(tn, "audio output")) aout_Release((audio_output_t *)o); if (!strcmp(tn, "video output")) vout_Release((vout_thread_t *)o); } #define vlc_object_release(o) vlc_object_release(VLC_OBJECT(o)) /** * @defgroup objres Object resources * * The object resource functions tie resource allocation to an instance of * a module through a VLC object. * Such resource will be automatically freed, in first in last out order, * when the module instance associated with the VLC object is terminated. * * Specifically, if the module instance activation/probe function fails, the * resource will be freed immediately after the failure. If the activation * succeeds, the resource will be freed when the module instance is terminated. * * This is a convenience mechanism to save explicit clean-up function calls * in modules. * * @{ */ /** * Allocates memory for a module. * * This function allocates memory from the heap for a module instance. * The memory is uninitialized. * * @param obj VLC object to tie the memory allocation to * @param size byte size of the memory allocation * * @return a pointer to the allocated memory, or NULL on error (errno is set). */ VLC_API VLC_MALLOC void *vlc_obj_malloc(vlc_object_t *obj, size_t size); #define vlc_obj_malloc(o, s) vlc_obj_malloc(VLC_OBJECT(o), s) /** * Allocates a zero-initialized table for a module. * * This function allocates a table from the heap for a module instance. * The memory is initialized to all zeroes. * * @param obj VLC object to tie the memory allocation to * @param nmemb number of table entries * @param size byte size of a table entry * * @return a pointer to the allocated memory, or NULL on error (errno is set). */ VLC_API VLC_MALLOC void *vlc_obj_calloc(vlc_object_t *obj, size_t nmemb, size_t size); #define vlc_obj_calloc(o, n, s) vlc_obj_calloc(VLC_OBJECT(o), n, s) /** * Duplicates a string for a module. * * This function allocates a copy of a nul-terminated string for a module * instance. * * @param obj VLC object to tie the memory allocation to * @param str string to copy * * @return a pointer to the copy, or NULL on error (errno is set). */ VLC_API VLC_MALLOC char *vlc_obj_strdup(vlc_object_t *obj, const char *str); #define vlc_obj_strdup(o, s) vlc_obj_strdup(VLC_OBJECT(o), s) /** * Manually frees module memory. * * This function manually frees a resource allocated with vlc_obj_malloc(), * vlc_obj_calloc() or vlc_obj_strdup() before the module instance is * terminated. This is seldom necessary. * * @param obj VLC object that the allocation was tied to * @param ptr pointer to the allocated resource */ VLC_API void vlc_obj_free(vlc_object_t *obj, void *ptr); #define vlc_obj_free(o, p) vlc_obj_free(VLC_OBJECT(o), p) #ifdef __cplusplus } /* extern "C" */ # include #undef vlc_object_create template VLC_MALLOC VLC_USED static inline void* vlc_object_create(O *obj, size_t size) { return vlc_object_create(VLC_OBJECT(obj), size); } template VLC_MALLOC VLC_USED static inline T* vlc_object_create(O *obj) { static_assert(std::is_pointer::value == false, "vlc_object_create can only create objects"); void *object = vlc_object_create(VLC_OBJECT(obj), sizeof(T)); if (object == nullptr) return nullptr; return new(object) T; } #undef vlc_object_delete template static inline void vlc_object_delete(O *obj) { if (!std::is_trivially_destructible::value) obj->~O(); vlc_object_delete(VLC_OBJECT(obj)); } #endif /** @} */ /** @} */ #endif