mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 09:55:36 +00:00
uri-asset: Implement multi threading support
Making sure to have 1 GstDiscoverer per thread. Use that new feature in gesdemux by loading the timeline directly from the streaming thread. Modifying the timeline is not supported allowed anyway.
This commit is contained in:
parent
3d11893fd7
commit
61c952c714
3 changed files with 80 additions and 51 deletions
|
@ -42,7 +42,43 @@
|
||||||
|
|
||||||
static GHashTable *parent_newparent_table = NULL;
|
static GHashTable *parent_newparent_table = NULL;
|
||||||
|
|
||||||
static GstDiscoverer *discoverer = NULL;
|
G_LOCK_DEFINE_STATIC (discoverers_lock);
|
||||||
|
static GstClockTime discovering_timeout = DEFAULT_DISCOVERY_TIMEOUT;
|
||||||
|
static GHashTable *discoverers = NULL; /* Thread ID -> GstDiscoverer */
|
||||||
|
static void discoverer_discovered_cb (GstDiscoverer * discoverer,
|
||||||
|
GstDiscovererInfo * info, GError * err, gpointer user_data);
|
||||||
|
|
||||||
|
/* WITH discoverers_lock */
|
||||||
|
static GstDiscoverer *
|
||||||
|
create_discoverer ()
|
||||||
|
{
|
||||||
|
GstDiscoverer *disco = gst_discoverer_new (discovering_timeout, NULL);
|
||||||
|
|
||||||
|
g_signal_connect (disco, "discovered", G_CALLBACK (discoverer_discovered_cb),
|
||||||
|
NULL);
|
||||||
|
GST_INFO_OBJECT (disco, "Creating new discoverer");
|
||||||
|
g_hash_table_insert (discoverers, g_thread_self (), disco);
|
||||||
|
gst_discoverer_start (disco);
|
||||||
|
|
||||||
|
return disco;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstDiscoverer *
|
||||||
|
get_discoverer ()
|
||||||
|
{
|
||||||
|
GstDiscoverer *disco;
|
||||||
|
|
||||||
|
G_LOCK (discoverers_lock);
|
||||||
|
g_assert (discoverers);
|
||||||
|
disco = g_hash_table_lookup (discoverers, g_thread_self ());
|
||||||
|
if (!disco) {
|
||||||
|
disco = create_discoverer ();
|
||||||
|
}
|
||||||
|
disco = gst_object_ref (disco);
|
||||||
|
G_UNLOCK (discoverers_lock);
|
||||||
|
|
||||||
|
return disco;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
initable_iface_init (GInitableIface * initable_iface)
|
initable_iface_init (GInitableIface * initable_iface)
|
||||||
|
@ -62,9 +98,6 @@ enum
|
||||||
};
|
};
|
||||||
static GParamSpec *properties[PROP_LAST];
|
static GParamSpec *properties[PROP_LAST];
|
||||||
|
|
||||||
static void discoverer_discovered_cb (GstDiscoverer * discoverer,
|
|
||||||
GstDiscovererInfo * info, GError * err, gpointer user_data);
|
|
||||||
|
|
||||||
struct _GESUriClipAssetPrivate
|
struct _GESUriClipAssetPrivate
|
||||||
{
|
{
|
||||||
GstDiscovererInfo *info;
|
GstDiscovererInfo *info;
|
||||||
|
@ -128,12 +161,14 @@ _start_loading (GESAsset * asset, GError ** error)
|
||||||
{
|
{
|
||||||
gboolean ret;
|
gboolean ret;
|
||||||
const gchar *uri;
|
const gchar *uri;
|
||||||
GESUriClipAssetClass *class = GES_URI_CLIP_ASSET_GET_CLASS (asset);
|
GstDiscoverer *discoverer = get_discoverer ();
|
||||||
|
|
||||||
uri = ges_asset_get_id (asset);
|
uri = ges_asset_get_id (asset);
|
||||||
|
GST_DEBUG_OBJECT (discoverer, "Started loading %s", uri);
|
||||||
|
|
||||||
|
ret = gst_discoverer_discover_uri_async (discoverer, uri);
|
||||||
|
gst_object_unref (discoverer);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (asset, "Started loading %s", uri);
|
|
||||||
ret = gst_discoverer_discover_uri_async (class->discoverer, uri);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
return GES_ASSET_LOADING_ASYNC;
|
return GES_ASSET_LOADING_ASYNC;
|
||||||
|
|
||||||
|
@ -568,9 +603,7 @@ ges_uri_clip_asset_request_sync (const gchar * uri, GError ** error)
|
||||||
GError *lerror = NULL;
|
GError *lerror = NULL;
|
||||||
GESUriClipAsset *asset;
|
GESUriClipAsset *asset;
|
||||||
RequestSyncData data = { 0, };
|
RequestSyncData data = { 0, };
|
||||||
GESUriClipAssetClass *klass = g_type_class_peek (GES_TYPE_URI_CLIP_ASSET);
|
GstDiscoverer *previous_discoverer;
|
||||||
GstClockTime timeout;
|
|
||||||
GstDiscoverer *previous_discoverer = klass->discoverer;
|
|
||||||
|
|
||||||
asset = GES_URI_CLIP_ASSET (ges_asset_request (GES_TYPE_URI_CLIP, uri,
|
asset = GES_URI_CLIP_ASSET (ges_asset_request (GES_TYPE_URI_CLIP, uri,
|
||||||
&lerror));
|
&lerror));
|
||||||
|
@ -579,25 +612,17 @@ ges_uri_clip_asset_request_sync (const gchar * uri, GError ** error)
|
||||||
return asset;
|
return asset;
|
||||||
|
|
||||||
data.ml = g_main_loop_new (NULL, TRUE);
|
data.ml = g_main_loop_new (NULL, TRUE);
|
||||||
g_object_get (previous_discoverer, "timeout", &timeout, NULL);
|
previous_discoverer = get_discoverer ();
|
||||||
klass->discoverer = gst_discoverer_new (timeout, error);
|
create_discoverer ();
|
||||||
g_object_set (klass->discoverer, "use-cache", TRUE, NULL);
|
|
||||||
if (!klass->discoverer) {
|
|
||||||
klass->discoverer = previous_discoverer;
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
g_signal_connect (klass->discoverer, "discovered",
|
|
||||||
G_CALLBACK (klass->discovered), NULL);
|
|
||||||
gst_discoverer_start (klass->discoverer);
|
|
||||||
ges_asset_request_async (GES_TYPE_URI_CLIP, uri, NULL,
|
ges_asset_request_async (GES_TYPE_URI_CLIP, uri, NULL,
|
||||||
(GAsyncReadyCallback) asset_ready_cb, &data);
|
(GAsyncReadyCallback) asset_ready_cb, &data);
|
||||||
g_main_loop_run (data.ml);
|
g_main_loop_run (data.ml);
|
||||||
g_main_loop_unref (data.ml);
|
g_main_loop_unref (data.ml);
|
||||||
|
|
||||||
gst_object_unref (klass->discoverer);
|
G_LOCK (discoverers_lock);
|
||||||
klass->discoverer = previous_discoverer;
|
g_hash_table_insert (discoverers, g_thread_self (), previous_discoverer);
|
||||||
|
G_UNLOCK (discoverers_lock);
|
||||||
|
|
||||||
if (data.error) {
|
if (data.error) {
|
||||||
GST_ERROR ("Got an error requesting asset: %s", data.error->message);
|
GST_ERROR ("Got an error requesting asset: %s", data.error->message);
|
||||||
|
@ -621,9 +646,18 @@ void
|
||||||
ges_uri_clip_asset_class_set_timeout (GESUriClipAssetClass * klass,
|
ges_uri_clip_asset_class_set_timeout (GESUriClipAssetClass * klass,
|
||||||
GstClockTime timeout)
|
GstClockTime timeout)
|
||||||
{
|
{
|
||||||
|
GHashTableIter iter;
|
||||||
|
gpointer value;
|
||||||
|
|
||||||
g_return_if_fail (GES_IS_URI_CLIP_ASSET_CLASS (klass));
|
g_return_if_fail (GES_IS_URI_CLIP_ASSET_CLASS (klass));
|
||||||
|
|
||||||
g_object_set (klass->discoverer, "timeout", timeout, NULL);
|
discovering_timeout = timeout;
|
||||||
|
|
||||||
|
G_LOCK (discoverers_lock);
|
||||||
|
g_hash_table_iter_init (&iter, discoverers);
|
||||||
|
while (g_hash_table_iter_next (&iter, NULL, &value))
|
||||||
|
g_object_set (value, "timeout", timeout, NULL);
|
||||||
|
G_UNLOCK (discoverers_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -774,13 +808,17 @@ ges_uri_source_asset_get_filesource_asset (GESUriSourceAsset * asset)
|
||||||
void
|
void
|
||||||
_ges_uri_asset_cleanup (void)
|
_ges_uri_asset_cleanup (void)
|
||||||
{
|
{
|
||||||
if (discoverer)
|
|
||||||
gst_discoverer_stop (discoverer);
|
|
||||||
g_clear_object (&discoverer);
|
|
||||||
if (parent_newparent_table) {
|
if (parent_newparent_table) {
|
||||||
g_hash_table_destroy (parent_newparent_table);
|
g_hash_table_destroy (parent_newparent_table);
|
||||||
parent_newparent_table = NULL;
|
parent_newparent_table = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
G_LOCK (discoverers_lock);
|
||||||
|
if (discoverers) {
|
||||||
|
g_hash_table_destroy (discoverers);
|
||||||
|
discoverers = NULL;
|
||||||
|
}
|
||||||
|
G_UNLOCK (discoverers_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
gboolean
|
gboolean
|
||||||
|
@ -790,6 +828,7 @@ _ges_uri_asset_ensure_setup (gpointer uriasset_class)
|
||||||
GError *err;
|
GError *err;
|
||||||
GstClockTime timeout;
|
GstClockTime timeout;
|
||||||
const gchar *timeout_str;
|
const gchar *timeout_str;
|
||||||
|
GstDiscoverer *discoverer = NULL;
|
||||||
|
|
||||||
g_return_val_if_fail (GES_IS_URI_CLIP_ASSET_CLASS (uriasset_class), FALSE);
|
g_return_val_if_fail (GES_IS_URI_CLIP_ASSET_CLASS (uriasset_class), FALSE);
|
||||||
|
|
||||||
|
@ -826,10 +865,17 @@ _ges_uri_asset_ensure_setup (gpointer uriasset_class)
|
||||||
|
|
||||||
g_signal_connect (klass->discoverer, "discovered",
|
g_signal_connect (klass->discoverer, "discovered",
|
||||||
G_CALLBACK (klass->discovered), NULL);
|
G_CALLBACK (klass->discovered), NULL);
|
||||||
|
gst_discoverer_start (klass->discoverer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
G_LOCK (discoverers_lock);
|
||||||
|
if (discoverers == NULL) {
|
||||||
|
discoverers = g_hash_table_new_full (g_direct_hash,
|
||||||
|
(GEqualFunc) g_direct_equal, NULL, g_object_unref);
|
||||||
|
}
|
||||||
|
G_UNLOCK (discoverers_lock);
|
||||||
|
|
||||||
/* We just start the discoverer and let it live */
|
/* We just start the discoverer and let it live */
|
||||||
gst_discoverer_start (klass->discoverer);
|
|
||||||
if (parent_newparent_table == NULL) {
|
if (parent_newparent_table == NULL) {
|
||||||
parent_newparent_table = g_hash_table_new_full (g_file_hash,
|
parent_newparent_table = g_hash_table_new_full (g_file_hash,
|
||||||
(GEqualFunc) g_file_equal, g_object_unref, g_object_unref);
|
(GEqualFunc) g_file_equal, g_object_unref, g_object_unref);
|
||||||
|
|
|
@ -62,11 +62,13 @@ struct _GESUriClipAssetClass
|
||||||
GESClipAssetClass parent_class;
|
GESClipAssetClass parent_class;
|
||||||
|
|
||||||
/* <private> */
|
/* <private> */
|
||||||
GstDiscoverer *discoverer;
|
GstDiscoverer *discoverer; /* Unused */
|
||||||
GstDiscoverer *sync_discoverer;
|
GstDiscoverer *sync_discoverer; /* Unused */
|
||||||
|
|
||||||
void (*discovered) (GstDiscoverer * discoverer, GstDiscovererInfo * info,
|
void (*discovered) (GstDiscoverer * discoverer, /* Unused */
|
||||||
GError * err, gpointer user_data);
|
GstDiscovererInfo * info,
|
||||||
|
GError * err,
|
||||||
|
gpointer user_data);
|
||||||
|
|
||||||
gpointer _ges_reserved[GES_PADDING -1];
|
gpointer _ges_reserved[GES_PADDING -1];
|
||||||
};
|
};
|
||||||
|
|
|
@ -171,26 +171,16 @@ error_loading_asset_cb (GESProject * project, GError * error, gchar * id,
|
||||||
g_main_loop_quit (data->ml);
|
g_main_loop_quit (data->ml);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO: Add a way to run a function in the right GES thread */
|
|
||||||
static gboolean
|
static gboolean
|
||||||
ges_timeline_new_from_uri_from_main_thread (TimelineConstructionData * data)
|
ges_timeline_new_from_uri_from_main_thread (TimelineConstructionData * data)
|
||||||
{
|
{
|
||||||
GESProject *project = ges_project_new (data->uri);
|
GESProject *project = ges_project_new (data->uri);
|
||||||
GESUriClipAssetClass *klass = g_type_class_peek (GES_TYPE_URI_CLIP_ASSET);
|
|
||||||
GstDiscoverer *previous_discoverer = klass->discoverer;
|
|
||||||
GstClockTime timeout;
|
|
||||||
G_GNUC_UNUSED void *unused;
|
G_GNUC_UNUSED void *unused;
|
||||||
|
|
||||||
g_object_get (previous_discoverer, "timeout", &timeout, NULL);
|
|
||||||
|
|
||||||
/* Make sure to use a new discoverer in case we are being discovered,
|
|
||||||
* as discovering is done one by one, and the global discoverer won't
|
|
||||||
* have the chance to discover the project assets */
|
|
||||||
g_mutex_lock (&data->lock);
|
g_mutex_lock (&data->lock);
|
||||||
klass->discoverer = gst_discoverer_new (timeout, &data->error);
|
klass->discoverer = gst_discoverer_new (timeout, &data->error);
|
||||||
g_object_set (klass->discoverer, "use-cache", TRUE, NULL);
|
g_object_set (klass->discoverer, "use-cache", TRUE, NULL);
|
||||||
if (data->error) {
|
if (data->error) {
|
||||||
klass->discoverer = previous_discoverer;
|
|
||||||
g_mutex_unlock (&data->lock);
|
g_mutex_unlock (&data->lock);
|
||||||
|
|
||||||
goto done;
|
goto done;
|
||||||
|
@ -222,15 +212,6 @@ done:
|
||||||
|
|
||||||
g_mutex_lock (&data->lock);
|
g_mutex_lock (&data->lock);
|
||||||
|
|
||||||
/* Set previous discoverer back! */
|
|
||||||
|
|
||||||
if (klass->discoverer)
|
|
||||||
gst_object_unref (klass->discoverer);
|
|
||||||
klass->discoverer = previous_discoverer;
|
|
||||||
|
|
||||||
if (data->timeline)
|
|
||||||
ges_timeline_commit (data->timeline);
|
|
||||||
|
|
||||||
if (data->loaded_sigid)
|
if (data->loaded_sigid)
|
||||||
g_signal_handler_disconnect (project, data->loaded_sigid);
|
g_signal_handler_disconnect (project, data->loaded_sigid);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue