[vlc-devel] [PATCH 1/3] add a new type of module: vlc_renderer_discovery
Steve Lhomme
robux4 at videolabs.io
Fri Apr 22 14:58:49 CEST 2016
From: Thomas Guillem <thomas at gllm.fr>
This new type of module will be used to discover renderers (chromecast, UPnP
Renderer, miracast, airport, DIAL, ConeCast).
These modules will send new vlc_renderer_item via an event manager callback.
This new kind of item contain the necessary information to setup a new sout.
Also-by: Steve Lhomme <robux4 at videolabs.io>
---
include/vlc_common.h | 2 +
include/vlc_events.h | 14 ++
include/vlc_plugin.h | 1 +
include/vlc_renderer_discovery.h | 230 +++++++++++++++++++++++++++++
src/Makefile.am | 2 +
src/libvlccore.sym | 17 +++
src/misc/renderer_discovery.c | 303 +++++++++++++++++++++++++++++++++++++++
7 files changed, 569 insertions(+)
create mode 100644 include/vlc_renderer_discovery.h
create mode 100644 src/misc/renderer_discovery.c
diff --git a/include/vlc_common.h b/include/vlc_common.h
index 296439e..f05d9c6 100644
--- a/include/vlc_common.h
+++ b/include/vlc_common.h
@@ -208,6 +208,8 @@ typedef struct playlist_item_t playlist_item_t;
typedef struct services_discovery_t services_discovery_t;
typedef struct services_discovery_sys_t services_discovery_sys_t;
typedef struct playlist_add_t playlist_add_t;
+typedef struct vlc_renderer_discovery vlc_renderer_discovery;
+typedef struct vlc_renderer_item vlc_renderer_item;
/* Modules */
typedef struct module_t module_t;
diff --git a/include/vlc_events.h b/include/vlc_events.h
index 9ca571a..328e2bd 100644
--- a/include/vlc_events.h
+++ b/include/vlc_events.h
@@ -132,6 +132,10 @@ typedef enum vlc_event_type_t {
vlc_ServicesDiscoveryStarted,
vlc_ServicesDiscoveryEnded,
+ /* Renderer Discovery events */
+ vlc_RendererDiscoveryItemAdded,
+ vlc_RendererDiscoveryItemRemoved,
+
/* Addons Manager events */
vlc_AddonFound,
vlc_AddonsDiscoveryEnded,
@@ -212,6 +216,16 @@ typedef struct vlc_event_t
void * unused;
} services_discovery_ended;
+ /* Renderer discovery events */
+ struct vlc_renderer_discovery_item_added
+ {
+ vlc_renderer_item * p_new_item;
+ } renderer_discovery_item_added;
+ struct vlc_renderer_discovery_item_removed
+ {
+ vlc_renderer_item * p_item;
+ } renderer_discovery_item_removed;
+
/* Addons */
struct vlc_addon_generic_event
{
diff --git a/include/vlc_plugin.h b/include/vlc_plugin.h
index e73ab91..42c3288 100644
--- a/include/vlc_plugin.h
+++ b/include/vlc_plugin.h
@@ -173,6 +173,7 @@ enum vlc_module_properties
#define SUBCAT_SOUT_ACO 504
#define SUBCAT_SOUT_PACKETIZER 505
#define SUBCAT_SOUT_VOD 507
+#define SUBCAT_SOUT_RENDERER 508
#define CAT_ADVANCED 6
#define SUBCAT_ADVANCED_MISC 602
diff --git a/include/vlc_renderer_discovery.h b/include/vlc_renderer_discovery.h
new file mode 100644
index 0000000..61cfa45
--- /dev/null
+++ b/include/vlc_renderer_discovery.h
@@ -0,0 +1,230 @@
+/*****************************************************************************
+ * vlc_renderer_discovery.h : Renderer Discovery functions
+ *****************************************************************************
+ * Copyright (C) 2016 VLC authors and VideoLAN
+ *
+ * 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.
+ *****************************************************************************/
+
+/**
+ * @file
+ * This file declares VLC renderer discvoery structures and functions
+ */
+
+#ifndef VLC_RENDERER_DISCOVERY_H
+#define VLC_RENDERER_DISCOVERY_H 1
+
+#include <vlc_input.h>
+#include <vlc_events.h>
+#include <vlc_probe.h>
+#include <vlc_url.h>
+
+/**
+ * @defgroup vlc_renderer VLC renderer discovery
+ * @{
+ * @defgroup vlc_renderer_item VLC renderer items returned by the discovery
+ * @{
+ */
+
+typedef struct vlc_renderer_item vlc_renderer_item;
+
+#define VLC_RENDERER_CAN_AUDIO 0x0001
+#define VLC_RENDERER_CAN_VIDEO 0x0002
+
+/**
+ * Create a new renderer item
+ *
+ * @param psz_name name of the item
+ * @param psz_uri uri of the renderer item, must contains a valid protocol and
+ * a valid host
+ * @param psz_extra_sout extra sout options
+ * @param psz_icon_uri icon uri of the renderer item
+ * @param i_flags flags for the item
+ * @return a renderer item or NULL in case of error
+ */
+VLC_API vlc_renderer_item *
+vlc_renderer_item_new(const char *psz_name, const char *psz_uri,
+ const char *psz_extra_sout, const char *psz_icon_uri,
+ int i_flags) VLC_USED;
+
+/**
+ * Hold a renderer item, i.e. creates a new reference
+ */
+VLC_API vlc_renderer_item *
+vlc_renderer_item_hold(vlc_renderer_item *p_item);
+
+/**
+ * Releases a renderer item, i.e. decrements its reference counter
+ */
+VLC_API void
+vlc_renderer_item_release(vlc_renderer_item *p_item);
+
+/**
+ * Get the human readable name of a renderer item
+ */
+VLC_API const char *
+vlc_renderer_item_name(const vlc_renderer_item *p_item);
+
+/**
+ * Get the sout command of a renderer item
+ */
+VLC_API const char *
+vlc_renderer_item_sout(const vlc_renderer_item *p_item);
+
+/**
+ * Get the icon uri of a renderer item
+ */
+VLC_API const char *
+vlc_renderer_item_icon_uri(const vlc_renderer_item *p_item);
+
+/**
+ * Get the flags of a renderer item
+ */
+VLC_API int
+vlc_renderer_item_flags(const vlc_renderer_item *p_item);
+
+/**
+ * Set an opaque context
+ */
+VLC_API void
+vlc_renderer_item_set_ctx(vlc_renderer_item *p_item, void *p_ctx);
+
+/**
+ * Get the opaque context previously set
+ */
+VLC_API void*
+vlc_renderer_item_ctx(const vlc_renderer_item *p_item);
+
+/**
+ * @}
+ * @defgroup vlc_renderer_discovery VLC renderer discovery interface
+ * @{
+ */
+
+typedef struct vlc_renderer_discovery vlc_renderer_discovery;
+typedef struct vlc_renderer_discovery_sys vlc_renderer_discovery_sys;
+
+/**
+ * Return a list of renderer discovery modules
+ *
+ * @param pppsz_names a pointer to a list of module name, NULL terminated
+ * @param pppsz_longnames a pointer to a list of module longname, NULL
+ * terminated
+ *
+ * @return VLC_SUCCESS on success, or VLC_EGENERIC on error
+ */
+VLC_API int
+vlc_rd_get_names(vlc_object_t *p_obj, char ***pppsz_names,
+ char ***pppsz_longnames) VLC_USED;
+#define vlc_rd_get_names(a, b, c) \
+ vlc_rd_get_names(VLC_OBJECT(a), b, c)
+
+/**
+ * Create a new renderer discovery module
+ *
+ * @param psz_name name of the module to load, see vlc_rd_get_names() to get
+ * the list of names
+ *
+ * @return a valid vlc_renderer_discovery, need to be released with
+ * vlc_rd_release()
+ */
+VLC_API vlc_renderer_discovery *
+vlc_rd_new(vlc_object_t *p_obj, const char *psz_name) VLC_USED;
+
+#define vlc_rd_release(p_rd) vlc_object_release(p_rd)
+
+/**
+ * Get the event manager of the renderer discovery module
+ *
+ * @see vlc_RendererDiscoveryItemAdded
+ * @see vlc_RendererDiscoveryItemRemoved
+ */
+VLC_API vlc_event_manager_t *
+vlc_rd_event_manager(vlc_renderer_discovery *p_rd);
+
+/**
+ * Start the renderer discovery module
+ *
+ * Once started, the module can send new vlc_renderer_item via the
+ * vlc_RendererDiscoveryItemAdded event.
+ */
+VLC_API int
+vlc_rd_start(vlc_renderer_discovery *p_rd);
+
+/**
+ * Stop the renderer discovery module
+ */
+VLC_API void
+vlc_rd_stop(vlc_renderer_discovery *p_rd);
+
+/**
+ * @}
+ * @defgroup vlc_renderer_discovery_module VLC renderer module
+ * @{
+ */
+
+struct vlc_renderer_discovery
+{
+ VLC_COMMON_MEMBERS
+ module_t * p_module;
+
+ vlc_event_manager_t event_manager;
+
+ char * psz_name;
+ config_chain_t * p_cfg;
+
+ vlc_renderer_discovery_sys *p_sys;
+};
+
+/**
+ * Add a new renderer item
+ *
+ * This will send the vlc_RendererDiscoveryItemAdded event
+ */
+VLC_API void
+vlc_rd_add_item(vlc_renderer_discovery * p_rd, vlc_renderer_item * p_item);
+
+/**
+ * Add a new renderer item
+ *
+ * This will send the vlc_RendererDiscoveryItemRemoved event
+ */
+VLC_API void
+vlc_rd_remove_item(vlc_renderer_discovery * p_rd, vlc_renderer_item * p_item);
+
+/**
+ * Renderer Discovery proble helpers
+ */
+VLC_API int
+vlc_rd_probe_add(vlc_probe_t *p_probe, const char *psz_name,
+ const char *psz_longname);
+
+#define VLC_RD_PROBE_HELPER(name, longname) \
+static int vlc_rd_probe_open(vlc_object_t *obj) \
+{ \
+ return vlc_rd_probe_add((struct vlc_probe_t *)obj, name, longname); \
+}
+
+#define VLC_RD_PROBE_SUBMODULE \
+ add_submodule() \
+ set_capability("renderer probe", 100) \
+ set_callbacks(vlc_rd_probe_open, NULL)
+
+/**
+ * @}
+ * @}
+ */
+
+#endif
diff --git a/src/Makefile.am b/src/Makefile.am
index 5ee185f..94b317b 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -82,6 +82,7 @@ pluginsinclude_HEADERS = \
../include/vlc_services_discovery.h \
../include/vlc_fingerprinter.h \
../include/vlc_interrupt.h \
+ ../include/vlc_renderer_discovery.h \
../include/vlc_sout.h \
../include/vlc_spu.h \
../include/vlc_stream.h \
@@ -444,6 +445,7 @@ SOURCES_libvlc_common = \
misc/interrupt.h \
misc/interrupt.c \
misc/keystore.c \
+ misc/renderer_discovery.c \
modules/modules.h \
modules/modules.c \
modules/bank.c \
diff --git a/src/libvlccore.sym b/src/libvlccore.sym
index 5cde08e..32dbaad 100644
--- a/src/libvlccore.sym
+++ b/src/libvlccore.sym
@@ -734,3 +734,20 @@ addons_manager_Remove
addon_entry_New
addon_entry_Hold
addon_entry_Release
+vlc_renderer_item_new
+vlc_renderer_item_hold
+vlc_renderer_item_release
+vlc_renderer_item_name
+vlc_renderer_item_sout
+vlc_renderer_item_icon_uri
+vlc_renderer_item_flags
+vlc_renderer_item_set_ctx
+vlc_renderer_item_ctx
+vlc_rd_get_names
+vlc_rd_new
+vlc_rd_event_manager
+vlc_rd_start
+vlc_rd_stop
+vlc_rd_add_item
+vlc_rd_remove_item
+vlc_rd_probe_add
diff --git a/src/misc/renderer_discovery.c b/src/misc/renderer_discovery.c
new file mode 100644
index 0000000..975417c
--- /dev/null
+++ b/src/misc/renderer_discovery.c
@@ -0,0 +1,303 @@
+/*****************************************************************************
+ * renderer_discovery.c : Renderer Discovery functions
+ *****************************************************************************
+ * Copyright(C) 2016 VLC authors and VideoLAN
+ *
+ * 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.
+ *****************************************************************************/
+
+/** @ingroup vlc_renderer */
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <assert.h>
+
+#include <vlc_common.h>
+#include <vlc_atomic.h>
+#include <vlc_events.h>
+#include <vlc_renderer_discovery.h>
+#include <vlc_probe.h>
+#include <vlc_modules.h>
+#include <libvlc.h>
+
+struct vlc_renderer_item
+{
+ char *psz_name;
+ char *psz_sout;
+ char *psz_icon_uri;
+ void *p_ctx;
+ int i_flags;
+ atomic_uint refs;
+};
+
+vlc_renderer_item *
+vlc_renderer_item_new(const char *psz_name, const char *psz_uri,
+ const char *psz_extra_sout, const char *psz_icon_uri,
+ int i_flags)
+{
+ assert(psz_uri != NULL);
+ vlc_renderer_item *p_item = NULL;
+ vlc_url_t url;
+ vlc_UrlParse(&url, psz_uri);
+
+ if (url.psz_protocol == NULL || url.psz_host == NULL)
+ goto error;
+
+ p_item = calloc(1, sizeof(vlc_renderer_item));
+ if (unlikely(p_item == NULL))
+ goto error;
+
+ if (psz_name != NULL)
+ p_item->psz_name = strdup(psz_name);
+ else if (asprintf(&p_item->psz_name, "%s (%s)", url.psz_protocol,
+ url.psz_host) == -1)
+ p_item->psz_name = NULL;
+ if (p_item->psz_name == NULL)
+ goto error;
+
+ if (asprintf(&p_item->psz_sout, "%s{ip=%s,port=%d%s%s}",
+ url.psz_protocol, url.psz_host, url.i_port,
+ psz_extra_sout != NULL ? "," : "",
+ psz_extra_sout != NULL ? psz_extra_sout : "") == -1)
+ goto error;
+
+ if ((p_item->psz_icon_uri = strdup(psz_icon_uri)) == NULL)
+ goto error;
+
+ p_item->i_flags = i_flags;
+ atomic_init(&p_item->refs, 1);
+ vlc_UrlClean(&url);
+ return p_item;
+
+error:
+ vlc_UrlClean(&url);
+ if (p_item != NULL)
+ {
+ free(p_item->psz_name);
+ free(p_item->psz_sout);
+ free(p_item->psz_icon_uri);
+ free(p_item);
+ }
+ return NULL;
+}
+
+const char *
+vlc_renderer_item_name(const vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ return p_item->psz_name;
+}
+
+const char *
+vlc_renderer_item_sout(const vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ return p_item->psz_sout;
+}
+
+const char *
+vlc_renderer_item_icon_uri(const vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ return p_item->psz_icon_uri;
+}
+
+int
+vlc_renderer_item_flags(const vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ return p_item->i_flags;
+}
+
+void
+vlc_renderer_item_set_ctx(vlc_renderer_item *p_item, void *p_ctx)
+{
+ assert(p_item != NULL);
+
+ p_item->p_ctx = p_ctx;
+}
+
+void*
+vlc_renderer_item_ctx(const vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ return p_item->p_ctx;
+}
+
+vlc_renderer_item *
+vlc_renderer_item_hold(vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ atomic_fetch_add(&p_item->refs, 1);
+ return p_item;
+}
+
+void
+vlc_renderer_item_release(vlc_renderer_item *p_item)
+{
+ assert(p_item != NULL);
+
+ if (atomic_fetch_sub(&p_item->refs, 1) != 1)
+ return;
+ free(p_item->psz_name);
+ free(p_item->psz_sout);
+ free(p_item->psz_icon_uri);
+ free(p_item);
+}
+
+struct vlc_rd_probe
+{
+ char *psz_name;
+ char *psz_longname;
+};
+
+int
+vlc_rd_probe_add(vlc_probe_t *probe, const char *psz_name,
+ const char *psz_longname)
+{
+ struct vlc_rd_probe names = { strdup(psz_name), strdup(psz_longname) };
+
+ if (unlikely(names.psz_name == NULL || names.psz_longname == NULL
+ || vlc_probe_add(probe, &names, sizeof(names))))
+ {
+ free(names.psz_name);
+ free(names.psz_longname);
+ return VLC_ENOMEM;
+ }
+ return VLC_PROBE_CONTINUE;
+}
+
+#undef vlc_rd_get_names
+int
+vlc_rd_get_names(vlc_object_t *p_obj, char ***pppsz_names,
+ char ***pppsz_longnames)
+{
+ size_t i_count;
+ struct vlc_rd_probe *p_tab = vlc_probe(p_obj, "renderer probe", &i_count);
+
+ if (i_count == 0)
+ {
+ free(p_tab);
+ return VLC_EGENERIC;
+ }
+
+ char **ppsz_names = malloc(sizeof(char *) * (i_count + 1));
+ char **ppsz_longnames = malloc(sizeof(char *) * (i_count + 1));
+
+ if (unlikely(ppsz_names == NULL || ppsz_longnames == NULL))
+ {
+ free(ppsz_names);
+ free(ppsz_longnames);
+ free(p_tab);
+ return VLC_EGENERIC;
+ }
+
+ for (size_t i = 0; i < i_count; i++)
+ {
+ ppsz_names[i] = p_tab[i].psz_name;
+ ppsz_longnames[i] = p_tab[i].psz_longname;
+ }
+ ppsz_names[i_count] = ppsz_longnames[i_count] = NULL;
+ free(p_tab);
+ *pppsz_names = ppsz_names;
+ *pppsz_longnames = ppsz_longnames;
+ return VLC_SUCCESS;
+}
+
+static void
+rd_destructor(vlc_object_t *p_obj)
+{
+ vlc_renderer_discovery * p_rd =(vlc_renderer_discovery *)p_obj;
+ assert(!p_rd->p_module); /* Forgot to call Stop */
+
+ config_ChainDestroy(p_rd->p_cfg);
+ free(p_rd->psz_name);
+ vlc_event_manager_fini(&p_rd->event_manager);
+}
+
+vlc_renderer_discovery *
+vlc_rd_new(vlc_object_t *p_obj, const char *psz_name)
+{
+ vlc_renderer_discovery *p_rd;
+
+ p_rd = vlc_custom_create(p_obj, sizeof(*p_rd), "renderer discovery");
+ if(!p_rd)
+ return NULL;
+ free(config_ChainCreate(&p_rd->psz_name, &p_rd->p_cfg, psz_name));
+
+ vlc_event_manager_t *p_em = &p_rd->event_manager;
+ vlc_event_manager_init(p_em, p_rd);
+ vlc_event_manager_register_event_type(p_em, vlc_RendererDiscoveryItemAdded);
+ vlc_event_manager_register_event_type(p_em, vlc_RendererDiscoveryItemRemoved);
+
+ vlc_object_set_destructor(p_rd, rd_destructor);
+ return p_rd;
+}
+
+VLC_API vlc_event_manager_t *
+vlc_rd_event_manager(vlc_renderer_discovery *p_rd)
+{
+ return &p_rd->event_manager;
+}
+
+int
+vlc_rd_start(vlc_renderer_discovery *p_rd)
+{
+ assert(!p_rd->p_module);
+
+ p_rd->p_module = module_need(p_rd, "renderer_discovery",
+ p_rd->psz_name, true);
+ if (p_rd->p_module == NULL)
+ {
+ msg_Err(p_rd, "no suitable renderer discovery module");
+ return VLC_EGENERIC;
+ }
+
+ return VLC_SUCCESS;
+}
+
+void
+vlc_rd_stop(vlc_renderer_discovery * p_rd)
+{
+ module_unneed(p_rd, p_rd->p_module);
+ p_rd->p_module = NULL;
+}
+
+void
+vlc_rd_add_item(vlc_renderer_discovery * p_rd, vlc_renderer_item * p_item)
+{
+ vlc_event_t event;
+ event.type = vlc_RendererDiscoveryItemAdded;
+ event.u.renderer_discovery_item_added.p_new_item = p_item;
+
+ vlc_event_send(&p_rd->event_manager, &event);
+}
+
+void
+vlc_rd_remove_item(vlc_renderer_discovery * p_rd, vlc_renderer_item * p_item)
+{
+ vlc_event_t event;
+ event.type = vlc_RendererDiscoveryItemRemoved;
+ event.u.renderer_discovery_item_removed.p_item = p_item;
+
+ vlc_event_send(&p_rd->event_manager, &event);
+}
--
2.7.0
More information about the vlc-devel
mailing list