* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
-
/**
* SECTION:element-playbin3
* @title: playbin3
*
*/
-/* FIXME 0.11: suppress warnings for deprecated API such as GValueArray
- * with newer GLib versions (>= 2.31.0) */
-#define GLIB_DISABLE_DEPRECATION_WARNINGS
-
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
typedef struct _GstPlayBin3Class GstPlayBin3Class;
typedef struct _GstSourceGroup GstSourceGroup;
typedef struct _GstSourceCombine GstSourceCombine;
+typedef struct _SourcePad SourcePad;
typedef GstCaps *(*SourceCombineGetMediaCapsFunc) (void);
-/* has the info for a combiner and provides the link to the sink */
+/* GstSourceCombine controls all the information regarding a certain
+ * media type.
+ *
+ * It can control a custom combiner element (by default none)
+ */
struct _GstSourceCombine
{
const gchar *media_type; /* the media type for the combiner */
SourceCombineGetMediaCapsFunc get_media_caps; /* more complex caps for the combiner */
GstPlaySinkType type; /* the sink pad type of the combiner */
+ GstStreamType stream_type; /* The GstStreamType of the combiner */
GstElement *combiner; /* the combiner */
- GPtrArray *channels;
+ GPtrArray *channels; /* Array of GstPad ? */
+
GstPad *srcpad; /* the source pad of the combiner */
GstPad *sinkpad; /* the sinkpad of the sink when the combiner
- * is linked
- */
- gulong block_id;
+ * is linked */
GPtrArray *streams; /* Sorted array of GstStream for the given type */
- gint current_stream; /* Currently selected GstStream */
gboolean has_active_pad; /* stream combiner has the "active-pad" property */
"audio", "video", "text"
};
+
+#define STREAM_TYPES_FORMAT "s%s%s"
+#define STREAM_TYPES_ARGS(s) (s) & GST_STREAM_TYPE_AUDIO ? "audio " : "", \
+ (s) & GST_STREAM_TYPE_VIDEO ? "video " : "", \
+ (s) & GST_STREAM_TYPE_TEXT ? "text " : ""
+
+
+
#if 0 /* AUTOPLUG DISABLED */
static void avelements_free (gpointer data);
static GSequence *avelements_create (GstPlayBin3 * playbin,
guint n_comm_cf; /* number of common caps features */
} GstAVElement;
+/* a structure to hold information about a uridecodebin pad */
+struct _SourcePad
+{
+ GstPad *pad; /* The controlled pad */
+ gulong event_probe_id;
+};
+
/* a structure to hold the objects for decoding a uri and the subtitle uri
*/
struct _GstSourceGroup
/* properties */
gchar *uri;
gchar *suburi;
- GValueArray *streaminfo;
- /* urisourcebins for uri and subtitle uri */
- /* FIXME: Just make this an array of uris */
- GstElement *urisourcebin;
- GstElement *suburisourcebin;
+ /* Bit-wise set of stream types we have requested from uridecodebin3 */
+ GstStreamType selected_stream_types;
+
+ /* uridecodebin to handle uri and suburi */
+ GstElement *uridecodebin;
/* Active sinks for each media type. These are initialized with
* the configured or currently used sink, otherwise
GstElement *video_sink;
GstElement *text_sink;
- gint pending;
- gboolean sub_pending;
+ /* List of source pads */
+ GList *source_pads;
+
+ /* uridecodebin signals */
+ gulong pad_added_id;
+ gulong pad_removed_id;
+ gulong select_stream_id;
+ gulong source_setup_id;
+ gulong about_to_finish_id;
- /* primary uri signals */
- gulong urisrc_pad_added_id;
- gulong urisrc_pad_removed_id;
- gulong urisrc_source_setup_id;
#if 0 /* AUTOPLUG DISABLED */
gulong autoplug_factories_id;
gulong autoplug_select_id;
gulong autoplug_query_id;
#endif
- /* subtitle uri signals */
- gulong sub_pad_added_id;
- gulong sub_pad_removed_id;
-#if 0 /* AUTOPLUG DISABLED */
- gulong sub_autoplug_continue_id;
- gulong sub_autoplug_query_id;
-#endif
+ gboolean stream_changed_pending;
- gulong block_id;
+ /* Active stream collection */
+ GstStreamCollection *collection;
- gboolean stream_changed_pending;
/* buffering message stored for after switching */
GstMessage *pending_buffering_msg;
#define GST_PLAY_BIN3_DYN_UNLOCK(bin) g_mutex_unlock (&(bin)->dyn_lock)
/* lock for shutdown */
-#define GST_PLAY_BIN3_SHUTDOWN_LOCK(bin,label) \
-G_STMT_START { \
- if (G_UNLIKELY (g_atomic_int_get (&bin->shutdown))) \
- goto label; \
- GST_PLAY_BIN3_DYN_LOCK (bin); \
- if (G_UNLIKELY (g_atomic_int_get (&bin->shutdown))) { \
- GST_PLAY_BIN3_DYN_UNLOCK (bin); \
- goto label; \
- } \
-} G_STMT_END
+#define GST_PLAY_BIN3_SHUTDOWN_LOCK(bin,label) \
+ G_STMT_START { \
+ if (G_UNLIKELY (g_atomic_int_get (&bin->shutdown))) \
+ goto label; \
+ GST_PLAY_BIN3_DYN_LOCK (bin); \
+ if (G_UNLIKELY (g_atomic_int_get (&bin->shutdown))) { \
+ GST_PLAY_BIN3_DYN_UNLOCK (bin); \
+ goto label; \
+ } \
+ } G_STMT_END
/* unlock for shutdown */
-#define GST_PLAY_BIN3_SHUTDOWN_UNLOCK(bin) \
- GST_PLAY_BIN3_DYN_UNLOCK (bin); \
+#define GST_PLAY_BIN3_SHUTDOWN_UNLOCK(bin) \
+ GST_PLAY_BIN3_DYN_UNLOCK (bin); \
/**
* GstPlayBin3:
GstSourceGroup *curr_group; /* pointer to the currently playing group */
GstSourceGroup *next_group; /* pointer to the next group */
- /* combiners for different streams */
+ /* Array of GstPad controlled by each combiner */
GPtrArray *channels[PLAYBIN_STREAM_LAST]; /* links to combiner pads */
+
+ /* combiners for different streams */
GstSourceCombine combiner[PLAYBIN_STREAM_LAST];
- /* A global decodebin3 that's used to actually do decoding */
- gboolean decodebin_active;
- GstElement *decodebin;
- /* Bit-wise set of stream types we have
- * requested from decodebin vs stream types
- * decodebin has provided */
+ /* Bit-wise set of stream types we have requested from uridecodebin3.
+ * Calculated as the combination of the 'selected_stream_types' of
+ * each sourcegroup */
GstStreamType selected_stream_types;
- GstStreamType active_stream_types;
- /* Decodebin signals */
- gulong db_pad_added_id;
- gulong db_pad_removed_id;
- gulong db_drained_id;
- gulong db_select_stream_id;
+ /* Bit-wise set of configured output stream types (i.e. active
+ playsink inputs and combiners) */
+ GstStreamType active_stream_types;
/* properties */
guint64 connection_speed; /* connection speed in bits/sec (0 = unknown) */
GSequence *velements; /* a list of GstAVElements for video stream */
guint64 ring_buffer_max_size; /* 0 means disabled */
-
- /* Active stream collection */
- GstStreamCollection *collection;
};
struct _GstPlayBin3Class
static GstSample *gst_play_bin3_convert_sample (GstPlayBin3 * playbin,
GstCaps * caps);
-static GstStateChangeReturn setup_next_source (GstPlayBin3 * playbin,
- GstState target);
+static GstStateChangeReturn setup_next_source (GstPlayBin3 * playbin);
-static void no_more_pads (GstPlayBin3 * playbin);
+static void reconfigure_output (GstPlayBin3 * playbin);
static void pad_removed_cb (GstElement * decodebin, GstPad * pad,
- GstPlayBin3 * playbin);
+ GstSourceGroup * group);
static gint select_stream_cb (GstElement * decodebin,
GstStreamCollection * collection, GstStream * stream,
- GstPlayBin3 * playbin);
+ GstSourceGroup * group);
-static void do_stream_selection (GstPlayBin3 * playbin);
+static void do_stream_selection (GstPlayBin3 * playbin, GstSourceGroup * group);
static GstElementClass *parent_class;
static guint gst_play_bin3_signals[LAST_SIGNAL] = { 0 };
-#define REMOVE_SIGNAL(obj,id) \
-if (id) { \
- g_signal_handler_disconnect (obj, id); \
- id = 0; \
-}
+#define REMOVE_SIGNAL(obj,id) \
+ if (id) { \
+ g_signal_handler_disconnect (obj, id); \
+ id = 0; \
+ }
static void gst_play_bin3_overlay_init (gpointer g_iface,
gpointer g_iface_data);
playbin->combiner[PLAYBIN_STREAM_AUDIO].media_type = "audio";
playbin->combiner[PLAYBIN_STREAM_AUDIO].type = GST_PLAY_SINK_TYPE_AUDIO;
+ playbin->combiner[PLAYBIN_STREAM_AUDIO].stream_type = GST_STREAM_TYPE_AUDIO;
playbin->combiner[PLAYBIN_STREAM_AUDIO].channels = playbin->channels[0];
playbin->combiner[PLAYBIN_STREAM_AUDIO].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_AUDIO].current_stream = -1;
playbin->combiner[PLAYBIN_STREAM_VIDEO].media_type = "video";
playbin->combiner[PLAYBIN_STREAM_VIDEO].type = GST_PLAY_SINK_TYPE_VIDEO;
+ playbin->combiner[PLAYBIN_STREAM_VIDEO].stream_type = GST_STREAM_TYPE_VIDEO;
playbin->combiner[PLAYBIN_STREAM_VIDEO].channels = playbin->channels[1];
playbin->combiner[PLAYBIN_STREAM_VIDEO].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_VIDEO].current_stream = -1;
playbin->combiner[PLAYBIN_STREAM_TEXT].media_type = "text";
playbin->combiner[PLAYBIN_STREAM_TEXT].get_media_caps =
gst_subtitle_overlay_create_factory_caps;
playbin->combiner[PLAYBIN_STREAM_TEXT].type = GST_PLAY_SINK_TYPE_TEXT;
+ playbin->combiner[PLAYBIN_STREAM_TEXT].stream_type = GST_STREAM_TYPE_TEXT;
playbin->combiner[PLAYBIN_STREAM_TEXT].channels = playbin->channels[2];
playbin->combiner[PLAYBIN_STREAM_TEXT].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_TEXT].current_stream = -1;
}
-/* Update the combiner information to be in sync with the current collection */
+/* Update the combiner information to be in sync with the current collection
+ *
+ * FIXME : "current" collection doesn't mean anything until we have a "combined"
+ * collection of all groups */
static void
-update_combiner_info (GstPlayBin3 * playbin)
+update_combiner_info (GstPlayBin3 * playbin, GstStreamCollection * collection)
{
guint i, len;
- if (playbin->collection == NULL)
+ if (collection == NULL)
return;
GST_DEBUG_OBJECT (playbin, "Updating combiner info");
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_TEXT].streams, TRUE);
playbin->combiner[PLAYBIN_STREAM_AUDIO].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_AUDIO].current_stream = -1;
playbin->combiner[PLAYBIN_STREAM_VIDEO].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_VIDEO].current_stream = -1;
playbin->combiner[PLAYBIN_STREAM_TEXT].streams =
g_ptr_array_new_with_free_func ((GDestroyNotify) gst_object_unref);
- playbin->combiner[PLAYBIN_STREAM_TEXT].current_stream = -1;
- len = gst_stream_collection_get_size (playbin->collection);
+ len = gst_stream_collection_get_size (collection);
for (i = 0; i < len; i++) {
- GstStream *stream =
- gst_stream_collection_get_stream (playbin->collection, i);
+ GstStream *stream = gst_stream_collection_get_stream (collection, i);
GstStreamType stype = gst_stream_get_stream_type (stream);
if (stype & GST_STREAM_TYPE_AUDIO) {
playbin->combiner[PLAYBIN_STREAM_TEXT].streams->len);
}
-/* Set the given stream as the selected stream */
-static void
-set_selected_stream (GstPlayBin3 * playbin, GstStream * stream)
-{
- GstSourceCombine *combine = NULL;
- GstStreamType stype = gst_stream_get_stream_type (stream);
-
- if (stype & GST_STREAM_TYPE_AUDIO)
- combine = &playbin->combiner[PLAYBIN_STREAM_AUDIO];
- else if (stype & GST_STREAM_TYPE_VIDEO)
- combine = &playbin->combiner[PLAYBIN_STREAM_VIDEO];
- else if (stype & GST_STREAM_TYPE_TEXT)
- combine = &playbin->combiner[PLAYBIN_STREAM_TEXT];
-
- if (combine) {
- if (combine->combiner == NULL) {
- guint i, len;
-
- GST_DEBUG_OBJECT (playbin, "Called for %s (%p)",
- gst_stream_get_stream_id (stream), combine->combiner);
-
- combine->current_stream = -1;
- len = combine->streams->len;
- for (i = 0; i < len; i++) {
- GstStream *cand = g_ptr_array_index (combine->streams, i);
- if (cand == stream) {
- GST_DEBUG_OBJECT (playbin, "Setting current to %d", i);
- combine->current_stream = i;
- break;
- }
- }
- }
- }
-}
-
static void
init_group (GstPlayBin3 * playbin, GstSourceGroup * group)
{
gst_message_unref (group->pending_buffering_msg);
group->pending_buffering_msg = NULL;
+ gst_object_replace ((GstObject **) & group->collection, NULL);
+
gst_object_replace ((GstObject **) & group->audio_sink, NULL);
gst_object_replace ((GstObject **) & group->video_sink, NULL);
gst_object_replace ((GstObject **) & group->text_sink, NULL);
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_VIDEO].streams, TRUE);
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_TEXT].streams, TRUE);
- if (playbin->decodebin)
- gst_object_unref (playbin->decodebin);
-
if (playbin->elements)
gst_plugin_feature_list_free (playbin->elements);
if (playbin->velements)
g_sequence_free (playbin->velements);
- if (playbin->collection)
- gst_object_unref (playbin->collection);
-
g_rec_mutex_clear (&playbin->lock);
g_mutex_clear (&playbin->dyn_lock);
g_mutex_clear (&playbin->elements_lock);
return gst_play_sink_convert_sample (playbin->playsink, caps);
}
-/* Returns current stream number, or -1 if none has been selected yet */
-static int
-get_current_stream_number (GstPlayBin3 * playbin, GstSourceCombine * combine,
- GPtrArray * channels)
-{
- /* Internal API cleanup would make this easier... */
- int i;
- GstPad *pad, *current;
- GstObject *combiner = NULL;
- int ret = -1;
-
- if (!combine->has_active_pad) {
- GST_WARNING_OBJECT (playbin,
- "combiner doesn't have the \"active-pad\" property");
- return ret;
- }
-
- for (i = 0; i < channels->len; i++) {
- pad = g_ptr_array_index (channels, i);
- if ((combiner = gst_pad_get_parent (pad))) {
- g_object_get (combiner, "active-pad", ¤t, NULL);
- gst_object_unref (combiner);
-
- if (pad == current) {
- gst_object_unref (current);
- ret = i;
- break;
- }
-
- if (current)
- gst_object_unref (current);
- }
- }
-
- return ret;
-}
-
static gboolean
gst_play_bin3_send_custom_event (GstObject * combiner, const gchar * event_name)
{
if (combine->combiner == NULL) {
/* FIXME: Check that the current_value is within range */
*current_value = stream;
- do_stream_selection (playbin);
+ do_stream_selection (playbin, playbin->curr_group);
GST_PLAY_BIN3_UNLOCK (playbin);
return TRUE;
}
&playbin->current_text, stream, &playbin->text_pending_flush_finish);
}
-static void
-source_combine_remove_pads (GstPlayBin3 * playbin, GstSourceCombine * combine)
-{
- if (combine->sinkpad) {
- GST_LOG_OBJECT (playbin, "unlinking from sink");
- gst_pad_unlink (combine->srcpad, combine->sinkpad);
-
- /* release back */
- GST_LOG_OBJECT (playbin, "release sink pad");
- gst_play_sink_release_pad (playbin->playsink, combine->sinkpad);
- gst_object_unref (combine->sinkpad);
- combine->sinkpad = NULL;
- }
- gst_object_unref (combine->srcpad);
- combine->srcpad = NULL;
-}
-
-static GstPadProbeReturn
-block_serialized_data_cb (GstPad * pad, GstPadProbeInfo * info,
- gpointer user_data)
-{
- if (GST_IS_EVENT (info->data) && !GST_EVENT_IS_SERIALIZED (info->data)) {
- GST_DEBUG_OBJECT (pad, "Letting non-serialized event %s pass",
- GST_EVENT_TYPE_NAME (info->data));
- return GST_PAD_PROBE_PASS;
- }
-
- return GST_PAD_PROBE_OK;
-}
static void
gst_play_bin3_set_sink (GstPlayBin3 * playbin, GstPlaySinkType type,
gst_play_bin3_set_flags (playbin, g_value_get_flags (value));
if (playbin->curr_group) {
GST_SOURCE_GROUP_LOCK (playbin->curr_group);
- if (playbin->curr_group->urisourcebin) {
- g_object_set (playbin->curr_group->urisourcebin, "download",
+ if (playbin->curr_group->uridecodebin) {
+ g_object_set (playbin->curr_group->uridecodebin, "download",
(g_value_get_flags (value) & GST_PLAY_FLAG_DOWNLOAD) != 0, NULL);
}
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
playbin->ring_buffer_max_size = g_value_get_uint64 (value);
if (playbin->curr_group) {
GST_SOURCE_GROUP_LOCK (playbin->curr_group);
- if (playbin->curr_group->urisourcebin) {
- g_object_set (playbin->curr_group->urisourcebin,
+ if (playbin->curr_group->uridecodebin) {
+ g_object_set (playbin->curr_group->uridecodebin,
"ring-buffer-max-size", playbin->ring_buffer_max_size, NULL);
}
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
static GList *
extend_list_of_streams (GstPlayBin3 * playbin, GstStreamType stype,
- GList * list)
+ GList * list, GstStreamCollection * collection)
{
GList *tmp, *res;
gint i, nb;
res = list;
- nb = gst_stream_collection_get_size (playbin->collection);
+ nb = gst_stream_collection_get_size (collection);
for (i = 0; i < nb; i++) {
- GstStream *stream =
- gst_stream_collection_get_stream (playbin->collection, i);
+ GstStream *stream = gst_stream_collection_get_stream (collection, i);
GstStreamType curtype = gst_stream_get_stream_type (stream);
if (stype == curtype) {
gboolean already_there = FALSE;
}
static GstEvent *
-update_select_streams_event (GstPlayBin3 * playbin, GstEvent * event)
+update_select_streams_event (GstPlayBin3 * playbin, GstEvent * event,
+ GstSourceGroup * group)
{
GList *streams = NULL;
GList *to_use;
return event;
}
+ if (!group->collection) {
+ GST_DEBUG_OBJECT (playbin,
+ "No stream collection for group, no need to modify SELECT_STREAMS event");
+ return event;
+ }
+
gst_event_parse_select_streams (event, &streams);
to_use = g_list_copy_deep (streams, (GCopyFunc) g_strdup, NULL);
/* For each combiner, we want to add all streams of that type to the
* selection */
if (playbin->audio_stream_combiner) {
- to_use = extend_list_of_streams (playbin, GST_STREAM_TYPE_AUDIO, to_use);
+ to_use =
+ extend_list_of_streams (playbin, GST_STREAM_TYPE_AUDIO, to_use,
+ group->collection);
combine_id =
get_combiner_stream_id (playbin,
&playbin->combiner[PLAYBIN_STREAM_AUDIO], streams);
gst_play_bin3_set_current_audio_stream (playbin, combine_id);
}
if (playbin->video_stream_combiner) {
- to_use = extend_list_of_streams (playbin, GST_STREAM_TYPE_VIDEO, to_use);
+ to_use =
+ extend_list_of_streams (playbin, GST_STREAM_TYPE_VIDEO, to_use,
+ group->collection);
combine_id =
get_combiner_stream_id (playbin,
&playbin->combiner[PLAYBIN_STREAM_VIDEO], streams);
gst_play_bin3_set_current_video_stream (playbin, combine_id);
}
if (playbin->text_stream_combiner) {
- to_use = extend_list_of_streams (playbin, GST_STREAM_TYPE_TEXT, to_use);
+ to_use =
+ extend_list_of_streams (playbin, GST_STREAM_TYPE_TEXT, to_use,
+ group->collection);
combine_id =
get_combiner_stream_id (playbin,
&playbin->combiner[PLAYBIN_STREAM_TEXT], streams);
return event;
}
+/* Returns TRUE if the given list of streams belongs to the stream collection */
+static gboolean
+gst_streams_belong_to_collection (GList * streams,
+ GstStreamCollection * collection)
+{
+ GList *tmp;
+ guint i, nb;
+
+ if (streams == NULL || collection == NULL)
+ return FALSE;
+ nb = gst_stream_collection_get_size (collection);
+ if (nb == 0)
+ return FALSE;
+
+ for (tmp = streams; tmp; tmp = tmp->next) {
+ const gchar *cand = (const gchar *) tmp->data;
+ gboolean found = FALSE;
+
+ for (i = 0; i < nb; i++) {
+ GstStream *stream = gst_stream_collection_get_stream (collection, i);
+ if (!g_strcmp0 (cand, gst_stream_get_stream_id (stream))) {
+ found = TRUE;
+ break;
+ }
+ }
+ if (!found)
+ return FALSE;
+ }
+ return TRUE;
+}
+
+static GstSourceGroup *
+get_source_group_for_streams (GstPlayBin3 * playbin, GstEvent * event)
+{
+ GList *streams;
+ GstSourceGroup *res = NULL;
+
+ gst_event_parse_select_streams (event, &streams);
+ if (playbin->curr_group->collection &&
+ gst_streams_belong_to_collection (streams,
+ playbin->curr_group->collection))
+ res = playbin->curr_group;
+ else if (playbin->next_group->collection &&
+ gst_streams_belong_to_collection (streams,
+ playbin->next_group->collection))
+ res = playbin->next_group;
+ g_list_free_full (streams, g_free);
+
+ return res;
+}
+
+static GstStreamType
+get_stream_type_for_event (GstStreamCollection * collection, GstEvent * event)
+{
+ GList *stream_list = NULL;
+ GList *tmp;
+ GstStreamType res = 0;
+ guint i, len;
+
+ gst_event_parse_select_streams (event, &stream_list);
+ len = gst_stream_collection_get_size (collection);
+ for (tmp = stream_list; tmp; tmp = tmp->next) {
+ gchar *stid = (gchar *) tmp->data;
+
+ for (i = 0; i < len; i++) {
+ GstStream *stream = gst_stream_collection_get_stream (collection, i);
+ if (!g_strcmp0 (stid, gst_stream_get_stream_id (stream))) {
+ res |= gst_stream_get_stream_type (stream);
+ }
+ }
+ }
+ g_list_free_full (stream_list, g_free);
+
+ return res;
+}
+
static gboolean
gst_play_bin3_send_event (GstElement * element, GstEvent * event)
{
if (GST_EVENT_TYPE (event) == GST_EVENT_SELECT_STREAMS) {
gboolean res;
+ GstSourceGroup *group;
GST_PLAY_BIN3_LOCK (playbin);
GST_LOG_OBJECT (playbin,
/* This is probably already false, but it doesn't hurt to be sure */
playbin->do_stream_selections = FALSE;
+ group = get_source_group_for_streams (playbin, event);
+ if (group == NULL) {
+ GST_WARNING_OBJECT (playbin,
+ "Can't figure out to which uridecodebin the select-streams event should be sent to");
+ GST_PLAY_BIN3_UNLOCK (playbin);
+ return FALSE;
+ }
+
/* If we have custom combiners, we need to extend the selection with
* the list of all streams for that given type since we will be handling
* the selection with that combiner */
- event = update_select_streams_event (playbin, event);
+ event = update_select_streams_event (playbin, event, group);
+
+ if (group->collection) {
+ group->selected_stream_types =
+ get_stream_type_for_event (group->collection, event);
+ playbin->selected_stream_types =
+ playbin->groups[0].selected_stream_types | playbin->
+ groups[1].selected_stream_types;
+ if (playbin->active_stream_types != playbin->selected_stream_types)
+ reconfigure_output (playbin);
+ }
- /* Send this event directly to decodebin, so it works even
- * if decodebin didn't add any pads yet */
- res = gst_element_send_event (playbin->decodebin, event);
+ /* Send this event directly to uridecodebin, so it works even
+ * if uridecodebin didn't add any pads yet */
+ res = gst_element_send_event (group->uridecodebin, event);
GST_PLAY_BIN3_UNLOCK (playbin);
return res;
/* Called with playbin lock held */
static void
-do_stream_selection (GstPlayBin3 * playbin)
+do_stream_selection (GstPlayBin3 * playbin, GstSourceGroup * group)
{
GstStreamCollection *collection;
guint i, nb_streams;
gint nb_video = 0, nb_audio = 0, nb_text = 0;
GstStreamType chosen_stream_types = 0;
- collection = playbin->collection;
+ if (group == NULL)
+ return;
+
+ collection = group->collection;
if (collection == NULL) {
GST_LOG_OBJECT (playbin, "No stream collection. Not doing stream-select");
return;
GST_INFO_OBJECT (playbin, "Empty collection received! Ignoring");
}
+ GST_DEBUG_OBJECT (playbin, "Doing selection on collection with %d streams",
+ nb_streams);
+
/* Iterate the collection and choose the streams that match
* either the current-* setting, or all streams of a type if there's
* a combiner for that type */
gint pb_stream_type = -1;
gboolean select_this = FALSE;
+ GST_LOG_OBJECT (playbin, "Looking at stream #%d : %s", i, stream_id);
+
if (stream_type & GST_STREAM_TYPE_AUDIO) {
pb_stream_type = PLAYBIN_STREAM_AUDIO;
/* Select the stream if it's the current one or if there's a custom selector */
}
if (streams) {
- GstEvent *ev = gst_event_new_select_streams (streams);
- gst_element_send_event (playbin->decodebin, ev);
+ if (group->uridecodebin) {
+ GstEvent *ev = gst_event_new_select_streams (streams);
+ gst_element_send_event (group->uridecodebin, ev);
+ }
g_list_free (streams);
}
- playbin->selected_stream_types = chosen_stream_types;
+
+ group->selected_stream_types = chosen_stream_types;
+ /* Update global selected_stream_types */
+ playbin->selected_stream_types =
+ playbin->groups[0].selected_stream_types | playbin->
+ groups[1].selected_stream_types;
+ if (playbin->active_stream_types != playbin->selected_stream_types)
+ reconfigure_output (playbin);
+}
+
+/* Return the GstSourceGroup to which this element belongs
+ * Can be NULL (if it belongs to playsink for example) */
+static GstSourceGroup *
+find_source_group_owner (GstPlayBin3 * playbin, GstObject * element)
+{
+ if (playbin->curr_group->uridecodebin
+ && gst_object_has_as_ancestor (element,
+ GST_OBJECT_CAST (playbin->curr_group->uridecodebin)))
+ return playbin->curr_group;
+ if (playbin->next_group->uridecodebin
+ && gst_object_has_as_ancestor (element,
+ GST_OBJECT_CAST (playbin->next_group->uridecodebin)))
+ return playbin->next_group;
+ return NULL;
}
static void
#endif
if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_STREAM_COLLECTION) {
GstStreamCollection *collection = NULL;
- GstObject *src = GST_MESSAGE_SRC (msg);
- gboolean pstate = playbin->do_stream_selections;
gst_message_parse_stream_collection (msg, &collection);
if (collection) {
+ gboolean pstate = playbin->do_stream_selections;
+ GstSourceGroup *target_group = NULL;
+
GST_PLAY_BIN3_LOCK (playbin);
GST_DEBUG_OBJECT (playbin,
- "STREAM_COLLECTION: Got a collection from %" GST_PTR_FORMAT, src);
- gst_object_replace ((GstObject **) & playbin->collection,
- (GstObject *) collection);
- update_combiner_info (playbin);
+ "STREAM_COLLECTION: Got a collection from %" GST_PTR_FORMAT,
+ msg->src);
+ target_group = find_source_group_owner (playbin, msg->src);
+ if (target_group)
+ gst_object_replace ((GstObject **) & target_group->collection,
+ (GstObject *) collection);
+ /* FIXME: Only do the following if it's the current group? */
+ if (target_group == playbin->curr_group)
+ update_combiner_info (playbin, target_group->collection);
if (pstate)
playbin->do_stream_selections = FALSE;
- do_stream_selection (playbin);
+ do_stream_selection (playbin, target_group);
if (pstate)
playbin->do_stream_selections = TRUE;
GST_PLAY_BIN3_UNLOCK (playbin);
gst_object_unref (collection);
}
- } else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_STREAMS_SELECTED) {
- GstStreamCollection *collection = NULL;
- GstObject *src = GST_MESSAGE_SRC (msg);
- GstStreamType chosen_stream_types = 0;
- gboolean reconfigure_playsink = FALSE;
-
- gst_message_parse_streams_selected (msg, &collection);
- if (collection) {
- guint i, len;
- GST_PLAY_BIN3_LOCK (playbin);
- GST_DEBUG_OBJECT (playbin,
- "STREAMS_SELECTED: Got a collection from %" GST_PTR_FORMAT, src);
- gst_object_replace ((GstObject **) & playbin->collection,
- (GstObject *) collection);
- update_combiner_info (playbin);
- len = gst_message_streams_selected_get_size (msg);
- for (i = 0; i < len; i++) {
- GstStream *stream;
-
- stream = gst_message_streams_selected_get_stream (msg, i);
- set_selected_stream (playbin, stream);
- chosen_stream_types |= gst_stream_get_stream_type (stream);
- gst_object_unref (stream);
- }
-
- if (playbin->selected_stream_types != chosen_stream_types) {
- GST_DEBUG_OBJECT (playbin, "Chosen stream types are changed");
- reconfigure_playsink = TRUE;
- }
- playbin->selected_stream_types = chosen_stream_types;
- GST_PLAY_BIN3_UNLOCK (playbin);
-
- gst_object_unref (collection);
-
- if (reconfigure_playsink &&
- (playbin->selected_stream_types & ~playbin->active_stream_types &
- (GST_STREAM_TYPE_VIDEO | GST_STREAM_TYPE_AUDIO)) == 0) {
- no_more_pads_cb (playbin->decodebin, playbin);
- }
- }
}
if (msg)
return out_caps;
}
+static SourcePad *
+find_source_pad (GstSourceGroup * group, GstPad * target)
+{
+ GList *tmp;
+
+ for (tmp = group->source_pads; tmp; tmp = tmp->next) {
+ SourcePad *res = (SourcePad *) tmp->data;
+ if (res->pad == target)
+ return res;
+ }
+ return NULL;
+}
+
static GstPadProbeReturn
_decodebin_event_probe (GstPad * pad, GstPadProbeInfo * info, gpointer udata)
{
GstPadProbeReturn ret = GST_PAD_PROBE_OK;
- GstPlayBin3 *playbin = (GstPlayBin3 *) udata;
+ GstSourceGroup *group = (GstSourceGroup *) udata;
+ GstPlayBin3 *playbin = group->playbin;
GstEvent *event = GST_PAD_PROBE_INFO_DATA (info);
switch (GST_EVENT_TYPE (event)) {
return ret;
}
-/* this function is called when a new pad is added to decodebin. We check the
- * type of the pad and add it to the combiner element
- */
static void
-pad_added_cb (GstElement * decodebin, GstPad * pad, GstPlayBin3 * playbin)
+control_source_pad (GstSourceGroup * group, GstPad * pad)
{
- GstPad *sinkpad;
- GstPadLinkReturn res;
- GstSourceCombine *combine = NULL;
- GstStreamType stream_type;
- gint pb_stream_type = -1;
- GstElement *custom_combiner = NULL;
- gulong event_probe_handler;
- gchar *pad_name;
-
- GST_PLAY_BIN3_SHUTDOWN_LOCK (playbin, shutdown);
+ SourcePad *sourcepad = g_slice_new0 (SourcePad);
- pad_name = gst_object_get_name (GST_OBJECT (pad));
+ sourcepad->pad = pad;
+ sourcepad->event_probe_id =
+ gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM,
+ _decodebin_event_probe, group, NULL);
+ group->source_pads = g_list_append (group->source_pads, sourcepad);
+}
- GST_DEBUG_OBJECT (playbin, "decoded pad %s:%s added",
- GST_DEBUG_PAD_NAME (pad));
+static void
+remove_combiner (GstPlayBin3 * playbin, GstSourceCombine * combine)
+{
+ gint n;
- /* major type of the pad, this determines the combiner to use,
- try exact match first */
- if (g_str_has_prefix (pad_name, "video")) {
- stream_type = GST_STREAM_TYPE_VIDEO;
- pb_stream_type = PLAYBIN_STREAM_VIDEO;
- custom_combiner = playbin->video_stream_combiner;
- } else if (g_str_has_prefix (pad_name, "audio")) {
- stream_type = GST_STREAM_TYPE_AUDIO;
- pb_stream_type = PLAYBIN_STREAM_AUDIO;
- custom_combiner = playbin->audio_stream_combiner;
- } else if (g_str_has_prefix (pad_name, "text")) {
- stream_type = GST_STREAM_TYPE_TEXT;
- pb_stream_type = PLAYBIN_STREAM_TEXT;
- custom_combiner = playbin->text_stream_combiner;
+ if (combine->combiner == NULL) {
+ GST_DEBUG_OBJECT (playbin, "No combiner element to remove");
+ return;
}
- g_free (pad_name);
+ /* Go over all sink pads and release them ! */
+ for (n = 0; n < combine->channels->len; n++) {
+ GstPad *sinkpad = g_ptr_array_index (combine->channels, n);
- /* no stream type found for the media type, don't bother linking it to a
- * combiner. This will leave the pad unlinked and thus ignored. */
- if (pb_stream_type < 0) {
- GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
- goto unknown_type;
+ gst_element_release_request_pad (combine->combiner, sinkpad);
+ gst_object_unref (sinkpad);
}
+ g_ptr_array_set_size (combine->channels, 0);
- combine = &playbin->combiner[pb_stream_type];
+ gst_element_set_state (combine->combiner, GST_STATE_NULL);
+ gst_bin_remove (GST_BIN_CAST (playbin), combine->combiner);
+ combine->combiner = NULL;
- if (custom_combiner && combine->combiner == NULL) {
- combine->combiner = custom_combiner;
- /* find out which properties the stream combiner supports */
- combine->has_active_pad =
- g_object_class_find_property (G_OBJECT_GET_CLASS (combine->combiner),
- "active-pad") != NULL;
+}
- if (combine->has_active_pad)
- g_signal_connect (combine->combiner, "notify::active-pad",
- G_CALLBACK (combiner_active_pad_changed), playbin);
+/* Create the combiner element if needed for the given combine */
+static void
+create_combiner (GstPlayBin3 * playbin, GstSourceCombine * combine)
+{
+ GstElement *custom_combiner = NULL;
- GST_DEBUG_OBJECT (playbin, "adding new stream combiner %p",
- combine->combiner);
- gst_element_set_state (combine->combiner, GST_STATE_PAUSED);
- gst_bin_add (GST_BIN_CAST (playbin), combine->combiner);
+ if (combine->combiner) {
+ GST_WARNING_OBJECT (playbin, "Combiner element already exists!");
+ return;
}
- GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
+ if (combine->stream_type == GST_STREAM_TYPE_VIDEO)
+ custom_combiner = playbin->video_stream_combiner;
+ else if (combine->stream_type == GST_STREAM_TYPE_AUDIO)
+ custom_combiner = playbin->audio_stream_combiner;
+ else if (combine->stream_type == GST_STREAM_TYPE_TEXT)
+ custom_combiner = playbin->text_stream_combiner;
- if (combine->srcpad == NULL) {
- if (combine->combiner) {
- /* save source pad of the combiner */
- combine->srcpad = gst_element_get_static_pad (combine->combiner, "src");
- } else {
- /* no combiner, use the pad as the source pad then */
- combine->srcpad = gst_object_ref (pad);
- }
+ combine->combiner = custom_combiner;
- /* block the combiner srcpad. It's possible that multiple source elements
- * pushing data into the combiners before we have a chance to collect all
- * streams and connect the sinks, resulting in not-linked errors. After we
- * configure the sinks we will unblock them all. */
- GST_DEBUG_OBJECT (playbin, "blocking %" GST_PTR_FORMAT, combine->srcpad);
- combine->block_id =
- gst_pad_add_probe (combine->srcpad, GST_PAD_PROBE_TYPE_BLOCK_DOWNSTREAM,
- block_serialized_data_cb, NULL, NULL);
+ if (!combine->combiner) {
+ GST_DEBUG_OBJECT (playbin, "No custom combiner requested");
+ return;
}
- /* get sinkpad for the new stream */
- if (combine->combiner) {
- if ((sinkpad = gst_element_get_request_pad (combine->combiner, "sink_%u"))) {
- GST_DEBUG_OBJECT (playbin, "got pad %s:%s from combiner",
- GST_DEBUG_PAD_NAME (sinkpad));
+ combine->srcpad = gst_element_get_static_pad (combine->combiner, "src");
+
+ combine->has_active_pad =
+ g_object_class_find_property (G_OBJECT_GET_CLASS (combine->combiner),
+ "active-pad") != NULL;
- /* find out which properties the sink pad supports */
- combine->has_always_ok =
- g_object_class_find_property (G_OBJECT_GET_CLASS (sinkpad),
- "always-ok") != NULL;
+ if (combine->has_active_pad)
+ g_signal_connect (combine->combiner, "notify::active-pad",
+ G_CALLBACK (combiner_active_pad_changed), playbin);
- /* store the combiner for the pad */
- g_object_set_data (G_OBJECT (sinkpad), "playbin.combine", combine);
+ GST_DEBUG_OBJECT (playbin, "adding new stream combiner %" GST_PTR_FORMAT,
+ combine->combiner);
+ gst_bin_add (GST_BIN_CAST (playbin), combine->combiner);
+ gst_element_sync_state_with_parent (combine->combiner);
+}
- /* store the pad in the array */
- GST_DEBUG_OBJECT (playbin, "pad %p added to array", sinkpad);
- g_ptr_array_add (combine->channels, sinkpad);
+static gboolean
+combiner_control_pad (GstPlayBin3 * playbin, GstSourceCombine * combine,
+ GstPad * srcpad)
+{
+ GstPadLinkReturn res;
- res = gst_pad_link (pad, sinkpad);
- if (GST_PAD_LINK_FAILED (res))
- goto link_failed;
+ GST_DEBUG_OBJECT (playbin, "srcpad %" GST_PTR_FORMAT, srcpad);
- /* store combiner pad so we can release it */
- g_object_set_data (G_OBJECT (pad), "playbin.sinkpad", sinkpad);
+ if (combine->combiner) {
+ GstPad *sinkpad =
+ gst_element_get_request_pad (combine->combiner, "sink_%u");
- GST_DEBUG_OBJECT (playbin, "linked pad %s:%s to combiner %p",
- GST_DEBUG_PAD_NAME (pad), combine->combiner);
- } else {
+ if (sinkpad == NULL)
goto request_pad_failed;
- }
- } else {
- /* no combiner, don't configure anything, we'll link the new pad directly to
- * the sink. */
- sinkpad = NULL;
- /* store the combiner for the pad */
- g_object_set_data (G_OBJECT (pad), "playbin.combine", combine);
- }
+ GST_DEBUG_OBJECT (playbin, "Got new combiner pad %" GST_PTR_FORMAT,
+ sinkpad);
- event_probe_handler =
- gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM,
- _decodebin_event_probe, playbin, NULL);
- g_object_set_data (G_OBJECT (pad), "playbin.event_probe_id",
- ULONG_TO_POINTER (event_probe_handler));
-
- playbin->active_stream_types |= stream_type;
-
- /* If we're expecting either audio or video,
- * wait for them to appear before configuring playsink */
- if ((playbin->active_stream_types & GST_STREAM_TYPE_TEXT) &&
- !(playbin->active_stream_types & GST_STREAM_TYPE_VIDEO)) {
- GST_LOG_OBJECT (playbin, "Have text stream but no video yet, Wating");
- } else if ((playbin->selected_stream_types & ~playbin->active_stream_types &
- (GST_STREAM_TYPE_VIDEO | GST_STREAM_TYPE_AUDIO))
- == 0) {
- no_more_pads (playbin);
- } else {
- GST_LOG_OBJECT (playbin, "Active stream types 0x%x, want 0x%x. Waiting",
- playbin->active_stream_types, playbin->selected_stream_types);
- }
+ /* find out which properties the sink pad supports */
+ combine->has_always_ok =
+ g_object_class_find_property (G_OBJECT_GET_CLASS (sinkpad),
+ "always-ok") != NULL;
- return;
+ /* store the pad in the array */
+ GST_DEBUG_OBJECT (playbin, "pad %" GST_PTR_FORMAT " added to array",
+ sinkpad);
+ g_ptr_array_add (combine->channels, sinkpad);
- /* ERRORS */
-unknown_type:
- GST_DEBUG_OBJECT (playbin, "Ignoring pad with unknown type");
- return;
+ res = gst_pad_link (srcpad, sinkpad);
+ if (GST_PAD_LINK_FAILED (res))
+ goto failed_combiner_link;
-link_failed:
- {
- GST_ERROR_OBJECT (playbin,
- "failed to link pad %s:%s to combiner, reason %s (%d)",
- GST_DEBUG_PAD_NAME (pad), gst_pad_link_get_name (res), res);
- return;
+ GST_DEBUG_OBJECT (playbin,
+ "linked pad %" GST_PTR_FORMAT " to combiner %" GST_PTR_FORMAT, srcpad,
+ combine->combiner);
+
+ } else {
+ GST_LOG_OBJECT (playbin, "combine->sinkpad:%" GST_PTR_FORMAT,
+ combine->sinkpad);
+ g_assert (combine->sinkpad != NULL);
+ /* Connect directly to playsink */
+ if (gst_pad_is_linked (combine->sinkpad))
+ goto sinkpad_already_linked;
+
+ GST_DEBUG_OBJECT (playbin, "Linking new pad straight to playsink");
+ res = gst_pad_link (srcpad, combine->sinkpad);
+
+ if (res != GST_PAD_LINK_OK)
+ goto failed_sinkpad_link;
}
+
+ return TRUE;
+
+ /* Failure cases */
request_pad_failed:
GST_ELEMENT_ERROR (playbin, CORE, PAD,
("Internal playbin error."),
("Failed to get request pad from combiner %p.", combine->combiner));
- return;
-shutdown:
- {
- GST_DEBUG ("ignoring, we are shutting down. Pad will be left unlinked");
- /* not going to done as we didn't request the caps */
- return;
+ return FALSE;
+
+
+sinkpad_already_linked:
+ GST_ELEMENT_ERROR (playbin, CORE, PAD,
+ ("Internal playbin error."), ("playsink pad already used !"));
+ return FALSE;
+
+failed_sinkpad_link:
+ GST_ELEMENT_ERROR (playbin, CORE, PAD,
+ ("Internal playbin error."),
+ ("Failed to link pad to sink. Error %d", res));
+ return FALSE;
+
+failed_combiner_link:
+ GST_ELEMENT_ERROR (playbin, CORE, PAD,
+ ("Internal playbin error."),
+ ("Failed to link pad to combiner. Error %d", res));
+ return FALSE;
+}
+
+static void
+combiner_release_pad (GstPlayBin3 * playbin, GstSourceCombine * combine,
+ GstPad * pad)
+{
+ if (combine->combiner) {
+ GstPad *peer = gst_pad_get_peer (pad);
+
+ if (peer) {
+ GST_DEBUG_OBJECT (playbin, "Removing combiner pad %" GST_PTR_FORMAT,
+ peer);
+ g_ptr_array_remove (combine->channels, peer);
+
+ gst_element_release_request_pad (combine->combiner, peer);
+ gst_object_unref (peer);
+ }
+ } else {
+ /* Release direct link if present */
+ if (combine->sinkpad) {
+ GST_DEBUG_OBJECT (playbin, "Unlinking pad from playsink sinkpad");
+ gst_pad_unlink (pad, combine->sinkpad);
+ }
}
}
-/* called when a pad is removed from the decodebin. We unlink the pad from
- * the combiner. This will make the combiner select a new pad. */
+/* Call after pad was unlinked from (potential) combiner */
static void
-pad_removed_cb (GstElement * decodebin, GstPad * pad, GstPlayBin3 * playbin)
+release_source_pad (GstPlayBin3 * playbin, GstSourceGroup * group, GstPad * pad)
{
- GstPad *peer;
- GstElement *combiner;
- GstSourceCombine *combine;
- gulong event_probe_handler;
- GstStreamType stream_type = GST_STREAM_TYPE_UNKNOWN;
- gchar *pad_name;
+ SourcePad *sourcepad;
- if (GST_PAD_IS_SINK (pad))
+ sourcepad = find_source_pad (group, pad);
+ if (!sourcepad) {
+ GST_DEBUG_OBJECT (playbin, "Not a pad controlled by us ?");
return;
+ }
- GST_DEBUG_OBJECT (playbin,
- "decoded pad %s:%s removed", GST_DEBUG_PAD_NAME (pad));
+ if (sourcepad->event_probe_id) {
+ gst_pad_remove_probe (pad, sourcepad->event_probe_id);
+ sourcepad->event_probe_id = 0;
+ }
- GST_PLAY_BIN3_LOCK (playbin);
+ /* Remove from list of controlled pads and check again for EOS status */
+ group->source_pads = g_list_remove (group->source_pads, sourcepad);
+ g_slice_free (SourcePad, sourcepad);
+}
- if ((event_probe_handler =
- POINTER_TO_ULONG (g_object_get_data (G_OBJECT (pad),
- "playbin.event_probe_id")))) {
- gst_pad_remove_probe (pad, event_probe_handler);
- g_object_set_data (G_OBJECT (pad), "playbin.event_probe_id", NULL);
- }
+/* this function is called when a new pad is added to decodebin. We check the
+ * type of the pad and add it to the combiner element
+ */
+static void
+pad_added_cb (GstElement * uridecodebin, GstPad * pad, GstSourceGroup * group)
+{
+ GstSourceCombine *combine = NULL;
+ gint pb_stream_type = -1;
+ gchar *pad_name;
+ GstPlayBin3 *playbin = group->playbin;
+
+ GST_PLAY_BIN3_SHUTDOWN_LOCK (playbin, shutdown);
pad_name = gst_object_get_name (GST_OBJECT (pad));
+ GST_DEBUG_OBJECT (playbin, "decoded pad %s:%s added",
+ GST_DEBUG_PAD_NAME (pad));
+
+ /* major type of the pad, this determines the combiner to use,
+ try exact match first */
if (g_str_has_prefix (pad_name, "video")) {
- stream_type = GST_STREAM_TYPE_VIDEO;
+ pb_stream_type = PLAYBIN_STREAM_VIDEO;
} else if (g_str_has_prefix (pad_name, "audio")) {
- stream_type = GST_STREAM_TYPE_AUDIO;
+ pb_stream_type = PLAYBIN_STREAM_AUDIO;
} else if (g_str_has_prefix (pad_name, "text")) {
- stream_type = GST_STREAM_TYPE_TEXT;
+ pb_stream_type = PLAYBIN_STREAM_TEXT;
}
g_free (pad_name);
- if ((combine = g_object_get_data (G_OBJECT (pad), "playbin.combine"))) {
- g_assert (combine->combiner == NULL);
- g_assert (combine->srcpad == pad);
- source_combine_remove_pads (playbin, combine);
- playbin->active_stream_types &= ~stream_type;
- goto exit;
+ /* no stream type found for the media type, don't bother linking it to a
+ * combiner. This will leave the pad unlinked and thus ignored. */
+ if (pb_stream_type < 0) {
+ GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
+ goto unknown_type;
}
- /* get the combiner sinkpad */
- if (!(peer = g_object_get_data (G_OBJECT (pad), "playbin.sinkpad")))
- goto not_linked;
-
- /* unlink the pad now (can fail, the pad is unlinked before it's removed) */
- gst_pad_unlink (pad, peer);
-
- /* get combiner */
- combiner = GST_ELEMENT_CAST (gst_pad_get_parent (peer));
- g_assert (combiner != NULL);
-
- if ((combine = g_object_get_data (G_OBJECT (peer), "playbin.combine"))) {
- /* remove the pad from the array */
- g_ptr_array_remove (combine->channels, peer);
- GST_DEBUG_OBJECT (playbin, "pad %p removed from array", peer);
-
- if (!combine->channels->len && combine->combiner) {
- GST_DEBUG_OBJECT (playbin, "all combiner sinkpads removed");
- GST_DEBUG_OBJECT (playbin, "removing combiner %p", combine->combiner);
- source_combine_remove_pads (playbin, combine);
- gst_element_set_state (combine->combiner, GST_STATE_NULL);
- gst_bin_remove (GST_BIN_CAST (playbin), combine->combiner);
- combine->combiner = NULL;
- playbin->active_stream_types &= ~stream_type;
- }
- }
+ combine = &playbin->combiner[pb_stream_type];
- /* release the pad to the combiner, this will make the combiner choose a new
- * pad. */
- gst_element_release_request_pad (combiner, peer);
- gst_object_unref (peer);
+ combiner_control_pad (playbin, combine, pad);
- gst_object_unref (combiner);
-exit:
- GST_PLAY_BIN3_UNLOCK (playbin);
+ control_source_pad (group, pad);
- if ((playbin->selected_stream_types & ~playbin->active_stream_types &
- (GST_STREAM_TYPE_VIDEO | GST_STREAM_TYPE_AUDIO))
- == 0) {
- no_more_pads (playbin);
- }
+ GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
return;
/* ERRORS */
-not_linked:
+unknown_type:
+ GST_DEBUG_OBJECT (playbin, "Ignoring pad with unknown type");
+ return;
+
+shutdown:
{
- GST_DEBUG_OBJECT (playbin, "pad not linked");
- goto exit;
+ GST_DEBUG ("ignoring, we are shutting down. Pad will be left unlinked");
+ /* not going to done as we didn't request the caps */
+ return;
}
}
+/* called when a pad is removed from the decodebin. We unlink the pad from
+ * the combiner. This will make the combiner select a new pad. */
+static void
+pad_removed_cb (GstElement * decodebin, GstPad * pad, GstSourceGroup * group)
+{
+ GstSourceCombine *combine;
+ GstPlayBin3 *playbin = group->playbin;
+
+ GST_DEBUG_OBJECT (playbin,
+ "decoded pad %s:%s removed", GST_DEBUG_PAD_NAME (pad));
+
+ GST_PLAY_BIN3_LOCK (playbin);
+
+ /* Get combiner for pad */
+ if (g_str_has_prefix (GST_PAD_NAME (pad), "video"))
+ combine = &playbin->combiner[PLAYBIN_STREAM_VIDEO];
+ else if (g_str_has_prefix (GST_PAD_NAME (pad), "audio"))
+ combine = &playbin->combiner[PLAYBIN_STREAM_AUDIO];
+ else if (g_str_has_prefix (GST_PAD_NAME (pad), "text"))
+ combine = &playbin->combiner[PLAYBIN_STREAM_TEXT];
+ else
+ return;
+
+ combiner_release_pad (playbin, combine, pad);
+ release_source_pad (playbin, group, pad);
+
+ GST_PLAY_BIN3_UNLOCK (playbin);
+}
+
static gint
select_stream_cb (GstElement * decodebin, GstStreamCollection * collection,
- GstStream * stream, GstPlayBin3 * playbin)
+ GstStream * stream, GstSourceGroup * group)
{
GstStreamType stype = gst_stream_get_stream_type (stream);
GstElement *combiner = NULL;
+ GstPlayBin3 *playbin = group->playbin;
if (stype & GST_STREAM_TYPE_AUDIO)
combiner = playbin->audio_stream_combiner;
return -1;
}
-/* we get called when all pads are available and we must connect the sinks to
- * them.
- * The main purpose of the code is to see if we have video/audio and subtitles
- * and pick the right pipelines to display them.
- *
- * The combiners installed on the group tell us about the presence of
- * audio/video and subtitle streams. This allows us to see if we need
- * visualisation, video or/and audio.
+/* We get called when the selected stream types change and
+ * reconfiguration of output (i.e. playsink and potential combiners)
+ * are required.
*/
static void
-no_more_pads (GstPlayBin3 * playbin)
+reconfigure_output (GstPlayBin3 * playbin)
{
- GstSourceGroup *group;
GstPadLinkReturn res;
gint i;
- gboolean configure;
- GST_DEBUG_OBJECT (playbin, "no more pads");
+ g_assert (playbin->selected_stream_types != playbin->active_stream_types);
- GST_PLAY_BIN3_SHUTDOWN_LOCK (playbin, shutdown);
+ GST_DEBUG_OBJECT (playbin, "selected_stream_types : %" STREAM_TYPES_FORMAT,
+ STREAM_TYPES_ARGS (playbin->selected_stream_types));
+ GST_DEBUG_OBJECT (playbin, "active_stream_types : %" STREAM_TYPES_FORMAT,
+ STREAM_TYPES_ARGS (playbin->active_stream_types));
GST_PLAY_BIN3_LOCK (playbin);
- group = playbin->curr_group;
+ /* Make sure combiners/playsink are in sync with selected stream types */
for (i = 0; i < PLAYBIN_STREAM_LAST; i++) {
GstSourceCombine *combine = &playbin->combiner[i];
-
- /* check if the specific media type was detected and thus has a combiner
- * created for it. If there is the media type, get a sinkpad from the sink
- * and link it. We only do this if we have not yet requested the sinkpad
- * before. */
- if (combine->srcpad && combine->sinkpad == NULL) {
- GST_DEBUG_OBJECT (playbin, "requesting new sink pad %d", combine->type);
- combine->sinkpad =
- gst_play_sink_request_pad (playbin->playsink, combine->type);
- gst_object_ref (combine->sinkpad);
- } else if (combine->srcpad && combine->sinkpad) {
- GST_DEBUG_OBJECT (playbin, "re-using sink pad %d", combine->type);
- } else if (combine->sinkpad && combine->srcpad == NULL) {
- GST_DEBUG_OBJECT (playbin, "releasing sink pad %d", combine->type);
- gst_play_sink_release_pad (playbin->playsink, combine->sinkpad);
- gst_object_unref (combine->sinkpad);
- combine->sinkpad = NULL;
- }
- if (combine->sinkpad && combine->srcpad &&
- !gst_pad_is_linked (combine->srcpad)) {
- res = gst_pad_link (combine->srcpad, combine->sinkpad);
- GST_DEBUG_OBJECT (playbin, "linked type %s, result: %d",
- combine->media_type, res);
- if (res != GST_PAD_LINK_OK) {
- GST_ELEMENT_ERROR (playbin, CORE, PAD,
- ("Internal playbin error."),
- ("Failed to link combiner to sink. Error %d", res));
+ gboolean is_selected =
+ (combine->stream_type & playbin->selected_stream_types) ==
+ combine->stream_type;
+ gboolean is_active =
+ (combine->stream_type & playbin->active_stream_types) ==
+ combine->stream_type;
+
+ GST_DEBUG_OBJECT (playbin, "Stream type status: '%s' %s %s",
+ combine->media_type, is_selected ? "selected" : "NOT selected",
+ is_active ? "active" : "NOT active");
+ /* FIXME : Remove asserts below once enough testing has been done */
+
+ if (is_selected && is_active) {
+ GST_DEBUG_OBJECT (playbin, "Stream type '%s' already active",
+ combine->media_type);
+ } else if (is_active && !is_selected) {
+ GST_DEBUG_OBJECT (playbin, "Stream type '%s' is no longer requested",
+ combine->media_type);
+
+ /* Unlink combiner from sink */
+ if (combine->srcpad) {
+ GST_LOG_OBJECT (playbin, "Unlinking from sink");
+ if (combine->sinkpad)
+ gst_pad_unlink (combine->srcpad, combine->sinkpad);
+ gst_object_unref (combine->srcpad);
+ combine->srcpad = NULL;
}
- }
- }
- GST_PLAY_BIN3_UNLOCK (playbin);
-
- GST_SOURCE_GROUP_LOCK (group);
- GST_DEBUG_OBJECT (playbin, "pending %d > %d", group->pending,
- group->pending - 1);
-
- if (group->pending > 0)
- group->pending--;
- if (group->pending == 0) {
- /* we are the last group to complete, we will configure the output and then
- * signal the other waiters. */
- GST_LOG_OBJECT (playbin, "last group complete");
- configure = TRUE;
- } else {
- GST_LOG_OBJECT (playbin, "have more pending groups");
- configure = FALSE;
- }
- GST_SOURCE_GROUP_UNLOCK (group);
-
- if (configure) {
- /* if we have custom sinks, configure them now */
- GST_SOURCE_GROUP_LOCK (group);
-
- if (group->audio_sink) {
- GST_INFO_OBJECT (playbin, "setting custom audio sink %" GST_PTR_FORMAT,
- group->audio_sink);
- gst_play_sink_set_sink (playbin->playsink, GST_PLAY_SINK_TYPE_AUDIO,
- group->audio_sink);
- }
-
- if (group->video_sink) {
- GST_INFO_OBJECT (playbin, "setting custom video sink %" GST_PTR_FORMAT,
- group->video_sink);
- gst_play_sink_set_sink (playbin->playsink, GST_PLAY_SINK_TYPE_VIDEO,
- group->video_sink);
- }
+ if (combine->sinkpad) {
+ /* Release playsink sink pad */
+ GST_LOG_OBJECT (playbin, "Releasing playsink pad");
+ gst_play_sink_release_pad (playbin->playsink, combine->sinkpad);
+ gst_object_unref (combine->sinkpad);
+ combine->sinkpad = NULL;
+ }
- if (group->text_sink) {
- GST_INFO_OBJECT (playbin, "setting custom text sink %" GST_PTR_FORMAT,
- group->text_sink);
- gst_play_sink_set_sink (playbin->playsink, GST_PLAY_SINK_TYPE_TEXT,
- group->text_sink);
- }
+ /* Release combiner */
+ GST_FIXME_OBJECT (playbin, "Release combiner");
+ remove_combiner (playbin, combine);
+ } else if (!is_active && is_selected) {
+ GST_DEBUG_OBJECT (playbin, "Stream type '%s' is now requested",
+ combine->media_type);
- GST_SOURCE_GROUP_UNLOCK (group);
+ /* If we are shutting down, do *not* add more combiners */
+ if (g_atomic_int_get (&playbin->shutdown))
+ continue;
- /* signal the other combiners that they can continue now. */
- GST_PLAY_BIN3_LOCK (playbin);
- /* unblock all combiners */
- for (i = 0; i < PLAYBIN_STREAM_LAST; i++) {
- GstSourceCombine *combine = &playbin->combiner[i];
+ g_assert (combine->sinkpad == NULL);
- if (combine->srcpad) {
- GST_DEBUG_OBJECT (playbin, "unblocking %" GST_PTR_FORMAT,
- combine->srcpad);
- if (combine->block_id) {
- gst_pad_remove_probe (combine->srcpad, combine->block_id);
- combine->block_id = 0;
+ /* Request playsink sink pad */
+ combine->sinkpad =
+ gst_play_sink_request_pad (playbin->playsink, combine->type);
+ gst_object_ref (combine->sinkpad);
+ /* Create combiner if needed and link it */
+ create_combiner (playbin, combine);
+ if (combine->combiner) {
+ res = gst_pad_link (combine->srcpad, combine->sinkpad);
+ GST_DEBUG_OBJECT (playbin, "linked type %s, result: %d",
+ combine->media_type, res);
+ if (res != GST_PAD_LINK_OK) {
+ GST_ELEMENT_ERROR (playbin, CORE, PAD,
+ ("Internal playbin error."),
+ ("Failed to link combiner to sink. Error %d", res));
}
+
}
}
- GST_PLAY_BIN3_UNLOCK (playbin);
- gst_play_sink_reconfigure (playbin->playsink);
}
- GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
+ playbin->active_stream_types = playbin->selected_stream_types;
- if (configure) {
- do_async_done (playbin);
- }
+ GST_PLAY_BIN3_UNLOCK (playbin);
- return;
+ gst_play_sink_reconfigure (playbin->playsink);
-shutdown:
- {
- GST_DEBUG ("ignoring, we are shutting down");
- /* Request a flushing pad from playsink that we then link to the combiner.
- * Then we unblock the combiners so that they stop with a WRONG_STATE
- * instead of a NOT_LINKED error.
- */
- GST_PLAY_BIN3_LOCK (playbin);
- for (i = 0; i < PLAYBIN_STREAM_LAST; i++) {
- GstSourceCombine *combine = &playbin->combiner[i];
+ do_async_done (playbin);
- if (combine->srcpad) {
- if (combine->sinkpad == NULL) {
- GST_DEBUG_OBJECT (playbin, "requesting new flushing sink pad");
- combine->sinkpad =
- gst_play_sink_request_pad (playbin->playsink,
- GST_PLAY_SINK_TYPE_FLUSHING);
- gst_object_ref (combine->sinkpad);
- res = gst_pad_link (combine->srcpad, combine->sinkpad);
- GST_DEBUG_OBJECT (playbin, "linked flushing, result: %d", res);
- }
- GST_DEBUG_OBJECT (playbin, "unblocking %" GST_PTR_FORMAT,
- combine->srcpad);
- if (combine->block_id) {
- gst_pad_remove_probe (combine->srcpad, combine->block_id);
- combine->block_id = 0;
- }
- }
- }
- GST_PLAY_BIN3_UNLOCK (playbin);
- return;
- }
+ GST_DEBUG_OBJECT (playbin, "selected_stream_types : %" STREAM_TYPES_FORMAT,
+ STREAM_TYPES_ARGS (playbin->selected_stream_types));
+ GST_DEBUG_OBJECT (playbin, "active_stream_types : %" STREAM_TYPES_FORMAT,
+ STREAM_TYPES_ARGS (playbin->active_stream_types));
+
+ return;
}
-/* FIXME : DISABLED UNTIL GAPLESS IS IMPLEMENTED */
-#if 0
static void
-drained_cb (GstElement * decodebin, GstSourceGroup * group)
+about_to_finish_cb (GstElement * uridecodebin, GstSourceGroup * group)
{
- GstPlayBin3 *playbin;
-
- playbin = group->playbin;
-
+ GstPlayBin3 *playbin = group->playbin;
GST_DEBUG_OBJECT (playbin, "about to finish in group %p", group);
/* after this call, we should have a next group to activate or we EOS */
g_signal_emit (G_OBJECT (playbin),
gst_play_bin3_signals[SIGNAL_ABOUT_TO_FINISH], 0, NULL);
+#if 0
/* now activate the next group. If the app did not set a uri, this will
* fail and we can do EOS */
- setup_next_source (playbin, GST_STATE_PAUSED);
-}
+ setup_next_source (playbin);
#endif
+}
#if 0 /* AUTOPLUG DISABLED */
/* Like gst_element_factory_can_sink_any_caps() but doesn't
{
GST_DEBUG_OBJECT (playbin, "locked_state %d on group %p", locked, group);
- if (group->urisourcebin)
- gst_element_set_locked_state (group->urisourcebin, locked);
- if (group->suburisourcebin)
- gst_element_set_locked_state (group->suburisourcebin, locked);
+ if (group->uridecodebin)
+ gst_element_set_locked_state (group->uridecodebin, locked);
return TRUE;
}
return TRUE;
}
-static void
-urisrc_pad_added (GstElement * urisrc, GstPad * pad, GstSourceGroup * group)
-{
- GstPadLinkReturn res;
- GstPad *sinkpad = NULL;
- GstPlayBin3 *playbin;
-
- GST_SOURCE_GROUP_LOCK (group);
- playbin = group->playbin;
- if (urisrc == group->urisourcebin) {
- /* Primary stream, link to the main pad of decodebin3 */
- sinkpad = gst_element_get_static_pad (playbin->decodebin, "sink");
- if (gst_pad_is_linked (sinkpad)) {
- gst_object_unref (GST_OBJECT (sinkpad));
- sinkpad = NULL;
- }
- }
- GST_SOURCE_GROUP_UNLOCK (group);
-
- if (sinkpad == NULL) {
- /* Auxiliary stream, request a new pad from decodebin */
- if ((sinkpad = gst_element_get_request_pad (playbin->decodebin, "sink_%u"))) {
- g_object_set_data (G_OBJECT (pad), "playbin.sinkpad", sinkpad);
- }
- }
- if (sinkpad) {
- GST_DEBUG_OBJECT (playbin, "New pad %" GST_PTR_FORMAT
- " from urisourcebin %" GST_PTR_FORMAT " linking to %"
- GST_PTR_FORMAT, pad, urisrc, sinkpad);
-
- res = gst_pad_link (pad, sinkpad);
- gst_object_unref (sinkpad);
-
- if (GST_PAD_LINK_FAILED (res))
- goto link_failed;
- }
- return;
-
-link_failed:
- {
- GST_ERROR_OBJECT (playbin,
- "failed to link pad %s:%s to decodebin, reason %s (%d)",
- GST_DEBUG_PAD_NAME (pad), gst_pad_link_get_name (res), res);
- return;
- }
-}
-
-static void
-urisrc_pad_removed_cb (GstElement * urisrc, GstPad * pad,
- GstSourceGroup * group)
-{
-}
-
-/* must be called with PLAY_BIN_LOCK */
-static GstStateChangeReturn
-activate_decodebin (GstPlayBin3 * playbin, GstState target)
-{
- GstStateChangeReturn state_ret;
- GstElement *decodebin = NULL;
-
- if (playbin->decodebin_active)
- return GST_STATE_CHANGE_SUCCESS;
-
- GST_LOG_OBJECT (playbin, "Adding and activating decodebin");
-
- if (!make_or_reuse_element (playbin, "decodebin3", &playbin->decodebin))
- goto no_decodebin;
- decodebin = playbin->decodebin;
-
- /* connect pads and other things */
- playbin->db_pad_added_id = g_signal_connect (decodebin, "pad-added",
- G_CALLBACK (pad_added_cb), playbin);
- playbin->db_pad_removed_id = g_signal_connect (decodebin, "pad-removed",
- G_CALLBACK (pad_removed_cb), playbin);
- playbin->db_select_stream_id = g_signal_connect (decodebin, "select-stream",
- G_CALLBACK (select_stream_cb), playbin);
- /* is called when the decodebin is out of data and we can switch to the
- * next uri */
-#if 0
- /* FIXME: Re-enable if/when decodebin3 supports 'drained' */
- playbin->db_drained_id =
- g_signal_connect (decodebin, "drained", G_CALLBACK (drained_cb), playbin);
-#endif
-
- gst_element_set_locked_state (decodebin, TRUE);
- if ((state_ret =
- gst_element_set_state (decodebin,
- target)) == GST_STATE_CHANGE_FAILURE)
- goto decodebin_failure;
- gst_element_set_locked_state (decodebin, FALSE);
-
- playbin->decodebin_active = TRUE;
-
- return state_ret;
-
-
-no_decodebin:
- {
- GstMessage *msg;
-
- msg =
- gst_missing_element_message_new (GST_ELEMENT_CAST (playbin),
- "decodebin3");
- gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
-
- GST_ELEMENT_ERROR (playbin, CORE, MISSING_PLUGIN,
- (_("Could not create \"decodebin3\" element.")), (NULL));
-
- goto error_cleanup;
- }
-decodebin_failure:
- {
- GST_DEBUG_OBJECT (playbin, "failed state change of decodebin");
- goto error_cleanup;
- }
-error_cleanup:{
- if (decodebin) {
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_pad_added_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_pad_removed_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_drained_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_select_stream_id);
- gst_element_set_state (decodebin, GST_STATE_NULL);
- gst_bin_remove (GST_BIN_CAST (playbin), decodebin);
- }
- return GST_STATE_CHANGE_FAILURE;
- }
-}
-
-/* must be called with PLAY_BIN_LOCK */
-static void
-deactivate_decodebin (GstPlayBin3 * playbin)
-{
- if (playbin->decodebin) {
- GST_LOG_OBJECT (playbin, "Deactivating and removing decodebin");
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_pad_added_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_pad_removed_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_drained_id);
- REMOVE_SIGNAL (playbin->decodebin, playbin->db_select_stream_id);
- gst_bin_remove (GST_BIN_CAST (playbin), playbin->decodebin);
- playbin->decodebin_active = FALSE;
- playbin->active_stream_types = 0;
- }
-}
static void
source_setup_cb (GstElement * element, GstElement * source,
/* must be called with PLAY_BIN_LOCK */
static GstStateChangeReturn
-activate_group (GstPlayBin3 * playbin, GstSourceGroup * group, GstState target)
+activate_group (GstPlayBin3 * playbin, GstSourceGroup * group)
{
- GstElement *urisrcbin = NULL;
- GstElement *suburisrcbin = NULL;
+ GstElement *uridecodebin = NULL;
GstPlayFlags flags;
gboolean audio_sink_activated = FALSE;
gboolean video_sink_activated = FALSE;
}
- if (!make_or_reuse_element (playbin, "urisourcebin", &group->urisourcebin))
- goto no_urisrcbin;
- urisrcbin = group->urisourcebin;
+ if (!make_or_reuse_element (playbin, "uridecodebin3", &group->uridecodebin))
+ goto no_uridecodebin;
+ uridecodebin = group->uridecodebin;
flags = gst_play_sink_get_flags (playbin->playsink);
- g_object_set (urisrcbin,
+ g_object_set (uridecodebin,
/* configure connection speed */
"connection-speed", playbin->connection_speed / 1000,
/* configure uri */
"buffer-size", playbin->buffer_size,
"ring-buffer-max-size", playbin->ring_buffer_max_size, NULL);
- /* we have 1 pending no-more-pads */
- group->pending = 1;
-
- group->urisrc_source_setup_id = g_signal_connect (urisrcbin, "source-setup",
+ group->pad_added_id = g_signal_connect (uridecodebin, "pad-added",
+ G_CALLBACK (pad_added_cb), group);
+ group->pad_removed_id = g_signal_connect (uridecodebin,
+ "pad-removed", G_CALLBACK (pad_removed_cb), group);
+ group->select_stream_id = g_signal_connect (uridecodebin, "select-stream",
+ G_CALLBACK (select_stream_cb), group);
+ group->source_setup_id = g_signal_connect (uridecodebin, "source-setup",
G_CALLBACK (source_setup_cb), group);
+ group->about_to_finish_id =
+ g_signal_connect (uridecodebin, "about-to-finish",
+ G_CALLBACK (about_to_finish_cb), group);
-#if 0 /* AUTOPLUG DISABLED */
- /* will be called when a new media type is found. We return a list of decoders
- * including sinks for decodebin to try */
- group->autoplug_factories_id =
- g_signal_connect (urisrcbin, "autoplug-factories",
- G_CALLBACK (autoplug_factories_cb), group);
- group->autoplug_select_id =
- g_signal_connect (urisrcbin, "autoplug-select",
- G_CALLBACK (autoplug_select_cb), group);
- group->autoplug_continue_id =
- g_signal_connect (urisrcbin, "autoplug-continue",
- G_CALLBACK (autoplug_continue_cb), group);
- group->autoplug_query_id =
- g_signal_connect (urisrcbin, "autoplug-query",
- G_CALLBACK (autoplug_query_cb), group);
-#endif
-
- group->urisrc_pad_added_id = g_signal_connect (urisrcbin, "pad-added",
- G_CALLBACK (urisrc_pad_added), group);
- group->urisrc_pad_removed_id = g_signal_connect (urisrcbin,
- "pad-removed", G_CALLBACK (urisrc_pad_removed_cb), group);
-
- if (group->suburi) {
- /* subtitles */
- if (!make_or_reuse_element (playbin, "urisourcebin",
- &group->suburisourcebin))
- goto no_urisrcbin;
- suburisrcbin = group->suburisourcebin;
-
- g_object_set (suburisrcbin,
- /* configure connection speed */
- "connection-speed", playbin->connection_speed,
- /* configure uri */
- "uri", group->suburi, NULL);
-
- /* connect pads and other things */
- group->sub_pad_added_id = g_signal_connect (suburisrcbin, "pad-added",
- G_CALLBACK (urisrc_pad_added), group);
- group->sub_pad_removed_id = g_signal_connect (suburisrcbin,
- "pad-removed", G_CALLBACK (urisrc_pad_removed_cb), group);
-
-#if 0
- group->sub_autoplug_continue_id =
- g_signal_connect (suburisrcbin, "autoplug-continue",
- G_CALLBACK (autoplug_continue_cb), group);
-
- group->sub_autoplug_query_id =
- g_signal_connect (suburisrcbin, "autoplug-query",
- G_CALLBACK (autoplug_query_cb), group);
-#endif
-
- /* we have 2 pending no-more-pads */
- group->pending = 2;
- group->sub_pending = TRUE;
- } else {
- group->sub_pending = FALSE;
- }
+ if (group->suburi)
+ g_object_set (group->uridecodebin, "suburi", group->suburi, NULL);
/* release the group lock before setting the state of the source bins, they
* might fire signals in this thread that we need to handle with the
* group_lock taken. */
GST_SOURCE_GROUP_UNLOCK (group);
- if (suburisrcbin) {
- if (gst_element_set_state (suburisrcbin,
- target) == GST_STATE_CHANGE_FAILURE) {
- GST_DEBUG_OBJECT (playbin,
- "failed state change of subtitle urisourcebin");
- GST_SOURCE_GROUP_LOCK (group);
-
- REMOVE_SIGNAL (suburisrcbin, group->sub_pad_added_id);
- REMOVE_SIGNAL (suburisrcbin, group->sub_pad_removed_id);
-#if 0
- REMOVE_SIGNAL (suburisrcbin, group->sub_autoplug_continue_id);
- REMOVE_SIGNAL (suburisrcbin, group->sub_autoplug_query_id);
-#endif
- /* Might already be removed because of an error message */
- if (GST_OBJECT_PARENT (suburisrcbin) == GST_OBJECT_CAST (playbin))
- gst_bin_remove (GST_BIN_CAST (playbin), suburisrcbin);
- if (group->sub_pending) {
- group->pending--;
- group->sub_pending = FALSE;
- }
- gst_element_set_state (suburisrcbin, GST_STATE_READY);
- g_free (group->suburi);
- group->suburi = NULL;
- GST_SOURCE_GROUP_UNLOCK (group);
- }
- }
if ((state_ret =
- gst_element_set_state (urisrcbin,
- target)) == GST_STATE_CHANGE_FAILURE)
- goto urisrcbin_failure;
+ gst_element_set_state (uridecodebin,
+ GST_STATE_PAUSED)) == GST_STATE_CHANGE_FAILURE)
+ goto uridecodebin_failure;
GST_SOURCE_GROUP_LOCK (group);
/* allow state changes of the playbin affect the group elements now */
return state_ret;
/* ERRORS */
-no_urisrcbin:
+no_uridecodebin:
{
GstMessage *msg;
GST_SOURCE_GROUP_UNLOCK (group);
msg =
gst_missing_element_message_new (GST_ELEMENT_CAST (playbin),
- "urisourcebin");
+ "uridecodebin3");
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
GST_ELEMENT_ERROR (playbin, CORE, MISSING_PLUGIN,
- (_("Could not create \"urisourcebin\" element.")), (NULL));
+ (_("Could not create \"uridecodebin3\" element.")), (NULL));
GST_SOURCE_GROUP_LOCK (group);
goto error_cleanup;
}
-urisrcbin_failure:
+uridecodebin_failure:
{
- GST_DEBUG_OBJECT (playbin, "failed state change of urisrcbin");
+ GST_DEBUG_OBJECT (playbin, "failed state change of uridecodebin");
GST_SOURCE_GROUP_LOCK (group);
goto error_cleanup;
}
error_cleanup:
{
+ group->selected_stream_types = 0;
+
/* delete any custom sinks we might have */
if (group->audio_sink) {
/* If this is a automatically created sink set it to NULL */
}
group->text_sink = NULL;
- if (urisrcbin) {
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_pad_added_id);
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_pad_removed_id);
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_source_setup_id);
+ if (uridecodebin) {
+ REMOVE_SIGNAL (group->uridecodebin, group->pad_added_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->pad_removed_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->select_stream_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->source_setup_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->about_to_finish_id);
#if 0
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_factories_id);
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_select_id);
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_query_id);
#endif
- gst_element_set_state (urisrcbin, GST_STATE_NULL);
- gst_bin_remove (GST_BIN_CAST (playbin), urisrcbin);
+ gst_element_set_state (uridecodebin, GST_STATE_NULL);
+ gst_bin_remove (GST_BIN_CAST (playbin), uridecodebin);
}
GST_SOURCE_GROUP_UNLOCK (group);
}
}
-/* unlink a group of urisrcbin from the decodebin.
+/* unlink a group of uridecodebin from the decodebin.
* must be called with PLAY_BIN_LOCK */
static gboolean
deactivate_group (GstPlayBin3 * playbin, GstSourceGroup * group)
{
- gint i;
-
g_return_val_if_fail (group->active, FALSE);
g_return_val_if_fail (group->valid, FALSE);
GST_SOURCE_GROUP_LOCK (group);
group->active = FALSE;
- for (i = 0; i < PLAYBIN_STREAM_LAST; i++) {
- GstSourceCombine *combine = &playbin->combiner[i];
-
- GST_DEBUG_OBJECT (playbin, "unlinking combiner %s", combine->media_type);
-
- if (combine->srcpad) {
- source_combine_remove_pads (playbin, combine);
- }
-
- if (combine->combiner) {
- gint n;
- /* release and unref requests pad from the combiner */
- for (n = 0; n < combine->channels->len; n++) {
- GstPad *sinkpad = g_ptr_array_index (combine->channels, n);
+ group->selected_stream_types = 0;
+ /* Update global selected_stream_types */
+ playbin->selected_stream_types =
+ playbin->groups[0].selected_stream_types | playbin->
+ groups[1].selected_stream_types;
+ if (playbin->active_stream_types != playbin->selected_stream_types)
+ reconfigure_output (playbin);
- gst_element_release_request_pad (combine->combiner, sinkpad);
- gst_object_unref (sinkpad);
- }
- g_ptr_array_set_size (combine->channels, 0);
-
- gst_element_set_state (combine->combiner, GST_STATE_NULL);
- gst_bin_remove (GST_BIN_CAST (playbin), combine->combiner);
- combine->combiner = NULL;
- }
- }
#if 0
/* delete any custom sinks we might have.
* conditionally set them to null if they aren't inside playsink yet */
group->text_sink = NULL;
#endif
- if (group->urisourcebin) {
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_pad_added_id);
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_pad_removed_id);
- REMOVE_SIGNAL (group->urisourcebin, group->urisrc_source_setup_id);
+ if (group->uridecodebin) {
+ REMOVE_SIGNAL (group->uridecodebin, group->pad_added_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->pad_removed_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->select_stream_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->source_setup_id);
+ REMOVE_SIGNAL (group->uridecodebin, group->about_to_finish_id);
#if 0
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_factories_id);
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_select_id);
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_continue_id);
REMOVE_SIGNAL (group->urisourcebin, group->autoplug_query_id);
#endif
- gst_bin_remove (GST_BIN_CAST (playbin), group->urisourcebin);
- }
-
- if (group->suburisourcebin) {
- REMOVE_SIGNAL (group->suburisourcebin, group->sub_pad_added_id);
- REMOVE_SIGNAL (group->suburisourcebin, group->sub_pad_removed_id);
-#if 0
- REMOVE_SIGNAL (group->suburisourcebin, group->sub_autoplug_continue_id);
- REMOVE_SIGNAL (group->suburisourcebin, group->sub_autoplug_query_id);
-#endif
-
- /* Might already be removed because of errors */
- if (GST_OBJECT_PARENT (group->suburisourcebin) == GST_OBJECT_CAST (playbin))
- gst_bin_remove (GST_BIN_CAST (playbin), group->suburisourcebin);
+ gst_element_set_state (group->uridecodebin, GST_STATE_NULL);
+ gst_bin_remove (GST_BIN_CAST (playbin), group->uridecodebin);
}
GST_SOURCE_GROUP_UNLOCK (group);
* configured. It swaps out the current_group and activates the valid
* next_group. */
static GstStateChangeReturn
-setup_next_source (GstPlayBin3 * playbin, GstState target)
+setup_next_source (GstPlayBin3 * playbin)
{
GstSourceGroup *new_group, *old_group;
GstStateChangeReturn state_ret;
playbin->curr_group = new_group;
playbin->next_group = old_group;
- /* Get decodebin ready now */
- if ((state_ret =
- activate_decodebin (playbin, target)) == GST_STATE_CHANGE_FAILURE)
- goto activate_failed;
-
/* activate the new group */
- if ((state_ret =
- activate_group (playbin, new_group,
- target)) == GST_STATE_CHANGE_FAILURE)
+ state_ret = activate_group (playbin, new_group);
+ if (state_ret == GST_STATE_CHANGE_FAILURE)
goto activate_failed;
GST_PLAY_BIN3_UNLOCK (playbin);
no_next_group:
{
GST_DEBUG_OBJECT (playbin, "no next group");
- if (target == GST_STATE_READY && new_group && new_group->uri == NULL)
- GST_ELEMENT_ERROR (playbin, RESOURCE, NOT_FOUND, ("No URI set"), (NULL));
GST_PLAY_BIN3_UNLOCK (playbin);
return GST_STATE_CHANGE_FAILURE;
}
break;
case GST_STATE_CHANGE_READY_TO_NULL:
/* we go async to PAUSED, so if that fails, we never make it to PAUSED
- * and no state change PAUSED to READY passes here,
- * though it is a nice-to-have ... */
+ * and we will never be called with the GST_STATE_CHANGE_PAUSED_TO_READY.
+ * Make sure we do go through the same steps (see above) for
+ * proper cleanup */
if (!g_atomic_int_get (&playbin->shutdown)) {
do_save = TRUE;
goto async_down;
switch (transition) {
case GST_STATE_CHANGE_READY_TO_PAUSED:
- if ((ret =
- setup_next_source (playbin,
- GST_STATE_PAUSED)) == GST_STATE_CHANGE_FAILURE)
+ if ((ret = setup_next_source (playbin)) == GST_STATE_CHANGE_FAILURE)
goto failure;
if (ret == GST_STATE_CHANGE_SUCCESS)
ret = GST_STATE_CHANGE_ASYNC;
/* also do missed state change down to READY */
if (do_save)
save_current_group (playbin);
- /* Deactive the groups, set the urisrcbins to NULL
- * and unref them.
- */
+ /* Deactive the groups, set uridecodebin to NULL and unref it */
for (i = 0; i < 2; i++) {
if (playbin->groups[i].active && playbin->groups[i].valid) {
deactivate_group (playbin, &playbin->groups[i]);
playbin->groups[i].valid = FALSE;
}
- if (playbin->groups[i].urisourcebin) {
- gst_element_set_state (playbin->groups[i].urisourcebin,
+ if (playbin->groups[i].uridecodebin) {
+ gst_element_set_state (playbin->groups[i].uridecodebin,
GST_STATE_NULL);
- gst_object_unref (playbin->groups[i].urisourcebin);
- playbin->groups[i].urisourcebin = NULL;
+ gst_object_unref (playbin->groups[i].uridecodebin);
+ playbin->groups[i].uridecodebin = NULL;
}
- if (playbin->groups[i].suburisourcebin) {
- gst_element_set_state (playbin->groups[i].suburisourcebin,
- GST_STATE_NULL);
- gst_object_unref (playbin->groups[i].suburisourcebin);
- playbin->groups[i].suburisourcebin = NULL;
- }
- }
-
- deactivate_decodebin (playbin);
- if (playbin->decodebin) {
- gst_object_unref (playbin->decodebin);
- playbin->decodebin = NULL;
- playbin->decodebin_active = FALSE;
}
/* Set our sinks back to NULL, they might not be child of playbin */