mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-01 14:11:15 +00:00
d514e79bee
With contributions from Jan Schmidt <jan@centricular.com> * decodebin3 and playbin3 have the same purpose as the decodebin and playbin elements, except make usage of more 1.x features and the new GstStream API. This allows them to be more memory/cpu efficient. * parsebin is a new element that demuxers/depayloads/parses an incoming stream and exposes elementary streams. It is used by decodebin3. It also automatically creates GstStream and GstStreamCollection for elements that don't natively create them and sends the corresponding events and messages * Any application using playbin can use playbin3 by setting the env variable USE_PLAYBIN3=1 without reconfiguration/recompilation.
6079 lines
198 KiB
C
6079 lines
198 KiB
C
/* GStreamer
|
|
* Copyright (C) <2007> Wim Taymans <wim.taymans@gmail.com>
|
|
* Copyright (C) <2011> Sebastian Dröge <sebastian.droege@collabora.co.uk>
|
|
* Copyright (C) <2013> Collabora Ltd.
|
|
* Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
|
|
* Copyright (C) <2015> Jan Schmidt <jan@centricular.com>
|
|
*
|
|
* 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 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., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:element-playbin3
|
|
*
|
|
* Playbin provides a stand-alone everything-in-one abstraction for an
|
|
* audio and/or video player.
|
|
*
|
|
* Playbin can handle both audio and video files and features
|
|
* <itemizedlist>
|
|
* <listitem>
|
|
* automatic file type recognition and based on that automatic
|
|
* selection and usage of the right audio/video/subtitle demuxers/decoders
|
|
* </listitem>
|
|
* <listitem>
|
|
* visualisations for audio files
|
|
* </listitem>
|
|
* <listitem>
|
|
* subtitle support for video files. Subtitles can be store in external
|
|
* files.
|
|
* </listitem>
|
|
* <listitem>
|
|
* stream selection between different video/audio/subtitles streams
|
|
* </listitem>
|
|
* <listitem>
|
|
* meta info (tag) extraction
|
|
* </listitem>
|
|
* <listitem>
|
|
* easy access to the last video sample
|
|
* </listitem>
|
|
* <listitem>
|
|
* buffering when playing streams over a network
|
|
* </listitem>
|
|
* <listitem>
|
|
* volume control with mute option
|
|
* </listitem>
|
|
* </itemizedlist>
|
|
*
|
|
* <refsect2>
|
|
* <title>Usage</title>
|
|
* <para>
|
|
* A playbin element can be created just like any other element using
|
|
* gst_element_factory_make(). The file/URI to play should be set via the #GstPlayBin3:uri
|
|
* property. This must be an absolute URI, relative file paths are not allowed.
|
|
* Example URIs are file:///home/joe/movie.avi or http://www.joedoe.com/foo.ogg
|
|
*
|
|
* Playbin is a #GstPipeline. It will notify the application of everything
|
|
* that's happening (errors, end of stream, tags found, state changes, etc.)
|
|
* by posting messages on its #GstBus. The application needs to watch the
|
|
* bus.
|
|
*
|
|
* Playback can be initiated by setting the element to PLAYING state using
|
|
* gst_element_set_state(). Note that the state change will take place in
|
|
* the background in a separate thread, when the function returns playback
|
|
* is probably not happening yet and any errors might not have occured yet.
|
|
* Applications using playbin should ideally be written to deal with things
|
|
* completely asynchroneous.
|
|
*
|
|
* When playback has finished (an EOS message has been received on the bus)
|
|
* or an error has occured (an ERROR message has been received on the bus) or
|
|
* the user wants to play a different track, playbin should be set back to
|
|
* READY or NULL state, then the #GstPlayBin3:uri property should be set to the
|
|
* new location and then playbin be set to PLAYING state again.
|
|
*
|
|
* Seeking can be done using gst_element_seek_simple() or gst_element_seek()
|
|
* on the playbin element. Again, the seek will not be executed
|
|
* instantaneously, but will be done in a background thread. When the seek
|
|
* call returns the seek will most likely still be in process. An application
|
|
* may wait for the seek to finish (or fail) using gst_element_get_state() with
|
|
* -1 as the timeout, but this will block the user interface and is not
|
|
* recommended at all.
|
|
*
|
|
* Applications may query the current position and duration of the stream
|
|
* via gst_element_query_position() and gst_element_query_duration() and
|
|
* setting the format passed to GST_FORMAT_TIME. If the query was successful,
|
|
* the duration or position will have been returned in units of nanoseconds.
|
|
* </para>
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Advanced Usage: specifying the audio and video sink</title>
|
|
* <para>
|
|
* By default, if no audio sink or video sink has been specified via the
|
|
* #GstPlayBin3:audio-sink or #GstPlayBin3:video-sink property, playbin will use the autoaudiosink
|
|
* and autovideosink elements to find the first-best available output method.
|
|
* This should work in most cases, but is not always desirable. Often either
|
|
* the user or application might want to specify more explicitly what to use
|
|
* for audio and video output.
|
|
*
|
|
* If the application wants more control over how audio or video should be
|
|
* output, it may create the audio/video sink elements itself (for example
|
|
* using gst_element_factory_make()) and provide them to playbin using the
|
|
* #GstPlayBin3:audio-sink or #GstPlayBin3:video-sink property.
|
|
*
|
|
* GNOME-based applications, for example, will usually want to create
|
|
* gconfaudiosink and gconfvideosink elements and make playbin use those,
|
|
* so that output happens to whatever the user has configured in the GNOME
|
|
* Multimedia System Selector configuration dialog.
|
|
*
|
|
* The sink elements do not necessarily need to be ready-made sinks. It is
|
|
* possible to create container elements that look like a sink to playbin,
|
|
* but in reality contain a number of custom elements linked together. This
|
|
* can be achieved by creating a #GstBin and putting elements in there and
|
|
* linking them, and then creating a sink #GstGhostPad for the bin and pointing
|
|
* it to the sink pad of the first element within the bin. This can be used
|
|
* for a number of purposes, for example to force output to a particular
|
|
* format or to modify or observe the data before it is output.
|
|
*
|
|
* It is also possible to 'suppress' audio and/or video output by using
|
|
* 'fakesink' elements (or capture it from there using the fakesink element's
|
|
* "handoff" signal, which, nota bene, is fired from the streaming thread!).
|
|
* </para>
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Retrieving Tags and Other Meta Data</title>
|
|
* <para>
|
|
* Most of the common meta data (artist, title, etc.) can be retrieved by
|
|
* watching for TAG messages on the pipeline's bus (see above).
|
|
*
|
|
* Other more specific meta information like width/height/framerate of video
|
|
* streams or samplerate/number of channels of audio streams can be obtained
|
|
* from the negotiated caps on the sink pads of the sinks.
|
|
* </para>
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Buffering</title>
|
|
* Playbin handles buffering automatically for the most part, but applications
|
|
* need to handle parts of the buffering process as well. Whenever playbin is
|
|
* buffering, it will post BUFFERING messages on the bus with a percentage
|
|
* value that shows the progress of the buffering process. Applications need
|
|
* to set playbin to PLAYING or PAUSED state in response to these messages.
|
|
* They may also want to convey the buffering progress to the user in some
|
|
* way. Here is how to extract the percentage information from the message:
|
|
* |[
|
|
* switch (GST_MESSAGE_TYPE (msg)) {
|
|
* case GST_MESSAGE_BUFFERING: {
|
|
* gint percent = 0;
|
|
* gst_message_parse_buffering (msg, &percent);
|
|
* g_print ("Buffering (%u percent done)", percent);
|
|
* break;
|
|
* }
|
|
* ...
|
|
* }
|
|
* ]|
|
|
* Note that applications should keep/set the pipeline in the PAUSED state when
|
|
* a BUFFERING message is received with a buffer percent value < 100 and set
|
|
* the pipeline back to PLAYING state when a BUFFERING message with a value
|
|
* of 100 percent is received (if PLAYING is the desired state, that is).
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Embedding the video window in your application</title>
|
|
* By default, playbin (or rather the video sinks used) will create their own
|
|
* window. Applications will usually want to force output to a window of their
|
|
* own, however. This can be done using the #GstVideoOverlay interface, which most
|
|
* video sinks implement. See the documentation there for more details.
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Specifying which CD/DVD device to use</title>
|
|
* The device to use for CDs/DVDs needs to be set on the source element
|
|
* playbin creates before it is opened. The most generic way of doing this
|
|
* is to connect to playbin's "source-setup" (or "notify::source") signal,
|
|
* which will be emitted by playbin when it has created the source element
|
|
* for a particular URI. In the signal callback you can check if the source
|
|
* element has a "device" property and set it appropriately. In some cases
|
|
* the device can also be set as part of the URI, but it depends on the
|
|
* elements involved if this will work or not. For example, for DVD menu
|
|
* playback, the following syntax might work (if the resindvd plugin is used):
|
|
* dvd://[/path/to/device]
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Handling redirects</title>
|
|
* <para>
|
|
* Some elements may post 'redirect' messages on the bus to tell the
|
|
* application to open another location. These are element messages containing
|
|
* a structure named 'redirect' along with a 'new-location' field of string
|
|
* type. The new location may be a relative or an absolute URI. Examples
|
|
* for such redirects can be found in many quicktime movie trailers.
|
|
* </para>
|
|
* </refsect2>
|
|
* <refsect2>
|
|
* <title>Examples</title>
|
|
* |[
|
|
* gst-launch-1.0 -v playbin uri=file:///path/to/somefile.mp4
|
|
* ]| This will play back the given AVI video file, given that the video and
|
|
* audio decoders required to decode the content are installed. Since no
|
|
* special audio sink or video sink is supplied (via playbin's audio-sink or
|
|
* video-sink properties) playbin will try to find a suitable audio and
|
|
* video sink automatically using the autoaudiosink and autovideosink elements.
|
|
* |[
|
|
* gst-launch-1.0 -v playbin uri=cdda://4
|
|
* ]| This will play back track 4 on an audio CD in your disc drive (assuming
|
|
* the drive is detected automatically by the plugin).
|
|
* |[
|
|
* gst-launch-1.0 -v playbin uri=dvd://
|
|
* ]| This will play back the DVD in your disc drive (assuming
|
|
* the drive is detected automatically by the plugin).
|
|
* </refsect2>
|
|
*/
|
|
|
|
/* 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
|
|
|
|
#include <string.h>
|
|
#include <gst/gst.h>
|
|
|
|
#include <gst/gst-i18n-plugin.h>
|
|
#include <gst/pbutils/pbutils.h>
|
|
#include <gst/audio/streamvolume.h>
|
|
#include <gst/video/video-info.h>
|
|
#include <gst/video/video-multiview.h>
|
|
#include <gst/video/videooverlay.h>
|
|
#include <gst/video/navigation.h>
|
|
#include <gst/video/colorbalance.h>
|
|
#include "gstplay-enum.h"
|
|
#include "gstplayback.h"
|
|
#include "gstplaysink.h"
|
|
#include "gstsubtitleoverlay.h"
|
|
#include "gstplaybackutils.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_play_bin3_debug);
|
|
#define GST_CAT_DEFAULT gst_play_bin3_debug
|
|
|
|
#define GST_TYPE_PLAY_BIN (gst_play_bin3_get_type())
|
|
#define GST_PLAY_BIN3(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_PLAY_BIN,GstPlayBin3))
|
|
#define GST_PLAY_BIN3_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_PLAY_BIN,GstPlayBin3Class))
|
|
#define GST_IS_PLAY_BIN(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_PLAY_BIN))
|
|
#define GST_IS_PLAY_BIN_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_PLAY_BIN))
|
|
|
|
#define ULONG_TO_POINTER(number) ((gpointer) (guintptr) (number))
|
|
#define POINTER_TO_ULONG(number) ((guintptr) (number))
|
|
|
|
#define VOLUME_MAX_DOUBLE 10.0
|
|
|
|
typedef struct _GstPlayBin3 GstPlayBin3;
|
|
typedef struct _GstPlayBin3Class GstPlayBin3Class;
|
|
typedef struct _GstSourceGroup GstSourceGroup;
|
|
typedef struct _GstSourceCombine GstSourceCombine;
|
|
|
|
typedef GstCaps *(*SourceCombineGetMediaCapsFunc) (void);
|
|
|
|
/* has the info for a combiner and provides the link to the sink */
|
|
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 */
|
|
|
|
GstElement *combiner; /* the combiner */
|
|
GPtrArray *channels;
|
|
GstPad *srcpad; /* the source pad of the combiner */
|
|
GstPad *sinkpad; /* the sinkpad of the sink when the combiner
|
|
* is linked
|
|
*/
|
|
gulong block_id;
|
|
|
|
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 */
|
|
|
|
gboolean has_always_ok; /* stream combiner's sink pads have the "always-ok" property */
|
|
};
|
|
|
|
#define GST_SOURCE_GROUP_GET_LOCK(group) (&((GstSourceGroup*)(group))->lock)
|
|
#define GST_SOURCE_GROUP_LOCK(group) (g_mutex_lock (GST_SOURCE_GROUP_GET_LOCK(group)))
|
|
#define GST_SOURCE_GROUP_UNLOCK(group) (g_mutex_unlock (GST_SOURCE_GROUP_GET_LOCK(group)))
|
|
|
|
enum
|
|
{
|
|
PLAYBIN_STREAM_AUDIO = 0,
|
|
PLAYBIN_STREAM_VIDEO,
|
|
PLAYBIN_STREAM_TEXT,
|
|
PLAYBIN_STREAM_LAST
|
|
};
|
|
|
|
/* names matching the enum above */
|
|
static const gchar *stream_type_names[] = {
|
|
"audio", "video", "text"
|
|
};
|
|
|
|
static void avelements_free (gpointer data);
|
|
static GSequence *avelements_create (GstPlayBin3 * playbin,
|
|
gboolean isaudioelement);
|
|
|
|
/* The GstAudioVideoElement structure holding the audio/video decoder
|
|
* and the audio/video sink factories together with field indicating
|
|
* the number of common caps features */
|
|
typedef struct
|
|
{
|
|
GstElementFactory *dec; /* audio:video decoder */
|
|
GstElementFactory *sink; /* audio:video sink */
|
|
guint n_comm_cf; /* number of common caps features */
|
|
} GstAVElement;
|
|
|
|
/* a structure to hold the objects for decoding a uri and the subtitle uri
|
|
*/
|
|
struct _GstSourceGroup
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
|
|
GMutex lock;
|
|
|
|
gboolean valid; /* the group has valid info to start playback */
|
|
gboolean active; /* the group is active */
|
|
|
|
/* properties */
|
|
gchar *uri;
|
|
gchar *suburi;
|
|
GValueArray *streaminfo;
|
|
GstElement *source;
|
|
|
|
|
|
/* urisourcebins for uri and subtitle uri */
|
|
/* FIXME: Just make this an array of uris */
|
|
GstElement *urisourcebin;
|
|
GstElement *suburisourcebin;
|
|
|
|
/* Active sinks for each media type. These are initialized with
|
|
* the configured or currently used sink, otherwise
|
|
* left as NULL and playbin tries to automatically
|
|
* select a good sink */
|
|
GstElement *audio_sink;
|
|
GstElement *video_sink;
|
|
GstElement *text_sink;
|
|
|
|
gint pending;
|
|
gboolean sub_pending;
|
|
|
|
/* primary uri signals */
|
|
gulong urisrc_pad_added_id;
|
|
gulong urisrc_pad_removed_id;
|
|
gulong notify_source_id;
|
|
gulong autoplug_factories_id;
|
|
gulong autoplug_select_id;
|
|
gulong autoplug_continue_id;
|
|
gulong autoplug_query_id;
|
|
|
|
/* subtitle uri signals */
|
|
gulong sub_pad_added_id;
|
|
gulong sub_pad_removed_id;
|
|
gulong sub_autoplug_continue_id;
|
|
gulong sub_autoplug_query_id;
|
|
|
|
gulong block_id;
|
|
|
|
GMutex stream_changed_pending_lock;
|
|
gboolean stream_changed_pending;
|
|
|
|
/* buffering message stored for after switching */
|
|
GstMessage *pending_buffering_msg;
|
|
};
|
|
|
|
#define GST_PLAY_BIN3_GET_LOCK(bin) (&((GstPlayBin3*)(bin))->lock)
|
|
#define GST_PLAY_BIN3_LOCK(bin) (g_rec_mutex_lock (GST_PLAY_BIN3_GET_LOCK(bin)))
|
|
#define GST_PLAY_BIN3_UNLOCK(bin) (g_rec_mutex_unlock (GST_PLAY_BIN3_GET_LOCK(bin)))
|
|
|
|
/* lock to protect dynamic callbacks, like no-more-pads */
|
|
#define GST_PLAY_BIN3_DYN_LOCK(bin) g_mutex_lock (&(bin)->dyn_lock)
|
|
#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
|
|
|
|
/* unlock for shutdown */
|
|
#define GST_PLAY_BIN3_SHUTDOWN_UNLOCK(bin) \
|
|
GST_PLAY_BIN3_DYN_UNLOCK (bin); \
|
|
|
|
/**
|
|
* GstPlayBin3:
|
|
*
|
|
* playbin element structure
|
|
*/
|
|
struct _GstPlayBin3
|
|
{
|
|
GstPipeline parent;
|
|
|
|
GRecMutex lock; /* to protect group switching */
|
|
|
|
/* the input groups, we use a double buffer to switch between current and next */
|
|
GstSourceGroup groups[2]; /* array with group info */
|
|
GstSourceGroup *curr_group; /* pointer to the currently playing group */
|
|
GstSourceGroup *next_group; /* pointer to the next group */
|
|
|
|
/* combiners for different streams */
|
|
GPtrArray *channels[PLAYBIN_STREAM_LAST]; /* links to combiner pads */
|
|
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 */
|
|
GstStreamType selected_stream_types;
|
|
GstStreamType active_stream_types;
|
|
|
|
/* Decodebin signals */
|
|
gulong db_pad_added_id;
|
|
gulong db_pad_removed_id;
|
|
gulong db_no_more_pads_id;
|
|
gulong db_drained_id;
|
|
gulong db_select_stream_id;
|
|
|
|
/* properties */
|
|
guint64 connection_speed; /* connection speed in bits/sec (0 = unknown) */
|
|
gint current_video; /* the currently selected stream */
|
|
gint current_audio; /* the currently selected stream */
|
|
gint current_text; /* the currently selected stream */
|
|
|
|
gboolean do_stream_selections; /* Set to TRUE when any of current-{video|audio|text} are set to
|
|
say playbin should do backwards-compatibility behaviours */
|
|
|
|
guint64 buffer_duration; /* When buffering, the max buffer duration (ns) */
|
|
guint buffer_size; /* When buffering, the max buffer size (bytes) */
|
|
gboolean force_aspect_ratio;
|
|
|
|
/* Multiview/stereoscopic overrides */
|
|
GstVideoMultiviewFramePacking multiview_mode;
|
|
GstVideoMultiviewFlags multiview_flags;
|
|
|
|
/* our play sink */
|
|
GstPlaySink *playsink;
|
|
|
|
/* the last activated source */
|
|
GstElement *source;
|
|
|
|
/* lock protecting dynamic adding/removing */
|
|
GMutex dyn_lock;
|
|
/* if we are shutting down or not */
|
|
gint shutdown;
|
|
gboolean async_pending; /* async-start has been emitted */
|
|
|
|
GMutex elements_lock;
|
|
guint32 elements_cookie;
|
|
GList *elements; /* factories we can use for selecting elements */
|
|
|
|
gboolean have_selector; /* set to FALSE when we fail to create an
|
|
* input-selector, so that we only post a
|
|
* warning once */
|
|
|
|
gboolean video_pending_flush_finish; /* whether we are pending to send a custom
|
|
* custom-video-flush-finish event
|
|
* on pad activation */
|
|
gboolean audio_pending_flush_finish; /* whether we are pending to send a custom
|
|
* custom-audio-flush-finish event
|
|
* on pad activation */
|
|
gboolean text_pending_flush_finish; /* whether we are pending to send a custom
|
|
* custom-subtitle-flush-finish event
|
|
* on pad activation */
|
|
|
|
GstElement *audio_sink; /* configured audio sink, or NULL */
|
|
GstElement *video_sink; /* configured video sink, or NULL */
|
|
GstElement *text_sink; /* configured text sink, or NULL */
|
|
|
|
GstElement *audio_stream_combiner; /* configured audio stream combiner, or NULL */
|
|
GstElement *video_stream_combiner; /* configured video stream combiner, or NULL */
|
|
GstElement *text_stream_combiner; /* configured text stream combiner, or NULL */
|
|
|
|
GSequence *aelements; /* a list of GstAVElements for audio stream */
|
|
GSequence *velements; /* a list of GstAVElements for video stream */
|
|
|
|
struct
|
|
{
|
|
gboolean valid;
|
|
GstFormat format;
|
|
gint64 duration;
|
|
} duration[5]; /* cached durations */
|
|
|
|
guint64 ring_buffer_max_size; /* 0 means disabled */
|
|
|
|
GList *contexts;
|
|
|
|
/* Active stream collection */
|
|
GstStreamCollection *collection;
|
|
guint collection_notify_id;
|
|
};
|
|
|
|
struct _GstPlayBin3Class
|
|
{
|
|
GstPipelineClass parent_class;
|
|
|
|
/* notify app that the current uri finished decoding and it is possible to
|
|
* queue a new one for gapless playback */
|
|
void (*about_to_finish) (GstPlayBin3 * playbin);
|
|
|
|
/* notify app that number of audio/video/text streams changed */
|
|
void (*video_changed) (GstPlayBin3 * playbin);
|
|
void (*audio_changed) (GstPlayBin3 * playbin);
|
|
void (*text_changed) (GstPlayBin3 * playbin);
|
|
|
|
/* notify app that the tags of audio/video/text streams changed */
|
|
void (*video_tags_changed) (GstPlayBin3 * playbin, gint stream);
|
|
void (*audio_tags_changed) (GstPlayBin3 * playbin, gint stream);
|
|
void (*text_tags_changed) (GstPlayBin3 * playbin, gint stream);
|
|
|
|
/* get audio/video/text tags for a stream */
|
|
GstTagList *(*get_video_tags) (GstPlayBin3 * playbin, gint stream);
|
|
GstTagList *(*get_audio_tags) (GstPlayBin3 * playbin, gint stream);
|
|
GstTagList *(*get_text_tags) (GstPlayBin3 * playbin, gint stream);
|
|
|
|
/* get the last video sample and convert it to the given caps */
|
|
GstSample *(*convert_sample) (GstPlayBin3 * playbin, GstCaps * caps);
|
|
|
|
/* get audio/video/text pad for a stream */
|
|
GstPad *(*get_video_pad) (GstPlayBin3 * playbin, gint stream);
|
|
GstPad *(*get_audio_pad) (GstPlayBin3 * playbin, gint stream);
|
|
GstPad *(*get_text_pad) (GstPlayBin3 * playbin, gint stream);
|
|
};
|
|
|
|
/* props */
|
|
#define DEFAULT_URI NULL
|
|
#define DEFAULT_SUBURI NULL
|
|
#define DEFAULT_SOURCE NULL
|
|
#define DEFAULT_FLAGS GST_PLAY_FLAG_AUDIO | GST_PLAY_FLAG_VIDEO | GST_PLAY_FLAG_TEXT | \
|
|
GST_PLAY_FLAG_SOFT_VOLUME | GST_PLAY_FLAG_DEINTERLACE | \
|
|
GST_PLAY_FLAG_SOFT_COLORBALANCE
|
|
#define DEFAULT_N_VIDEO 0
|
|
#define DEFAULT_CURRENT_VIDEO -1
|
|
#define DEFAULT_N_AUDIO 0
|
|
#define DEFAULT_CURRENT_AUDIO -1
|
|
#define DEFAULT_N_TEXT 0
|
|
#define DEFAULT_CURRENT_TEXT -1
|
|
#define DEFAULT_AUTO_SELECT_STREAMS TRUE
|
|
#define DEFAULT_SUBTITLE_ENCODING NULL
|
|
#define DEFAULT_AUDIO_SINK NULL
|
|
#define DEFAULT_VIDEO_SINK NULL
|
|
#define DEFAULT_VIS_PLUGIN NULL
|
|
#define DEFAULT_TEXT_SINK NULL
|
|
#define DEFAULT_VOLUME 1.0
|
|
#define DEFAULT_MUTE FALSE
|
|
#define DEFAULT_FRAME NULL
|
|
#define DEFAULT_FONT_DESC NULL
|
|
#define DEFAULT_CONNECTION_SPEED 0
|
|
#define DEFAULT_BUFFER_DURATION -1
|
|
#define DEFAULT_BUFFER_SIZE -1
|
|
#define DEFAULT_RING_BUFFER_MAX_SIZE 0
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_URI,
|
|
PROP_CURRENT_URI,
|
|
PROP_SUBURI,
|
|
PROP_CURRENT_SUBURI,
|
|
PROP_SOURCE,
|
|
PROP_FLAGS,
|
|
PROP_N_VIDEO,
|
|
PROP_CURRENT_VIDEO,
|
|
PROP_N_AUDIO,
|
|
PROP_CURRENT_AUDIO,
|
|
PROP_N_TEXT,
|
|
PROP_CURRENT_TEXT,
|
|
PROP_AUTO_SELECT_STREAMS,
|
|
PROP_SUBTITLE_ENCODING,
|
|
PROP_AUDIO_SINK,
|
|
PROP_VIDEO_SINK,
|
|
PROP_VIS_PLUGIN,
|
|
PROP_TEXT_SINK,
|
|
PROP_VIDEO_STREAM_COMBINER,
|
|
PROP_AUDIO_STREAM_COMBINER,
|
|
PROP_TEXT_STREAM_COMBINER,
|
|
PROP_VOLUME,
|
|
PROP_MUTE,
|
|
PROP_SAMPLE,
|
|
PROP_FONT_DESC,
|
|
PROP_CONNECTION_SPEED,
|
|
PROP_BUFFER_SIZE,
|
|
PROP_BUFFER_DURATION,
|
|
PROP_AV_OFFSET,
|
|
PROP_RING_BUFFER_MAX_SIZE,
|
|
PROP_FORCE_ASPECT_RATIO,
|
|
PROP_AUDIO_FILTER,
|
|
PROP_VIDEO_FILTER,
|
|
PROP_MULTIVIEW_MODE,
|
|
PROP_MULTIVIEW_FLAGS
|
|
};
|
|
|
|
/* signals */
|
|
enum
|
|
{
|
|
SIGNAL_ABOUT_TO_FINISH,
|
|
SIGNAL_CONVERT_SAMPLE,
|
|
SIGNAL_VIDEO_CHANGED,
|
|
SIGNAL_AUDIO_CHANGED,
|
|
SIGNAL_TEXT_CHANGED,
|
|
SIGNAL_VIDEO_TAGS_CHANGED,
|
|
SIGNAL_AUDIO_TAGS_CHANGED,
|
|
SIGNAL_TEXT_TAGS_CHANGED,
|
|
SIGNAL_GET_VIDEO_TAGS,
|
|
SIGNAL_GET_AUDIO_TAGS,
|
|
SIGNAL_GET_TEXT_TAGS,
|
|
SIGNAL_GET_VIDEO_PAD,
|
|
SIGNAL_GET_AUDIO_PAD,
|
|
SIGNAL_GET_TEXT_PAD,
|
|
SIGNAL_SOURCE_SETUP,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static GstStaticCaps raw_audio_caps = GST_STATIC_CAPS ("audio/x-raw(ANY)");
|
|
static GstStaticCaps raw_video_caps = GST_STATIC_CAPS ("video/x-raw(ANY)");
|
|
|
|
static void gst_play_bin3_class_init (GstPlayBin3Class * klass);
|
|
static void gst_play_bin3_init (GstPlayBin3 * playbin);
|
|
static void gst_play_bin3_finalize (GObject * object);
|
|
|
|
static void gst_play_bin3_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * spec);
|
|
static void gst_play_bin3_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * spec);
|
|
|
|
static GstStateChangeReturn gst_play_bin3_change_state (GstElement * element,
|
|
GstStateChange transition);
|
|
|
|
static void gst_play_bin3_handle_message (GstBin * bin, GstMessage * message);
|
|
static gboolean gst_play_bin3_query (GstElement * element, GstQuery * query);
|
|
static void gst_play_bin3_set_context (GstElement * element,
|
|
GstContext * context);
|
|
static gboolean gst_play_bin3_send_event (GstElement * element,
|
|
GstEvent * event);
|
|
|
|
static GstTagList *gst_play_bin3_get_video_tags (GstPlayBin3 * playbin,
|
|
gint stream);
|
|
static GstTagList *gst_play_bin3_get_audio_tags (GstPlayBin3 * playbin,
|
|
gint stream);
|
|
static GstTagList *gst_play_bin3_get_text_tags (GstPlayBin3 * playbin,
|
|
gint stream);
|
|
|
|
static GstSample *gst_play_bin3_convert_sample (GstPlayBin3 * playbin,
|
|
GstCaps * caps);
|
|
|
|
static GstPad *gst_play_bin3_get_video_pad (GstPlayBin3 * playbin, gint stream);
|
|
static GstPad *gst_play_bin3_get_audio_pad (GstPlayBin3 * playbin, gint stream);
|
|
static GstPad *gst_play_bin3_get_text_pad (GstPlayBin3 * playbin, gint stream);
|
|
|
|
static GstStateChangeReturn setup_next_source (GstPlayBin3 * playbin,
|
|
GstState target);
|
|
|
|
static void no_more_pads_cb (GstElement * decodebin, GstPlayBin3 * playbin);
|
|
static void pad_removed_cb (GstElement * decodebin, GstPad * pad,
|
|
GstPlayBin3 * playbin);
|
|
|
|
static gint select_stream_cb (GstElement * decodebin,
|
|
GstStreamCollection * collection, GstStream * stream,
|
|
GstPlayBin3 * playbin);
|
|
|
|
static void do_stream_selection (GstPlayBin3 * playbin);
|
|
static void notify_tags_cb (GstStreamCollection * collection,
|
|
GstStream * stream, GParamSpec * pspec, GstPlayBin3 * playbin);
|
|
static void notify_tags_for_stream (GstPlayBin3 * playbin,
|
|
GstStreamCollection * collection, GstStream * stream);
|
|
|
|
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; \
|
|
}
|
|
|
|
static void gst_play_bin3_overlay_init (gpointer g_iface,
|
|
gpointer g_iface_data);
|
|
static void gst_play_bin3_navigation_init (gpointer g_iface,
|
|
gpointer g_iface_data);
|
|
static void gst_play_bin3_colorbalance_init (gpointer g_iface,
|
|
gpointer g_iface_data);
|
|
|
|
static GType
|
|
gst_play_bin3_get_type (void)
|
|
{
|
|
static GType gst_play_bin3_type = 0;
|
|
|
|
if (!gst_play_bin3_type) {
|
|
static const GTypeInfo gst_play_bin3_info = {
|
|
sizeof (GstPlayBin3Class),
|
|
NULL,
|
|
NULL,
|
|
(GClassInitFunc) gst_play_bin3_class_init,
|
|
NULL,
|
|
NULL,
|
|
sizeof (GstPlayBin3),
|
|
0,
|
|
(GInstanceInitFunc) gst_play_bin3_init,
|
|
NULL
|
|
};
|
|
static const GInterfaceInfo svol_info = {
|
|
NULL, NULL, NULL
|
|
};
|
|
static const GInterfaceInfo ov_info = {
|
|
gst_play_bin3_overlay_init,
|
|
NULL, NULL
|
|
};
|
|
static const GInterfaceInfo nav_info = {
|
|
gst_play_bin3_navigation_init,
|
|
NULL, NULL
|
|
};
|
|
static const GInterfaceInfo col_info = {
|
|
gst_play_bin3_colorbalance_init,
|
|
NULL, NULL
|
|
};
|
|
|
|
gst_play_bin3_type = g_type_register_static (GST_TYPE_PIPELINE,
|
|
"GstPlayBin3", &gst_play_bin3_info, 0);
|
|
|
|
g_type_add_interface_static (gst_play_bin3_type, GST_TYPE_STREAM_VOLUME,
|
|
&svol_info);
|
|
g_type_add_interface_static (gst_play_bin3_type, GST_TYPE_VIDEO_OVERLAY,
|
|
&ov_info);
|
|
g_type_add_interface_static (gst_play_bin3_type, GST_TYPE_NAVIGATION,
|
|
&nav_info);
|
|
g_type_add_interface_static (gst_play_bin3_type, GST_TYPE_COLOR_BALANCE,
|
|
&col_info);
|
|
}
|
|
|
|
return gst_play_bin3_type;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_class_init (GstPlayBin3Class * klass)
|
|
{
|
|
GObjectClass *gobject_klass;
|
|
GstElementClass *gstelement_klass;
|
|
GstBinClass *gstbin_klass;
|
|
|
|
gobject_klass = (GObjectClass *) klass;
|
|
gstelement_klass = (GstElementClass *) klass;
|
|
gstbin_klass = (GstBinClass *) klass;
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
gobject_klass->set_property = gst_play_bin3_set_property;
|
|
gobject_klass->get_property = gst_play_bin3_get_property;
|
|
|
|
gobject_klass->finalize = gst_play_bin3_finalize;
|
|
|
|
/**
|
|
* GstPlayBin3:uri
|
|
*
|
|
* Set the next URI that playbin will play. This property can be set from the
|
|
* about-to-finish signal to queue the next media file.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_URI,
|
|
g_param_spec_string ("uri", "URI", "URI of the media to play",
|
|
NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:current-uri
|
|
*
|
|
* The currently playing uri.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_CURRENT_URI,
|
|
g_param_spec_string ("current-uri", "Current URI",
|
|
"The currently playing URI", NULL,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:suburi
|
|
*
|
|
* Set the next subtitle URI that playbin will play. This property can be
|
|
* set from the about-to-finish signal to queue the next subtitle media file.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_SUBURI,
|
|
g_param_spec_string ("suburi", ".sub-URI", "Optional URI of a subtitle",
|
|
NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:current-suburi
|
|
*
|
|
* The currently playing subtitle uri.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_CURRENT_SUBURI,
|
|
g_param_spec_string ("current-suburi", "Current .sub-URI",
|
|
"The currently playing URI of a subtitle",
|
|
NULL, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_SOURCE,
|
|
g_param_spec_object ("source", "Source", "Source element",
|
|
GST_TYPE_ELEMENT, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:flags
|
|
*
|
|
* Control the behaviour of playbin.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_FLAGS,
|
|
g_param_spec_flags ("flags", "Flags", "Flags to control behaviour",
|
|
GST_TYPE_PLAY_FLAGS, DEFAULT_FLAGS,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:n-video
|
|
*
|
|
* Get the total number of available video streams.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_N_VIDEO,
|
|
g_param_spec_int ("n-video", "Number Video",
|
|
"Total number of video streams", 0, G_MAXINT, 0,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:current-video
|
|
*
|
|
* Get or set the currently playing video stream. By default the first video
|
|
* stream with data is played.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_CURRENT_VIDEO,
|
|
g_param_spec_int ("current-video", "Current Video",
|
|
"Currently playing video stream (-1 = auto)",
|
|
-1, G_MAXINT, -1, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:n-audio
|
|
*
|
|
* Get the total number of available audio streams.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_N_AUDIO,
|
|
g_param_spec_int ("n-audio", "Number Audio",
|
|
"Total number of audio streams", 0, G_MAXINT, 0,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:current-audio
|
|
*
|
|
* Get or set the currently playing audio stream. By default the first audio
|
|
* stream with data is played.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_CURRENT_AUDIO,
|
|
g_param_spec_int ("current-audio", "Current audio",
|
|
"Currently playing audio stream (-1 = auto)",
|
|
-1, G_MAXINT, -1, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:n-text
|
|
*
|
|
* Get the total number of available subtitle streams.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_N_TEXT,
|
|
g_param_spec_int ("n-text", "Number Text",
|
|
"Total number of text streams", 0, G_MAXINT, 0,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:current-text:
|
|
*
|
|
* Get or set the currently playing subtitle stream. By default the first
|
|
* subtitle stream with data is played.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_CURRENT_TEXT,
|
|
g_param_spec_int ("current-text", "Current Text",
|
|
"Currently playing text stream (-1 = auto)",
|
|
-1, G_MAXINT, -1, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3::auto-select-streams:
|
|
*
|
|
* If TRUE the playbin will respond to stream-collection messages
|
|
* by sending a SELECT_STREAMS event to decodebin. Set to FALSE
|
|
* if the application will manage stream selection. This property
|
|
* will automatically be set to FALSE if playbin receives a select-streams
|
|
* event from the application, but setting it explicitly avoids any
|
|
* races where playbin mind send a select-streams event before the
|
|
* application.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_AUTO_SELECT_STREAMS,
|
|
g_param_spec_boolean ("auto-select-streams", "Automatic Select-Streams",
|
|
"Whether playbin should respond to stream-collection messags with select-streams events",
|
|
DEFAULT_AUTO_SELECT_STREAMS,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_SUBTITLE_ENCODING,
|
|
g_param_spec_string ("subtitle-encoding", "subtitle encoding",
|
|
"Encoding to assume if input subtitles are not in UTF-8 encoding. "
|
|
"If not set, the GST_SUBTITLE_ENCODING environment variable will "
|
|
"be checked for an encoding to use. If that is not set either, "
|
|
"ISO-8859-15 will be assumed.", NULL,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_VIDEO_FILTER,
|
|
g_param_spec_object ("video-filter", "Video filter",
|
|
"the video filter(s) to apply, if possible",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_AUDIO_FILTER,
|
|
g_param_spec_object ("audio-filter", "Audio filter",
|
|
"the audio filter(s) to apply, if possible",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_VIDEO_SINK,
|
|
g_param_spec_object ("video-sink", "Video Sink",
|
|
"the video output element to use (NULL = default sink)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_AUDIO_SINK,
|
|
g_param_spec_object ("audio-sink", "Audio Sink",
|
|
"the audio output element to use (NULL = default sink)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_VIS_PLUGIN,
|
|
g_param_spec_object ("vis-plugin", "Vis plugin",
|
|
"the visualization element to use (NULL = default)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_TEXT_SINK,
|
|
g_param_spec_object ("text-sink", "Text plugin",
|
|
"the text output element to use (NULL = default subtitleoverlay)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:video-stream-combiner
|
|
*
|
|
* Get or set the current video stream combiner. By default, an input-selector
|
|
* is created and deleted as-needed.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_VIDEO_STREAM_COMBINER,
|
|
g_param_spec_object ("video-stream-combiner", "Video stream combiner",
|
|
"Current video stream combiner (NULL = input-selector)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:audio-stream-combiner
|
|
*
|
|
* Get or set the current audio stream combiner. By default, an input-selector
|
|
* is created and deleted as-needed.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_AUDIO_STREAM_COMBINER,
|
|
g_param_spec_object ("audio-stream-combiner", "Audio stream combiner",
|
|
"Current audio stream combiner (NULL = input-selector)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:text-stream-combiner
|
|
*
|
|
* Get or set the current text stream combiner. By default, an input-selector
|
|
* is created and deleted as-needed.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_TEXT_STREAM_COMBINER,
|
|
g_param_spec_object ("text-stream-combiner", "Text stream combiner",
|
|
"Current text stream combiner (NULL = input-selector)",
|
|
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:volume:
|
|
*
|
|
* Get or set the current audio stream volume. 1.0 means 100%,
|
|
* 0.0 means mute. This uses a linear volume scale.
|
|
*
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_VOLUME,
|
|
g_param_spec_double ("volume", "Volume", "The audio volume, 1.0=100%",
|
|
0.0, VOLUME_MAX_DOUBLE, 1.0,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_MUTE,
|
|
g_param_spec_boolean ("mute", "Mute",
|
|
"Mute the audio channel without changing the volume", FALSE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:sample:
|
|
* @playbin: a #GstPlayBin3
|
|
*
|
|
* Get the currently rendered or prerolled sample in the video sink.
|
|
* The #GstCaps in the sample will describe the format of the buffer.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_SAMPLE,
|
|
g_param_spec_boxed ("sample", "Sample",
|
|
"The last sample (NULL = no video available)",
|
|
GST_TYPE_SAMPLE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_FONT_DESC,
|
|
g_param_spec_string ("subtitle-font-desc",
|
|
"Subtitle font description",
|
|
"Pango font description of font "
|
|
"to be used for subtitle rendering", NULL,
|
|
G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_CONNECTION_SPEED,
|
|
g_param_spec_uint64 ("connection-speed", "Connection Speed",
|
|
"Network connection speed in kbps (0 = unknown)",
|
|
0, G_MAXUINT64 / 1000, DEFAULT_CONNECTION_SPEED,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_klass, PROP_BUFFER_SIZE,
|
|
g_param_spec_int ("buffer-size", "Buffer size (bytes)",
|
|
"Buffer size when buffering network streams",
|
|
-1, G_MAXINT, DEFAULT_BUFFER_SIZE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_klass, PROP_BUFFER_DURATION,
|
|
g_param_spec_int64 ("buffer-duration", "Buffer duration (ns)",
|
|
"Buffer duration when buffering network streams",
|
|
-1, G_MAXINT64, DEFAULT_BUFFER_DURATION,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
/**
|
|
* GstPlayBin3:av-offset:
|
|
*
|
|
* Control the synchronisation offset between the audio and video streams.
|
|
* Positive values make the audio ahead of the video and negative values make
|
|
* the audio go behind the video.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_AV_OFFSET,
|
|
g_param_spec_int64 ("av-offset", "AV Offset",
|
|
"The synchronisation offset between audio and video in nanoseconds",
|
|
G_MININT64, G_MAXINT64, 0,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3:ring-buffer-max-size
|
|
*
|
|
* The maximum size of the ring buffer in bytes. If set to 0, the ring
|
|
* buffer is disabled. Default 0.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_RING_BUFFER_MAX_SIZE,
|
|
g_param_spec_uint64 ("ring-buffer-max-size",
|
|
"Max. ring buffer size (bytes)",
|
|
"Max. amount of data in the ring buffer (bytes, 0 = ring buffer disabled)",
|
|
0, G_MAXUINT, DEFAULT_RING_BUFFER_MAX_SIZE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3::force-aspect-ratio:
|
|
*
|
|
* Requests the video sink to enforce the video display aspect ratio.
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_FORCE_ASPECT_RATIO,
|
|
g_param_spec_boolean ("force-aspect-ratio", "Force Aspect Ratio",
|
|
"When enabled, scaling will respect original aspect ratio", TRUE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3::video-multiview-mode:
|
|
*
|
|
* Set the stereoscopic mode for video streams that don't contain
|
|
* any information in the stream, so they can be correctly played
|
|
* as 3D streams. If a video already has multiview information
|
|
* encoded, this property can override other modes in the set,
|
|
* but cannot be used to re-interpret MVC or mixed-mono streams.
|
|
*
|
|
* See Also: The #GstPlayBin3::video-multiview-flags property
|
|
*
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_MULTIVIEW_MODE,
|
|
g_param_spec_enum ("video-multiview-mode",
|
|
"Multiview Mode Override",
|
|
"Re-interpret a video stream as one of several frame-packed stereoscopic modes.",
|
|
GST_TYPE_VIDEO_MULTIVIEW_FRAME_PACKING,
|
|
GST_VIDEO_MULTIVIEW_FRAME_PACKING_NONE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3::video-multiview-flags:
|
|
*
|
|
* When overriding the multiview mode of an input stream,
|
|
* these flags modify details of the view layout.
|
|
*
|
|
* See Also: The #GstPlayBin3::video-multiview-mode property
|
|
*/
|
|
g_object_class_install_property (gobject_klass, PROP_MULTIVIEW_FLAGS,
|
|
g_param_spec_flags ("video-multiview-flags",
|
|
"Multiview Flags Override",
|
|
"Override details of the multiview frame layout",
|
|
GST_TYPE_VIDEO_MULTIVIEW_FLAGS, GST_VIDEO_MULTIVIEW_FLAGS_NONE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstPlayBin3::about-to-finish
|
|
* @playbin: a #GstPlayBin3
|
|
*
|
|
* This signal is emitted when the current uri is about to finish. You can
|
|
* set the uri and suburi to make sure that playback continues.
|
|
*
|
|
* This signal is emitted from the context of a GStreamer streaming thread.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_ABOUT_TO_FINISH] =
|
|
g_signal_new ("about-to-finish", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, about_to_finish), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 0, G_TYPE_NONE);
|
|
|
|
/**
|
|
* GstPlayBin3::video-changed
|
|
* @playbin: a #GstPlayBin3
|
|
*
|
|
* This signal is emitted whenever the number or order of the video
|
|
* streams has changed. The application will most likely want to select
|
|
* a new video stream.
|
|
*
|
|
* This signal is usually emitted from the context of a GStreamer streaming
|
|
* thread. You can use gst_message_new_application() and
|
|
* gst_element_post_message() to notify your application's main thread.
|
|
*/
|
|
/* FIXME 0.11: turn video-changed signal into message? */
|
|
gst_play_bin3_signals[SIGNAL_VIDEO_CHANGED] =
|
|
g_signal_new ("video-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, video_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 0, G_TYPE_NONE);
|
|
/**
|
|
* GstPlayBin3::audio-changed
|
|
* @playbin: a #GstPlayBin3
|
|
*
|
|
* This signal is emitted whenever the number or order of the audio
|
|
* streams has changed. The application will most likely want to select
|
|
* a new audio stream.
|
|
*
|
|
* This signal may be emitted from the context of a GStreamer streaming thread.
|
|
* You can use gst_message_new_application() and gst_element_post_message()
|
|
* to notify your application's main thread.
|
|
*/
|
|
/* FIXME 0.11: turn audio-changed signal into message? */
|
|
gst_play_bin3_signals[SIGNAL_AUDIO_CHANGED] =
|
|
g_signal_new ("audio-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, audio_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 0, G_TYPE_NONE);
|
|
/**
|
|
* GstPlayBin3::text-changed
|
|
* @playbin: a #GstPlayBin3
|
|
*
|
|
* This signal is emitted whenever the number or order of the text
|
|
* streams has changed. The application will most likely want to select
|
|
* a new text stream.
|
|
*
|
|
* This signal may be emitted from the context of a GStreamer streaming thread.
|
|
* You can use gst_message_new_application() and gst_element_post_message()
|
|
* to notify your application's main thread.
|
|
*/
|
|
/* FIXME 0.11: turn text-changed signal into message? */
|
|
gst_play_bin3_signals[SIGNAL_TEXT_CHANGED] =
|
|
g_signal_new ("text-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, text_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 0, G_TYPE_NONE);
|
|
|
|
/**
|
|
* GstPlayBin3::video-tags-changed
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: stream index with changed tags
|
|
*
|
|
* This signal is emitted whenever the tags of a video stream have changed.
|
|
* The application will most likely want to get the new tags.
|
|
*
|
|
* This signal may be emitted from the context of a GStreamer streaming thread.
|
|
* You can use gst_message_new_application() and gst_element_post_message()
|
|
* to notify your application's main thread.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_VIDEO_TAGS_CHANGED] =
|
|
g_signal_new ("video-tags-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, video_tags_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 1, G_TYPE_INT);
|
|
|
|
/**
|
|
* GstPlayBin3::audio-tags-changed
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: stream index with changed tags
|
|
*
|
|
* This signal is emitted whenever the tags of an audio stream have changed.
|
|
* The application will most likely want to get the new tags.
|
|
*
|
|
* This signal may be emitted from the context of a GStreamer streaming thread.
|
|
* You can use gst_message_new_application() and gst_element_post_message()
|
|
* to notify your application's main thread.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_AUDIO_TAGS_CHANGED] =
|
|
g_signal_new ("audio-tags-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, audio_tags_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 1, G_TYPE_INT);
|
|
|
|
/**
|
|
* GstPlayBin3::text-tags-changed
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: stream index with changed tags
|
|
*
|
|
* This signal is emitted whenever the tags of a text stream have changed.
|
|
* The application will most likely want to get the new tags.
|
|
*
|
|
* This signal may be emitted from the context of a GStreamer streaming thread.
|
|
* You can use gst_message_new_application() and gst_element_post_message()
|
|
* to notify your application's main thread.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_TEXT_TAGS_CHANGED] =
|
|
g_signal_new ("text-tags-changed", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, text_tags_changed), NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 1, G_TYPE_INT);
|
|
|
|
/**
|
|
* GstPlayBin3::source-setup:
|
|
* @playbin: a #GstPlayBin3
|
|
* @source: source element
|
|
*
|
|
* This signal is emitted after the source element has been created, so
|
|
* it can be configured by setting additional properties (e.g. set a
|
|
* proxy server for an http source, or set the device and read speed for
|
|
* an audio cd source). This is functionally equivalent to connecting to
|
|
* the notify::source signal, but more convenient.
|
|
*
|
|
* This signal is usually emitted from the context of a GStreamer streaming
|
|
* thread.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_SOURCE_SETUP] =
|
|
g_signal_new ("source-setup", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
|
|
g_cclosure_marshal_generic, G_TYPE_NONE, 1, GST_TYPE_ELEMENT);
|
|
|
|
/**
|
|
* GstPlayBin3::get-video-tags
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: a video stream number
|
|
*
|
|
* Action signal to retrieve the tags of a specific video stream number.
|
|
* This information can be used to select a stream.
|
|
*
|
|
* Returns: a GstTagList with tags or NULL when the stream number does not
|
|
* exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_VIDEO_TAGS] =
|
|
g_signal_new ("get-video-tags", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_video_tags), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_TAG_LIST, 1, G_TYPE_INT);
|
|
/**
|
|
* GstPlayBin3::get-audio-tags
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: an audio stream number
|
|
*
|
|
* Action signal to retrieve the tags of a specific audio stream number.
|
|
* This information can be used to select a stream.
|
|
*
|
|
* Returns: a GstTagList with tags or NULL when the stream number does not
|
|
* exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_AUDIO_TAGS] =
|
|
g_signal_new ("get-audio-tags", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_audio_tags), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_TAG_LIST, 1, G_TYPE_INT);
|
|
/**
|
|
* GstPlayBin3::get-text-tags
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: a text stream number
|
|
*
|
|
* Action signal to retrieve the tags of a specific text stream number.
|
|
* This information can be used to select a stream.
|
|
*
|
|
* Returns: a GstTagList with tags or NULL when the stream number does not
|
|
* exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_TEXT_TAGS] =
|
|
g_signal_new ("get-text-tags", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_text_tags), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_TAG_LIST, 1, G_TYPE_INT);
|
|
/**
|
|
* GstPlayBin3::convert-sample
|
|
* @playbin: a #GstPlayBin3
|
|
* @caps: the target format of the frame
|
|
*
|
|
* Action signal to retrieve the currently playing video frame in the format
|
|
* specified by @caps.
|
|
* If @caps is %NULL, no conversion will be performed and this function is
|
|
* equivalent to the #GstPlayBin3::frame property.
|
|
*
|
|
* Returns: a #GstSample of the current video frame converted to #caps.
|
|
* The caps on the sample will describe the final layout of the buffer data.
|
|
* %NULL is returned when no current buffer can be retrieved or when the
|
|
* conversion failed.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_CONVERT_SAMPLE] =
|
|
g_signal_new ("convert-sample", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, convert_sample), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_SAMPLE, 1, GST_TYPE_CAPS);
|
|
|
|
/**
|
|
* GstPlayBin3::get-video-pad
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: a video stream number
|
|
*
|
|
* Action signal to retrieve the stream-combiner sinkpad for a specific
|
|
* video stream.
|
|
* This pad can be used for notifications of caps changes, stream-specific
|
|
* queries, etc.
|
|
*
|
|
* Returns: a #GstPad, or NULL when the stream number does not exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_VIDEO_PAD] =
|
|
g_signal_new ("get-video-pad", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_video_pad), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_PAD, 1, G_TYPE_INT);
|
|
/**
|
|
* GstPlayBin3::get-audio-pad
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: an audio stream number
|
|
*
|
|
* Action signal to retrieve the stream-combiner sinkpad for a specific
|
|
* audio stream.
|
|
* This pad can be used for notifications of caps changes, stream-specific
|
|
* queries, etc.
|
|
*
|
|
* Returns: a #GstPad, or NULL when the stream number does not exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_AUDIO_PAD] =
|
|
g_signal_new ("get-audio-pad", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_audio_pad), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_PAD, 1, G_TYPE_INT);
|
|
/**
|
|
* GstPlayBin3::get-text-pad
|
|
* @playbin: a #GstPlayBin3
|
|
* @stream: a text stream number
|
|
*
|
|
* Action signal to retrieve the stream-combiner sinkpad for a specific
|
|
* text stream.
|
|
* This pad can be used for notifications of caps changes, stream-specific
|
|
* queries, etc.
|
|
*
|
|
* Returns: a #GstPad, or NULL when the stream number does not exist.
|
|
*/
|
|
gst_play_bin3_signals[SIGNAL_GET_TEXT_PAD] =
|
|
g_signal_new ("get-text-pad", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (GstPlayBin3Class, get_text_pad), NULL, NULL,
|
|
g_cclosure_marshal_generic, GST_TYPE_PAD, 1, G_TYPE_INT);
|
|
|
|
klass->get_video_tags = gst_play_bin3_get_video_tags;
|
|
klass->get_audio_tags = gst_play_bin3_get_audio_tags;
|
|
klass->get_text_tags = gst_play_bin3_get_text_tags;
|
|
|
|
klass->convert_sample = gst_play_bin3_convert_sample;
|
|
|
|
klass->get_video_pad = gst_play_bin3_get_video_pad;
|
|
klass->get_audio_pad = gst_play_bin3_get_audio_pad;
|
|
klass->get_text_pad = gst_play_bin3_get_text_pad;
|
|
|
|
gst_element_class_set_static_metadata (gstelement_klass,
|
|
"Player Bin 3", "Generic/Bin/Player",
|
|
"Autoplug and play media from an uri",
|
|
"Wim Taymans <wim.taymans@gmail.com>");
|
|
|
|
gstelement_klass->change_state =
|
|
GST_DEBUG_FUNCPTR (gst_play_bin3_change_state);
|
|
gstelement_klass->query = GST_DEBUG_FUNCPTR (gst_play_bin3_query);
|
|
gstelement_klass->set_context = GST_DEBUG_FUNCPTR (gst_play_bin3_set_context);
|
|
gstelement_klass->send_event = GST_DEBUG_FUNCPTR (gst_play_bin3_send_event);
|
|
|
|
gstbin_klass->handle_message =
|
|
GST_DEBUG_FUNCPTR (gst_play_bin3_handle_message);
|
|
}
|
|
|
|
static void
|
|
do_async_start (GstPlayBin3 * playbin)
|
|
{
|
|
GstMessage *message;
|
|
|
|
playbin->async_pending = TRUE;
|
|
|
|
message = gst_message_new_async_start (GST_OBJECT_CAST (playbin));
|
|
GST_BIN_CLASS (parent_class)->handle_message (GST_BIN_CAST (playbin),
|
|
message);
|
|
}
|
|
|
|
static void
|
|
do_async_done (GstPlayBin3 * playbin)
|
|
{
|
|
GstMessage *message;
|
|
|
|
if (playbin->async_pending) {
|
|
GST_DEBUG_OBJECT (playbin, "posting ASYNC_DONE");
|
|
message =
|
|
gst_message_new_async_done (GST_OBJECT_CAST (playbin),
|
|
GST_CLOCK_TIME_NONE);
|
|
GST_BIN_CLASS (parent_class)->handle_message (GST_BIN_CAST (playbin),
|
|
message);
|
|
|
|
playbin->async_pending = FALSE;
|
|
}
|
|
}
|
|
|
|
/* init combiners. The combiner is found by finding the first prefix that
|
|
* matches the media. */
|
|
static void
|
|
init_combiners (GstPlayBin3 * playbin)
|
|
{
|
|
gint i;
|
|
|
|
/* store the array for the different channels */
|
|
for (i = 0; i < PLAYBIN_STREAM_LAST; i++)
|
|
playbin->channels[i] = g_ptr_array_new ();
|
|
|
|
playbin->combiner[PLAYBIN_STREAM_AUDIO].media_type = "audio";
|
|
playbin->combiner[PLAYBIN_STREAM_AUDIO].type = GST_PLAY_SINK_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].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].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 */
|
|
static void
|
|
update_combiner_info (GstPlayBin3 * playbin)
|
|
{
|
|
guint i, len;
|
|
|
|
if (playbin->collection == NULL)
|
|
return;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Updating combiner info");
|
|
|
|
/* Wipe current combiner streams */
|
|
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_AUDIO].streams, TRUE);
|
|
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_VIDEO].streams, TRUE);
|
|
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);
|
|
for (i = 0; i < len; i++) {
|
|
GstStream *stream =
|
|
gst_stream_collection_get_stream (playbin->collection, i);
|
|
GstStreamType stype = gst_stream_get_stream_type (stream);
|
|
|
|
switch (stype) {
|
|
case GST_STREAM_TYPE_AUDIO:
|
|
g_ptr_array_add (playbin->combiner[PLAYBIN_STREAM_AUDIO].streams,
|
|
gst_object_ref (stream));
|
|
break;
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
g_ptr_array_add (playbin->combiner[PLAYBIN_STREAM_VIDEO].streams,
|
|
gst_object_ref (stream));
|
|
break;
|
|
case GST_STREAM_TYPE_TEXT:
|
|
g_ptr_array_add (playbin->combiner[PLAYBIN_STREAM_TEXT].streams,
|
|
gst_object_ref (stream));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (playbin, "There are %d audio streams",
|
|
playbin->combiner[PLAYBIN_STREAM_AUDIO].streams->len);
|
|
GST_DEBUG_OBJECT (playbin, "There are %d video streams",
|
|
playbin->combiner[PLAYBIN_STREAM_VIDEO].streams->len);
|
|
GST_DEBUG_OBJECT (playbin, "There are %d text streams",
|
|
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;
|
|
|
|
switch (gst_stream_get_stream_type (stream)) {
|
|
case GST_STREAM_TYPE_AUDIO:
|
|
combine = &playbin->combiner[PLAYBIN_STREAM_AUDIO];
|
|
break;
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
combine = &playbin->combiner[PLAYBIN_STREAM_VIDEO];
|
|
break;
|
|
case GST_STREAM_TYPE_TEXT:
|
|
combine = &playbin->combiner[PLAYBIN_STREAM_TEXT];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
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)
|
|
{
|
|
g_mutex_init (&group->lock);
|
|
|
|
group->stream_changed_pending = FALSE;
|
|
g_mutex_init (&group->stream_changed_pending_lock);
|
|
|
|
group->playbin = playbin;
|
|
}
|
|
|
|
static void
|
|
free_group (GstPlayBin3 * playbin, GstSourceGroup * group)
|
|
{
|
|
g_free (group->uri);
|
|
g_free (group->suburi);
|
|
|
|
g_mutex_clear (&group->lock);
|
|
group->stream_changed_pending = FALSE;
|
|
g_mutex_clear (&group->stream_changed_pending_lock);
|
|
|
|
if (group->pending_buffering_msg)
|
|
gst_message_unref (group->pending_buffering_msg);
|
|
group->pending_buffering_msg = 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);
|
|
}
|
|
|
|
static void
|
|
notify_volume_cb (GObject * combiner, GParamSpec * pspec, GstPlayBin3 * playbin)
|
|
{
|
|
g_object_notify (G_OBJECT (playbin), "volume");
|
|
}
|
|
|
|
static void
|
|
notify_mute_cb (GObject * combiner, GParamSpec * pspec, GstPlayBin3 * playbin)
|
|
{
|
|
g_object_notify (G_OBJECT (playbin), "mute");
|
|
}
|
|
|
|
static void
|
|
colorbalance_value_changed_cb (GstColorBalance * balance,
|
|
GstColorBalanceChannel * channel, gint value, GstPlayBin3 * playbin)
|
|
{
|
|
gst_color_balance_value_changed (GST_COLOR_BALANCE (playbin), channel, value);
|
|
}
|
|
|
|
static gint
|
|
compare_factories_func (gconstpointer p1, gconstpointer p2)
|
|
{
|
|
GstPluginFeature *f1, *f2;
|
|
gboolean is_sink1, is_sink2;
|
|
gboolean is_parser1, is_parser2;
|
|
|
|
f1 = (GstPluginFeature *) p1;
|
|
f2 = (GstPluginFeature *) p2;
|
|
|
|
is_sink1 = gst_element_factory_list_is_type (GST_ELEMENT_FACTORY_CAST (f1),
|
|
GST_ELEMENT_FACTORY_TYPE_SINK);
|
|
is_sink2 = gst_element_factory_list_is_type (GST_ELEMENT_FACTORY_CAST (f2),
|
|
GST_ELEMENT_FACTORY_TYPE_SINK);
|
|
is_parser1 = gst_element_factory_list_is_type (GST_ELEMENT_FACTORY_CAST (f1),
|
|
GST_ELEMENT_FACTORY_TYPE_PARSER);
|
|
is_parser2 = gst_element_factory_list_is_type (GST_ELEMENT_FACTORY_CAST (f2),
|
|
GST_ELEMENT_FACTORY_TYPE_PARSER);
|
|
|
|
/* First we want all sinks as we prefer a sink if it directly
|
|
* supports the current caps */
|
|
if (is_sink1 && !is_sink2)
|
|
return -1;
|
|
else if (!is_sink1 && is_sink2)
|
|
return 1;
|
|
|
|
/* Then we want all parsers as we always want to plug parsers
|
|
* before decoders */
|
|
if (is_parser1 && !is_parser2)
|
|
return -1;
|
|
else if (!is_parser1 && is_parser2)
|
|
return 1;
|
|
|
|
/* And if it's a both a parser or sink we first sort by rank
|
|
* and then by factory name */
|
|
return gst_plugin_feature_rank_compare_func (p1, p2);
|
|
}
|
|
|
|
/* Must be called with elements lock! */
|
|
static void
|
|
gst_play_bin3_update_elements_list (GstPlayBin3 * playbin)
|
|
{
|
|
GList *res, *tmp;
|
|
guint cookie;
|
|
|
|
cookie = gst_registry_get_feature_list_cookie (gst_registry_get ());
|
|
|
|
if (!playbin->elements || playbin->elements_cookie != cookie) {
|
|
if (playbin->elements)
|
|
gst_plugin_feature_list_free (playbin->elements);
|
|
res =
|
|
gst_element_factory_list_get_elements
|
|
(GST_ELEMENT_FACTORY_TYPE_DECODABLE, GST_RANK_MARGINAL);
|
|
tmp =
|
|
gst_element_factory_list_get_elements
|
|
(GST_ELEMENT_FACTORY_TYPE_AUDIOVIDEO_SINKS, GST_RANK_MARGINAL);
|
|
playbin->elements = g_list_concat (res, tmp);
|
|
playbin->elements = g_list_sort (playbin->elements, compare_factories_func);
|
|
}
|
|
|
|
if (!playbin->aelements || playbin->elements_cookie != cookie) {
|
|
if (playbin->aelements)
|
|
g_sequence_free (playbin->aelements);
|
|
playbin->aelements = avelements_create (playbin, TRUE);
|
|
}
|
|
|
|
if (!playbin->velements || playbin->elements_cookie != cookie) {
|
|
if (playbin->velements)
|
|
g_sequence_free (playbin->velements);
|
|
playbin->velements = avelements_create (playbin, FALSE);
|
|
}
|
|
|
|
playbin->elements_cookie = cookie;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_init (GstPlayBin3 * playbin)
|
|
{
|
|
g_rec_mutex_init (&playbin->lock);
|
|
g_mutex_init (&playbin->dyn_lock);
|
|
|
|
/* assume we can create an input-selector */
|
|
playbin->have_selector = TRUE;
|
|
|
|
playbin->do_stream_selections = DEFAULT_AUTO_SELECT_STREAMS;
|
|
|
|
init_combiners (playbin);
|
|
|
|
/* init groups */
|
|
playbin->curr_group = &playbin->groups[0];
|
|
playbin->next_group = &playbin->groups[1];
|
|
init_group (playbin, &playbin->groups[0]);
|
|
init_group (playbin, &playbin->groups[1]);
|
|
|
|
/* first filter out the interesting element factories */
|
|
g_mutex_init (&playbin->elements_lock);
|
|
|
|
/* add sink */
|
|
playbin->playsink =
|
|
g_object_new (GST_TYPE_PLAY_SINK, "name", "playsink", "send-event-mode",
|
|
1, NULL);
|
|
gst_bin_add (GST_BIN_CAST (playbin), GST_ELEMENT_CAST (playbin->playsink));
|
|
gst_play_sink_set_flags (playbin->playsink, DEFAULT_FLAGS);
|
|
/* Connect to notify::volume and notify::mute signals for proxying */
|
|
g_signal_connect (playbin->playsink, "notify::volume",
|
|
G_CALLBACK (notify_volume_cb), playbin);
|
|
g_signal_connect (playbin->playsink, "notify::mute",
|
|
G_CALLBACK (notify_mute_cb), playbin);
|
|
g_signal_connect (playbin->playsink, "value-changed",
|
|
G_CALLBACK (colorbalance_value_changed_cb), playbin);
|
|
|
|
playbin->current_video = DEFAULT_CURRENT_VIDEO;
|
|
playbin->current_audio = DEFAULT_CURRENT_AUDIO;
|
|
playbin->current_text = DEFAULT_CURRENT_TEXT;
|
|
|
|
playbin->buffer_duration = DEFAULT_BUFFER_DURATION;
|
|
playbin->buffer_size = DEFAULT_BUFFER_SIZE;
|
|
playbin->ring_buffer_max_size = DEFAULT_RING_BUFFER_MAX_SIZE;
|
|
|
|
playbin->force_aspect_ratio = TRUE;
|
|
|
|
playbin->multiview_mode = GST_VIDEO_MULTIVIEW_FRAME_PACKING_NONE;
|
|
playbin->multiview_flags = GST_VIDEO_MULTIVIEW_FLAGS_NONE;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_finalize (GObject * object)
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
gint i;
|
|
|
|
playbin = GST_PLAY_BIN3 (object);
|
|
|
|
free_group (playbin, &playbin->groups[0]);
|
|
free_group (playbin, &playbin->groups[1]);
|
|
|
|
for (i = 0; i < PLAYBIN_STREAM_LAST; i++)
|
|
g_ptr_array_free (playbin->channels[i], TRUE);
|
|
|
|
if (playbin->source)
|
|
gst_object_unref (playbin->source);
|
|
|
|
/* Setting states to NULL is safe here because playsink
|
|
* will already be gone and none of these sinks will be
|
|
* a child of playsink
|
|
*/
|
|
if (playbin->video_sink) {
|
|
gst_element_set_state (playbin->video_sink, GST_STATE_NULL);
|
|
gst_object_unref (playbin->video_sink);
|
|
}
|
|
if (playbin->audio_sink) {
|
|
gst_element_set_state (playbin->audio_sink, GST_STATE_NULL);
|
|
gst_object_unref (playbin->audio_sink);
|
|
}
|
|
if (playbin->text_sink) {
|
|
gst_element_set_state (playbin->text_sink, GST_STATE_NULL);
|
|
gst_object_unref (playbin->text_sink);
|
|
}
|
|
|
|
if (playbin->video_stream_combiner) {
|
|
gst_element_set_state (playbin->video_stream_combiner, GST_STATE_NULL);
|
|
gst_object_unref (playbin->video_stream_combiner);
|
|
}
|
|
if (playbin->audio_stream_combiner) {
|
|
gst_element_set_state (playbin->audio_stream_combiner, GST_STATE_NULL);
|
|
gst_object_unref (playbin->audio_stream_combiner);
|
|
}
|
|
if (playbin->text_stream_combiner) {
|
|
gst_element_set_state (playbin->text_stream_combiner, GST_STATE_NULL);
|
|
gst_object_unref (playbin->text_stream_combiner);
|
|
}
|
|
|
|
g_ptr_array_free (playbin->combiner[PLAYBIN_STREAM_AUDIO].streams, TRUE);
|
|
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->aelements)
|
|
g_sequence_free (playbin->aelements);
|
|
|
|
if (playbin->velements)
|
|
g_sequence_free (playbin->velements);
|
|
|
|
g_list_free_full (playbin->contexts, (GDestroyNotify) gst_context_unref);
|
|
|
|
g_rec_mutex_clear (&playbin->lock);
|
|
g_mutex_clear (&playbin->dyn_lock);
|
|
g_mutex_clear (&playbin->elements_lock);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static gboolean
|
|
gst_playbin_uri_is_valid (GstPlayBin3 * playbin, const gchar * uri)
|
|
{
|
|
const gchar *c;
|
|
|
|
GST_LOG_OBJECT (playbin, "checking uri '%s'", uri);
|
|
|
|
/* this just checks the protocol */
|
|
if (!gst_uri_is_valid (uri))
|
|
return FALSE;
|
|
|
|
for (c = uri; *c != '\0'; ++c) {
|
|
if (!g_ascii_isprint (*c))
|
|
goto invalid;
|
|
if (*c == ' ')
|
|
goto invalid;
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
invalid:
|
|
{
|
|
GST_WARNING_OBJECT (playbin, "uri '%s' not valid, character #%u",
|
|
uri, (guint) ((guintptr) c - (guintptr) uri));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_uri (GstPlayBin3 * playbin, const gchar * uri)
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
if (uri == NULL) {
|
|
g_warning ("cannot set NULL uri");
|
|
return;
|
|
}
|
|
|
|
if (!gst_playbin_uri_is_valid (playbin, uri)) {
|
|
if (g_str_has_prefix (uri, "file:")) {
|
|
GST_WARNING_OBJECT (playbin, "not entirely correct file URI '%s' - make "
|
|
"sure to escape spaces and non-ASCII characters properly and specify "
|
|
"an absolute path. Use gst_filename_to_uri() to convert filenames "
|
|
"to URIs", uri);
|
|
} else {
|
|
/* GST_ERROR_OBJECT (playbin, "malformed URI '%s'", uri); */
|
|
}
|
|
}
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = playbin->next_group;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
/* store the uri in the next group we will play */
|
|
g_free (group->uri);
|
|
group->uri = g_strdup (uri);
|
|
group->valid = TRUE;
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
GST_DEBUG ("set new uri to %s", uri);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_suburi (GstPlayBin3 * playbin, const gchar * suburi)
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = playbin->next_group;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
g_free (group->suburi);
|
|
group->suburi = g_strdup (suburi);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
GST_DEBUG ("setting new .sub uri to %s", suburi);
|
|
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_flags (GstPlayBin3 * playbin, GstPlayFlags flags)
|
|
{
|
|
GstPlayFlags old_flags;
|
|
old_flags = gst_play_sink_get_flags (playbin->playsink);
|
|
|
|
if (flags != old_flags) {
|
|
gst_play_sink_set_flags (playbin->playsink, flags);
|
|
gst_play_sink_reconfigure (playbin->playsink);
|
|
}
|
|
}
|
|
|
|
static GstPlayFlags
|
|
gst_play_bin3_get_flags (GstPlayBin3 * playbin)
|
|
{
|
|
GstPlayFlags flags;
|
|
|
|
flags = gst_play_sink_get_flags (playbin->playsink);
|
|
|
|
return flags;
|
|
}
|
|
|
|
/* get the currently playing group or if nothing is playing, the next
|
|
* group. Must be called with the PLAY_BIN_LOCK. */
|
|
static GstSourceGroup *
|
|
get_group (GstPlayBin3 * playbin)
|
|
{
|
|
GstSourceGroup *result;
|
|
|
|
if (!(result = playbin->curr_group))
|
|
result = playbin->next_group;
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstPad *
|
|
gst_play_bin3_get_pad_of_type (GstPlayBin3 * playbin, gint stream_type,
|
|
gint stream)
|
|
{
|
|
GstPad *sinkpad = NULL;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (playbin->combiner[stream_type].combiner == NULL) {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"get-pad of type %d w/o custom-combiner. Returning playsink pad",
|
|
stream_type);
|
|
sinkpad = playbin->combiner[stream_type].sinkpad;
|
|
if (sinkpad) {
|
|
sinkpad = gst_object_ref (sinkpad);
|
|
goto done;
|
|
}
|
|
}
|
|
if (stream < playbin->channels[stream_type]->len) {
|
|
sinkpad = g_ptr_array_index (playbin->channels[stream_type], stream);
|
|
gst_object_ref (sinkpad);
|
|
}
|
|
|
|
done:
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return sinkpad;
|
|
}
|
|
|
|
static GstPad *
|
|
gst_play_bin3_get_video_pad (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_get_pad_of_type (playbin, PLAYBIN_STREAM_VIDEO, stream);
|
|
}
|
|
|
|
static GstPad *
|
|
gst_play_bin3_get_audio_pad (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_get_pad_of_type (playbin, PLAYBIN_STREAM_AUDIO, stream);
|
|
}
|
|
|
|
static GstPad *
|
|
gst_play_bin3_get_text_pad (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_get_pad_of_type (playbin, PLAYBIN_STREAM_TEXT, stream);
|
|
}
|
|
|
|
|
|
static GstTagList *
|
|
get_tags (GstPlayBin3 * playbin, GstStreamType type, gint stream_num)
|
|
{
|
|
GstTagList *result = NULL;
|
|
gint nb_streams = gst_stream_collection_get_size (playbin->collection);
|
|
gint i, cur_idx = 0;
|
|
|
|
/* Count the streams of the type we want to find the one numbered 'stream' */
|
|
for (i = 0; i < nb_streams; i++) {
|
|
GstStream *stream =
|
|
gst_stream_collection_get_stream (playbin->collection, i);
|
|
GstStreamType stream_type = gst_stream_get_stream_type (stream);
|
|
if (stream_type != type)
|
|
continue;
|
|
if (cur_idx == stream_num)
|
|
return gst_stream_get_tags (stream);
|
|
cur_idx++;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstTagList *
|
|
gst_play_bin3_get_video_tags (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
GstTagList *result;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
result = get_tags (playbin, GST_STREAM_TYPE_VIDEO, stream);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstTagList *
|
|
gst_play_bin3_get_audio_tags (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
GstTagList *result;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
result = get_tags (playbin, GST_STREAM_TYPE_AUDIO, stream);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstTagList *
|
|
gst_play_bin3_get_text_tags (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
GstTagList *result;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
result = get_tags (playbin, GST_STREAM_TYPE_TEXT, stream);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GstSample *
|
|
gst_play_bin3_convert_sample (GstPlayBin3 * playbin, GstCaps * caps)
|
|
{
|
|
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)
|
|
{
|
|
GstPad *src;
|
|
GstPad *peer;
|
|
GstStructure *s;
|
|
GstEvent *event;
|
|
gboolean ret = FALSE;
|
|
|
|
src = gst_element_get_static_pad (GST_ELEMENT_CAST (combiner), "src");
|
|
peer = gst_pad_get_peer (src);
|
|
if (peer) {
|
|
s = gst_structure_new_empty (event_name);
|
|
event = gst_event_new_custom (GST_EVENT_CUSTOM_DOWNSTREAM_OOB, s);
|
|
gst_pad_send_event (peer, event);
|
|
gst_object_unref (peer);
|
|
ret = TRUE;
|
|
}
|
|
gst_object_unref (src);
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_set_current_stream (GstPlayBin3 * playbin,
|
|
gint stream_type, gint * current_value, gint stream,
|
|
gboolean * flush_marker)
|
|
{
|
|
GstSourceCombine *combine;
|
|
GPtrArray *channels;
|
|
GstPad *sinkpad;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
/* This function is only called if the app sets
|
|
* one of the current-* properties, which means it doesn't
|
|
* handle collections or select-streams yet */
|
|
playbin->do_stream_selections = TRUE;
|
|
|
|
combine = playbin->combiner + stream_type;
|
|
channels = playbin->channels[stream_type];
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Changing current %s stream %d -> %d",
|
|
stream_type_names[stream_type], *current_value, stream);
|
|
|
|
if (combine->combiner == NULL) {
|
|
/* FIXME: Check that the current_value is within range */
|
|
*current_value = stream;
|
|
do_stream_selection (playbin);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
return TRUE;
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Using old style combiner");
|
|
|
|
if (!combine->has_active_pad)
|
|
goto no_active_pad;
|
|
if (channels == NULL)
|
|
goto no_channels;
|
|
|
|
if (stream == -1 || channels->len <= stream) {
|
|
sinkpad = NULL;
|
|
} else {
|
|
/* take channel from selected stream */
|
|
sinkpad = g_ptr_array_index (channels, stream);
|
|
}
|
|
|
|
if (sinkpad)
|
|
gst_object_ref (sinkpad);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
if (sinkpad) {
|
|
GstObject *combiner;
|
|
|
|
if ((combiner = gst_pad_get_parent (sinkpad))) {
|
|
GstPad *old_sinkpad;
|
|
|
|
g_object_get (combiner, "active-pad", &old_sinkpad, NULL);
|
|
|
|
if (old_sinkpad != sinkpad) {
|
|
/* FIXME: Is there actually any reason playsink
|
|
* needs special names for each type of stream we flush? */
|
|
gchar *flush_event_name = g_strdup_printf ("playsink-custom-%s-flush",
|
|
stream_type_names[stream_type]);
|
|
if (gst_play_bin3_send_custom_event (combiner, flush_event_name))
|
|
*flush_marker = TRUE;
|
|
g_free (flush_event_name);
|
|
|
|
/* activate the selected pad */
|
|
g_object_set (combiner, "active-pad", sinkpad, NULL);
|
|
}
|
|
|
|
if (old_sinkpad)
|
|
gst_object_unref (old_sinkpad);
|
|
|
|
gst_object_unref (combiner);
|
|
}
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
return TRUE;
|
|
|
|
no_active_pad:
|
|
{
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
GST_WARNING_OBJECT (playbin,
|
|
"can't switch %s, the stream combiner's sink pads don't have the \"active-pad\" property",
|
|
stream_type_names[stream_type]);
|
|
return FALSE;
|
|
}
|
|
no_channels:
|
|
{
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
GST_DEBUG_OBJECT (playbin, "can't switch video, we have no channels");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_set_current_video_stream (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_set_current_stream (playbin, PLAYBIN_STREAM_VIDEO,
|
|
&playbin->current_video, stream, &playbin->video_pending_flush_finish);
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_set_current_audio_stream (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_set_current_stream (playbin, PLAYBIN_STREAM_AUDIO,
|
|
&playbin->current_audio, stream, &playbin->audio_pending_flush_finish);
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_set_current_text_stream (GstPlayBin3 * playbin, gint stream)
|
|
{
|
|
return gst_play_bin3_set_current_stream (playbin, PLAYBIN_STREAM_TEXT,
|
|
&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,
|
|
const gchar * dbg, GstElement ** elem, GstElement * sink)
|
|
{
|
|
GST_INFO_OBJECT (playbin, "Setting %s sink to %" GST_PTR_FORMAT, dbg, sink);
|
|
|
|
gst_play_sink_set_sink (playbin->playsink, type, sink);
|
|
|
|
if (*elem)
|
|
gst_object_unref (*elem);
|
|
*elem = sink ? gst_object_ref (sink) : NULL;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_stream_combiner (GstPlayBin3 * playbin, GstElement ** elem,
|
|
const gchar * dbg, GstElement * combiner)
|
|
{
|
|
GST_INFO_OBJECT (playbin, "Setting %s stream combiner to %" GST_PTR_FORMAT,
|
|
dbg, combiner);
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (*elem != combiner) {
|
|
GstElement *old;
|
|
|
|
old = *elem;
|
|
if (combiner)
|
|
gst_object_ref_sink (combiner);
|
|
|
|
*elem = combiner;
|
|
if (old)
|
|
gst_object_unref (old);
|
|
}
|
|
GST_LOG_OBJECT (playbin, "%s stream combiner now %" GST_PTR_FORMAT, dbg,
|
|
*elem);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_encoding (GstPlayBin3 * playbin, const gchar * encoding)
|
|
{
|
|
GstElement *elem;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
|
|
/* set subtitles on decodebin. */
|
|
if ((elem = playbin->decodebin))
|
|
g_object_set (G_OBJECT (elem), "subtitle-encoding", encoding, NULL);
|
|
|
|
gst_play_sink_set_subtitle_encoding (playbin->playsink, encoding);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_URI:
|
|
gst_play_bin3_set_uri (playbin, g_value_get_string (value));
|
|
break;
|
|
case PROP_SUBURI:
|
|
gst_play_bin3_set_suburi (playbin, g_value_get_string (value));
|
|
break;
|
|
case PROP_FLAGS:
|
|
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",
|
|
(g_value_get_flags (value) & GST_PLAY_FLAG_DOWNLOAD) != 0, NULL);
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
|
|
}
|
|
break;
|
|
case PROP_CURRENT_VIDEO:
|
|
gst_play_bin3_set_current_video_stream (playbin, g_value_get_int (value));
|
|
break;
|
|
case PROP_CURRENT_AUDIO:
|
|
gst_play_bin3_set_current_audio_stream (playbin, g_value_get_int (value));
|
|
break;
|
|
case PROP_CURRENT_TEXT:
|
|
gst_play_bin3_set_current_text_stream (playbin, g_value_get_int (value));
|
|
break;
|
|
case PROP_AUTO_SELECT_STREAMS:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
playbin->do_stream_selections = g_value_get_boolean (value);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_SUBTITLE_ENCODING:
|
|
gst_play_bin3_set_encoding (playbin, g_value_get_string (value));
|
|
break;
|
|
case PROP_VIDEO_FILTER:
|
|
gst_play_sink_set_filter (playbin->playsink, GST_PLAY_SINK_TYPE_VIDEO,
|
|
GST_ELEMENT (g_value_get_object (value)));
|
|
break;
|
|
case PROP_AUDIO_FILTER:
|
|
gst_play_sink_set_filter (playbin->playsink, GST_PLAY_SINK_TYPE_AUDIO,
|
|
GST_ELEMENT (g_value_get_object (value)));
|
|
break;
|
|
case PROP_VIDEO_SINK:
|
|
gst_play_bin3_set_sink (playbin, GST_PLAY_SINK_TYPE_VIDEO, "video",
|
|
&playbin->video_sink, g_value_get_object (value));
|
|
break;
|
|
case PROP_AUDIO_SINK:
|
|
gst_play_bin3_set_sink (playbin, GST_PLAY_SINK_TYPE_AUDIO, "audio",
|
|
&playbin->audio_sink, g_value_get_object (value));
|
|
break;
|
|
case PROP_VIS_PLUGIN:
|
|
gst_play_sink_set_vis_plugin (playbin->playsink,
|
|
g_value_get_object (value));
|
|
break;
|
|
case PROP_TEXT_SINK:
|
|
gst_play_bin3_set_sink (playbin, GST_PLAY_SINK_TYPE_TEXT, "text",
|
|
&playbin->text_sink, g_value_get_object (value));
|
|
break;
|
|
case PROP_VIDEO_STREAM_COMBINER:
|
|
gst_play_bin3_set_stream_combiner (playbin,
|
|
&playbin->video_stream_combiner, "video", g_value_get_object (value));
|
|
break;
|
|
case PROP_AUDIO_STREAM_COMBINER:
|
|
gst_play_bin3_set_stream_combiner (playbin,
|
|
&playbin->audio_stream_combiner, "audio", g_value_get_object (value));
|
|
break;
|
|
case PROP_TEXT_STREAM_COMBINER:
|
|
gst_play_bin3_set_stream_combiner (playbin,
|
|
&playbin->text_stream_combiner, "text", g_value_get_object (value));
|
|
break;
|
|
case PROP_VOLUME:
|
|
gst_play_sink_set_volume (playbin->playsink, g_value_get_double (value));
|
|
break;
|
|
case PROP_MUTE:
|
|
gst_play_sink_set_mute (playbin->playsink, g_value_get_boolean (value));
|
|
break;
|
|
case PROP_FONT_DESC:
|
|
gst_play_sink_set_font_desc (playbin->playsink,
|
|
g_value_get_string (value));
|
|
break;
|
|
case PROP_CONNECTION_SPEED:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
playbin->connection_speed = g_value_get_uint64 (value) * 1000;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_BUFFER_SIZE:
|
|
playbin->buffer_size = g_value_get_int (value);
|
|
break;
|
|
case PROP_BUFFER_DURATION:
|
|
playbin->buffer_duration = g_value_get_int64 (value);
|
|
break;
|
|
case PROP_AV_OFFSET:
|
|
gst_play_sink_set_av_offset (playbin->playsink,
|
|
g_value_get_int64 (value));
|
|
break;
|
|
case PROP_RING_BUFFER_MAX_SIZE:
|
|
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,
|
|
"ring-buffer-max-size", playbin->ring_buffer_max_size, NULL);
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
|
|
}
|
|
break;
|
|
case PROP_FORCE_ASPECT_RATIO:
|
|
g_object_set (playbin->playsink, "force-aspect-ratio",
|
|
g_value_get_boolean (value), NULL);
|
|
break;
|
|
case PROP_MULTIVIEW_MODE:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
playbin->multiview_mode = g_value_get_enum (value);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_MULTIVIEW_FLAGS:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
playbin->multiview_flags = g_value_get_flags (value);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static GstElement *
|
|
gst_play_bin3_get_current_sink (GstPlayBin3 * playbin, GstElement ** elem,
|
|
const gchar * dbg, GstPlaySinkType type)
|
|
{
|
|
GstElement *sink = gst_play_sink_get_sink (playbin->playsink, type);
|
|
|
|
GST_LOG_OBJECT (playbin, "play_sink_get_sink() returned %s sink %"
|
|
GST_PTR_FORMAT ", the originally set %s sink is %" GST_PTR_FORMAT,
|
|
dbg, sink, dbg, *elem);
|
|
|
|
if (sink == NULL) {
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if ((sink = *elem))
|
|
gst_object_ref (sink);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
}
|
|
|
|
return sink;
|
|
}
|
|
|
|
static GstElement *
|
|
gst_play_bin3_get_current_stream_combiner (GstPlayBin3 * playbin,
|
|
GstElement ** elem, const gchar * dbg, int stream_type)
|
|
{
|
|
GstElement *combiner;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if ((combiner = playbin->combiner[stream_type].combiner))
|
|
gst_object_ref (combiner);
|
|
else if ((combiner = *elem))
|
|
gst_object_ref (combiner);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return combiner;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_get_property (GObject * object, guint prop_id, GValue * value,
|
|
GParamSpec * pspec)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_URI:
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = playbin->next_group;
|
|
g_value_set_string (value, group->uri);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
case PROP_CURRENT_URI:
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = get_group (playbin);
|
|
g_value_set_string (value, group->uri);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
case PROP_SUBURI:
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = playbin->next_group;
|
|
g_value_set_string (value, group->suburi);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
case PROP_CURRENT_SUBURI:
|
|
{
|
|
GstSourceGroup *group;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = get_group (playbin);
|
|
g_value_set_string (value, group->suburi);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
case PROP_SOURCE:
|
|
{
|
|
GST_OBJECT_LOCK (playbin);
|
|
g_value_set_object (value, playbin->source);
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
case PROP_FLAGS:
|
|
g_value_set_flags (value, gst_play_bin3_get_flags (playbin));
|
|
break;
|
|
case PROP_N_VIDEO:
|
|
{
|
|
gint n_video;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
n_video =
|
|
playbin->combiner[PLAYBIN_STREAM_VIDEO].
|
|
streams ? playbin->combiner[PLAYBIN_STREAM_VIDEO].streams->len : 0;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
g_value_set_int (value, n_video);
|
|
break;
|
|
}
|
|
case PROP_CURRENT_VIDEO:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (playbin->combiner[PLAYBIN_STREAM_VIDEO].current_stream != -1)
|
|
g_value_set_int (value,
|
|
playbin->combiner[PLAYBIN_STREAM_VIDEO].current_stream);
|
|
else
|
|
g_value_set_int (value, playbin->current_video);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_N_AUDIO:
|
|
{
|
|
gint n_audio;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
n_audio =
|
|
playbin->combiner[PLAYBIN_STREAM_AUDIO].
|
|
streams ? playbin->combiner[PLAYBIN_STREAM_AUDIO].streams->len : 0;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
g_value_set_int (value, n_audio);
|
|
break;
|
|
}
|
|
case PROP_CURRENT_AUDIO:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (playbin->combiner[PLAYBIN_STREAM_AUDIO].current_stream != -1)
|
|
g_value_set_int (value,
|
|
playbin->combiner[PLAYBIN_STREAM_AUDIO].current_stream);
|
|
else
|
|
g_value_set_int (value, playbin->current_audio);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_N_TEXT:
|
|
{
|
|
gint n_text;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
n_text =
|
|
playbin->combiner[PLAYBIN_STREAM_TEXT].
|
|
streams ? playbin->combiner[PLAYBIN_STREAM_TEXT].streams->len : 0;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
g_value_set_int (value, n_text);
|
|
break;
|
|
}
|
|
case PROP_CURRENT_TEXT:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (playbin->combiner[PLAYBIN_STREAM_TEXT].current_stream != -1)
|
|
g_value_set_int (value,
|
|
playbin->combiner[PLAYBIN_STREAM_TEXT].current_stream);
|
|
else
|
|
g_value_set_int (value, playbin->current_text);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_AUTO_SELECT_STREAMS:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
g_value_set_boolean (value, playbin->do_stream_selections);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_SUBTITLE_ENCODING:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
g_value_take_string (value,
|
|
gst_play_sink_get_subtitle_encoding (playbin->playsink));
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_VIDEO_FILTER:
|
|
g_value_take_object (value,
|
|
gst_play_sink_get_filter (playbin->playsink,
|
|
GST_PLAY_SINK_TYPE_VIDEO));
|
|
break;
|
|
case PROP_AUDIO_FILTER:
|
|
g_value_take_object (value,
|
|
gst_play_sink_get_filter (playbin->playsink,
|
|
GST_PLAY_SINK_TYPE_AUDIO));
|
|
break;
|
|
case PROP_VIDEO_SINK:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_sink (playbin, &playbin->video_sink,
|
|
"video", GST_PLAY_SINK_TYPE_VIDEO));
|
|
break;
|
|
case PROP_AUDIO_SINK:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_sink (playbin, &playbin->audio_sink,
|
|
"audio", GST_PLAY_SINK_TYPE_AUDIO));
|
|
break;
|
|
case PROP_VIS_PLUGIN:
|
|
g_value_take_object (value,
|
|
gst_play_sink_get_vis_plugin (playbin->playsink));
|
|
break;
|
|
case PROP_TEXT_SINK:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_sink (playbin, &playbin->text_sink,
|
|
"text", GST_PLAY_SINK_TYPE_TEXT));
|
|
break;
|
|
case PROP_VIDEO_STREAM_COMBINER:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_stream_combiner (playbin,
|
|
&playbin->video_stream_combiner, "video", PLAYBIN_STREAM_VIDEO));
|
|
break;
|
|
case PROP_AUDIO_STREAM_COMBINER:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_stream_combiner (playbin,
|
|
&playbin->audio_stream_combiner, "audio", PLAYBIN_STREAM_AUDIO));
|
|
break;
|
|
case PROP_TEXT_STREAM_COMBINER:
|
|
g_value_take_object (value,
|
|
gst_play_bin3_get_current_stream_combiner (playbin,
|
|
&playbin->text_stream_combiner, "text", PLAYBIN_STREAM_TEXT));
|
|
break;
|
|
case PROP_VOLUME:
|
|
g_value_set_double (value, gst_play_sink_get_volume (playbin->playsink));
|
|
break;
|
|
case PROP_MUTE:
|
|
g_value_set_boolean (value, gst_play_sink_get_mute (playbin->playsink));
|
|
break;
|
|
case PROP_SAMPLE:
|
|
gst_value_take_sample (value,
|
|
gst_play_sink_get_last_sample (playbin->playsink));
|
|
break;
|
|
case PROP_FONT_DESC:
|
|
g_value_take_string (value,
|
|
gst_play_sink_get_font_desc (playbin->playsink));
|
|
break;
|
|
case PROP_CONNECTION_SPEED:
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
g_value_set_uint64 (value, playbin->connection_speed / 1000);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
break;
|
|
case PROP_BUFFER_SIZE:
|
|
GST_OBJECT_LOCK (playbin);
|
|
g_value_set_int (value, playbin->buffer_size);
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
case PROP_BUFFER_DURATION:
|
|
GST_OBJECT_LOCK (playbin);
|
|
g_value_set_int64 (value, playbin->buffer_duration);
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
case PROP_AV_OFFSET:
|
|
g_value_set_int64 (value,
|
|
gst_play_sink_get_av_offset (playbin->playsink));
|
|
break;
|
|
case PROP_RING_BUFFER_MAX_SIZE:
|
|
g_value_set_uint64 (value, playbin->ring_buffer_max_size);
|
|
break;
|
|
case PROP_FORCE_ASPECT_RATIO:{
|
|
gboolean v;
|
|
|
|
g_object_get (playbin->playsink, "force-aspect-ratio", &v, NULL);
|
|
g_value_set_boolean (value, v);
|
|
break;
|
|
}
|
|
case PROP_MULTIVIEW_MODE:
|
|
GST_OBJECT_LOCK (playbin);
|
|
g_value_set_enum (value, playbin->multiview_mode);
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
case PROP_MULTIVIEW_FLAGS:
|
|
GST_OBJECT_LOCK (playbin);
|
|
g_value_set_flags (value, playbin->multiview_flags);
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_update_cached_duration_from_query (GstPlayBin3 * playbin,
|
|
gboolean valid, GstQuery * query)
|
|
{
|
|
GstFormat fmt;
|
|
gint64 duration;
|
|
gint i;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Updating cached duration from query");
|
|
gst_query_parse_duration (query, &fmt, &duration);
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (playbin->duration); i++) {
|
|
if (playbin->duration[i].format == 0 || fmt == playbin->duration[i].format) {
|
|
playbin->duration[i].valid = valid;
|
|
playbin->duration[i].format = fmt;
|
|
playbin->duration[i].duration = valid ? duration : -1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_update_cached_duration (GstPlayBin3 * playbin)
|
|
{
|
|
const GstFormat formats[] =
|
|
{ GST_FORMAT_TIME, GST_FORMAT_BYTES, GST_FORMAT_DEFAULT };
|
|
gboolean ret;
|
|
GstQuery *query;
|
|
gint i;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Updating cached durations before group switch");
|
|
for (i = 0; i < G_N_ELEMENTS (formats); i++) {
|
|
query = gst_query_new_duration (formats[i]);
|
|
ret =
|
|
GST_ELEMENT_CLASS (parent_class)->query (GST_ELEMENT_CAST (playbin),
|
|
query);
|
|
gst_play_bin3_update_cached_duration_from_query (playbin, ret, query);
|
|
gst_query_unref (query);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_query (GstElement * element, GstQuery * query)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (element);
|
|
gboolean ret;
|
|
|
|
/* During a group switch we shouldn't allow duration queries
|
|
* because it's not clear if the old or new group's duration
|
|
* is returned and if the sinks are already playing new data
|
|
* or old data. See bug #585969
|
|
*
|
|
* While we're at it, also don't do any other queries during
|
|
* a group switch or any other event that causes topology changes
|
|
* by taking the playbin lock in any case.
|
|
*/
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
|
|
if (GST_QUERY_TYPE (query) == GST_QUERY_DURATION) {
|
|
GstSourceGroup *group = playbin->curr_group;
|
|
gboolean pending;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
|
|
pending = group->pending || group->stream_changed_pending;
|
|
|
|
if (pending) {
|
|
GstFormat fmt;
|
|
gint i;
|
|
|
|
ret = FALSE;
|
|
gst_query_parse_duration (query, &fmt, NULL);
|
|
for (i = 0; i < G_N_ELEMENTS (playbin->duration); i++) {
|
|
if (fmt == playbin->duration[i].format) {
|
|
ret = playbin->duration[i].valid;
|
|
gst_query_set_duration (query, fmt,
|
|
(ret ? playbin->duration[i].duration : -1));
|
|
break;
|
|
}
|
|
}
|
|
/* if nothing cached yet, we might as well request duration,
|
|
* such as during initial startup */
|
|
if (ret) {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"Taking cached duration because of pending group switch: %d", ret);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
return ret;
|
|
}
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
}
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->query (element, query);
|
|
|
|
if (GST_QUERY_TYPE (query) == GST_QUERY_DURATION)
|
|
gst_play_bin3_update_cached_duration_from_query (playbin, ret, query);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gint
|
|
get_combiner_stream_id (GstPlayBin3 * playbin, GstSourceCombine * combine,
|
|
GList * full_list)
|
|
{
|
|
gint i;
|
|
GList *tmp;
|
|
|
|
for (i = 0; combine->streams->len; i++) {
|
|
GstStream *stream = (GstStream *) g_ptr_array_index (combine->streams, i);
|
|
const gchar *sid = gst_stream_get_stream_id (stream);
|
|
for (tmp = full_list; tmp; tmp = tmp->next) {
|
|
gchar *orig = (gchar *) tmp->data;
|
|
if (!g_strcmp0 (orig, sid))
|
|
return i;
|
|
}
|
|
}
|
|
|
|
/* Fallback */
|
|
return -1;
|
|
}
|
|
|
|
static GList *
|
|
extend_list_of_streams (GstPlayBin3 * playbin, GstStreamType stype,
|
|
GList * list)
|
|
{
|
|
GList *tmp, *res;
|
|
gint i, nb;
|
|
|
|
res = list;
|
|
|
|
nb = gst_stream_collection_get_size (playbin->collection);
|
|
for (i = 0; i < nb; i++) {
|
|
GstStream *stream =
|
|
gst_stream_collection_get_stream (playbin->collection, i);
|
|
GstStreamType curtype = gst_stream_get_stream_type (stream);
|
|
if (stype == curtype) {
|
|
gboolean already_there = FALSE;
|
|
const gchar *sid = gst_stream_get_stream_id (stream);
|
|
for (tmp = res; tmp; tmp = tmp->next) {
|
|
const gchar *other = (const gchar *) tmp->data;
|
|
if (!g_strcmp0 (sid, other)) {
|
|
already_there = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
if (!already_there) {
|
|
GST_DEBUG_OBJECT (playbin, "Adding stream %s", sid);
|
|
res = g_list_append (res, g_strdup (sid));
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static GstEvent *
|
|
update_select_streams_event (GstPlayBin3 * playbin, GstEvent * event)
|
|
{
|
|
GList *streams = NULL;
|
|
GList *to_use;
|
|
gint combine_id;
|
|
|
|
if (!playbin->audio_stream_combiner && !playbin->video_stream_combiner &&
|
|
!playbin->text_stream_combiner) {
|
|
/* Nothing to do */
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"No custom combiners, 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);
|
|
combine_id =
|
|
get_combiner_stream_id (playbin,
|
|
&playbin->combiner[PLAYBIN_STREAM_AUDIO], streams);
|
|
if (combine_id != -1)
|
|
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);
|
|
combine_id =
|
|
get_combiner_stream_id (playbin,
|
|
&playbin->combiner[PLAYBIN_STREAM_VIDEO], streams);
|
|
if (combine_id != -1)
|
|
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);
|
|
combine_id =
|
|
get_combiner_stream_id (playbin,
|
|
&playbin->combiner[PLAYBIN_STREAM_TEXT], streams);
|
|
if (combine_id != -1)
|
|
gst_play_bin3_set_current_text_stream (playbin, combine_id);
|
|
}
|
|
|
|
gst_event_unref (event);
|
|
return gst_event_new_select_streams (to_use);
|
|
}
|
|
|
|
static gboolean
|
|
gst_play_bin3_send_event (GstElement * element, GstEvent * event)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (element);
|
|
|
|
if (GST_EVENT_TYPE (event) == GST_EVENT_SELECT_STREAMS) {
|
|
gboolean res;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
GST_LOG_OBJECT (playbin,
|
|
"App sent select-streams, we won't do anything ourselves now");
|
|
/* This is probably already false, but it doesn't hurt to be sure */
|
|
playbin->do_stream_selections = 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);
|
|
|
|
/* 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);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return res;
|
|
}
|
|
|
|
/* Send event directly to playsink instead of letting GstBin iterate
|
|
* over all sink elements. The latter might send the event multiple times
|
|
* in case the SEEK causes a reconfiguration of the pipeline, as can easily
|
|
* happen with adaptive streaming demuxers.
|
|
*
|
|
* What would then happen is that the iterator would be reset, we send the
|
|
* event again, and on the second time it will fail in the majority of cases
|
|
* because the pipeline is still being reconfigured
|
|
*/
|
|
if (GST_EVENT_IS_UPSTREAM (event)) {
|
|
return gst_element_send_event (GST_ELEMENT_CAST (playbin->playsink), event);
|
|
}
|
|
|
|
return GST_ELEMENT_CLASS (parent_class)->send_event (element, event);
|
|
}
|
|
|
|
/* Called with playbin lock held */
|
|
static void
|
|
do_stream_selection (GstPlayBin3 * playbin)
|
|
{
|
|
GstStreamCollection *collection;
|
|
guint i, nb_streams;
|
|
GList *streams = NULL;
|
|
gint nb_video = 0, nb_audio = 0, nb_text = 0;
|
|
GstStreamType chosen_stream_types = 0;
|
|
|
|
collection = playbin->collection;
|
|
if (collection == NULL) {
|
|
GST_LOG_OBJECT (playbin, "No stream collection. Not doing stream-select");
|
|
return;
|
|
}
|
|
|
|
nb_streams = gst_stream_collection_get_size (collection);
|
|
if (nb_streams == 0) {
|
|
GST_INFO_OBJECT (playbin, "Empty collection received! Ignoring");
|
|
}
|
|
|
|
/* 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 */
|
|
for (i = 0; i < nb_streams; i++) {
|
|
GstStream *stream = gst_stream_collection_get_stream (collection, i);
|
|
GstStreamType stream_type = gst_stream_get_stream_type (stream);
|
|
const gchar *stream_id = gst_stream_get_stream_id (stream);
|
|
gint pb_stream_type = -1;
|
|
gboolean select_this = FALSE;
|
|
|
|
switch (stream_type) {
|
|
case 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 */
|
|
select_this =
|
|
(nb_audio == playbin->current_audio ||
|
|
(playbin->current_audio == -1 && nb_audio == 0) ||
|
|
playbin->audio_stream_combiner != NULL);
|
|
nb_audio++;
|
|
break;
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
pb_stream_type = PLAYBIN_STREAM_AUDIO;
|
|
select_this =
|
|
(nb_video == playbin->current_video ||
|
|
(playbin->current_video == -1 && nb_video == 0) ||
|
|
playbin->video_stream_combiner != NULL);
|
|
nb_video++;
|
|
break;
|
|
case GST_STREAM_TYPE_TEXT:
|
|
pb_stream_type = PLAYBIN_STREAM_TEXT;
|
|
select_this =
|
|
(nb_text == playbin->current_text ||
|
|
(playbin->current_text == -1 && nb_text == 0) ||
|
|
playbin->text_stream_combiner != NULL);
|
|
nb_text++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (pb_stream_type < 0) {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"Stream %d (id %s) of unhandled type %s. Ignoring", i, stream_id,
|
|
gst_stream_type_get_name (stream_type));
|
|
continue;
|
|
}
|
|
if (select_this) {
|
|
GST_DEBUG_OBJECT (playbin, "Selecting stream %s of type %s",
|
|
stream_id, gst_stream_type_get_name (stream_type));
|
|
/* Don't build the list if we're not in charge of stream selection */
|
|
if (playbin->do_stream_selections)
|
|
streams = g_list_append (streams, (gpointer) stream_id);
|
|
chosen_stream_types |= stream_type;
|
|
}
|
|
}
|
|
|
|
if (streams) {
|
|
GstEvent *ev = gst_event_new_select_streams (streams);
|
|
gst_element_send_event (playbin->decodebin, ev);
|
|
g_list_free (streams);
|
|
}
|
|
playbin->selected_stream_types = chosen_stream_types;
|
|
}
|
|
|
|
/* mime types we are not handling on purpose right now, don't post a
|
|
* missing-plugin message for these */
|
|
static const gchar *blacklisted_mimes[] = {
|
|
NULL
|
|
};
|
|
|
|
static void
|
|
notify_all_streams (GstPlayBin3 * playbin, GstStreamCollection * collection)
|
|
{
|
|
gint i, nb_streams;
|
|
nb_streams = gst_stream_collection_get_size (collection);
|
|
for (i = 0; i < nb_streams; i++)
|
|
notify_tags_for_stream (playbin, collection,
|
|
gst_stream_collection_get_stream (collection, i));
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_handle_message (GstBin * bin, GstMessage * msg)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (bin);
|
|
|
|
if (gst_is_missing_plugin_message (msg)) {
|
|
gchar *detail;
|
|
guint i;
|
|
|
|
detail = gst_missing_plugin_message_get_installer_detail (msg);
|
|
for (i = 0; detail != NULL && blacklisted_mimes[i] != NULL; ++i) {
|
|
if (strstr (detail, "|decoder-") && strstr (detail, blacklisted_mimes[i])) {
|
|
GST_LOG_OBJECT (bin, "suppressing message %" GST_PTR_FORMAT, msg);
|
|
gst_message_unref (msg);
|
|
g_free (detail);
|
|
return;
|
|
}
|
|
}
|
|
g_free (detail);
|
|
} else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_STREAM_START) {
|
|
GstSourceGroup *new_group = playbin->curr_group;
|
|
GstMessage *buffering_msg = NULL;
|
|
|
|
GST_SOURCE_GROUP_LOCK (new_group);
|
|
new_group->stream_changed_pending = FALSE;
|
|
if (new_group->pending_buffering_msg) {
|
|
buffering_msg = new_group->pending_buffering_msg;
|
|
new_group->pending_buffering_msg = NULL;
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (new_group);
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Stream start from new group %p", new_group);
|
|
|
|
if (buffering_msg) {
|
|
GST_DEBUG_OBJECT (playbin, "Posting pending buffering message: %"
|
|
GST_PTR_FORMAT, buffering_msg);
|
|
GST_BIN_CLASS (parent_class)->handle_message (bin, buffering_msg);
|
|
}
|
|
|
|
} else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_BUFFERING) {
|
|
GstSourceGroup *group = playbin->curr_group;
|
|
gboolean pending;
|
|
|
|
/* drop buffering messages from child queues while we are switching
|
|
* groups (because the application set a new uri in about-to-finish)
|
|
* if the playsink queue still has buffers to play */
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
pending = group->stream_changed_pending;
|
|
|
|
if (pending) {
|
|
GST_DEBUG_OBJECT (playbin, "Storing buffering message from pending group "
|
|
"%p %" GST_PTR_FORMAT, group, msg);
|
|
gst_message_replace (&group->pending_buffering_msg, msg);
|
|
gst_message_unref (msg);
|
|
msg = NULL;
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
} else 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) {
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"STREAM_COLLECTION: Got a collection from %" GST_PTR_FORMAT, src);
|
|
if (playbin->collection && playbin->collection_notify_id) {
|
|
g_signal_handler_disconnect (playbin->collection,
|
|
playbin->collection_notify_id);
|
|
playbin->collection_notify_id = 0;
|
|
}
|
|
gst_object_replace ((GstObject **) & playbin->collection,
|
|
(GstObject *) collection);
|
|
playbin->collection_notify_id =
|
|
g_signal_connect (collection, "stream-notify::tags",
|
|
(GCallback) notify_tags_cb, playbin);
|
|
update_combiner_info (playbin);
|
|
if (pstate)
|
|
playbin->do_stream_selections = FALSE;
|
|
do_stream_selection (playbin);
|
|
if (pstate)
|
|
playbin->do_stream_selections = TRUE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
notify_all_streams (playbin, collection);
|
|
}
|
|
} else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_STREAMS_SELECTED) {
|
|
GstStreamCollection *collection = NULL;
|
|
GstObject *src = GST_MESSAGE_SRC (msg);
|
|
gboolean pstate = playbin->do_stream_selections;
|
|
|
|
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);
|
|
if (playbin->collection && playbin->collection_notify_id) {
|
|
g_signal_handler_disconnect (playbin->collection,
|
|
playbin->collection_notify_id);
|
|
playbin->collection_notify_id = 0;
|
|
}
|
|
gst_object_replace ((GstObject **) & playbin->collection,
|
|
(GstObject *) collection);
|
|
playbin->collection_notify_id =
|
|
g_signal_connect (collection, "stream-notify::tags",
|
|
(GCallback) notify_tags_cb, playbin);
|
|
update_combiner_info (playbin);
|
|
len = gst_message_streams_selected_get_size (msg);
|
|
for (i = 0; i < len; i++) {
|
|
set_selected_stream (playbin,
|
|
gst_message_streams_selected_get_stream (msg, i));
|
|
}
|
|
if (pstate)
|
|
playbin->do_stream_selections = FALSE;
|
|
do_stream_selection (playbin);
|
|
if (pstate)
|
|
playbin->do_stream_selections = TRUE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
notify_all_streams (playbin, collection);
|
|
}
|
|
}
|
|
|
|
if (msg)
|
|
GST_BIN_CLASS (parent_class)->handle_message (bin, msg);
|
|
}
|
|
|
|
static void
|
|
combiner_active_pad_changed (GObject * combiner, GParamSpec * pspec,
|
|
GstPlayBin3 * playbin)
|
|
{
|
|
const gchar *property;
|
|
GstSourceCombine *combine = NULL;
|
|
GPtrArray *channels = NULL;
|
|
int i;
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
|
|
for (i = 0; i < PLAYBIN_STREAM_LAST; i++) {
|
|
if (combiner == G_OBJECT (playbin->combiner[i].combiner)) {
|
|
combine = &playbin->combiner[i];
|
|
channels = playbin->channels[i];
|
|
}
|
|
}
|
|
|
|
/* We got a pad-change after our group got switched out; no need to notify */
|
|
if (!combine) {
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
return;
|
|
}
|
|
|
|
switch (combine->type) {
|
|
case GST_PLAY_SINK_TYPE_VIDEO:
|
|
case GST_PLAY_SINK_TYPE_VIDEO_RAW:
|
|
property = "current-video";
|
|
playbin->current_video = get_current_stream_number (playbin,
|
|
combine, channels);
|
|
|
|
if (playbin->video_pending_flush_finish) {
|
|
playbin->video_pending_flush_finish = FALSE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
gst_play_bin3_send_custom_event (GST_OBJECT (combiner),
|
|
"playsink-custom-video-flush-finish");
|
|
goto notify;
|
|
}
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_AUDIO:
|
|
case GST_PLAY_SINK_TYPE_AUDIO_RAW:
|
|
property = "current-audio";
|
|
playbin->current_audio = get_current_stream_number (playbin,
|
|
combine, channels);
|
|
|
|
if (playbin->audio_pending_flush_finish) {
|
|
playbin->audio_pending_flush_finish = FALSE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
gst_play_bin3_send_custom_event (GST_OBJECT (combiner),
|
|
"playsink-custom-audio-flush-finish");
|
|
goto notify;
|
|
}
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_TEXT:
|
|
property = "current-text";
|
|
playbin->current_text = get_current_stream_number (playbin,
|
|
combine, channels);
|
|
|
|
if (playbin->text_pending_flush_finish) {
|
|
playbin->text_pending_flush_finish = FALSE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
gst_play_bin3_send_custom_event (GST_OBJECT (combiner),
|
|
"playsink-custom-subtitle-flush-finish");
|
|
goto notify;
|
|
}
|
|
break;
|
|
default:
|
|
property = NULL;
|
|
}
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
notify:
|
|
if (property)
|
|
g_object_notify (G_OBJECT (playbin), property);
|
|
}
|
|
|
|
static GstCaps *
|
|
update_video_multiview_caps (GstPlayBin3 * playbin, GstCaps * caps)
|
|
{
|
|
GstVideoMultiviewMode mv_mode;
|
|
GstVideoMultiviewMode cur_mv_mode;
|
|
GstVideoMultiviewFlags mv_flags, cur_mv_flags;
|
|
GstStructure *s;
|
|
const gchar *mview_mode_str;
|
|
GstCaps *out_caps;
|
|
|
|
GST_OBJECT_LOCK (playbin);
|
|
mv_mode = (GstVideoMultiviewMode) playbin->multiview_mode;
|
|
mv_flags = playbin->multiview_flags;
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
|
|
if (mv_mode == GST_VIDEO_MULTIVIEW_MODE_NONE)
|
|
return NULL;
|
|
|
|
cur_mv_mode = GST_VIDEO_MULTIVIEW_MODE_NONE;
|
|
cur_mv_flags = GST_VIDEO_MULTIVIEW_FLAGS_NONE;
|
|
|
|
s = gst_caps_get_structure (caps, 0);
|
|
|
|
gst_structure_get_flagset (s, "multiview-flags", &cur_mv_flags, NULL);
|
|
if ((mview_mode_str = gst_structure_get_string (s, "multiview-mode")))
|
|
cur_mv_mode = gst_video_multiview_mode_from_caps_string (mview_mode_str);
|
|
|
|
/* We can't override an existing annotated multiview mode, except
|
|
* maybe (in the future) we could change some flags. */
|
|
if ((gint) cur_mv_mode > GST_VIDEO_MULTIVIEW_MAX_FRAME_PACKING) {
|
|
GST_INFO_OBJECT (playbin, "Cannot override existing multiview mode");
|
|
return NULL;
|
|
}
|
|
|
|
mview_mode_str = gst_video_multiview_mode_to_caps_string (mv_mode);
|
|
g_assert (mview_mode_str != NULL);
|
|
out_caps = gst_caps_copy (caps);
|
|
s = gst_caps_get_structure (out_caps, 0);
|
|
|
|
gst_structure_set (s, "multiview-mode", G_TYPE_STRING, mview_mode_str,
|
|
"multiview-flags", GST_TYPE_VIDEO_MULTIVIEW_FLAGSET, mv_flags,
|
|
GST_FLAG_SET_MASK_EXACT, NULL);
|
|
|
|
return out_caps;
|
|
}
|
|
|
|
static GstPadProbeReturn
|
|
_decodebin_event_probe (GstPad * pad, GstPadProbeInfo * info, gpointer udata)
|
|
{
|
|
GstPadProbeReturn ret = GST_PAD_PROBE_OK;
|
|
GstPlayBin3 *playbin = (GstPlayBin3 *) udata;
|
|
GstEvent *event = GST_PAD_PROBE_INFO_DATA (info);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_CAPS:{
|
|
GstCaps *caps = NULL;
|
|
const GstStructure *s;
|
|
const gchar *name;
|
|
|
|
gst_event_parse_caps (event, &caps);
|
|
/* If video caps, check if we should override multiview flags */
|
|
s = gst_caps_get_structure (caps, 0);
|
|
name = gst_structure_get_name (s);
|
|
if (g_str_has_prefix (name, "video/")) {
|
|
caps = update_video_multiview_caps (playbin, caps);
|
|
if (caps) {
|
|
gst_event_unref (event);
|
|
event = gst_event_new_caps (caps);
|
|
GST_PAD_PROBE_INFO_DATA (info) = event;
|
|
gst_caps_unref (caps);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gint
|
|
find_index_for_stream_by_type (GstStreamCollection * collection,
|
|
GstStream * stream)
|
|
{
|
|
gint nb_streams = gst_stream_collection_get_size (collection);
|
|
gint i, cur_idx = 0;
|
|
GstStreamType target_type = gst_stream_get_stream_type (stream);
|
|
|
|
/* Count the streams of the type we want to find the index of the one we want */
|
|
for (i = 0; i < nb_streams; i++) {
|
|
GstStream *cur_stream = gst_stream_collection_get_stream (collection, i);
|
|
if (stream == cur_stream)
|
|
return cur_idx;
|
|
if (gst_stream_get_stream_type (cur_stream) == target_type)
|
|
cur_idx++;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static void
|
|
notify_tags_for_stream (GstPlayBin3 * playbin, GstStreamCollection * collection,
|
|
GstStream * stream)
|
|
{
|
|
GstStreamType stream_type = gst_stream_get_stream_type (stream);
|
|
gint stream_idx = find_index_for_stream_by_type (collection, stream);
|
|
gint signal;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "Tags on stream %" GST_PTR_FORMAT
|
|
" with stream idx %d and type %d have changed",
|
|
stream, stream_idx, stream_type);
|
|
|
|
switch (stream_type) {
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
signal = SIGNAL_VIDEO_TAGS_CHANGED;
|
|
break;
|
|
case GST_STREAM_TYPE_AUDIO:
|
|
signal = SIGNAL_AUDIO_TAGS_CHANGED;
|
|
break;
|
|
case GST_STREAM_TYPE_TEXT:
|
|
signal = SIGNAL_TEXT_TAGS_CHANGED;
|
|
break;
|
|
default:
|
|
signal = -1;
|
|
break;
|
|
}
|
|
|
|
if (signal >= 0)
|
|
g_signal_emit (G_OBJECT (playbin), gst_play_bin3_signals[signal], 0,
|
|
stream_idx);
|
|
}
|
|
|
|
static void
|
|
notify_tags_cb (GstStreamCollection * collection, GstStream * stream,
|
|
GParamSpec * pspec, GstPlayBin3 * playbin)
|
|
{
|
|
notify_tags_for_stream (playbin, collection, stream);
|
|
}
|
|
|
|
/* 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)
|
|
{
|
|
GstPad *sinkpad;
|
|
GstPadLinkReturn res;
|
|
GstSourceCombine *combine = NULL;
|
|
GstStreamType stream_type;
|
|
gint pb_stream_type = -1;
|
|
gboolean changed = FALSE;
|
|
GstElement *custom_combiner = NULL;
|
|
gulong event_probe_handler;
|
|
gchar *pad_name;
|
|
|
|
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;
|
|
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;
|
|
}
|
|
|
|
g_free (pad_name);
|
|
|
|
/* 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;
|
|
}
|
|
|
|
combine = &playbin->combiner[pb_stream_type];
|
|
|
|
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 (!custom_combiner) {
|
|
/* sync-mode=1, use clock */
|
|
if (combine->type == GST_PLAY_SINK_TYPE_TEXT)
|
|
g_object_set (combine->combiner, "sync-streams", TRUE,
|
|
"sync-mode", 1, "cache-buffers", TRUE, NULL);
|
|
else
|
|
g_object_set (combine->combiner, "sync-streams", TRUE, NULL);
|
|
}
|
|
|
|
if (combine->has_active_pad)
|
|
g_signal_connect (combine->combiner, "notify::active-pad",
|
|
G_CALLBACK (combiner_active_pad_changed), playbin);
|
|
|
|
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);
|
|
}
|
|
|
|
GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
|
|
|
|
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);
|
|
}
|
|
|
|
/* 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);
|
|
}
|
|
|
|
/* 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));
|
|
|
|
/* 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;
|
|
|
|
/* store the combiner for the pad */
|
|
g_object_set_data (G_OBJECT (sinkpad), "playbin.combine", combine);
|
|
|
|
/* store the pad in the array */
|
|
GST_DEBUG_OBJECT (playbin, "pad %p added to array", sinkpad);
|
|
g_ptr_array_add (combine->channels, sinkpad);
|
|
|
|
res = gst_pad_link (pad, sinkpad);
|
|
if (GST_PAD_LINK_FAILED (res))
|
|
goto link_failed;
|
|
|
|
/* store combiner pad so we can release it */
|
|
g_object_set_data (G_OBJECT (pad), "playbin.sinkpad", sinkpad);
|
|
|
|
changed = TRUE;
|
|
GST_DEBUG_OBJECT (playbin, "linked pad %s:%s to combiner %p",
|
|
GST_DEBUG_PAD_NAME (pad), combine->combiner);
|
|
} else {
|
|
goto request_pad_failed;
|
|
}
|
|
} else {
|
|
/* no combiner, don't configure anything, we'll link the new pad directly to
|
|
* the sink. */
|
|
changed = FALSE;
|
|
sinkpad = NULL;
|
|
|
|
/* store the combiner for the pad */
|
|
g_object_set_data (G_OBJECT (pad), "playbin.combine", combine);
|
|
}
|
|
|
|
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));
|
|
|
|
if (changed) {
|
|
int signal;
|
|
|
|
switch (combine->type) {
|
|
case GST_PLAY_SINK_TYPE_VIDEO:
|
|
case GST_PLAY_SINK_TYPE_VIDEO_RAW:
|
|
signal = SIGNAL_VIDEO_CHANGED;
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_AUDIO:
|
|
case GST_PLAY_SINK_TYPE_AUDIO_RAW:
|
|
signal = SIGNAL_AUDIO_CHANGED;
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_TEXT:
|
|
signal = SIGNAL_TEXT_CHANGED;
|
|
break;
|
|
default:
|
|
signal = -1;
|
|
}
|
|
|
|
if (signal >= 0) {
|
|
g_signal_emit (G_OBJECT (playbin), gst_play_bin3_signals[signal], 0,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
playbin->active_stream_types |= stream_type;
|
|
|
|
/* If we're expecting either audio or video,
|
|
* wait for them to appear before configuring playsink */
|
|
if ((playbin->selected_stream_types & ~playbin->active_stream_types &
|
|
(GST_STREAM_TYPE_VIDEO | GST_STREAM_TYPE_AUDIO))
|
|
== 0) {
|
|
no_more_pads_cb (decodebin, playbin);
|
|
} else {
|
|
GST_LOG_OBJECT (playbin, "Active stream types 0x%x, want 0x%x. Waiting",
|
|
playbin->active_stream_types, playbin->selected_stream_types);
|
|
}
|
|
|
|
return;
|
|
|
|
/* ERRORS */
|
|
unknown_type:
|
|
GST_DEBUG_OBJECT (playbin, "Ignoring pad with unknown type");
|
|
return;
|
|
|
|
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;
|
|
}
|
|
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;
|
|
}
|
|
}
|
|
|
|
/* 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, GstPlayBin3 * playbin)
|
|
{
|
|
GstPad *peer;
|
|
GstElement *combiner;
|
|
GstSourceCombine *combine;
|
|
int signal = -1;
|
|
gulong event_probe_handler;
|
|
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"decoded pad %s:%s removed", GST_DEBUG_PAD_NAME (pad));
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
|
|
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);
|
|
}
|
|
|
|
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);
|
|
goto exit;
|
|
}
|
|
|
|
/* 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);
|
|
|
|
/* get the correct type-changed signal */
|
|
switch (combine->type) {
|
|
case GST_PLAY_SINK_TYPE_VIDEO:
|
|
case GST_PLAY_SINK_TYPE_VIDEO_RAW:
|
|
signal = SIGNAL_VIDEO_CHANGED;
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_AUDIO:
|
|
case GST_PLAY_SINK_TYPE_AUDIO_RAW:
|
|
signal = SIGNAL_AUDIO_CHANGED;
|
|
break;
|
|
case GST_PLAY_SINK_TYPE_TEXT:
|
|
signal = SIGNAL_TEXT_CHANGED;
|
|
break;
|
|
default:
|
|
signal = -1;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
/* 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);
|
|
|
|
gst_object_unref (combiner);
|
|
exit:
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
if (signal >= 0)
|
|
g_signal_emit (G_OBJECT (playbin), gst_play_bin3_signals[signal], 0, NULL);
|
|
|
|
return;
|
|
|
|
/* ERRORS */
|
|
not_linked:
|
|
{
|
|
GST_DEBUG_OBJECT (playbin, "pad not linked");
|
|
goto exit;
|
|
}
|
|
}
|
|
|
|
|
|
static gint
|
|
select_stream_cb (GstElement * decodebin, GstStreamCollection * collection,
|
|
GstStream * stream, GstPlayBin3 * playbin)
|
|
{
|
|
GstStreamType stype = gst_stream_get_stream_type (stream);
|
|
GstElement *combiner = NULL;
|
|
|
|
switch (stype) {
|
|
case GST_STREAM_TYPE_AUDIO:
|
|
combiner = playbin->audio_stream_combiner;
|
|
break;
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
combiner = playbin->video_stream_combiner;
|
|
break;
|
|
case GST_STREAM_TYPE_TEXT:
|
|
combiner = playbin->text_stream_combiner;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (combiner) {
|
|
GST_DEBUG_OBJECT (playbin, "Got a combiner, requesting stream activation");
|
|
return 1;
|
|
}
|
|
|
|
/* Let decodebin3 decide otherwise */
|
|
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.
|
|
*/
|
|
static void
|
|
no_more_pads_cb (GstElement * decodebin, GstPlayBin3 * playbin)
|
|
{
|
|
GstSourceGroup *group;
|
|
GstPadLinkReturn res;
|
|
gint i;
|
|
gboolean configure;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "no more pads");
|
|
|
|
GST_PLAY_BIN3_SHUTDOWN_LOCK (playbin, shutdown);
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
group = playbin->curr_group;
|
|
|
|
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));
|
|
}
|
|
}
|
|
}
|
|
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 (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);
|
|
}
|
|
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
/* 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];
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
gst_play_sink_reconfigure (playbin->playsink);
|
|
}
|
|
|
|
GST_PLAY_BIN3_SHUTDOWN_UNLOCK (playbin);
|
|
|
|
if (configure) {
|
|
do_async_done (playbin);
|
|
}
|
|
|
|
return;
|
|
|
|
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];
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
drained_cb (GstElement * decodebin, GstSourceGroup * group)
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
|
|
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);
|
|
|
|
/* 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);
|
|
}
|
|
#endif
|
|
|
|
/* Like gst_element_factory_can_sink_any_caps() but doesn't
|
|
* allow ANY caps on the sinkpad template */
|
|
static gboolean
|
|
_factory_can_sink_caps (GstElementFactory * factory, GstCaps * caps)
|
|
{
|
|
const GList *templs;
|
|
|
|
templs = gst_element_factory_get_static_pad_templates (factory);
|
|
|
|
while (templs) {
|
|
GstStaticPadTemplate *templ = (GstStaticPadTemplate *) templs->data;
|
|
|
|
if (templ->direction == GST_PAD_SINK) {
|
|
GstCaps *templcaps = gst_static_caps_get (&templ->static_caps);
|
|
|
|
if (!gst_caps_is_any (templcaps)
|
|
&& gst_caps_is_subset (caps, templcaps)) {
|
|
gst_caps_unref (templcaps);
|
|
return TRUE;
|
|
}
|
|
gst_caps_unref (templcaps);
|
|
}
|
|
templs = g_list_next (templs);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
avelements_free (gpointer avelement)
|
|
{
|
|
GstAVElement *elm = (GstAVElement *) avelement;
|
|
|
|
if (elm->dec)
|
|
gst_object_unref (elm->dec);
|
|
if (elm->sink)
|
|
gst_object_unref (elm->sink);
|
|
g_slice_free (GstAVElement, elm);
|
|
}
|
|
|
|
static gint
|
|
avelement_compare_decoder (gconstpointer p1, gconstpointer p2,
|
|
gpointer user_data)
|
|
{
|
|
GstAVElement *v1, *v2;
|
|
|
|
v1 = (GstAVElement *) p1;
|
|
v2 = (GstAVElement *) p2;
|
|
|
|
return strcmp (GST_OBJECT_NAME (v1->dec), GST_OBJECT_NAME (v2->dec));
|
|
}
|
|
|
|
static gint
|
|
avelement_lookup_decoder (gconstpointer p1, gconstpointer p2,
|
|
gpointer user_data)
|
|
{
|
|
GstAVElement *v1;
|
|
GstElementFactory *f2;
|
|
|
|
v1 = (GstAVElement *) p1;
|
|
f2 = (GstElementFactory *) p2;
|
|
|
|
return strcmp (GST_OBJECT_NAME (v1->dec), GST_OBJECT_NAME (f2));
|
|
}
|
|
|
|
static gint
|
|
avelement_compare (gconstpointer p1, gconstpointer p2)
|
|
{
|
|
GstAVElement *v1, *v2;
|
|
GstPluginFeature *fd1, *fd2, *fs1, *fs2;
|
|
gint64 diff, v1_rank, v2_rank;
|
|
|
|
v1 = (GstAVElement *) p1;
|
|
v2 = (GstAVElement *) p2;
|
|
|
|
fd1 = (GstPluginFeature *) v1->dec;
|
|
fd2 = (GstPluginFeature *) v2->dec;
|
|
|
|
/* If both have a sink, we also compare their ranks */
|
|
if (v1->sink && v2->sink) {
|
|
fs1 = (GstPluginFeature *) v1->sink;
|
|
fs2 = (GstPluginFeature *) v2->sink;
|
|
v1_rank =
|
|
gst_plugin_feature_get_rank (fd1) * gst_plugin_feature_get_rank (fs1);
|
|
v2_rank =
|
|
gst_plugin_feature_get_rank (fd2) * gst_plugin_feature_get_rank (fs2);
|
|
} else {
|
|
v1_rank = gst_plugin_feature_get_rank (fd1);
|
|
v2_rank = gst_plugin_feature_get_rank (fd2);
|
|
fs1 = fs2 = NULL;
|
|
}
|
|
|
|
/* comparison based on the rank */
|
|
diff = v2_rank - v1_rank;
|
|
if (diff < 0)
|
|
return -1;
|
|
else if (diff > 0)
|
|
return 1;
|
|
|
|
/* comparison based on number of common caps features */
|
|
diff = v2->n_comm_cf - v1->n_comm_cf;
|
|
if (diff != 0)
|
|
return diff;
|
|
|
|
if (fs1 && fs2) {
|
|
/* comparison based on the name of sink elements */
|
|
diff = strcmp (GST_OBJECT_NAME (fs1), GST_OBJECT_NAME (fs2));
|
|
if (diff != 0)
|
|
return diff;
|
|
}
|
|
|
|
/* comparison based on the name of decoder elements */
|
|
return strcmp (GST_OBJECT_NAME (fd1), GST_OBJECT_NAME (fd2));
|
|
}
|
|
|
|
static GSequence *
|
|
avelements_create (GstPlayBin3 * playbin, gboolean isaudioelement)
|
|
{
|
|
GstElementFactory *d_factory, *s_factory;
|
|
GList *dec_list, *sink_list, *dl, *sl;
|
|
GSequence *ave_seq = NULL;
|
|
GstAVElement *ave;
|
|
guint n_common_cf = 0;
|
|
|
|
if (isaudioelement) {
|
|
sink_list = gst_element_factory_list_get_elements
|
|
(GST_ELEMENT_FACTORY_TYPE_SINK |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO, GST_RANK_MARGINAL);
|
|
dec_list =
|
|
gst_element_factory_list_get_elements (GST_ELEMENT_FACTORY_TYPE_DECODER
|
|
| GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO, GST_RANK_MARGINAL);
|
|
} else {
|
|
sink_list = gst_element_factory_list_get_elements
|
|
(GST_ELEMENT_FACTORY_TYPE_SINK |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE, GST_RANK_MARGINAL);
|
|
|
|
dec_list =
|
|
gst_element_factory_list_get_elements (GST_ELEMENT_FACTORY_TYPE_DECODER
|
|
| GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE, GST_RANK_MARGINAL);
|
|
}
|
|
|
|
/* create a list of audio/video elements. Each element in the list
|
|
* is holding an audio/video decoder and an audio/video sink in which
|
|
* the decoders srcpad template caps and sink element's sinkpad template
|
|
* caps are compatible */
|
|
dl = dec_list;
|
|
sl = sink_list;
|
|
|
|
ave_seq = g_sequence_new ((GDestroyNotify) avelements_free);
|
|
|
|
for (; dl; dl = dl->next) {
|
|
d_factory = (GstElementFactory *) dl->data;
|
|
for (; sl; sl = sl->next) {
|
|
s_factory = (GstElementFactory *) sl->data;
|
|
|
|
n_common_cf =
|
|
gst_playback_utils_get_n_common_capsfeatures (d_factory, s_factory,
|
|
gst_play_bin3_get_flags (playbin), isaudioelement);
|
|
if (n_common_cf < 1)
|
|
continue;
|
|
|
|
ave = g_slice_new (GstAVElement);
|
|
ave->dec = gst_object_ref (d_factory);
|
|
ave->sink = gst_object_ref (s_factory);
|
|
ave->n_comm_cf = n_common_cf;
|
|
g_sequence_append (ave_seq, ave);
|
|
}
|
|
sl = sink_list;
|
|
}
|
|
g_sequence_sort (ave_seq, (GCompareDataFunc) avelement_compare_decoder, NULL);
|
|
|
|
gst_plugin_feature_list_free (dec_list);
|
|
gst_plugin_feature_list_free (sink_list);
|
|
|
|
return ave_seq;
|
|
}
|
|
|
|
static gboolean
|
|
avelement_iter_is_equal (GSequenceIter * iter, GstElementFactory * factory)
|
|
{
|
|
GstAVElement *ave;
|
|
|
|
if (!iter)
|
|
return FALSE;
|
|
|
|
ave = g_sequence_get (iter);
|
|
if (!ave)
|
|
return FALSE;
|
|
|
|
return strcmp (GST_OBJECT_NAME (ave->dec), GST_OBJECT_NAME (factory)) == 0;
|
|
}
|
|
|
|
static GList *
|
|
create_decoders_list (GList * factory_list, GSequence * avelements)
|
|
{
|
|
GList *dec_list = NULL, *tmp;
|
|
GList *ave_list = NULL;
|
|
GList *ave_free_list = NULL;
|
|
GstAVElement *ave, *best_ave;
|
|
|
|
g_return_val_if_fail (factory_list != NULL, NULL);
|
|
g_return_val_if_fail (avelements != NULL, NULL);
|
|
|
|
for (tmp = factory_list; tmp; tmp = tmp->next) {
|
|
GstElementFactory *factory = (GstElementFactory *) tmp->data;
|
|
|
|
/* if there are parsers or sink elements, add them first */
|
|
if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_PARSER) ||
|
|
gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_SINK)) {
|
|
dec_list = g_list_prepend (dec_list, gst_object_ref (factory));
|
|
} else {
|
|
GSequenceIter *seq_iter;
|
|
|
|
seq_iter =
|
|
g_sequence_lookup (avelements, factory,
|
|
(GCompareDataFunc) avelement_lookup_decoder, NULL);
|
|
if (!seq_iter) {
|
|
GstAVElement *ave = g_slice_new0 (GstAVElement);
|
|
|
|
ave->dec = factory;
|
|
ave->sink = NULL;
|
|
/* There's at least raw */
|
|
ave->n_comm_cf = 1;
|
|
|
|
ave_list = g_list_prepend (ave_list, ave);
|
|
|
|
/* We need to free these later */
|
|
ave_free_list = g_list_prepend (ave_free_list, ave);
|
|
continue;
|
|
}
|
|
|
|
/* Go to first iter with that decoder */
|
|
do {
|
|
GSequenceIter *tmp_seq_iter;
|
|
|
|
tmp_seq_iter = g_sequence_iter_prev (seq_iter);
|
|
if (!avelement_iter_is_equal (tmp_seq_iter, factory))
|
|
break;
|
|
seq_iter = tmp_seq_iter;
|
|
} while (!g_sequence_iter_is_begin (seq_iter));
|
|
|
|
/* Get the best ranked GstAVElement for that factory */
|
|
best_ave = NULL;
|
|
while (!g_sequence_iter_is_end (seq_iter)
|
|
&& avelement_iter_is_equal (seq_iter, factory)) {
|
|
ave = g_sequence_get (seq_iter);
|
|
|
|
if (!best_ave || avelement_compare (ave, best_ave) < 0)
|
|
best_ave = ave;
|
|
|
|
seq_iter = g_sequence_iter_next (seq_iter);
|
|
}
|
|
ave_list = g_list_prepend (ave_list, best_ave);
|
|
}
|
|
}
|
|
|
|
/* Sort all GstAVElements by their relative ranks and insert
|
|
* into the decoders list */
|
|
ave_list = g_list_sort (ave_list, (GCompareFunc) avelement_compare);
|
|
for (tmp = ave_list; tmp; tmp = tmp->next) {
|
|
ave = (GstAVElement *) tmp->data;
|
|
dec_list = g_list_prepend (dec_list, gst_object_ref (ave->dec));
|
|
}
|
|
g_list_free (ave_list);
|
|
gst_plugin_feature_list_free (factory_list);
|
|
|
|
for (tmp = ave_free_list; tmp; tmp = tmp->next)
|
|
g_slice_free (GstAVElement, tmp->data);
|
|
g_list_free (ave_free_list);
|
|
|
|
dec_list = g_list_reverse (dec_list);
|
|
|
|
return dec_list;
|
|
}
|
|
|
|
/* Called when we must provide a list of factories to plug to @pad with @caps.
|
|
* We first check if we have a sink that can handle the format and if we do, we
|
|
* return NULL, to expose the pad. If we have no sink (or the sink does not
|
|
* work), we return the list of elements that can connect. */
|
|
static GValueArray *
|
|
autoplug_factories_cb (GstElement * decodebin, GstPad * pad,
|
|
GstCaps * caps, GstSourceGroup * group)
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
GList *factory_list, *tmp;
|
|
GValueArray *result;
|
|
gboolean unref_caps = FALSE;
|
|
gboolean isaudiodeclist = FALSE;
|
|
gboolean isvideodeclist = FALSE;
|
|
|
|
if (!caps) {
|
|
caps = gst_caps_new_any ();
|
|
unref_caps = TRUE;
|
|
}
|
|
|
|
playbin = group->playbin;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "factories group %p for %s:%s, %" GST_PTR_FORMAT,
|
|
group, GST_DEBUG_PAD_NAME (pad), caps);
|
|
|
|
/* filter out the elements based on the caps. */
|
|
g_mutex_lock (&playbin->elements_lock);
|
|
gst_play_bin3_update_elements_list (playbin);
|
|
factory_list =
|
|
gst_element_factory_list_filter (playbin->elements, caps, GST_PAD_SINK,
|
|
gst_caps_is_fixed (caps));
|
|
g_mutex_unlock (&playbin->elements_lock);
|
|
|
|
GST_DEBUG_OBJECT (playbin, "found factories %p", factory_list);
|
|
GST_PLUGIN_FEATURE_LIST_DEBUG (factory_list);
|
|
|
|
/* check whether the caps are asking for a list of audio/video decoders */
|
|
tmp = factory_list;
|
|
if (!gst_caps_is_any (caps)) {
|
|
for (; tmp; tmp = tmp->next) {
|
|
GstElementFactory *factory = (GstElementFactory *) tmp->data;
|
|
|
|
isvideodeclist = gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_DECODER |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE);
|
|
isaudiodeclist = gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_DECODER |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO);
|
|
|
|
if (isaudiodeclist || isvideodeclist)
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (isaudiodeclist || isvideodeclist) {
|
|
GSequence **ave_list;
|
|
if (isaudiodeclist)
|
|
ave_list = &playbin->aelements;
|
|
else
|
|
ave_list = &playbin->velements;
|
|
|
|
g_mutex_lock (&playbin->elements_lock);
|
|
/* sort factory_list based on the GstAVElement list priority */
|
|
factory_list = create_decoders_list (factory_list, *ave_list);
|
|
g_mutex_unlock (&playbin->elements_lock);
|
|
}
|
|
|
|
/* 2 additional elements for the already set audio/video sinks */
|
|
result = g_value_array_new (g_list_length (factory_list) + 2);
|
|
|
|
/* Check if we already have an audio/video sink and if this is the case
|
|
* put it as the first element of the array */
|
|
if (group->audio_sink) {
|
|
GstElementFactory *factory = gst_element_get_factory (group->audio_sink);
|
|
|
|
if (factory && _factory_can_sink_caps (factory, caps)) {
|
|
GValue val = { 0, };
|
|
|
|
g_value_init (&val, G_TYPE_OBJECT);
|
|
g_value_set_object (&val, factory);
|
|
result = g_value_array_append (result, &val);
|
|
g_value_unset (&val);
|
|
}
|
|
}
|
|
|
|
if (group->video_sink) {
|
|
GstElementFactory *factory = gst_element_get_factory (group->video_sink);
|
|
|
|
if (factory && _factory_can_sink_caps (factory, caps)) {
|
|
GValue val = { 0, };
|
|
|
|
g_value_init (&val, G_TYPE_OBJECT);
|
|
g_value_set_object (&val, factory);
|
|
result = g_value_array_append (result, &val);
|
|
g_value_unset (&val);
|
|
}
|
|
}
|
|
|
|
for (tmp = factory_list; tmp; tmp = tmp->next) {
|
|
GstElementFactory *factory = GST_ELEMENT_FACTORY_CAST (tmp->data);
|
|
GValue val = { 0, };
|
|
|
|
if (group->audio_sink && gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_SINK |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO)) {
|
|
continue;
|
|
}
|
|
if (group->video_sink && gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_SINK | GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO
|
|
| GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE)) {
|
|
continue;
|
|
}
|
|
|
|
g_value_init (&val, G_TYPE_OBJECT);
|
|
g_value_set_object (&val, factory);
|
|
g_value_array_append (result, &val);
|
|
g_value_unset (&val);
|
|
}
|
|
gst_plugin_feature_list_free (factory_list);
|
|
|
|
if (unref_caps)
|
|
gst_caps_unref (caps);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_set_context (GstElement * element, GstContext * context)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (element);
|
|
|
|
/* Proxy contexts to the sinks, they might not be in playsink yet */
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
if (playbin->audio_sink)
|
|
gst_element_set_context (playbin->audio_sink, context);
|
|
if (playbin->video_sink)
|
|
gst_element_set_context (playbin->video_sink, context);
|
|
if (playbin->text_sink)
|
|
gst_element_set_context (playbin->text_sink, context);
|
|
|
|
GST_SOURCE_GROUP_LOCK (playbin->curr_group);
|
|
|
|
if (playbin->curr_group->audio_sink)
|
|
gst_element_set_context (playbin->curr_group->audio_sink, context);
|
|
if (playbin->curr_group->video_sink)
|
|
gst_element_set_context (playbin->curr_group->video_sink, context);
|
|
if (playbin->curr_group->text_sink)
|
|
gst_element_set_context (playbin->curr_group->text_sink, context);
|
|
|
|
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
GST_ELEMENT_CLASS (parent_class)->set_context (element, context);
|
|
}
|
|
|
|
/* Pass sink messages to the application, e.g. NEED_CONTEXT messages */
|
|
static void
|
|
gst_play_bin3_update_context (GstPlayBin3 * playbin, GstContext * context)
|
|
{
|
|
GList *l;
|
|
const gchar *context_type;
|
|
|
|
GST_OBJECT_LOCK (playbin);
|
|
context_type = gst_context_get_context_type (context);
|
|
for (l = playbin->contexts; l; l = l->next) {
|
|
GstContext *tmp = l->data;
|
|
const gchar *tmp_type = gst_context_get_context_type (tmp);
|
|
|
|
/* Always store newest context but never replace
|
|
* a persistent one by a non-persistent one */
|
|
if (strcmp (context_type, tmp_type) == 0 &&
|
|
(gst_context_is_persistent (context) ||
|
|
!gst_context_is_persistent (tmp))) {
|
|
gst_context_replace ((GstContext **) & l->data, context);
|
|
break;
|
|
}
|
|
}
|
|
/* Not found? Add */
|
|
if (l == NULL)
|
|
playbin->contexts =
|
|
g_list_prepend (playbin->contexts, gst_context_ref (context));
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
}
|
|
|
|
static GstBusSyncReply
|
|
activate_sink_bus_handler (GstBus * bus, GstMessage * msg,
|
|
GstPlayBin3 * playbin)
|
|
{
|
|
if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR) {
|
|
/* Only proxy errors from a fixed sink. If that fails we can just error out
|
|
* early as stuff will fail later anyway */
|
|
if (playbin->audio_sink
|
|
&& gst_object_has_as_ancestor (GST_MESSAGE_SRC (msg),
|
|
GST_OBJECT_CAST (playbin->audio_sink)))
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
else if (playbin->video_sink
|
|
&& gst_object_has_as_ancestor (GST_MESSAGE_SRC (msg),
|
|
GST_OBJECT_CAST (playbin->video_sink)))
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
else if (playbin->text_sink
|
|
&& gst_object_has_as_ancestor (GST_MESSAGE_SRC (msg),
|
|
GST_OBJECT_CAST (playbin->text_sink)))
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
else
|
|
gst_message_unref (msg);
|
|
} else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_NEED_CONTEXT) {
|
|
const gchar *context_type;
|
|
GList *l;
|
|
|
|
gst_message_parse_context_type (msg, &context_type);
|
|
GST_OBJECT_LOCK (playbin);
|
|
for (l = playbin->contexts; l; l = l->next) {
|
|
GstContext *tmp = l->data;
|
|
const gchar *tmp_type = gst_context_get_context_type (tmp);
|
|
|
|
if (strcmp (context_type, tmp_type) == 0) {
|
|
gst_element_set_context (GST_ELEMENT (GST_MESSAGE_SRC (msg)), l->data);
|
|
break;
|
|
}
|
|
}
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
|
|
/* Forward if we couldn't answer the message */
|
|
if (l == NULL) {
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
} else {
|
|
gst_message_unref (msg);
|
|
}
|
|
} else if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_HAVE_CONTEXT) {
|
|
GstContext *context;
|
|
|
|
gst_message_parse_have_context (msg, &context);
|
|
gst_play_bin3_update_context (playbin, context);
|
|
gst_context_unref (context);
|
|
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
} else {
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
}
|
|
|
|
/* Doesn't really matter, nothing is using this bus */
|
|
return GST_BUS_DROP;
|
|
}
|
|
|
|
static gboolean
|
|
activate_sink (GstPlayBin3 * playbin, GstElement * sink, gboolean * activated)
|
|
{
|
|
GstState state;
|
|
GstBus *bus = NULL;
|
|
GstStateChangeReturn sret;
|
|
gboolean ret = FALSE;
|
|
|
|
if (activated)
|
|
*activated = FALSE;
|
|
|
|
GST_OBJECT_LOCK (sink);
|
|
state = GST_STATE (sink);
|
|
GST_OBJECT_UNLOCK (sink);
|
|
if (state >= GST_STATE_READY) {
|
|
ret = TRUE;
|
|
goto done;
|
|
}
|
|
|
|
if (!GST_OBJECT_PARENT (sink)) {
|
|
bus = gst_bus_new ();
|
|
gst_bus_set_sync_handler (bus,
|
|
(GstBusSyncHandler) activate_sink_bus_handler, playbin, NULL);
|
|
gst_element_set_bus (sink, bus);
|
|
}
|
|
|
|
sret = gst_element_set_state (sink, GST_STATE_READY);
|
|
if (sret == GST_STATE_CHANGE_FAILURE)
|
|
goto done;
|
|
|
|
if (activated)
|
|
*activated = TRUE;
|
|
ret = TRUE;
|
|
|
|
done:
|
|
if (bus) {
|
|
gst_element_set_bus (sink, NULL);
|
|
gst_object_unref (bus);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* autoplug-continue decides, if a pad has raw caps that can be exposed
|
|
* directly or if further decoding is necessary. We use this to expose
|
|
* supported subtitles directly */
|
|
|
|
/* FIXME 0.11: Remove the checks for ANY caps, a sink should specify
|
|
* explicitly the caps it supports and if it claims to support ANY
|
|
* caps it really should support everything */
|
|
static gboolean
|
|
autoplug_continue_cb (GstElement * element, GstPad * pad, GstCaps * caps,
|
|
GstSourceGroup * group)
|
|
{
|
|
gboolean ret = TRUE;
|
|
GstPad *sinkpad = NULL;
|
|
gboolean activated_sink;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
|
|
if (group->text_sink &&
|
|
activate_sink (group->playbin, group->text_sink, &activated_sink)) {
|
|
sinkpad = gst_element_get_static_pad (group->text_sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, NULL);
|
|
if (!gst_caps_is_any (sinkcaps))
|
|
ret = !gst_pad_query_accept_caps (sinkpad, caps);
|
|
gst_caps_unref (sinkcaps);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
if (activated_sink)
|
|
gst_element_set_state (group->text_sink, GST_STATE_NULL);
|
|
} else {
|
|
GstCaps *subcaps = gst_subtitle_overlay_create_factory_caps ();
|
|
ret = !gst_caps_is_subset (caps, subcaps);
|
|
gst_caps_unref (subcaps);
|
|
}
|
|
/* If autoplugging can stop don't do additional checks */
|
|
if (!ret)
|
|
goto done;
|
|
|
|
if (group->audio_sink &&
|
|
activate_sink (group->playbin, group->audio_sink, &activated_sink)) {
|
|
|
|
sinkpad = gst_element_get_static_pad (group->audio_sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, NULL);
|
|
if (!gst_caps_is_any (sinkcaps))
|
|
ret = !gst_pad_query_accept_caps (sinkpad, caps);
|
|
gst_caps_unref (sinkcaps);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
if (activated_sink)
|
|
gst_element_set_state (group->audio_sink, GST_STATE_NULL);
|
|
}
|
|
if (!ret)
|
|
goto done;
|
|
|
|
if (group->video_sink
|
|
&& activate_sink (group->playbin, group->video_sink, &activated_sink)) {
|
|
sinkpad = gst_element_get_static_pad (group->video_sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, NULL);
|
|
if (!gst_caps_is_any (sinkcaps))
|
|
ret = !gst_pad_query_accept_caps (sinkpad, caps);
|
|
gst_caps_unref (sinkcaps);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
if (activated_sink)
|
|
gst_element_set_state (group->video_sink, GST_STATE_NULL);
|
|
}
|
|
|
|
done:
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
GST_DEBUG_OBJECT (group->playbin,
|
|
"continue autoplugging group %p for %s:%s, %" GST_PTR_FORMAT ": %d",
|
|
group, GST_DEBUG_PAD_NAME (pad), caps, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
sink_accepts_caps (GstPlayBin3 * playbin, GstElement * sink, GstCaps * caps)
|
|
{
|
|
GstPad *sinkpad;
|
|
|
|
if ((sinkpad = gst_element_get_static_pad (sink, "sink"))) {
|
|
/* Got the sink pad, now let's see if the element actually does accept the
|
|
* caps that we have */
|
|
if (!gst_pad_query_accept_caps (sinkpad, caps)) {
|
|
gst_object_unref (sinkpad);
|
|
return FALSE;
|
|
}
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* We are asked to select an element. See if the next element to check
|
|
* is a sink. If this is the case, we see if the sink works by setting it to
|
|
* READY. If the sink works, we return SELECT_EXPOSE to make decodebin
|
|
* expose the raw pad so that we can setup the mixers. */
|
|
static GstAutoplugSelectResult
|
|
autoplug_select_cb (GstElement * decodebin, GstPad * pad,
|
|
GstCaps * caps, GstElementFactory * factory, GstSourceGroup * group)
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
GstElement *element;
|
|
const gchar *klass;
|
|
GstPlaySinkType type;
|
|
GstElement **sinkp;
|
|
GList *ave_list = NULL, *l;
|
|
GstAVElement *ave = NULL;
|
|
GSequence *ave_seq = NULL;
|
|
GSequenceIter *seq_iter;
|
|
|
|
playbin = group->playbin;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "select group %p for %s:%s, %" GST_PTR_FORMAT,
|
|
group, GST_DEBUG_PAD_NAME (pad), caps);
|
|
|
|
GST_DEBUG_OBJECT (playbin, "checking factory %s", GST_OBJECT_NAME (factory));
|
|
|
|
/* if it's not a sink, we make sure the element is compatible with
|
|
* the fixed sink */
|
|
if (!gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_SINK)) {
|
|
gboolean isvideodec = gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_DECODER |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE);
|
|
gboolean isaudiodec = gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_DECODER |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO);
|
|
|
|
if (!isvideodec && !isaudiodec)
|
|
return GST_AUTOPLUG_SELECT_TRY;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
g_mutex_lock (&playbin->elements_lock);
|
|
|
|
if (isaudiodec) {
|
|
ave_seq = playbin->aelements;
|
|
sinkp = &group->audio_sink;
|
|
} else {
|
|
ave_seq = playbin->velements;
|
|
sinkp = &group->video_sink;
|
|
}
|
|
|
|
seq_iter =
|
|
g_sequence_lookup (ave_seq, factory,
|
|
(GCompareDataFunc) avelement_lookup_decoder, NULL);
|
|
if (seq_iter) {
|
|
/* Go to first iter with that decoder */
|
|
do {
|
|
GSequenceIter *tmp_seq_iter;
|
|
|
|
tmp_seq_iter = g_sequence_iter_prev (seq_iter);
|
|
if (!avelement_iter_is_equal (tmp_seq_iter, factory))
|
|
break;
|
|
seq_iter = tmp_seq_iter;
|
|
} while (!g_sequence_iter_is_begin (seq_iter));
|
|
|
|
while (!g_sequence_iter_is_end (seq_iter)
|
|
&& avelement_iter_is_equal (seq_iter, factory)) {
|
|
ave = g_sequence_get (seq_iter);
|
|
ave_list = g_list_prepend (ave_list, ave);
|
|
seq_iter = g_sequence_iter_next (seq_iter);
|
|
}
|
|
|
|
/* Sort all GstAVElements by their relative ranks and insert
|
|
* into the decoders list */
|
|
ave_list = g_list_sort (ave_list, (GCompareFunc) avelement_compare);
|
|
} else {
|
|
ave_list = g_list_prepend (ave_list, NULL);
|
|
}
|
|
|
|
/* if it is a decoder and we don't have a fixed sink, then find out
|
|
* the matching audio/video sink from GstAVElements list */
|
|
for (l = ave_list; l; l = l->next) {
|
|
gboolean created_sink = FALSE;
|
|
|
|
ave = (GstAVElement *) l->data;
|
|
|
|
if (((isaudiodec && !group->audio_sink) ||
|
|
(isvideodec && !group->video_sink))) {
|
|
if (ave && ave->sink) {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"Trying to create sink '%s' for decoder '%s'",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (ave->sink)),
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
|
if ((*sinkp = gst_element_factory_create (ave->sink, NULL)) == NULL) {
|
|
GST_WARNING_OBJECT (playbin,
|
|
"Could not create an element from %s",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (ave->sink)));
|
|
continue;
|
|
} else {
|
|
if (!activate_sink (playbin, *sinkp, NULL)) {
|
|
gst_object_unref (*sinkp);
|
|
*sinkp = NULL;
|
|
GST_WARNING_OBJECT (playbin,
|
|
"Could not activate sink %s",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (ave->sink)));
|
|
continue;
|
|
}
|
|
gst_object_ref_sink (*sinkp);
|
|
created_sink = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If it is a decoder and we have a fixed sink for the media
|
|
* type it outputs, check that the decoder is compatible with this sink */
|
|
if ((isaudiodec && group->audio_sink) || (isvideodec
|
|
&& group->video_sink)) {
|
|
gboolean compatible = FALSE;
|
|
GstPad *sinkpad;
|
|
GstCaps *caps;
|
|
GstElement *sink;
|
|
|
|
sink = *sinkp;
|
|
|
|
if ((sinkpad = gst_element_get_static_pad (sink, "sink"))) {
|
|
GstPlayFlags flags = gst_play_bin3_get_flags (playbin);
|
|
GstCaps *raw_caps =
|
|
(isaudiodec) ? gst_static_caps_get (&raw_audio_caps) :
|
|
gst_static_caps_get (&raw_video_caps);
|
|
|
|
caps = gst_pad_query_caps (sinkpad, NULL);
|
|
|
|
/* If the sink supports raw audio/video, we first check
|
|
* if the decoder could output any raw audio/video format
|
|
* and assume it is compatible with the sink then. We don't
|
|
* do a complete compatibility check here if converters
|
|
* are plugged between the decoder and the sink because
|
|
* the converters will convert between raw formats and
|
|
* even if the decoder format is not supported by the decoder
|
|
* a converter will convert it.
|
|
*
|
|
* We assume here that the converters can convert between
|
|
* any raw format.
|
|
*/
|
|
if ((isaudiodec && !(flags & GST_PLAY_FLAG_NATIVE_AUDIO)
|
|
&& gst_caps_can_intersect (caps, raw_caps)) || (!isaudiodec
|
|
&& !(flags & GST_PLAY_FLAG_NATIVE_VIDEO)
|
|
&& gst_caps_can_intersect (caps, raw_caps))) {
|
|
compatible =
|
|
gst_element_factory_can_src_any_caps (factory, raw_caps)
|
|
|| gst_element_factory_can_src_any_caps (factory, caps);
|
|
} else {
|
|
compatible = gst_element_factory_can_src_any_caps (factory, caps);
|
|
}
|
|
|
|
gst_object_unref (sinkpad);
|
|
gst_caps_unref (caps);
|
|
}
|
|
|
|
if (compatible)
|
|
break;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "%s not compatible with the fixed sink",
|
|
GST_OBJECT_NAME (factory));
|
|
|
|
/* If it is not compatible, either continue with the next possible
|
|
* sink or if we have a fixed sink, skip the decoder */
|
|
if (created_sink) {
|
|
gst_element_set_state (*sinkp, GST_STATE_NULL);
|
|
gst_object_unref (*sinkp);
|
|
*sinkp = NULL;
|
|
} else {
|
|
g_mutex_unlock (&playbin->elements_lock);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
}
|
|
}
|
|
g_list_free (ave_list);
|
|
g_mutex_unlock (&playbin->elements_lock);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_TRY;
|
|
}
|
|
|
|
/* it's a sink, see if an instance of it actually works */
|
|
GST_DEBUG_OBJECT (playbin, "we found a sink '%s'", GST_OBJECT_NAME (factory));
|
|
|
|
klass =
|
|
gst_element_factory_get_metadata (factory, GST_ELEMENT_METADATA_KLASS);
|
|
|
|
/* figure out the klass */
|
|
if (strstr (klass, "Audio")) {
|
|
GST_DEBUG_OBJECT (playbin, "we found an audio sink");
|
|
type = GST_PLAY_SINK_TYPE_AUDIO;
|
|
sinkp = &group->audio_sink;
|
|
} else if (strstr (klass, "Video")) {
|
|
GST_DEBUG_OBJECT (playbin, "we found a video sink");
|
|
type = GST_PLAY_SINK_TYPE_VIDEO;
|
|
sinkp = &group->video_sink;
|
|
} else {
|
|
/* unknown klass, skip this element */
|
|
GST_WARNING_OBJECT (playbin, "unknown sink klass %s found", klass);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
|
|
/* if we are asked to do visualisations and it's an audio sink, skip the
|
|
* element. We can only do visualisations with raw sinks */
|
|
if (gst_play_sink_get_flags (playbin->playsink) & GST_PLAY_FLAG_VIS) {
|
|
if (type == GST_PLAY_SINK_TYPE_AUDIO) {
|
|
GST_DEBUG_OBJECT (playbin, "skip audio sink because of vis");
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
}
|
|
|
|
/* now see if we already have a sink element */
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
if (*sinkp && GST_STATE (*sinkp) >= GST_STATE_READY) {
|
|
GstElement *sink = gst_object_ref (*sinkp);
|
|
|
|
if (sink_accepts_caps (playbin, sink, caps)) {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"Existing sink '%s' accepts caps: %" GST_PTR_FORMAT,
|
|
GST_ELEMENT_NAME (sink), caps);
|
|
gst_object_unref (sink);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_EXPOSE;
|
|
} else {
|
|
GST_DEBUG_OBJECT (playbin,
|
|
"Existing sink '%s' does not accept caps: %" GST_PTR_FORMAT,
|
|
GST_ELEMENT_NAME (sink), caps);
|
|
gst_object_unref (sink);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
}
|
|
GST_DEBUG_OBJECT (playbin, "we have no pending sink, try to create '%s'",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
|
|
|
if ((*sinkp = gst_element_factory_create (factory, NULL)) == NULL) {
|
|
GST_WARNING_OBJECT (playbin, "Could not create an element from %s",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
|
|
element = *sinkp;
|
|
|
|
if (!activate_sink (playbin, element, NULL)) {
|
|
GST_WARNING_OBJECT (playbin, "Could not activate sink %s",
|
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
|
*sinkp = NULL;
|
|
gst_object_unref (element);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
|
|
/* Check if the selected sink actually supports the
|
|
* caps and can be set to READY*/
|
|
if (!sink_accepts_caps (playbin, element, caps)) {
|
|
*sinkp = NULL;
|
|
gst_element_set_state (element, GST_STATE_NULL);
|
|
gst_object_unref (element);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
return GST_AUTOPLUG_SELECT_SKIP;
|
|
}
|
|
|
|
/* remember the sink in the group now, the element is floating, we take
|
|
* ownership now
|
|
*
|
|
* store the sink in the group, we will configure it later when we
|
|
* reconfigure the sink */
|
|
GST_DEBUG_OBJECT (playbin, "remember sink");
|
|
gst_object_ref_sink (element);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
/* tell decodebin to expose the pad because we are going to use this
|
|
* sink */
|
|
GST_DEBUG_OBJECT (playbin, "we found a working sink, expose pad");
|
|
|
|
return GST_AUTOPLUG_SELECT_EXPOSE;
|
|
}
|
|
|
|
#define GST_PLAY_BIN3_FILTER_CAPS(filter,caps) G_STMT_START { \
|
|
if ((filter)) { \
|
|
GstCaps *intersection = \
|
|
gst_caps_intersect_full ((filter), (caps), GST_CAPS_INTERSECT_FIRST); \
|
|
gst_caps_unref ((caps)); \
|
|
(caps) = intersection; \
|
|
} \
|
|
} G_STMT_END
|
|
|
|
static gboolean
|
|
autoplug_query_caps (GstElement * uridecodebin, GstPad * pad,
|
|
GstElement * element, GstQuery * query, GstSourceGroup * group)
|
|
{
|
|
GstCaps *filter, *result = NULL;
|
|
GstElement *sink;
|
|
GstPad *sinkpad = NULL;
|
|
GstElementFactory *factory;
|
|
GstElementFactoryListType factory_type;
|
|
gboolean have_sink = FALSE;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
gst_query_parse_caps (query, &filter);
|
|
|
|
factory = gst_element_get_factory (element);
|
|
if (!factory)
|
|
goto done;
|
|
|
|
if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE)) {
|
|
factory_type =
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE;
|
|
|
|
if ((sink = group->video_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, filter);
|
|
if (!gst_caps_is_any (sinkcaps)) {
|
|
if (!result)
|
|
result = sinkcaps;
|
|
else
|
|
result = gst_caps_merge (result, sinkcaps);
|
|
} else {
|
|
gst_caps_unref (sinkcaps);
|
|
}
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
have_sink = TRUE;
|
|
}
|
|
} else if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO)) {
|
|
factory_type = GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO;
|
|
|
|
if ((sink = group->audio_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, filter);
|
|
if (!gst_caps_is_any (sinkcaps)) {
|
|
if (!result)
|
|
result = sinkcaps;
|
|
else
|
|
result = gst_caps_merge (result, sinkcaps);
|
|
} else {
|
|
gst_caps_unref (sinkcaps);
|
|
}
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
have_sink = TRUE;
|
|
}
|
|
} else if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_SUBTITLE)) {
|
|
factory_type = GST_ELEMENT_FACTORY_TYPE_MEDIA_SUBTITLE;
|
|
|
|
if ((sink = group->playbin->text_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
GstCaps *sinkcaps;
|
|
|
|
sinkcaps = gst_pad_query_caps (sinkpad, filter);
|
|
if (!gst_caps_is_any (sinkcaps)) {
|
|
if (!result)
|
|
result = sinkcaps;
|
|
else
|
|
result = gst_caps_merge (result, sinkcaps);
|
|
} else {
|
|
gst_caps_unref (sinkcaps);
|
|
}
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
have_sink = TRUE;
|
|
} else {
|
|
GstCaps *subcaps = gst_subtitle_overlay_create_factory_caps ();
|
|
GST_PLAY_BIN3_FILTER_CAPS (filter, subcaps);
|
|
if (!result)
|
|
result = subcaps;
|
|
else
|
|
result = gst_caps_merge (result, subcaps);
|
|
}
|
|
} else {
|
|
goto done;
|
|
}
|
|
|
|
if (!have_sink) {
|
|
GValueArray *factories;
|
|
gint i, n;
|
|
|
|
factories = autoplug_factories_cb (uridecodebin, pad, NULL, group);
|
|
n = factories->n_values;
|
|
for (i = 0; i < n; i++) {
|
|
GValue *v = g_value_array_get_nth (factories, i);
|
|
GstElementFactory *f = g_value_get_object (v);
|
|
const GList *templates;
|
|
const GList *l;
|
|
GstCaps *templ_caps;
|
|
|
|
if (!gst_element_factory_list_is_type (f, factory_type))
|
|
continue;
|
|
|
|
templates = gst_element_factory_get_static_pad_templates (f);
|
|
|
|
for (l = templates; l; l = l->next) {
|
|
templ_caps = gst_static_pad_template_get_caps (l->data);
|
|
|
|
if (!gst_caps_is_any (templ_caps)) {
|
|
GST_PLAY_BIN3_FILTER_CAPS (filter, templ_caps);
|
|
if (!result)
|
|
result = templ_caps;
|
|
else
|
|
result = gst_caps_merge (result, templ_caps);
|
|
} else {
|
|
gst_caps_unref (templ_caps);
|
|
}
|
|
}
|
|
}
|
|
g_value_array_free (factories);
|
|
}
|
|
|
|
done:
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
if (!result)
|
|
return FALSE;
|
|
|
|
/* Add the actual decoder/parser/etc caps at the very end to
|
|
* make sure we don't cause empty caps to be returned, e.g.
|
|
* if a parser asks us but a decoder is required after it
|
|
* because no sink can handle the format directly.
|
|
*/
|
|
{
|
|
GstPad *target = gst_ghost_pad_get_target (GST_GHOST_PAD (pad));
|
|
|
|
if (target) {
|
|
GstCaps *target_caps = gst_pad_get_pad_template_caps (target);
|
|
GST_PLAY_BIN3_FILTER_CAPS (filter, target_caps);
|
|
result = gst_caps_merge (result, target_caps);
|
|
gst_object_unref (target);
|
|
}
|
|
}
|
|
|
|
|
|
gst_query_set_caps_result (query, result);
|
|
gst_caps_unref (result);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
autoplug_query_context (GstElement * uridecodebin, GstPad * pad,
|
|
GstElement * element, GstQuery * query, GstSourceGroup * group)
|
|
{
|
|
GstElement *sink;
|
|
GstPad *sinkpad = NULL;
|
|
GstElementFactory *factory;
|
|
gboolean res = FALSE;
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
|
|
factory = gst_element_get_factory (element);
|
|
if (!factory)
|
|
goto done;
|
|
|
|
if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_VIDEO |
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_IMAGE)) {
|
|
if ((sink = group->video_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
res = gst_pad_query (sinkpad, query);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
}
|
|
} else if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_AUDIO)) {
|
|
if ((sink = group->audio_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
res = gst_pad_query (sinkpad, query);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
}
|
|
} else if (gst_element_factory_list_is_type (factory,
|
|
GST_ELEMENT_FACTORY_TYPE_MEDIA_SUBTITLE)) {
|
|
if ((sink = group->playbin->text_sink)) {
|
|
sinkpad = gst_element_get_static_pad (sink, "sink");
|
|
if (sinkpad) {
|
|
res = gst_pad_query (sinkpad, query);
|
|
gst_object_unref (sinkpad);
|
|
}
|
|
}
|
|
} else {
|
|
goto done;
|
|
}
|
|
|
|
done:
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
return res;
|
|
}
|
|
|
|
static gboolean
|
|
autoplug_query_cb (GstElement * uridecodebin, GstPad * pad,
|
|
GstElement * element, GstQuery * query, GstSourceGroup * group)
|
|
{
|
|
|
|
switch (GST_QUERY_TYPE (query)) {
|
|
case GST_QUERY_CAPS:
|
|
return autoplug_query_caps (uridecodebin, pad, element, query, group);
|
|
case GST_QUERY_CONTEXT:
|
|
return autoplug_query_context (uridecodebin, pad, element, query, group);
|
|
default:
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
notify_source_cb (GstElement * urisourcebin, GParamSpec * pspec,
|
|
GstSourceGroup * group)
|
|
{
|
|
GstPlayBin3 *playbin;
|
|
GstElement *source;
|
|
|
|
playbin = group->playbin;
|
|
|
|
g_object_get (urisourcebin, "source", &source, NULL);
|
|
|
|
GST_OBJECT_LOCK (playbin);
|
|
if (playbin->source)
|
|
gst_object_unref (playbin->source);
|
|
playbin->source = source;
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
|
|
g_object_notify (G_OBJECT (playbin), "source");
|
|
|
|
g_signal_emit (playbin, gst_play_bin3_signals[SIGNAL_SOURCE_SETUP],
|
|
0, playbin->source);
|
|
}
|
|
|
|
/* must be called with the group lock */
|
|
static gboolean
|
|
group_set_locked_state_unlocked (GstPlayBin3 * playbin, GstSourceGroup * group,
|
|
gboolean locked)
|
|
{
|
|
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);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
make_or_reuse_element (GstPlayBin3 * playbin, const gchar * name,
|
|
GstElement ** elem)
|
|
{
|
|
if (*elem) {
|
|
GST_DEBUG_OBJECT (playbin, "reusing existing %s", name);
|
|
gst_element_set_state (*elem, GST_STATE_READY);
|
|
/* no need to take extra ref, we already have one
|
|
* and the bin will add one since it is no longer floating,
|
|
* as we added a non-floating ref when removing it from the
|
|
* bin earlier */
|
|
} else {
|
|
GstElement *new_elem;
|
|
GST_DEBUG_OBJECT (playbin, "making new %s", name);
|
|
new_elem = gst_element_factory_make (name, NULL);
|
|
if (!new_elem)
|
|
return FALSE;
|
|
*elem = gst_object_ref (new_elem);
|
|
}
|
|
|
|
if (GST_OBJECT_PARENT (*elem) != GST_OBJECT_CAST (playbin))
|
|
gst_bin_add (GST_BIN_CAST (playbin), *elem);
|
|
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;
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
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);
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
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_no_more_pads_id = g_signal_connect (decodebin, "no-more-pads",
|
|
G_CALLBACK (no_more_pads_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_no_more_pads_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_no_more_pads_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;
|
|
}
|
|
}
|
|
|
|
/* must be called with PLAY_BIN_LOCK */
|
|
static GstStateChangeReturn
|
|
activate_group (GstPlayBin3 * playbin, GstSourceGroup * group, GstState target)
|
|
{
|
|
GstElement *urisrcbin = NULL;
|
|
GstElement *suburisrcbin = NULL;
|
|
GstPlayFlags flags;
|
|
gboolean audio_sink_activated = FALSE;
|
|
gboolean video_sink_activated = FALSE;
|
|
gboolean text_sink_activated = FALSE;
|
|
GstStateChangeReturn state_ret;
|
|
|
|
g_return_val_if_fail (group->valid, GST_STATE_CHANGE_FAILURE);
|
|
g_return_val_if_fail (!group->active, GST_STATE_CHANGE_FAILURE);
|
|
|
|
GST_DEBUG_OBJECT (playbin, "activating group %p", group);
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
|
|
/* First set up the custom sinks */
|
|
if (playbin->audio_sink)
|
|
group->audio_sink = gst_object_ref (playbin->audio_sink);
|
|
else
|
|
group->audio_sink =
|
|
gst_play_sink_get_sink (playbin->playsink, GST_PLAY_SINK_TYPE_AUDIO);
|
|
|
|
if (group->audio_sink) {
|
|
if (!activate_sink (playbin, group->audio_sink, &audio_sink_activated)) {
|
|
if (group->audio_sink == playbin->audio_sink) {
|
|
goto sink_failure;
|
|
} else {
|
|
gst_object_unref (group->audio_sink);
|
|
group->audio_sink = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (playbin->video_sink)
|
|
group->video_sink = gst_object_ref (playbin->video_sink);
|
|
else
|
|
group->video_sink =
|
|
gst_play_sink_get_sink (playbin->playsink, GST_PLAY_SINK_TYPE_VIDEO);
|
|
|
|
if (group->video_sink) {
|
|
if (!activate_sink (playbin, group->video_sink, &video_sink_activated)) {
|
|
if (group->video_sink == playbin->video_sink) {
|
|
goto sink_failure;
|
|
} else {
|
|
gst_object_unref (group->video_sink);
|
|
group->video_sink = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (playbin->text_sink)
|
|
group->text_sink = gst_object_ref (playbin->text_sink);
|
|
else
|
|
group->text_sink =
|
|
gst_play_sink_get_sink (playbin->playsink, GST_PLAY_SINK_TYPE_TEXT);
|
|
|
|
if (group->text_sink) {
|
|
if (!activate_sink (playbin, group->text_sink, &text_sink_activated)) {
|
|
if (group->text_sink == playbin->text_sink) {
|
|
goto sink_failure;
|
|
} else {
|
|
gst_object_unref (group->text_sink);
|
|
group->text_sink = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (!make_or_reuse_element (playbin, "urisourcebin", &group->urisourcebin))
|
|
goto no_urisrcbin;
|
|
urisrcbin = group->urisourcebin;
|
|
|
|
flags = gst_play_sink_get_flags (playbin->playsink);
|
|
|
|
g_object_set (urisrcbin,
|
|
/* configure connection speed */
|
|
"connection-speed", playbin->connection_speed / 1000,
|
|
/* configure uri */
|
|
"uri", group->uri,
|
|
/* configure download buffering */
|
|
"download", ((flags & GST_PLAY_FLAG_DOWNLOAD) != 0),
|
|
/* configure buffering of demuxed/parsed data */
|
|
"use-buffering", ((flags & GST_PLAY_FLAG_BUFFERING) != 0),
|
|
/* configure buffering parameters */
|
|
"buffer-duration", playbin->buffer_duration,
|
|
"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->notify_source_id = g_signal_connect (urisrcbin, "notify::source",
|
|
G_CALLBACK (notify_source_cb), group);
|
|
|
|
/* 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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
/* we have 2 pending no-more-pads */
|
|
group->pending = 2;
|
|
group->sub_pending = TRUE;
|
|
} else {
|
|
group->sub_pending = FALSE;
|
|
}
|
|
|
|
/* 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);
|
|
REMOVE_SIGNAL (suburisrcbin, group->sub_autoplug_continue_id);
|
|
REMOVE_SIGNAL (suburisrcbin, group->sub_autoplug_query_id);
|
|
/* 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_SOURCE_GROUP_LOCK (group);
|
|
/* allow state changes of the playbin affect the group elements now */
|
|
group_set_locked_state_unlocked (playbin, group, FALSE);
|
|
group->active = TRUE;
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
return state_ret;
|
|
|
|
/* ERRORS */
|
|
no_urisrcbin:
|
|
{
|
|
GstMessage *msg;
|
|
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
msg =
|
|
gst_missing_element_message_new (GST_ELEMENT_CAST (playbin),
|
|
"urisourcebin");
|
|
gst_element_post_message (GST_ELEMENT_CAST (playbin), msg);
|
|
|
|
GST_ELEMENT_ERROR (playbin, CORE, MISSING_PLUGIN,
|
|
(_("Could not create \"urisourcebin\" element.")), (NULL));
|
|
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
|
|
goto error_cleanup;
|
|
}
|
|
urisrcbin_failure:
|
|
{
|
|
GST_DEBUG_OBJECT (playbin, "failed state change of urisrcbin");
|
|
GST_SOURCE_GROUP_LOCK (group);
|
|
goto error_cleanup;
|
|
}
|
|
sink_failure:
|
|
{
|
|
GST_ERROR_OBJECT (playbin, "failed to activate sinks");
|
|
goto error_cleanup;
|
|
}
|
|
|
|
error_cleanup:
|
|
{
|
|
/* delete any custom sinks we might have */
|
|
if (group->audio_sink) {
|
|
/* If this is a automatically created sink set it to NULL */
|
|
if (audio_sink_activated)
|
|
gst_element_set_state (group->audio_sink, GST_STATE_NULL);
|
|
gst_object_unref (group->audio_sink);
|
|
}
|
|
group->audio_sink = NULL;
|
|
|
|
if (group->video_sink) {
|
|
/* If this is a automatically created sink set it to NULL */
|
|
if (video_sink_activated)
|
|
gst_element_set_state (group->video_sink, GST_STATE_NULL);
|
|
gst_object_unref (group->video_sink);
|
|
}
|
|
group->video_sink = NULL;
|
|
|
|
if (group->text_sink) {
|
|
/* If this is a automatically created sink set it to NULL */
|
|
if (text_sink_activated)
|
|
gst_element_set_state (group->text_sink, GST_STATE_NULL);
|
|
gst_object_unref (group->text_sink);
|
|
}
|
|
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->notify_source_id);
|
|
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);
|
|
|
|
gst_element_set_state (urisrcbin, GST_STATE_NULL);
|
|
gst_bin_remove (GST_BIN_CAST (playbin), urisrcbin);
|
|
}
|
|
|
|
GST_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
return GST_STATE_CHANGE_FAILURE;
|
|
}
|
|
}
|
|
|
|
/* unlink a group of urisrcbin 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_DEBUG_OBJECT (playbin, "unlinking group %p", group);
|
|
|
|
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);
|
|
|
|
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 */
|
|
if (group->audio_sink) {
|
|
if (!gst_object_has_as_ancestor (GST_OBJECT_CAST (group->audio_sink),
|
|
GST_OBJECT_CAST (playbin->playsink))) {
|
|
gst_element_set_state (group->audio_sink, GST_STATE_NULL);
|
|
}
|
|
gst_object_unref (group->audio_sink);
|
|
}
|
|
group->audio_sink = NULL;
|
|
if (group->video_sink) {
|
|
if (!gst_object_has_as_ancestor (GST_OBJECT_CAST (group->video_sink),
|
|
GST_OBJECT_CAST (playbin->playsink))) {
|
|
gst_element_set_state (group->video_sink, GST_STATE_NULL);
|
|
}
|
|
gst_object_unref (group->video_sink);
|
|
}
|
|
group->video_sink = NULL;
|
|
if (group->text_sink) {
|
|
if (!gst_object_has_as_ancestor (GST_OBJECT_CAST (group->text_sink),
|
|
GST_OBJECT_CAST (playbin->playsink))) {
|
|
gst_element_set_state (group->text_sink, GST_STATE_NULL);
|
|
}
|
|
gst_object_unref (group->text_sink);
|
|
}
|
|
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->notify_source_id);
|
|
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);
|
|
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);
|
|
REMOVE_SIGNAL (group->suburisourcebin, group->sub_autoplug_continue_id);
|
|
REMOVE_SIGNAL (group->suburisourcebin, group->sub_autoplug_query_id);
|
|
|
|
/* 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_SOURCE_GROUP_UNLOCK (group);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* setup the next group to play, this assumes the next_group is valid and
|
|
* configured. It swaps out the current_group and activates the valid
|
|
* next_group. */
|
|
static GstStateChangeReturn
|
|
setup_next_source (GstPlayBin3 * playbin, GstState target)
|
|
{
|
|
GstSourceGroup *new_group, *old_group;
|
|
GstStateChangeReturn state_ret;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "setup sources");
|
|
|
|
/* see if there is a next group */
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
new_group = playbin->next_group;
|
|
if (!new_group || !new_group->valid)
|
|
goto no_next_group;
|
|
|
|
/* first unlink the current source, if any */
|
|
old_group = playbin->curr_group;
|
|
if (old_group && old_group->valid && old_group->active) {
|
|
new_group->stream_changed_pending = TRUE;
|
|
|
|
gst_play_bin3_update_cached_duration (playbin);
|
|
/* unlink our pads with the sink */
|
|
deactivate_group (playbin, old_group);
|
|
old_group->valid = FALSE;
|
|
}
|
|
|
|
/* swap old and new */
|
|
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)
|
|
goto activate_failed;
|
|
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return state_ret;
|
|
|
|
/* ERRORS */
|
|
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;
|
|
}
|
|
activate_failed:
|
|
{
|
|
new_group->stream_changed_pending = FALSE;
|
|
GST_DEBUG_OBJECT (playbin, "activate failed");
|
|
new_group->valid = FALSE;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
return GST_STATE_CHANGE_FAILURE;
|
|
}
|
|
}
|
|
|
|
/* The group that is currently playing is copied again to the
|
|
* next_group so that it will start playing the next time.
|
|
*/
|
|
static gboolean
|
|
save_current_group (GstPlayBin3 * playbin)
|
|
{
|
|
GstSourceGroup *curr_group;
|
|
|
|
GST_DEBUG_OBJECT (playbin, "save current group");
|
|
|
|
/* see if there is a current group */
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
curr_group = playbin->curr_group;
|
|
if (curr_group && curr_group->valid && curr_group->active) {
|
|
/* unlink our pads with the sink */
|
|
deactivate_group (playbin, curr_group);
|
|
}
|
|
/* swap old and new */
|
|
playbin->curr_group = playbin->next_group;
|
|
playbin->next_group = curr_group;
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* clear the locked state from all groups. This function is called before a
|
|
* state change to NULL is performed on them. */
|
|
static gboolean
|
|
groups_set_locked_state (GstPlayBin3 * playbin, gboolean locked)
|
|
{
|
|
GST_DEBUG_OBJECT (playbin, "setting locked state to %d on all groups",
|
|
locked);
|
|
|
|
GST_PLAY_BIN3_LOCK (playbin);
|
|
GST_SOURCE_GROUP_LOCK (playbin->curr_group);
|
|
group_set_locked_state_unlocked (playbin, playbin->curr_group, locked);
|
|
GST_SOURCE_GROUP_UNLOCK (playbin->curr_group);
|
|
GST_SOURCE_GROUP_LOCK (playbin->next_group);
|
|
group_set_locked_state_unlocked (playbin, playbin->next_group, locked);
|
|
GST_SOURCE_GROUP_UNLOCK (playbin->next_group);
|
|
GST_PLAY_BIN3_UNLOCK (playbin);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_play_bin3_change_state (GstElement * element, GstStateChange transition)
|
|
{
|
|
GstStateChangeReturn ret;
|
|
GstPlayBin3 *playbin;
|
|
gboolean do_save = FALSE;
|
|
|
|
playbin = GST_PLAY_BIN3 (element);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
memset (&playbin->duration, 0, sizeof (playbin->duration));
|
|
break;
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
|
GST_LOG_OBJECT (playbin, "clearing shutdown flag");
|
|
memset (&playbin->duration, 0, sizeof (playbin->duration));
|
|
g_atomic_int_set (&playbin->shutdown, 0);
|
|
do_async_start (playbin);
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
async_down:
|
|
/* FIXME unlock our waiting groups */
|
|
GST_LOG_OBJECT (playbin, "setting shutdown flag");
|
|
g_atomic_int_set (&playbin->shutdown, 1);
|
|
memset (&playbin->duration, 0, sizeof (playbin->duration));
|
|
|
|
/* wait for all callbacks to end by taking the lock.
|
|
* No dynamic (critical) new callbacks will
|
|
* be able to happen as we set the shutdown flag. */
|
|
GST_PLAY_BIN3_DYN_LOCK (playbin);
|
|
GST_LOG_OBJECT (playbin, "dynamic lock taken, we can continue shutdown");
|
|
GST_PLAY_BIN3_DYN_UNLOCK (playbin);
|
|
if (!do_save)
|
|
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 ... */
|
|
if (!g_atomic_int_get (&playbin->shutdown)) {
|
|
do_save = TRUE;
|
|
goto async_down;
|
|
}
|
|
memset (&playbin->duration, 0, sizeof (playbin->duration));
|
|
|
|
/* unlock so that all groups go to NULL */
|
|
groups_set_locked_state (playbin, FALSE);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
if (ret == GST_STATE_CHANGE_FAILURE)
|
|
goto failure;
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
|
if ((ret =
|
|
setup_next_source (playbin,
|
|
GST_STATE_PAUSED)) == GST_STATE_CHANGE_FAILURE)
|
|
goto failure;
|
|
if (ret == GST_STATE_CHANGE_SUCCESS)
|
|
ret = GST_STATE_CHANGE_ASYNC;
|
|
|
|
break;
|
|
case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
|
|
do_async_done (playbin);
|
|
/* FIXME Release audio device when we implement that */
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
save_current_group (playbin);
|
|
break;
|
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
|
{
|
|
guint i;
|
|
GList *l;
|
|
|
|
/* 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.
|
|
*/
|
|
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,
|
|
GST_STATE_NULL);
|
|
gst_object_unref (playbin->groups[i].urisourcebin);
|
|
playbin->groups[i].urisourcebin = 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 */
|
|
if (playbin->audio_sink)
|
|
gst_element_set_state (playbin->audio_sink, GST_STATE_NULL);
|
|
if (playbin->video_sink)
|
|
gst_element_set_state (playbin->video_sink, GST_STATE_NULL);
|
|
if (playbin->text_sink)
|
|
gst_element_set_state (playbin->text_sink, GST_STATE_NULL);
|
|
|
|
if (playbin->video_stream_combiner)
|
|
gst_element_set_state (playbin->video_stream_combiner, GST_STATE_NULL);
|
|
if (playbin->audio_stream_combiner)
|
|
gst_element_set_state (playbin->audio_stream_combiner, GST_STATE_NULL);
|
|
if (playbin->text_stream_combiner)
|
|
gst_element_set_state (playbin->text_stream_combiner, GST_STATE_NULL);
|
|
|
|
/* make sure the groups don't perform a state change anymore until we
|
|
* enable them again */
|
|
groups_set_locked_state (playbin, TRUE);
|
|
|
|
/* Remove all non-persistent contexts */
|
|
GST_OBJECT_LOCK (playbin);
|
|
for (l = playbin->contexts; l;) {
|
|
GstContext *context = l->data;
|
|
|
|
if (!gst_context_is_persistent (context)) {
|
|
GList *next;
|
|
|
|
gst_context_unref (context);
|
|
|
|
next = l->next;
|
|
playbin->contexts = g_list_delete_link (playbin->contexts, l);
|
|
l = next;
|
|
} else {
|
|
l = l->next;
|
|
}
|
|
}
|
|
|
|
if (playbin->source) {
|
|
gst_object_unref (playbin->source);
|
|
playbin->source = NULL;
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (playbin);
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (ret == GST_STATE_CHANGE_NO_PREROLL)
|
|
do_async_done (playbin);
|
|
|
|
return ret;
|
|
|
|
/* ERRORS */
|
|
failure:
|
|
{
|
|
do_async_done (playbin);
|
|
|
|
if (transition == GST_STATE_CHANGE_READY_TO_PAUSED) {
|
|
GstSourceGroup *curr_group;
|
|
|
|
curr_group = playbin->curr_group;
|
|
if (curr_group) {
|
|
if (curr_group->active && curr_group->valid) {
|
|
/* unlink our pads with the sink */
|
|
deactivate_group (playbin, curr_group);
|
|
}
|
|
curr_group->valid = FALSE;
|
|
}
|
|
|
|
/* Swap current and next group back */
|
|
playbin->curr_group = playbin->next_group;
|
|
playbin->next_group = curr_group;
|
|
}
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_overlay_expose (GstVideoOverlay * overlay)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (overlay);
|
|
|
|
gst_video_overlay_expose (GST_VIDEO_OVERLAY (playbin->playsink));
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_overlay_handle_events (GstVideoOverlay * overlay,
|
|
gboolean handle_events)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (overlay);
|
|
|
|
gst_video_overlay_handle_events (GST_VIDEO_OVERLAY (playbin->playsink),
|
|
handle_events);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_overlay_set_render_rectangle (GstVideoOverlay * overlay, gint x,
|
|
gint y, gint width, gint height)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (overlay);
|
|
|
|
gst_video_overlay_set_render_rectangle (GST_VIDEO_OVERLAY (playbin->playsink),
|
|
x, y, width, height);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_overlay_set_window_handle (GstVideoOverlay * overlay,
|
|
guintptr handle)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (overlay);
|
|
|
|
gst_video_overlay_set_window_handle (GST_VIDEO_OVERLAY (playbin->playsink),
|
|
handle);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_overlay_init (gpointer g_iface, gpointer g_iface_data)
|
|
{
|
|
GstVideoOverlayInterface *iface = (GstVideoOverlayInterface *) g_iface;
|
|
iface->expose = gst_play_bin3_overlay_expose;
|
|
iface->handle_events = gst_play_bin3_overlay_handle_events;
|
|
iface->set_render_rectangle = gst_play_bin3_overlay_set_render_rectangle;
|
|
iface->set_window_handle = gst_play_bin3_overlay_set_window_handle;
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_navigation_send_event (GstNavigation * navigation,
|
|
GstStructure * structure)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (navigation);
|
|
|
|
gst_navigation_send_event (GST_NAVIGATION (playbin->playsink), structure);
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_navigation_init (gpointer g_iface, gpointer g_iface_data)
|
|
{
|
|
GstNavigationInterface *iface = (GstNavigationInterface *) g_iface;
|
|
|
|
iface->send_event = gst_play_bin3_navigation_send_event;
|
|
}
|
|
|
|
static const GList *
|
|
gst_play_bin3_colorbalance_list_channels (GstColorBalance * balance)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (balance);
|
|
|
|
return
|
|
gst_color_balance_list_channels (GST_COLOR_BALANCE (playbin->playsink));
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_colorbalance_set_value (GstColorBalance * balance,
|
|
GstColorBalanceChannel * channel, gint value)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (balance);
|
|
|
|
gst_color_balance_set_value (GST_COLOR_BALANCE (playbin->playsink), channel,
|
|
value);
|
|
}
|
|
|
|
static gint
|
|
gst_play_bin3_colorbalance_get_value (GstColorBalance * balance,
|
|
GstColorBalanceChannel * channel)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (balance);
|
|
|
|
return gst_color_balance_get_value (GST_COLOR_BALANCE (playbin->playsink),
|
|
channel);
|
|
}
|
|
|
|
static GstColorBalanceType
|
|
gst_play_bin3_colorbalance_get_balance_type (GstColorBalance * balance)
|
|
{
|
|
GstPlayBin3 *playbin = GST_PLAY_BIN3 (balance);
|
|
|
|
return
|
|
gst_color_balance_get_balance_type (GST_COLOR_BALANCE
|
|
(playbin->playsink));
|
|
}
|
|
|
|
static void
|
|
gst_play_bin3_colorbalance_init (gpointer g_iface, gpointer g_iface_data)
|
|
{
|
|
GstColorBalanceInterface *iface = (GstColorBalanceInterface *) g_iface;
|
|
|
|
iface->list_channels = gst_play_bin3_colorbalance_list_channels;
|
|
iface->set_value = gst_play_bin3_colorbalance_set_value;
|
|
iface->get_value = gst_play_bin3_colorbalance_get_value;
|
|
iface->get_balance_type = gst_play_bin3_colorbalance_get_balance_type;
|
|
}
|
|
|
|
gboolean
|
|
gst_play_bin3_plugin_init (GstPlugin * plugin, gboolean as_playbin)
|
|
{
|
|
GST_DEBUG_CATEGORY_INIT (gst_play_bin3_debug, "playbin3", 0, "play bin");
|
|
|
|
if (as_playbin)
|
|
return gst_element_register (plugin, "playbin", GST_RANK_NONE,
|
|
GST_TYPE_PLAY_BIN);
|
|
|
|
return gst_element_register (plugin, "playbin3", GST_RANK_NONE,
|
|
GST_TYPE_PLAY_BIN);
|
|
}
|