2013-07-26 22:05:31 +00:00
|
|
|
/* GStreamer
|
|
|
|
* Copyright (C) 2013 Thiago Santos <thiago.sousa.santos@collabora.com>
|
|
|
|
*
|
|
|
|
* gst-qa-override-registry.c - QA Override Registry
|
|
|
|
*
|
|
|
|
* 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.1 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., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
2013-07-30 09:20:43 +00:00
|
|
|
#define __USE_GNU
|
|
|
|
#include <dlfcn.h>
|
|
|
|
|
2013-07-26 22:05:31 +00:00
|
|
|
#include "gst-qa-override-registry.h"
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
gchar *name;
|
|
|
|
GstQaOverride *override;
|
|
|
|
} GstQaOverrideRegistryNameEntry;
|
|
|
|
|
2013-07-29 16:17:50 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GType gtype;
|
|
|
|
GstQaOverride *override;
|
|
|
|
} GstQaOverrideRegistryGTypeEntry;
|
|
|
|
|
2013-07-26 22:05:31 +00:00
|
|
|
static GMutex _gst_qa_override_registry_mutex;
|
|
|
|
static GstQaOverrideRegistry *_registry_default;
|
|
|
|
|
|
|
|
#define GST_QA_OVERRIDE_REGISTRY_LOCK(r) g_mutex_lock (&r->mutex)
|
|
|
|
#define GST_QA_OVERRIDE_REGISTRY_UNLOCK(r) g_mutex_unlock (&r->mutex)
|
|
|
|
|
2013-07-30 09:20:43 +00:00
|
|
|
#define GST_QA_OVERRIDE_INIT_SYMBOL "gst_qa_create_overrides"
|
|
|
|
|
2013-07-26 22:05:31 +00:00
|
|
|
static GstQaOverrideRegistry *
|
|
|
|
gst_qa_override_registry_new (void)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistry *reg = g_slice_new0 (GstQaOverrideRegistry);
|
|
|
|
|
|
|
|
g_mutex_init (®->mutex);
|
|
|
|
g_queue_init (®->name_overrides);
|
2013-07-29 16:17:50 +00:00
|
|
|
g_queue_init (®->gtype_overrides);
|
|
|
|
g_queue_init (®->klass_overrides);
|
2013-07-26 22:05:31 +00:00
|
|
|
|
|
|
|
return reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
GstQaOverrideRegistry *
|
|
|
|
gst_qa_override_registry_get (void)
|
|
|
|
{
|
|
|
|
g_mutex_lock (&_gst_qa_override_registry_mutex);
|
|
|
|
if (G_UNLIKELY (!_registry_default)) {
|
|
|
|
_registry_default = gst_qa_override_registry_new ();
|
|
|
|
}
|
|
|
|
g_mutex_unlock (&_gst_qa_override_registry_mutex);
|
|
|
|
|
|
|
|
return _registry_default;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_qa_override_register_by_name (const gchar * name, GstQaOverride * override)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistry *registry = gst_qa_override_registry_get ();
|
|
|
|
GstQaOverrideRegistryNameEntry *entry =
|
|
|
|
g_slice_new (GstQaOverrideRegistryNameEntry);
|
|
|
|
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_LOCK (registry);
|
|
|
|
entry->name = g_strdup (name);
|
|
|
|
entry->override = override;
|
|
|
|
g_queue_push_tail (®istry->name_overrides, entry);
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_UNLOCK (registry);
|
|
|
|
}
|
|
|
|
|
2013-07-29 16:17:50 +00:00
|
|
|
void
|
|
|
|
gst_qa_override_register_by_type (GType gtype, GstQaOverride * override)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistry *registry = gst_qa_override_registry_get ();
|
|
|
|
GstQaOverrideRegistryGTypeEntry *entry =
|
|
|
|
g_slice_new (GstQaOverrideRegistryGTypeEntry);
|
|
|
|
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_LOCK (registry);
|
|
|
|
entry->gtype = gtype;
|
|
|
|
entry->override = override;
|
|
|
|
g_queue_push_tail (®istry->gtype_overrides, entry);
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_UNLOCK (registry);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_qa_override_register_by_klass (const gchar * klass,
|
|
|
|
GstQaOverride * override)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistry *registry = gst_qa_override_registry_get ();
|
|
|
|
GstQaOverrideRegistryNameEntry *entry =
|
|
|
|
g_slice_new (GstQaOverrideRegistryNameEntry);
|
|
|
|
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_LOCK (registry);
|
|
|
|
entry->name = g_strdup (klass);
|
|
|
|
entry->override = override;
|
|
|
|
g_queue_push_tail (®istry->klass_overrides, entry);
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_UNLOCK (registry);
|
|
|
|
}
|
|
|
|
|
2013-07-26 22:05:31 +00:00
|
|
|
static void
|
|
|
|
gst_qa_override_registry_attach_name_overrides_unlocked (GstQaOverrideRegistry *
|
|
|
|
registry, GstQaMonitor * monitor)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistryNameEntry *entry;
|
|
|
|
GList *iter;
|
|
|
|
const gchar *name;
|
|
|
|
|
|
|
|
name = gst_qa_monitor_get_element_name (monitor);
|
|
|
|
for (iter = registry->name_overrides.head; iter; iter = g_list_next (iter)) {
|
|
|
|
entry = iter->data;
|
|
|
|
if (strcmp (name, entry->name) == 0) {
|
|
|
|
gst_qa_monitor_attach_override (monitor, entry->override);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-29 16:17:50 +00:00
|
|
|
static void
|
|
|
|
gst_qa_override_registry_attach_gtype_overrides_unlocked (GstQaOverrideRegistry
|
|
|
|
* registry, GstQaMonitor * monitor)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistryGTypeEntry *entry;
|
|
|
|
GstElement *element;
|
|
|
|
GList *iter;
|
|
|
|
|
|
|
|
element = gst_qa_monitor_get_element (monitor);
|
|
|
|
if (!element)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (iter = registry->name_overrides.head; iter; iter = g_list_next (iter)) {
|
|
|
|
entry = iter->data;
|
|
|
|
if (G_TYPE_CHECK_INSTANCE_TYPE (element, entry->gtype)) {
|
|
|
|
gst_qa_monitor_attach_override (monitor, entry->override);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_qa_override_registry_attach_klass_overrides_unlocked (GstQaOverrideRegistry
|
|
|
|
* registry, GstQaMonitor * monitor)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistryNameEntry *entry;
|
|
|
|
GList *iter;
|
|
|
|
GstElement *element;
|
|
|
|
GstElementClass *klass;
|
|
|
|
|
|
|
|
element = gst_qa_monitor_get_element (monitor);
|
|
|
|
if (!element)
|
|
|
|
return;
|
|
|
|
|
|
|
|
klass = GST_ELEMENT_GET_CLASS (element);
|
|
|
|
|
|
|
|
for (iter = registry->name_overrides.head; iter; iter = g_list_next (iter)) {
|
|
|
|
entry = iter->data;
|
|
|
|
/* TODO It would be more correct to split it before comparing */
|
|
|
|
if (strstr (klass->details.klass, entry->name) != NULL) {
|
|
|
|
gst_qa_monitor_attach_override (monitor, entry->override);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-26 22:05:31 +00:00
|
|
|
void
|
|
|
|
gst_qa_override_registry_attach_overrides (GstQaMonitor * monitor)
|
|
|
|
{
|
|
|
|
GstQaOverrideRegistry *reg = gst_qa_override_registry_get ();
|
|
|
|
|
|
|
|
GST_QA_OVERRIDE_REGISTRY_LOCK (reg);
|
|
|
|
gst_qa_override_registry_attach_name_overrides_unlocked (reg, monitor);
|
2013-07-29 16:17:50 +00:00
|
|
|
gst_qa_override_registry_attach_gtype_overrides_unlocked (reg, monitor);
|
|
|
|
gst_qa_override_registry_attach_klass_overrides_unlocked (reg, monitor);
|
2013-07-26 22:05:31 +00:00
|
|
|
GST_QA_OVERRIDE_REGISTRY_UNLOCK (reg);
|
|
|
|
}
|
2013-07-30 09:20:43 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
gst_qa_override_registry_preload (void)
|
|
|
|
{
|
|
|
|
gchar **solist, *const *so;
|
|
|
|
const char *sos, *soerr;
|
|
|
|
void *sol;
|
|
|
|
int ret, (*entry) (void), nloaded = 0;
|
|
|
|
|
|
|
|
sos = g_getenv ("GST_QA_OVERRIDE");
|
|
|
|
if (!sos) {
|
|
|
|
GST_INFO ("No GST_QA_OVERRIDE found, no overrides to load");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
solist = g_strsplit (sos, ",", 0);
|
|
|
|
for (so = solist; *so; ++so) {
|
|
|
|
GST_INFO ("Loading overrides from %s", *so);
|
|
|
|
sol = dlopen (*so, RTLD_LAZY);
|
|
|
|
if (!sol) {
|
|
|
|
soerr = dlerror ();
|
|
|
|
GST_ERROR ("Failed to load %s %s", *so, soerr ? soerr : "no idea why");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
entry = dlsym (sol, GST_QA_OVERRIDE_INIT_SYMBOL);
|
|
|
|
if (entry) {
|
|
|
|
ret = (*entry) ();
|
|
|
|
if (ret > 0) {
|
|
|
|
GST_INFO ("Loaded %d overrides from %s", ret, *so);
|
|
|
|
nloaded += ret;
|
|
|
|
} else if (ret < 0) {
|
|
|
|
GST_WARNING ("Error loading overrides from %s", *so);
|
|
|
|
} else {
|
|
|
|
GST_INFO ("Loaded no overrides from %s", *so);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
GST_WARNING (GST_QA_OVERRIDE_INIT_SYMBOL " not found in %s", *so);
|
|
|
|
}
|
|
|
|
dlclose (sol);
|
|
|
|
}
|
|
|
|
g_strfreev (solist);
|
|
|
|
GST_INFO ("%d overrides loaded", nloaded);
|
|
|
|
return nloaded;
|
|
|
|
}
|