| /* GStreamer |
| * Copyright (C) <2007> Wim Taymans <wim.taymans@gmail.com> |
| * Copyright (C) <2011> Sebastian Dröge <sebastian.droege@collabora.co.uk> |
| * |
| * 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. |
| */ |
| |
| #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/video/video.h> |
| #include <gst/audio/streamvolume.h> |
| #include <gst/video/colorbalance.h> |
| #include <gst/video/videooverlay.h> |
| #include <gst/video/navigation.h> |
| |
| #include "gstplaysink.h" |
| #include "gststreamsynchronizer.h" |
| #include "gstplaysinkvideoconvert.h" |
| #include "gstplaysinkaudioconvert.h" |
| |
| GST_DEBUG_CATEGORY_STATIC (gst_play_sink_debug); |
| #define GST_CAT_DEFAULT gst_play_sink_debug |
| |
| #define VOLUME_MAX_DOUBLE 10.0 |
| |
| #define DEFAULT_FLAGS GST_PLAY_FLAG_AUDIO | GST_PLAY_FLAG_VIDEO | GST_PLAY_FLAG_TEXT | \ |
| GST_PLAY_FLAG_SOFT_VOLUME | GST_PLAY_FLAG_SOFT_COLORBALANCE |
| |
| #define GST_PLAY_CHAIN(c) ((GstPlayChain *)(c)) |
| |
| /* enum types */ |
| /** |
| * GstPlaySinkSendEventMode: |
| * @MODE_DEFAULT: default GstBin's send_event handling |
| * @MODE_FIRST: send event only to the first sink that return true |
| * |
| * Send event handling to use |
| */ |
| typedef enum |
| { |
| MODE_DEFAULT = 0, |
| MODE_FIRST = 1 |
| } GstPlaySinkSendEventMode; |
| |
| |
| #define GST_TYPE_PLAY_SINK_SEND_EVENT_MODE (gst_play_sink_send_event_mode_get_type ()) |
| static GType |
| gst_play_sink_send_event_mode_get_type (void) |
| { |
| static GType gtype = 0; |
| |
| if (gtype == 0) { |
| static const GEnumValue values[] = { |
| {MODE_DEFAULT, "Default GstBin's send_event handling (default)", |
| "default"}, |
| {MODE_FIRST, "Sends the event to sinks until the first one handles it", |
| "first"}, |
| {0, NULL, NULL} |
| }; |
| |
| gtype = g_enum_register_static ("GstPlaySinkSendEventMode", values); |
| } |
| return gtype; |
| } |
| |
| /* holds the common data fields for the audio and video pipelines. We keep them |
| * in a structure to more easily have all the info available. */ |
| typedef struct |
| { |
| GstPlaySink *playsink; |
| GstElement *bin; |
| gboolean added; |
| gboolean activated; |
| gboolean raw; |
| } GstPlayChain; |
| |
| typedef struct |
| { |
| GstPlayChain chain; |
| GstPad *sinkpad; |
| GstElement *queue; |
| GstElement *filter_conv; |
| GstElement *filter; |
| GstElement *conv; |
| GstElement *volume; /* element with the volume property */ |
| gboolean sink_volume; /* if the volume was provided by the sink */ |
| gulong notify_volume_id; |
| gulong notify_mute_id; |
| GstElement *sink; |
| GstElement *ts_offset; |
| } GstPlayAudioChain; |
| |
| typedef struct |
| { |
| GstPlayChain chain; |
| GstPad *sinkpad, *srcpad; |
| GstElement *conv; |
| GstElement *deinterlace; |
| } GstPlayVideoDeinterlaceChain; |
| |
| typedef struct |
| { |
| GstPlayChain chain; |
| GstPad *sinkpad; |
| GstElement *queue; |
| GstElement *filter_conv; |
| GstElement *filter; |
| GstElement *conv; |
| GstElement *sink; |
| gboolean async; |
| GstElement *ts_offset; |
| } GstPlayVideoChain; |
| |
| typedef struct |
| { |
| GstPlayChain chain; |
| GstPad *sinkpad; |
| GstElement *queue; |
| GstElement *conv; |
| GstElement *resample; |
| GstPad *blockpad; /* srcpad of queue, used for blocking the vis */ |
| GstPad *vispeerpad; /* srcpad of resample, used for unlinking the vis */ |
| GstPad *vissinkpad; /* visualisation sinkpad, */ |
| GstElement *vis; |
| GstPad *vissrcpad; /* visualisation srcpad, */ |
| GstPad *srcpad; /* outgoing srcpad, used to connect to the next |
| * chain */ |
| } GstPlayVisChain; |
| |
| typedef struct |
| { |
| GstPlayChain chain; |
| GstPad *sinkpad; |
| GstElement *queue; |
| GstElement *identity; |
| GstElement *overlay; |
| GstPad *videosinkpad; |
| GstPad *textsinkpad; |
| GstPad *srcpad; /* outgoing srcpad, used to connect to the next |
| * chain */ |
| GstElement *sink; /* custom sink to receive subtitle buffers */ |
| } GstPlayTextChain; |
| |
| #define GST_PLAY_SINK_GET_LOCK(playsink) (&((GstPlaySink *)playsink)->lock) |
| #define GST_PLAY_SINK_LOCK(playsink) G_STMT_START { \ |
| GST_LOG_OBJECT (playsink, "locking from thread %p", g_thread_self ()); \ |
| g_rec_mutex_lock (GST_PLAY_SINK_GET_LOCK (playsink)); \ |
| GST_LOG_OBJECT (playsink, "locked from thread %p", g_thread_self ()); \ |
| } G_STMT_END |
| #define GST_PLAY_SINK_UNLOCK(playsink) G_STMT_START { \ |
| GST_LOG_OBJECT (playsink, "unlocking from thread %p", g_thread_self ()); \ |
| g_rec_mutex_unlock (GST_PLAY_SINK_GET_LOCK (playsink)); \ |
| } G_STMT_END |
| |
| #define PENDING_FLAG_SET(playsink, flagtype) \ |
| ((playsink->pending_blocked_pads) |= (1 << flagtype)) |
| #define PENDING_FLAG_UNSET(playsink, flagtype) \ |
| ((playsink->pending_blocked_pads) &= ~(1 << flagtype)) |
| #define PENDING_FLAG_IS_SET(playsink, flagtype) \ |
| ((playsink->pending_blocked_pads) & (1 << flagtype)) |
| #define PENDING_VIDEO_BLOCK(playsink) \ |
| ((playsink->pending_blocked_pads) & (1 << GST_PLAY_SINK_TYPE_VIDEO_RAW | 1 << GST_PLAY_SINK_TYPE_VIDEO)) |
| #define PENDING_AUDIO_BLOCK(playsink) \ |
| ((playsink->pending_blocked_pads) & (1 << GST_PLAY_SINK_TYPE_AUDIO_RAW | 1 << GST_PLAY_SINK_TYPE_AUDIO)) |
| #define PENDING_TEXT_BLOCK(playsink) \ |
| PENDING_FLAG_IS_SET(playsink, GST_PLAY_SINK_TYPE_TEXT) |
| |
| struct _GstPlaySink |
| { |
| GstBin bin; |
| |
| GRecMutex lock; |
| |
| gboolean async_pending; |
| gboolean need_async_start; |
| |
| GstPlayFlags flags; |
| |
| GstStreamSynchronizer *stream_synchronizer; |
| |
| /* chains */ |
| GstPlayAudioChain *audiochain; |
| GstPlayVideoDeinterlaceChain *videodeinterlacechain; |
| GstPlayVideoChain *videochain; |
| GstPlayVisChain *vischain; |
| GstPlayTextChain *textchain; |
| |
| /* audio */ |
| GstPad *audio_pad; |
| gboolean audio_pad_raw; |
| gboolean audio_pad_blocked; |
| GstPad *audio_srcpad_stream_synchronizer; |
| GstPad *audio_sinkpad_stream_synchronizer; |
| GstElement *audio_ssync_queue; |
| GstPad *audio_ssync_queue_sinkpad; |
| gulong audio_block_id; |
| gulong audio_notify_caps_id; |
| /* audio tee */ |
| GstElement *audio_tee; |
| GstPad *audio_tee_sink; |
| GstPad *audio_tee_asrc; |
| GstPad *audio_tee_vissrc; |
| /* video */ |
| GstPad *video_pad; |
| gboolean video_pad_raw; |
| gboolean video_pad_blocked; |
| GstPad *video_srcpad_stream_synchronizer; |
| GstPad *video_sinkpad_stream_synchronizer; |
| gulong video_block_id; |
| gulong video_notify_caps_id; |
| /* text */ |
| GstPad *text_pad; |
| gboolean text_pad_blocked; |
| GstPad *text_srcpad_stream_synchronizer; |
| GstPad *text_sinkpad_stream_synchronizer; |
| gulong text_block_id; |
| |
| gulong vis_pad_block_id; |
| |
| guint32 pending_blocked_pads; |
| |
| /* properties */ |
| GstElement *audio_sink; |
| GstElement *video_sink; |
| GstElement *audio_filter; |
| GstElement *video_filter; |
| GstElement *visualisation; |
| GstElement *text_sink; |
| gdouble volume; |
| gboolean mute; |
| gchar *font_desc; /* font description */ |
| gchar *subtitle_encoding; /* subtitle encoding */ |
| guint connection_speed; /* connection speed in bits/sec (0 = unknown) */ |
| guint count; |
| gboolean volume_changed; /* volume/mute changed while no audiochain */ |
| gboolean mute_changed; /* ... has been created yet */ |
| gint64 av_offset; |
| GstPlaySinkSendEventMode send_event_mode; |
| gboolean force_aspect_ratio; |
| |
| /* videooverlay proxy interface */ |
| GstVideoOverlay *overlay_element; /* protected with LOCK */ |
| gboolean overlay_handle_set; |
| guintptr overlay_handle; |
| gboolean overlay_render_rectangle_set; |
| gint overlay_x, overlay_y, overlay_width, overlay_height; |
| gboolean overlay_handle_events_set; |
| gboolean overlay_handle_events; |
| |
| /* colorbalance proxy interface */ |
| GstColorBalance *colorbalance_element; |
| GList *colorbalance_channels; /* CONTRAST, BRIGHTNESS, HUE, SATURATION */ |
| gint colorbalance_values[4]; |
| gulong colorbalance_value_changed_id; |
| |
| /* sending audio/video flushes break stream changes when the pipeline |
| * is paused and played again in 0.10 */ |
| #if 0 |
| gboolean video_custom_flush_finished; |
| gboolean video_ignore_wrong_state; |
| gboolean video_pending_flush; |
| |
| gboolean audio_custom_flush_finished; |
| gboolean audio_ignore_wrong_state; |
| gboolean audio_pending_flush; |
| #endif |
| |
| gboolean text_custom_flush_finished; |
| gboolean text_ignore_wrong_state; |
| gboolean text_pending_flush; |
| }; |
| |
| struct _GstPlaySinkClass |
| { |
| GstBinClass parent_class; |
| |
| gboolean (*reconfigure) (GstPlaySink * playsink); |
| |
| GstSample *(*convert_sample) (GstPlaySink * playsink, GstCaps * caps); |
| }; |
| |
| |
| static GstStaticPadTemplate audiotemplate = |
| GST_STATIC_PAD_TEMPLATE ("audio_sink", |
| GST_PAD_SINK, |
| GST_PAD_REQUEST, |
| GST_STATIC_CAPS_ANY); |
| static GstStaticPadTemplate videotemplate = |
| GST_STATIC_PAD_TEMPLATE ("video_sink", |
| GST_PAD_SINK, |
| GST_PAD_REQUEST, |
| GST_STATIC_CAPS_ANY); |
| static GstStaticPadTemplate texttemplate = GST_STATIC_PAD_TEMPLATE ("text_sink", |
| GST_PAD_SINK, |
| GST_PAD_REQUEST, |
| GST_STATIC_CAPS_ANY); |
| |
| /* FIXME 0.11: Remove */ |
| static GstStaticPadTemplate audiorawtemplate = |
| GST_STATIC_PAD_TEMPLATE ("audio_raw_sink", |
| GST_PAD_SINK, |
| GST_PAD_REQUEST, |
| GST_STATIC_CAPS_ANY); |
| static GstStaticPadTemplate videorawtemplate = |
| GST_STATIC_PAD_TEMPLATE ("video_raw_sink", |
| GST_PAD_SINK, |
| GST_PAD_REQUEST, |
| GST_STATIC_CAPS_ANY); |
| |
| |
| /* props */ |
| enum |
| { |
| PROP_0, |
| PROP_FLAGS, |
| PROP_MUTE, |
| PROP_VOLUME, |
| PROP_FONT_DESC, |
| PROP_SUBTITLE_ENCODING, |
| PROP_VIS_PLUGIN, |
| PROP_SAMPLE, |
| PROP_AV_OFFSET, |
| PROP_VIDEO_SINK, |
| PROP_AUDIO_SINK, |
| PROP_TEXT_SINK, |
| PROP_SEND_EVENT_MODE, |
| PROP_FORCE_ASPECT_RATIO, |
| PROP_VIDEO_FILTER, |
| PROP_AUDIO_FILTER |
| }; |
| |
| /* signals */ |
| enum |
| { |
| LAST_SIGNAL |
| }; |
| |
| static void gst_play_sink_dispose (GObject * object); |
| static void gst_play_sink_finalize (GObject * object); |
| static void gst_play_sink_set_property (GObject * object, guint prop_id, |
| const GValue * value, GParamSpec * spec); |
| static void gst_play_sink_get_property (GObject * object, guint prop_id, |
| GValue * value, GParamSpec * spec); |
| |
| static GstPad *gst_play_sink_request_new_pad (GstElement * element, |
| GstPadTemplate * templ, const gchar * name, const GstCaps * caps); |
| static void gst_play_sink_release_request_pad (GstElement * element, |
| GstPad * pad); |
| static gboolean gst_play_sink_send_event (GstElement * element, |
| GstEvent * event); |
| static GstStateChangeReturn gst_play_sink_change_state (GstElement * element, |
| GstStateChange transition); |
| |
| static void gst_play_sink_handle_message (GstBin * bin, GstMessage * message); |
| |
| /* sending audio/video flushes break stream changes when the pipeline |
| * is paused and played again in 0.10 */ |
| #if 0 |
| static gboolean gst_play_sink_video_sink_event (GstPad * pad, GstEvent * event); |
| static GstFlowReturn gst_play_sink_video_sink_chain (GstPad * pad, |
| GstBuffer * buffer); |
| static gboolean gst_play_sink_audio_sink_event (GstPad * pad, GstEvent * event); |
| static GstFlowReturn gst_play_sink_audio_sink_chain (GstPad * pad, |
| GstBuffer * buffer); |
| #endif |
| static gboolean gst_play_sink_text_sink_event (GstPad * pad, GstObject * parent, |
| GstEvent * event); |
| static GstFlowReturn gst_play_sink_text_sink_chain (GstPad * pad, |
| GstObject * parent, GstBuffer * buffer); |
| |
| static void notify_volume_cb (GObject * object, GParamSpec * pspec, |
| GstPlaySink * playsink); |
| static void notify_mute_cb (GObject * object, GParamSpec * pspec, |
| GstPlaySink * playsink); |
| |
| static void update_av_offset (GstPlaySink * playsink); |
| |
| static gboolean gst_play_sink_do_reconfigure (GstPlaySink * playsink); |
| |
| static GQuark _playsink_reset_segment_event_marker_id = 0; |
| |
| /* static guint gst_play_sink_signals[LAST_SIGNAL] = { 0 }; */ |
| |
| static void gst_play_sink_overlay_init (gpointer g_iface, |
| gpointer g_iface_data); |
| static void gst_play_sink_navigation_init (gpointer g_iface, |
| gpointer g_iface_data); |
| static void gst_play_sink_colorbalance_init (gpointer g_iface, |
| gpointer g_iface_data); |
| |
| static void |
| _do_init (GType type) |
| { |
| static const GInterfaceInfo svol_info = { |
| NULL, NULL, NULL |
| }; |
| static const GInterfaceInfo ov_info = { |
| gst_play_sink_overlay_init, |
| NULL, NULL |
| }; |
| static const GInterfaceInfo nav_info = { |
| gst_play_sink_navigation_init, |
| NULL, NULL |
| }; |
| static const GInterfaceInfo col_info = { |
| gst_play_sink_colorbalance_init, |
| NULL, NULL |
| }; |
| |
| g_type_add_interface_static (type, GST_TYPE_STREAM_VOLUME, &svol_info); |
| g_type_add_interface_static (type, GST_TYPE_VIDEO_OVERLAY, &ov_info); |
| g_type_add_interface_static (type, GST_TYPE_NAVIGATION, &nav_info); |
| g_type_add_interface_static (type, GST_TYPE_COLOR_BALANCE, &col_info); |
| } |
| |
| G_DEFINE_TYPE_WITH_CODE (GstPlaySink, gst_play_sink, GST_TYPE_BIN, |
| _do_init (g_define_type_id)); |
| |
| static void |
| gst_play_sink_class_init (GstPlaySinkClass * klass) |
| { |
| GObjectClass *gobject_klass; |
| GstElementClass *gstelement_klass; |
| GstBinClass *gstbin_klass; |
| |
| gobject_klass = (GObjectClass *) klass; |
| gstelement_klass = (GstElementClass *) klass; |
| gstbin_klass = (GstBinClass *) klass; |
| |
| gobject_klass->dispose = gst_play_sink_dispose; |
| gobject_klass->finalize = gst_play_sink_finalize; |
| gobject_klass->set_property = gst_play_sink_set_property; |
| gobject_klass->get_property = gst_play_sink_get_property; |
| |
| |
| /** |
| * GstPlaySink:flags |
| * |
| * Control the behaviour of playsink. |
| */ |
| 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)); |
| |
| /** |
| * GstPlaySink: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)); |
| 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_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_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)); |
| /** |
| * GstPlaySink:sample: |
| * |
| * 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)); |
| /** |
| * GstPlaySink: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)); |
| |
| /** |
| * GstPlaySink:video-filter: |
| * |
| * Set the video filter element/bin to use. Will apply on a best-effort basis |
| * unless GST_PLAY_FLAG_FORCE_FILTERS is set. playsink must be in |
| * %GST_STATE_NULL |
| */ |
| 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)); |
| /** |
| * GstPlaySink:audio-filter: |
| * |
| * Set the audio filter element/bin to use. Will apply on a best-effort basis |
| * unless GST_PLAY_FLAG_FORCE_FILTERS is set. playsink must be in |
| * %GST_STATE_NULL |
| */ |
| 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)); |
| |
| /** |
| * GstPlaySink:video-sink: |
| * |
| * Set the used video sink element. NULL will use the default sink. playsink |
| * must be in %GST_STATE_NULL |
| */ |
| 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)); |
| /** |
| * GstPlaySink:audio-sink: |
| * |
| * Set the used audio sink element. NULL will use the default sink. playsink |
| * must be in %GST_STATE_NULL |
| */ |
| 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)); |
| |
| /** |
| * GstPlaySink:text-sink: |
| * |
| * Set the used text sink element. NULL will use the default sink. playsink |
| * must be in %GST_STATE_NULL |
| */ |
| g_object_class_install_property (gobject_klass, PROP_TEXT_SINK, |
| g_param_spec_object ("text-sink", "Text sink", |
| "the text output element to use (NULL = default subtitleoverlay)", |
| GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| /** |
| * GstPlaySink::send-event-mode: |
| * |
| * Sets the handling method used for events received from send_event |
| * function. The default is %MODE_DEFAULT, that uses %GstBin's default |
| * handling (push the event to all internal sinks). |
| */ |
| g_object_class_install_property (gobject_klass, PROP_SEND_EVENT_MODE, |
| g_param_spec_enum ("send-event-mode", "Send event mode", |
| "How to send events received in send_event function", |
| GST_TYPE_PLAY_SINK_SEND_EVENT_MODE, MODE_DEFAULT, |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| /** |
| * GstPlaySink::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)); |
| |
| g_signal_new ("reconfigure", G_TYPE_FROM_CLASS (klass), |
| G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (GstPlaySinkClass, |
| reconfigure), NULL, NULL, g_cclosure_marshal_generic, G_TYPE_BOOLEAN, |
| 0, G_TYPE_NONE); |
| /** |
| * GstPlaySink::convert-sample |
| * @playsink: a #GstPlaySink |
| * @caps: the target format of the sample |
| * |
| * Action signal to retrieve the currently playing video sample in the format |
| * specified by @caps. |
| * If @caps is %NULL, no conversion will be performed and this function is |
| * equivalent to the #GstPlaySink::sample property. |
| * |
| * Returns: a #GstSample of the current video sample converted to #caps. |
| * The caps in the sample will describe the final layout of the buffer data. |
| * %NULL is returned when no current sample can be retrieved or when the |
| * conversion failed. |
| */ |
| g_signal_new ("convert-sample", G_TYPE_FROM_CLASS (klass), |
| G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, |
| G_STRUCT_OFFSET (GstPlaySinkClass, convert_sample), NULL, NULL, |
| g_cclosure_marshal_generic, GST_TYPE_SAMPLE, 1, GST_TYPE_CAPS); |
| |
| gst_element_class_add_static_pad_template (gstelement_klass, |
| &audiorawtemplate); |
| gst_element_class_add_static_pad_template (gstelement_klass, &audiotemplate); |
| gst_element_class_add_static_pad_template (gstelement_klass, |
| &videorawtemplate); |
| gst_element_class_add_static_pad_template (gstelement_klass, &videotemplate); |
| gst_element_class_add_static_pad_template (gstelement_klass, &texttemplate); |
| gst_element_class_set_static_metadata (gstelement_klass, "Player Sink", |
| "Generic/Bin/Sink", |
| "Convenience sink for multiple streams", |
| "Wim Taymans <wim.taymans@gmail.com>"); |
| |
| gstelement_klass->change_state = |
| GST_DEBUG_FUNCPTR (gst_play_sink_change_state); |
| gstelement_klass->send_event = GST_DEBUG_FUNCPTR (gst_play_sink_send_event); |
| gstelement_klass->request_new_pad = |
| GST_DEBUG_FUNCPTR (gst_play_sink_request_new_pad); |
| gstelement_klass->release_pad = |
| GST_DEBUG_FUNCPTR (gst_play_sink_release_request_pad); |
| |
| gstbin_klass->handle_message = |
| GST_DEBUG_FUNCPTR (gst_play_sink_handle_message); |
| |
| klass->reconfigure = GST_DEBUG_FUNCPTR (gst_play_sink_reconfigure); |
| klass->convert_sample = GST_DEBUG_FUNCPTR (gst_play_sink_convert_sample); |
| |
| _playsink_reset_segment_event_marker_id = |
| g_quark_from_static_string ("gst-playsink-reset-segment-event-marker"); |
| |
| g_type_class_ref (GST_TYPE_STREAM_SYNCHRONIZER); |
| g_type_class_ref (GST_TYPE_COLOR_BALANCE_CHANNEL); |
| } |
| |
| static void |
| gst_play_sink_init (GstPlaySink * playsink) |
| { |
| GstColorBalanceChannel *channel; |
| |
| /* init groups */ |
| playsink->video_sink = NULL; |
| playsink->audio_sink = NULL; |
| playsink->visualisation = NULL; |
| playsink->text_sink = NULL; |
| playsink->volume = 1.0; |
| playsink->font_desc = NULL; |
| playsink->subtitle_encoding = NULL; |
| playsink->flags = DEFAULT_FLAGS; |
| playsink->send_event_mode = MODE_DEFAULT; |
| playsink->force_aspect_ratio = TRUE; |
| |
| playsink->stream_synchronizer = |
| g_object_new (GST_TYPE_STREAM_SYNCHRONIZER, NULL); |
| gst_bin_add (GST_BIN_CAST (playsink), |
| GST_ELEMENT_CAST (playsink->stream_synchronizer)); |
| |
| g_rec_mutex_init (&playsink->lock); |
| GST_OBJECT_FLAG_SET (playsink, GST_ELEMENT_FLAG_SINK); |
| gst_bin_set_suppressed_flags (GST_BIN (playsink), |
| GST_ELEMENT_FLAG_SOURCE | GST_ELEMENT_FLAG_SINK); |
| |
| channel = |
| GST_COLOR_BALANCE_CHANNEL (g_object_new (GST_TYPE_COLOR_BALANCE_CHANNEL, |
| NULL)); |
| channel->label = g_strdup ("CONTRAST"); |
| channel->min_value = -1000; |
| channel->max_value = 1000; |
| playsink->colorbalance_channels = |
| g_list_append (playsink->colorbalance_channels, channel); |
| playsink->colorbalance_values[0] = 0; |
| |
| channel = |
| GST_COLOR_BALANCE_CHANNEL (g_object_new (GST_TYPE_COLOR_BALANCE_CHANNEL, |
| NULL)); |
| channel->label = g_strdup ("BRIGHTNESS"); |
| channel->min_value = -1000; |
| channel->max_value = 1000; |
| playsink->colorbalance_channels = |
| g_list_append (playsink->colorbalance_channels, channel); |
| playsink->colorbalance_values[1] = 0; |
| |
| channel = |
| GST_COLOR_BALANCE_CHANNEL (g_object_new (GST_TYPE_COLOR_BALANCE_CHANNEL, |
| NULL)); |
| channel->label = g_strdup ("HUE"); |
| channel->min_value = -1000; |
| channel->max_value = 1000; |
| playsink->colorbalance_channels = |
| g_list_append (playsink->colorbalance_channels, channel); |
| playsink->colorbalance_values[2] = 0; |
| |
| channel = |
| GST_COLOR_BALANCE_CHANNEL (g_object_new (GST_TYPE_COLOR_BALANCE_CHANNEL, |
| NULL)); |
| channel->label = g_strdup ("SATURATION"); |
| channel->min_value = -1000; |
| channel->max_value = 1000; |
| playsink->colorbalance_channels = |
| g_list_append (playsink->colorbalance_channels, channel); |
| playsink->colorbalance_values[3] = 0; |
| } |
| |
| static void |
| disconnect_audio_chain (GstPlayAudioChain * chain, GstPlaySink * playsink) |
| { |
| if (chain) { |
| if (chain->notify_volume_id) |
| g_signal_handler_disconnect (chain->volume, chain->notify_volume_id); |
| if (chain->notify_mute_id) |
| g_signal_handler_disconnect (chain->volume, chain->notify_mute_id); |
| chain->notify_volume_id = chain->notify_mute_id = 0; |
| } |
| } |
| |
| static void |
| free_chain (GstPlayChain * chain) |
| { |
| if (chain) { |
| if (chain->bin) |
| gst_object_unref (chain->bin); |
| g_free (chain); |
| } |
| } |
| |
| static void |
| gst_play_sink_remove_audio_ssync_queue (GstPlaySink * playsink) |
| { |
| if (playsink->audio_ssync_queue) { |
| gst_element_set_state (playsink->audio_ssync_queue, GST_STATE_NULL); |
| gst_object_unref (playsink->audio_ssync_queue_sinkpad); |
| gst_bin_remove (GST_BIN_CAST (playsink), playsink->audio_ssync_queue); |
| playsink->audio_ssync_queue = NULL; |
| playsink->audio_ssync_queue_sinkpad = NULL; |
| } |
| } |
| |
| static void |
| gst_play_sink_dispose (GObject * object) |
| { |
| GstPlaySink *playsink; |
| |
| playsink = GST_PLAY_SINK (object); |
| |
| if (playsink->audio_filter != NULL) { |
| gst_element_set_state (playsink->audio_filter, GST_STATE_NULL); |
| gst_object_unref (playsink->audio_filter); |
| playsink->audio_filter = NULL; |
| } |
| if (playsink->video_filter != NULL) { |
| gst_element_set_state (playsink->video_filter, GST_STATE_NULL); |
| gst_object_unref (playsink->video_filter); |
| playsink->video_filter = NULL; |
| } |
| if (playsink->audio_sink != NULL) { |
| gst_element_set_state (playsink->audio_sink, GST_STATE_NULL); |
| gst_object_unref (playsink->audio_sink); |
| playsink->audio_sink = NULL; |
| } |
| if (playsink->video_sink != NULL) { |
| gst_element_set_state (playsink->video_sink, GST_STATE_NULL); |
| gst_object_unref (playsink->video_sink); |
| playsink->video_sink = NULL; |
| } |
| if (playsink->visualisation != NULL) { |
| gst_element_set_state (playsink->visualisation, GST_STATE_NULL); |
| gst_object_unref (playsink->visualisation); |
| playsink->visualisation = NULL; |
| } |
| if (playsink->text_sink != NULL) { |
| gst_element_set_state (playsink->text_sink, GST_STATE_NULL); |
| gst_object_unref (playsink->text_sink); |
| playsink->text_sink = NULL; |
| } |
| |
| free_chain ((GstPlayChain *) playsink->videodeinterlacechain); |
| playsink->videodeinterlacechain = NULL; |
| free_chain ((GstPlayChain *) playsink->videochain); |
| playsink->videochain = NULL; |
| free_chain ((GstPlayChain *) playsink->audiochain); |
| playsink->audiochain = NULL; |
| free_chain ((GstPlayChain *) playsink->vischain); |
| playsink->vischain = NULL; |
| free_chain ((GstPlayChain *) playsink->textchain); |
| playsink->textchain = NULL; |
| |
| if (playsink->audio_tee_sink) { |
| gst_object_unref (playsink->audio_tee_sink); |
| playsink->audio_tee_sink = NULL; |
| } |
| |
| if (playsink->audio_tee_vissrc) { |
| gst_element_release_request_pad (playsink->audio_tee, |
| playsink->audio_tee_vissrc); |
| gst_object_unref (playsink->audio_tee_vissrc); |
| playsink->audio_tee_vissrc = NULL; |
| } |
| |
| if (playsink->audio_tee_asrc) { |
| gst_element_release_request_pad (playsink->audio_tee, |
| playsink->audio_tee_asrc); |
| gst_object_unref (playsink->audio_tee_asrc); |
| playsink->audio_tee_asrc = NULL; |
| } |
| |
| g_free (playsink->font_desc); |
| playsink->font_desc = NULL; |
| |
| g_free (playsink->subtitle_encoding); |
| playsink->subtitle_encoding = NULL; |
| |
| playsink->stream_synchronizer = NULL; |
| |
| g_list_foreach (playsink->colorbalance_channels, (GFunc) gst_object_unref, |
| NULL); |
| g_list_free (playsink->colorbalance_channels); |
| playsink->colorbalance_channels = NULL; |
| |
| G_OBJECT_CLASS (gst_play_sink_parent_class)->dispose (object); |
| } |
| |
| static void |
| gst_play_sink_finalize (GObject * object) |
| { |
| GstPlaySink *playsink; |
| |
| playsink = GST_PLAY_SINK (object); |
| |
| g_rec_mutex_clear (&playsink->lock); |
| |
| G_OBJECT_CLASS (gst_play_sink_parent_class)->finalize (object); |
| } |
| |
| void |
| gst_play_sink_set_sink (GstPlaySink * playsink, GstPlaySinkType type, |
| GstElement * sink) |
| { |
| GstElement **elem = NULL, *old = NULL; |
| #ifndef GST_DISABLE_GST_DEBUG |
| GstPad *sink_pad; |
| const gchar *sink_type = NULL; |
| #endif |
| |
| GST_LOG ("Setting sink %" GST_PTR_FORMAT " as sink type %d", sink, type); |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| switch (type) { |
| case GST_PLAY_SINK_TYPE_AUDIO: |
| case GST_PLAY_SINK_TYPE_AUDIO_RAW: |
| elem = &playsink->audio_sink; |
| #ifndef GST_DISABLE_GST_DEBUG |
| sink_type = "audio"; |
| #endif |
| break; |
| case GST_PLAY_SINK_TYPE_VIDEO: |
| case GST_PLAY_SINK_TYPE_VIDEO_RAW: |
| elem = &playsink->video_sink; |
| #ifndef GST_DISABLE_GST_DEBUG |
| sink_type = "video"; |
| #endif |
| break; |
| case GST_PLAY_SINK_TYPE_TEXT: |
| elem = &playsink->text_sink; |
| #ifndef GST_DISABLE_GST_DEBUG |
| sink_type = "text"; |
| #endif |
| break; |
| default: |
| break; |
| } |
| if (elem) { |
| old = *elem; |
| if (sink) |
| gst_object_ref_sink (sink); |
| *elem = sink; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| #ifndef GST_DISABLE_GST_DEBUG |
| /* Check and warn if an application sets a sink with no 'sink' pad */ |
| if (sink && elem) { |
| if ((sink_pad = gst_element_get_static_pad (sink, "sink")) != NULL) { |
| gst_object_unref (sink_pad); |
| } else { |
| GST_ELEMENT_WARNING (playsink, CORE, FAILED, |
| ("Application error - playback can't work"), |
| ("custom %s sink has no pad named \"sink\"", sink_type)); |
| } |
| } |
| #endif |
| |
| if (old) { |
| /* Set the old sink to NULL if it is not used any longer */ |
| if (old != sink && !GST_OBJECT_PARENT (old)) |
| gst_element_set_state (old, GST_STATE_NULL); |
| gst_object_unref (old); |
| } |
| } |
| |
| GstElement * |
| gst_play_sink_get_sink (GstPlaySink * playsink, GstPlaySinkType type) |
| { |
| GstElement *result = NULL; |
| GstElement *elem = NULL, *chainp = NULL; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| switch (type) { |
| case GST_PLAY_SINK_TYPE_AUDIO: |
| case GST_PLAY_SINK_TYPE_AUDIO_RAW: |
| { |
| GstPlayAudioChain *chain; |
| if ((chain = (GstPlayAudioChain *) playsink->audiochain)) |
| chainp = chain->sink; |
| elem = playsink->audio_sink; |
| break; |
| } |
| case GST_PLAY_SINK_TYPE_VIDEO: |
| case GST_PLAY_SINK_TYPE_VIDEO_RAW: |
| { |
| GstPlayVideoChain *chain; |
| if ((chain = (GstPlayVideoChain *) playsink->videochain)) |
| chainp = chain->sink; |
| elem = playsink->video_sink; |
| break; |
| } |
| case GST_PLAY_SINK_TYPE_TEXT: |
| { |
| GstPlayTextChain *chain; |
| if ((chain = (GstPlayTextChain *) playsink->textchain)) |
| chainp = chain->sink; |
| elem = playsink->text_sink; |
| break; |
| } |
| default: |
| break; |
| } |
| if (chainp) { |
| /* we have an active chain with a sink, get the sink */ |
| result = gst_object_ref (chainp); |
| } |
| /* nothing found, return last configured sink */ |
| if (result == NULL && elem) |
| result = gst_object_ref (elem); |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return result; |
| } |
| |
| void |
| gst_play_sink_set_filter (GstPlaySink * playsink, GstPlaySinkType type, |
| GstElement * filter) |
| { |
| GstElement **elem = NULL, *old = NULL; |
| |
| GST_LOG_OBJECT (playsink, |
| "Setting filter %" GST_PTR_FORMAT " as filter type %d", filter, type); |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| switch (type) { |
| case GST_PLAY_SINK_TYPE_AUDIO: |
| case GST_PLAY_SINK_TYPE_AUDIO_RAW: |
| elem = &playsink->audio_filter; |
| break; |
| case GST_PLAY_SINK_TYPE_VIDEO: |
| case GST_PLAY_SINK_TYPE_VIDEO_RAW: |
| elem = &playsink->video_filter; |
| break; |
| default: |
| break; |
| } |
| if (elem) { |
| old = *elem; |
| if (filter) |
| gst_object_ref_sink (filter); |
| *elem = filter; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| if (old) { |
| /* Set the old filter to NULL if it is not used any longer */ |
| if (old != filter && !GST_OBJECT_PARENT (old)) |
| gst_element_set_state (old, GST_STATE_NULL); |
| gst_object_unref (old); |
| } |
| } |
| |
| GstElement * |
| gst_play_sink_get_filter (GstPlaySink * playsink, GstPlaySinkType type) |
| { |
| GstElement *result = NULL; |
| GstElement *elem = NULL, *chainp = NULL; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| switch (type) { |
| case GST_PLAY_SINK_TYPE_AUDIO: |
| case GST_PLAY_SINK_TYPE_AUDIO_RAW: |
| { |
| GstPlayAudioChain *chain; |
| if ((chain = (GstPlayAudioChain *) playsink->audiochain)) |
| chainp = chain->filter; |
| elem = playsink->audio_filter; |
| break; |
| } |
| case GST_PLAY_SINK_TYPE_VIDEO: |
| case GST_PLAY_SINK_TYPE_VIDEO_RAW: |
| { |
| GstPlayVideoChain *chain; |
| if ((chain = (GstPlayVideoChain *) playsink->videochain)) |
| chainp = chain->filter; |
| elem = playsink->video_filter; |
| break; |
| } |
| default: |
| break; |
| } |
| if (chainp) { |
| /* we have an active chain with a filter, get the filter */ |
| result = gst_object_ref (chainp); |
| } |
| /* nothing found, return last configured filter */ |
| if (result == NULL && elem) |
| result = gst_object_ref (elem); |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return result; |
| } |
| |
| static GstPadProbeReturn |
| gst_play_sink_vis_blocked (GstPad * tee_pad, GstPadProbeInfo * info, |
| gpointer user_data) |
| { |
| GstPlaySink *playsink; |
| GstPlayVisChain *chain; |
| |
| playsink = GST_PLAY_SINK (user_data); |
| |
| if (GST_IS_EVENT (info->data) && !GST_EVENT_IS_SERIALIZED (info->data)) { |
| GST_DEBUG_OBJECT (playsink, "Letting non-serialized event %s pass", |
| GST_EVENT_TYPE_NAME (info->data)); |
| return GST_PAD_PROBE_PASS; |
| } |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| GST_DEBUG_OBJECT (playsink, "vis pad blocked"); |
| /* now try to change the plugin in the running vis chain */ |
| if (!(chain = (GstPlayVisChain *) playsink->vischain)) |
| goto done; |
| |
| /* unlink the old plugin and unghost the pad */ |
| gst_pad_unlink (chain->vispeerpad, chain->vissinkpad); |
| gst_ghost_pad_set_target (GST_GHOST_PAD_CAST (chain->srcpad), NULL); |
| |
| /* set the old plugin to NULL and remove */ |
| gst_element_set_state (chain->vis, GST_STATE_NULL); |
| gst_bin_remove (GST_BIN_CAST (chain->chain.bin), chain->vis); |
| |
| /* add new plugin and set state to playing */ |
| chain->vis = playsink->visualisation; |
| gst_bin_add (GST_BIN_CAST (chain->chain.bin), chain->vis); |
| gst_element_set_state (chain->vis, GST_STATE_PLAYING); |
| |
| /* get pads */ |
| chain->vissinkpad = gst_element_get_static_pad (chain->vis, "sink"); |
| chain->vissrcpad = gst_element_get_static_pad (chain->vis, "src"); |
| |
| /* link pads */ |
| gst_pad_link_full (chain->vispeerpad, chain->vissinkpad, |
| GST_PAD_LINK_CHECK_NOTHING); |
| gst_ghost_pad_set_target (GST_GHOST_PAD_CAST (chain->srcpad), |
| chain->vissrcpad); |
| |
| done: |
| playsink->vis_pad_block_id = 0; |
| |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| /* remove the probe and unblock the pad */ |
| return GST_PAD_PROBE_REMOVE; |
| } |
| |
| void |
| gst_play_sink_set_vis_plugin (GstPlaySink * playsink, GstElement * vis) |
| { |
| GstPlayVisChain *chain; |
| |
| /* setting NULL means creating the default vis plugin */ |
| if (vis == NULL) |
| vis = gst_element_factory_make ("goom", "vis"); |
| |
| /* simply return if we don't have a vis plugin here */ |
| if (vis == NULL) |
| return; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| /* first store the new vis */ |
| if (playsink->visualisation) |
| gst_object_unref (playsink->visualisation); |
| /* take ownership */ |
| gst_object_ref_sink (vis); |
| playsink->visualisation = vis; |
| |
| /* now try to change the plugin in the running vis chain, if we have no chain, |
| * we don't bother, any future vis chain will be created with the new vis |
| * plugin. */ |
| if (!(chain = (GstPlayVisChain *) playsink->vischain)) |
| goto done; |
| |
| /* block the pad, the next time the callback is called we can change the |
| * visualisation. It's possible that this never happens or that the pad was |
| * already blocked. If the callback never happens, we don't have new data so |
| * we don't need the new vis plugin. If the pad was already blocked, the |
| * function returns FALSE but the previous pad block will do the right thing |
| * anyway. */ |
| GST_DEBUG_OBJECT (playsink, "blocking vis pad"); |
| if (!playsink->vis_pad_block_id && !playsink->audio_block_id |
| && !playsink->video_block_id && !playsink->text_block_id) |
| playsink->vis_pad_block_id = |
| gst_pad_add_probe (chain->blockpad, GST_PAD_PROBE_TYPE_BLOCK_DOWNSTREAM, |
| gst_play_sink_vis_blocked, playsink, NULL); |
| done: |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return; |
| } |
| |
| GstElement * |
| gst_play_sink_get_vis_plugin (GstPlaySink * playsink) |
| { |
| GstElement *result = NULL; |
| GstPlayVisChain *chain; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| if ((chain = (GstPlayVisChain *) playsink->vischain)) { |
| /* we have an active chain, get the sink */ |
| if (chain->vis) |
| result = gst_object_ref (chain->vis); |
| } |
| /* nothing found, return last configured sink */ |
| if (result == NULL && playsink->visualisation) |
| result = gst_object_ref (playsink->visualisation); |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return result; |
| } |
| |
| void |
| gst_play_sink_set_volume (GstPlaySink * playsink, gdouble volume) |
| { |
| GstPlayAudioChain *chain; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| playsink->volume = volume; |
| chain = (GstPlayAudioChain *) playsink->audiochain; |
| if (chain && chain->volume) { |
| GST_LOG_OBJECT (playsink, |
| "elements: volume=%" GST_PTR_FORMAT "; new volume=%.03f, mute=%d", |
| chain->volume, volume, playsink->mute); |
| g_object_set (chain->volume, "volume", volume, NULL); |
| } else { |
| GST_LOG_OBJECT (playsink, "no volume element"); |
| playsink->volume_changed = TRUE; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| } |
| |
| gdouble |
| gst_play_sink_get_volume (GstPlaySink * playsink) |
| { |
| gdouble result; |
| GstPlayAudioChain *chain; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| chain = (GstPlayAudioChain *) playsink->audiochain; |
| result = playsink->volume; |
| if (chain && chain->volume) { |
| g_object_get (chain->volume, "volume", &result, NULL); |
| playsink->volume = result; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return result; |
| } |
| |
| void |
| gst_play_sink_set_mute (GstPlaySink * playsink, gboolean mute) |
| { |
| GstPlayAudioChain *chain; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| playsink->mute = mute; |
| chain = (GstPlayAudioChain *) playsink->audiochain; |
| if (chain && chain->volume) { |
| g_object_set (chain->volume, "mute", mute, NULL); |
| } else { |
| playsink->mute_changed = TRUE; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| } |
| |
| gboolean |
| gst_play_sink_get_mute (GstPlaySink * playsink) |
| { |
| gboolean result; |
| GstPlayAudioChain *chain; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| chain = (GstPlayAudioChain *) playsink->audiochain; |
| if (chain && chain->volume) { |
| g_object_get (chain->volume, "mute", &result, NULL); |
| playsink->mute = result; |
| } else { |
| result = playsink->mute; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| return result; |
| } |
| |
| static void |
| post_missing_element_message (GstPlaySink * playsink, const gchar * name) |
| { |
| GstMessage *msg; |
| |
| msg = gst_missing_element_message_new (GST_ELEMENT_CAST (playsink), name); |
| gst_element_post_message (GST_ELEMENT_CAST (playsink), msg); |
| } |
| |
| static gboolean |
| add_chain (GstPlayChain * chain, gboolean add) |
| { |
| if (chain->added == add) |
| return TRUE; |
| |
| if (add) |
| gst_bin_add (GST_BIN_CAST (chain->playsink), chain->bin); |
| else { |
| gst_bin_remove (GST_BIN_CAST (chain->playsink), chain->bin); |
| } |
| |
| chain->added = add; |
| |
| return TRUE; |
| } |
| |
| static gboolean |
| activate_chain (GstPlayChain * chain, gboolean activate) |
| { |
| GstState state; |
| |
| if (chain->activated == activate) |
| return TRUE; |
| |
| GST_OBJECT_LOCK (chain->playsink); |
| state = GST_STATE_TARGET (chain->playsink); |
| GST_OBJECT_UNLOCK (chain->playsink); |
| |
| if (activate) |
| gst_element_set_state (chain->bin, state); |
| else |
| gst_element_set_state (chain->bin, GST_STATE_NULL); |
| |
| chain->activated = activate; |
| |
| return TRUE; |
| } |
| |
| static gboolean |
| element_is_sink (GstElement * element) |
| { |
| gboolean is_sink; |
| |
| GST_OBJECT_LOCK (element); |
| is_sink = GST_OBJECT_FLAG_IS_SET (element, GST_ELEMENT_FLAG_SINK); |
| GST_OBJECT_UNLOCK (element); |
| |
| GST_DEBUG_OBJECT (element, "is a sink: %s", (is_sink) ? "yes" : "no"); |
| return is_sink; |
| } |
| |
| static gboolean |
| element_has_property (GstElement * element, const gchar * pname, GType type) |
| { |
| GParamSpec *pspec; |
| |
| pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (element), pname); |
| |
| if (pspec == NULL) { |
| GST_DEBUG_OBJECT (element, "no %s property", pname); |
| return FALSE; |
| } |
| |
| if (type == G_TYPE_INVALID || type == pspec->value_type || |
| g_type_is_a (pspec->value_type, type)) { |
| GST_DEBUG_OBJECT (element, "has %s property of type %s", pname, |
| (type == G_TYPE_INVALID) ? "any type" : g_type_name (type)); |
| return TRUE; |
| } |
| |
| GST_WARNING_OBJECT (element, "has %s property, but property is of type %s " |
| "and we expected it to be of type %s", pname, |
| g_type_name (pspec->value_type), g_type_name (type)); |
| |
| return FALSE; |
| } |
| |
| typedef struct |
| { |
| const gchar *prop_name; |
| GType prop_type; |
| gboolean need_sink; |
| } FindPropertyHelper; |
| |
| static gint |
| find_property (const GValue * item, FindPropertyHelper * helper) |
| { |
| GstElement *element = g_value_get_object (item); |
| if (helper->need_sink && !element_is_sink (element)) { |
| return 1; |
| } |
| |
| if (!element_has_property (element, helper->prop_name, helper->prop_type)) { |
| return 1; |
| } |
| |
| GST_INFO_OBJECT (element, "found %s with %s property", helper->prop_name, |
| (helper->need_sink) ? "sink" : "element"); |
| return 0; /* keep it */ |
| } |
| |
| /* FIXME: why not move these functions into core? */ |
| /* find a sink in the hierarchy with a property named @name. This function does |
| * not increase the refcount of the returned object and thus remains valid as |
| * long as the bin is valid. */ |
| static GstElement * |
| gst_play_sink_find_property_sinks (GstPlaySink * playsink, GstElement * obj, |
| const gchar * name, GType expected_type) |
| { |
| GstElement *result = NULL; |
| GstIterator *it; |
| |
| if (element_has_property (obj, name, expected_type)) { |
| result = obj; |
| } else if (GST_IS_BIN (obj)) { |
| gboolean found; |
| GValue item = { 0, }; |
| FindPropertyHelper helper = { name, expected_type, TRUE }; |
| |
| it = gst_bin_iterate_recurse (GST_BIN_CAST (obj)); |
| found = gst_iterator_find_custom (it, |
| (GCompareFunc) find_property, &item, &helper); |
| gst_iterator_free (it); |
| if (found) { |
| result = g_value_get_object (&item); |
| /* we don't need the extra ref */ |
| g_value_unset (&item); |
| } |
| } |
| return result; |
| } |
| |
| /* find an object in the hierarchy with a property named @name */ |
| static GstElement * |
| gst_play_sink_find_property (GstPlaySink * playsink, GstElement * obj, |
| const gchar * name, GType expected_type) |
| { |
| GstElement *result = NULL; |
| GstIterator *it; |
| |
| if (GST_IS_BIN (obj)) { |
| gboolean found; |
| GValue item = { 0, }; |
| FindPropertyHelper helper = { name, expected_type, FALSE }; |
| |
| it = gst_bin_iterate_recurse (GST_BIN_CAST (obj)); |
| found = gst_iterator_find_custom (it, |
| (GCompareFunc) find_property, &item, &helper); |
| gst_iterator_free (it); |
| if (found) { |
| result = g_value_dup_object (&item); |
| g_value_unset (&item); |
| } |
| } else { |
| if (element_has_property (obj, name, expected_type)) { |
| result = obj; |
| gst_object_ref (obj); |
| } |
| } |
| return result; |
| } |
| |
| static void |
| do_async_start (GstPlaySink * playsink) |
| { |
| GstMessage *message; |
| |
| if (!playsink->need_async_start) { |
| GST_INFO_OBJECT (playsink, "no async_start needed"); |
| return; |
| } |
| |
| playsink->async_pending = TRUE; |
| |
| GST_INFO_OBJECT (playsink, "Sending async_start message"); |
| message = gst_message_new_async_start (GST_OBJECT_CAST (playsink)); |
| GST_BIN_CLASS (gst_play_sink_parent_class)->handle_message (GST_BIN_CAST |
| (playsink), message); |
| } |
| |
| static void |
| do_async_done (GstPlaySink * playsink) |
| { |
| GstMessage *message; |
| |
| if (playsink->async_pending) { |
| GST_INFO_OBJECT (playsink, "Sending async_done message"); |
| message = |
| gst_message_new_async_done (GST_OBJECT_CAST (playsink), |
| GST_CLOCK_TIME_NONE); |
| GST_BIN_CLASS (gst_play_sink_parent_class)->handle_message (GST_BIN_CAST |
| (playsink), message); |
| |
| playsink->async_pending = FALSE; |
| } |
| |
| playsink->need_async_start = FALSE; |
| } |
| |
| /* try to change the state of an element. This function returns the element when |
| * the state change could be performed. When this function returns NULL an error |
| * occured and the element is unreffed if @unref is TRUE. */ |
| static GstElement * |
| try_element (GstPlaySink * playsink, GstElement * element, gboolean unref) |
| { |
| GstStateChangeReturn ret; |
| |
| if (element) { |
| ret = gst_element_set_state (element, GST_STATE_READY); |
| if (ret == GST_STATE_CHANGE_FAILURE) { |
| GST_DEBUG_OBJECT (playsink, "failed state change.."); |
| gst_element_set_state (element, GST_STATE_NULL); |
| if (unref) |
| gst_object_unref (element); |
| element = NULL; |
| } |
| } |
| return element; |
| } |
| |
| /* make the element (bin) that contains the elements needed to perform |
| * video deinterlacing. Only used for *raw* video streams. |
| * |
| * +---------------------------------------+ |
| * | vbin | |
| * | +----------+ +-----------+ | |
| * | |colorspace| |deinterlace| | |
| * | +-sink src-sink src-+ | |
| * | | +----------+ +-----------+ | | |
| * sink-+ +-src |
| * +---------------------------------------+ |
| * |
| */ |
| static GstPlayVideoDeinterlaceChain * |
| gen_video_deinterlace_chain (GstPlaySink * playsink) |
| { |
| GstPlayVideoDeinterlaceChain *chain; |
| GstBin *bin; |
| GstPad *pad; |
| GstElement *head = NULL, *prev = NULL; |
| |
| chain = g_new0 (GstPlayVideoDeinterlaceChain, 1); |
| chain->chain.playsink = playsink; |
| |
| GST_DEBUG_OBJECT (playsink, "making video deinterlace chain %p", chain); |
| |
| /* create a bin to hold objects, as we create them we add them to this bin so |
| * that when something goes wrong we only need to unref the bin */ |
| chain->chain.bin = gst_bin_new ("vdbin"); |
| bin = GST_BIN_CAST (chain->chain.bin); |
| gst_object_ref_sink (bin); |
| |
| GST_DEBUG_OBJECT (playsink, "creating " COLORSPACE); |
| chain->conv = gst_element_factory_make (COLORSPACE, "vdconv"); |
| if (chain->conv == NULL) { |
| post_missing_element_message (playsink, COLORSPACE); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| COLORSPACE), ("video rendering might fail")); |
| } else { |
| gst_bin_add (bin, chain->conv); |
| head = chain->conv; |
| prev = chain->conv; |
| } |
| |
| GST_DEBUG_OBJECT (playsink, "creating deinterlace"); |
| chain->deinterlace = gst_element_factory_make ("deinterlace", "deinterlace"); |
| if (chain->deinterlace == NULL) { |
| post_missing_element_message (playsink, "deinterlace"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "deinterlace"), ("deinterlacing won't work")); |
| } else { |
| gst_bin_add (bin, chain->deinterlace); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->deinterlace, "sink", |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } else { |
| head = chain->deinterlace; |
| } |
| prev = chain->deinterlace; |
| } |
| |
| if (head) { |
| pad = gst_element_get_static_pad (head, "sink"); |
| chain->sinkpad = gst_ghost_pad_new ("sink", pad); |
| gst_object_unref (pad); |
| } else { |
| chain->sinkpad = gst_ghost_pad_new_no_target ("sink", GST_PAD_SINK); |
| } |
| |
| if (prev) { |
| pad = gst_element_get_static_pad (prev, "src"); |
| chain->srcpad = gst_ghost_pad_new ("src", pad); |
| gst_object_unref (pad); |
| } else { |
| chain->srcpad = gst_ghost_pad_new ("src", chain->sinkpad); |
| } |
| |
| gst_element_add_pad (chain->chain.bin, chain->sinkpad); |
| gst_element_add_pad (chain->chain.bin, chain->srcpad); |
| |
| return chain; |
| |
| link_failed: |
| { |
| GST_ELEMENT_ERROR (playsink, CORE, PAD, |
| (NULL), ("Failed to configure the video deinterlace chain.")); |
| free_chain ((GstPlayChain *) chain); |
| return NULL; |
| } |
| } |
| |
| static gboolean |
| is_valid_color_balance_element (GstColorBalance * bal) |
| { |
| gboolean have_brightness = FALSE; |
| gboolean have_contrast = FALSE; |
| gboolean have_hue = FALSE; |
| gboolean have_saturation = FALSE; |
| const GList *channels, *l; |
| |
| channels = gst_color_balance_list_channels (bal); |
| for (l = channels; l; l = l->next) { |
| GstColorBalanceChannel *ch = l->data; |
| |
| if (g_strrstr (ch->label, "BRIGHTNESS")) |
| have_brightness = TRUE; |
| else if (g_strrstr (ch->label, "CONTRAST")) |
| have_contrast = TRUE; |
| else if (g_strrstr (ch->label, "HUE")) |
| have_hue = TRUE; |
| else if (g_strrstr (ch->label, "SATURATION")) |
| have_saturation = TRUE; |
| } |
| |
| return have_brightness && have_contrast && have_hue && have_saturation; |
| } |
| |
| static void |
| iterate_color_balance_elements (const GValue * item, gpointer user_data) |
| { |
| gboolean valid; |
| GstColorBalance *cb, **cb_out = user_data; |
| |
| cb = GST_COLOR_BALANCE (g_value_get_object (item)); |
| valid = is_valid_color_balance_element (cb); |
| if (valid) { |
| if (*cb_out |
| && gst_color_balance_get_balance_type (*cb_out) == |
| GST_COLOR_BALANCE_SOFTWARE) { |
| gst_object_unref (*cb_out); |
| *cb_out = GST_COLOR_BALANCE (gst_object_ref (cb)); |
| } else if (!*cb_out) { |
| *cb_out = GST_COLOR_BALANCE (gst_object_ref (cb)); |
| } |
| } |
| } |
| |
| static GstColorBalance * |
| find_color_balance_element (GstElement * element) |
| { |
| GstIterator *it; |
| GstColorBalance *cb = NULL; |
| |
| if (GST_IS_COLOR_BALANCE (element) |
| && is_valid_color_balance_element (GST_COLOR_BALANCE (element))) |
| return GST_COLOR_BALANCE (gst_object_ref (element)); |
| else if (!GST_IS_BIN (element)) |
| return FALSE; |
| |
| it = gst_bin_iterate_all_by_interface (GST_BIN (element), |
| GST_TYPE_COLOR_BALANCE); |
| while (gst_iterator_foreach (it, iterate_color_balance_elements, |
| &cb) == GST_ITERATOR_RESYNC) |
| gst_iterator_resync (it); |
| gst_iterator_free (it); |
| |
| return cb; |
| } |
| |
| static void |
| colorbalance_value_changed_cb (GstColorBalance * balance, |
| GstColorBalanceChannel * channel, gint value, GstPlaySink * playsink) |
| { |
| GList *l; |
| gint i; |
| |
| for (i = 0, l = playsink->colorbalance_channels; l; l = l->next, i++) { |
| GstColorBalanceChannel *proxy = l->data; |
| |
| if (g_strrstr (channel->label, proxy->label)) { |
| gdouble new_val; |
| |
| /* Convert to [0, 1] range */ |
| new_val = |
| ((gdouble) value - |
| (gdouble) channel->min_value) / ((gdouble) channel->max_value - |
| (gdouble) channel->min_value); |
| /* Convert to proxy range */ |
| new_val = |
| proxy->min_value + new_val * ((gdouble) proxy->max_value - |
| (gdouble) proxy->min_value); |
| playsink->colorbalance_values[i] = (gint) (0.5 + new_val); |
| |
| gst_color_balance_value_changed (GST_COLOR_BALANCE (playsink), proxy, |
| playsink->colorbalance_values[i]); |
| break; |
| } |
| } |
| } |
| |
| static void |
| update_colorbalance (GstPlaySink * playsink) |
| { |
| GstColorBalance *balance = NULL; |
| GList *l; |
| gint i; |
| |
| GST_OBJECT_LOCK (playsink); |
| if (playsink->colorbalance_element) { |
| balance = |
| GST_COLOR_BALANCE (gst_object_ref (playsink->colorbalance_element)); |
| } |
| GST_OBJECT_UNLOCK (playsink); |
| if (!balance) |
| return; |
| |
| g_signal_handler_block (balance, playsink->colorbalance_value_changed_id); |
| |
| for (i = 0, l = playsink->colorbalance_channels; l; l = l->next, i++) { |
| GstColorBalanceChannel *proxy = l->data; |
| GstColorBalanceChannel *channel = NULL; |
| const GList *channels, *k; |
| gdouble new_val; |
| |
| channels = gst_color_balance_list_channels (balance); |
| for (k = channels; k; k = k->next) { |
| GstColorBalanceChannel *tmp = k->data; |
| |
| if (g_strrstr (tmp->label, proxy->label)) { |
| channel = tmp; |
| break; |
| } |
| } |
| |
| g_assert (channel); |
| |
| /* Convert to [0, 1] range */ |
| new_val = |
| ((gdouble) playsink->colorbalance_values[i] - |
| (gdouble) proxy->min_value) / ((gdouble) proxy->max_value - |
| (gdouble) proxy->min_value); |
| /* Convert to channel range */ |
| new_val = |
| channel->min_value + new_val * ((gdouble) channel->max_value - |
| (gdouble) channel->min_value); |
| |
| gst_color_balance_set_value (balance, channel, (gint) (new_val + 0.5)); |
| } |
| |
| g_signal_handler_unblock (balance, playsink->colorbalance_value_changed_id); |
| |
| gst_object_unref (balance); |
| } |
| |
| /* make the element (bin) that contains the elements needed to perform |
| * video display. |
| * |
| * +------------------------------------------------------------------------+ |
| * | vbin | |
| * | +--------+ +-------+ +----------+ +----------+ +---------+ | |
| * | | filter | | queue | |colorspace| |videoscale| |videosink| | |
| * | +-sink src-sink src-sink src-sink src-sink | | |
| * | | +--------+ +-------+ +----------+ +----------+ +---------+ | |
| * sink-+ | |
| * +------------------------------------------------------------------------+ |
| * |
| */ |
| static GstPlayVideoChain * |
| gen_video_chain (GstPlaySink * playsink, gboolean raw, gboolean async) |
| { |
| GstPlayVideoChain *chain; |
| GstBin *bin; |
| GstPad *pad; |
| GstElement *head = NULL, *prev = NULL, *elem = NULL; |
| |
| chain = g_new0 (GstPlayVideoChain, 1); |
| chain->chain.playsink = playsink; |
| chain->chain.raw = raw; |
| |
| GST_DEBUG_OBJECT (playsink, "making video chain %p", chain); |
| |
| if (playsink->video_sink) { |
| GST_DEBUG_OBJECT (playsink, "trying configured videosink"); |
| chain->sink = try_element (playsink, playsink->video_sink, FALSE); |
| } else { |
| /* only try fallback if no specific sink was chosen */ |
| if (chain->sink == NULL) { |
| GST_DEBUG_OBJECT (playsink, "trying autovideosink"); |
| elem = gst_element_factory_make ("autovideosink", "videosink"); |
| chain->sink = try_element (playsink, elem, TRUE); |
| } |
| if (chain->sink == NULL) { |
| /* if default sink from config.h is different then try it too */ |
| if (strcmp (DEFAULT_VIDEOSINK, "autovideosink")) { |
| GST_DEBUG_OBJECT (playsink, "trying " DEFAULT_VIDEOSINK); |
| elem = gst_element_factory_make (DEFAULT_VIDEOSINK, "videosink"); |
| chain->sink = try_element (playsink, elem, TRUE); |
| } |
| } |
| if (chain->sink) |
| playsink->video_sink = gst_object_ref (chain->sink); |
| } |
| if (chain->sink == NULL) |
| goto no_sinks; |
| head = chain->sink; |
| |
| /* if we can disable async behaviour of the sink, we can avoid adding a |
| * queue for the audio chain. */ |
| elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "async", |
| G_TYPE_BOOLEAN); |
| if (elem) { |
| GST_DEBUG_OBJECT (playsink, "setting async property to %d on element %s", |
| async, GST_ELEMENT_NAME (elem)); |
| g_object_set (elem, "async", async, NULL); |
| chain->async = async; |
| } else { |
| GST_DEBUG_OBJECT (playsink, "no async property on the sink"); |
| chain->async = TRUE; |
| } |
| |
| /* Make sure the aspect ratio is kept */ |
| elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, |
| "force-aspect-ratio", G_TYPE_BOOLEAN); |
| if (elem) |
| g_object_set (elem, "force-aspect-ratio", playsink->force_aspect_ratio, |
| NULL); |
| |
| /* find ts-offset element */ |
| gst_object_replace ((GstObject **) & chain->ts_offset, (GstObject *) |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "ts-offset", |
| G_TYPE_INT64)); |
| |
| /* create a bin to hold objects, as we create them we add them to this bin so |
| * that when something goes wrong we only need to unref the bin */ |
| chain->chain.bin = gst_bin_new ("vbin"); |
| bin = GST_BIN_CAST (chain->chain.bin); |
| gst_object_ref_sink (bin); |
| gst_bin_add (bin, chain->sink); |
| |
| /* Get the VideoOverlay element */ |
| { |
| GstVideoOverlay *overlay = NULL; |
| |
| GST_OBJECT_LOCK (playsink); |
| if (playsink->overlay_element) |
| gst_object_unref (playsink->overlay_element); |
| playsink->overlay_element = |
| GST_VIDEO_OVERLAY (gst_bin_get_by_interface (GST_BIN (chain->chain.bin), |
| GST_TYPE_VIDEO_OVERLAY)); |
| if (playsink->overlay_element) |
| overlay = GST_VIDEO_OVERLAY (gst_object_ref (playsink->overlay_element)); |
| GST_OBJECT_UNLOCK (playsink); |
| |
| if (overlay) { |
| if (playsink->overlay_handle_set) |
| gst_video_overlay_set_window_handle (overlay, playsink->overlay_handle); |
| if (playsink->overlay_handle_events_set) |
| gst_video_overlay_handle_events (overlay, |
| playsink->overlay_handle_events); |
| if (playsink->overlay_render_rectangle_set) |
| gst_video_overlay_set_render_rectangle (overlay, |
| playsink->overlay_x, playsink->overlay_y, |
| playsink->overlay_width, playsink->overlay_height); |
| gst_object_unref (overlay); |
| } |
| } |
| |
| head = chain->sink; |
| prev = NULL; |
| |
| /* add the video filter first, so everything is working with post-filter |
| * samples */ |
| chain->filter = gst_play_sink_get_filter (playsink, |
| GST_PLAY_SINK_TYPE_VIDEO_RAW); |
| if (chain->filter) { |
| if (!raw) { |
| gst_object_unref (chain->filter); |
| chain->filter = NULL; |
| |
| if (playsink->flags & GST_PLAY_FLAG_FORCE_FILTERS) { |
| goto filter_with_nonraw; |
| } else { |
| GST_DEBUG_OBJECT (playsink, |
| "skipping video filter since we're not raw"); |
| } |
| } else { |
| GST_DEBUG_OBJECT (playsink, "adding video filter"); |
| chain->filter_conv = |
| gst_element_factory_make ("videoconvert", "filter-convert"); |
| if (!chain->filter_conv) { |
| post_missing_element_message (playsink, "videoconvert"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "videoconvert"), |
| ("video playback and visualizations might not work")); |
| } else { |
| gst_bin_add (bin, chain->filter_conv); |
| head = prev = chain->filter_conv; |
| } |
| |
| gst_bin_add (bin, chain->filter); |
| /* Bin takes a new reference because we sinked any |
| * floating reference ourselves already */ |
| gst_object_unref (chain->filter); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->filter, NULL, |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) { |
| goto link_failed; |
| } |
| } else { |
| head = chain->filter; |
| } |
| prev = chain->filter; |
| } |
| } |
| |
| /* decouple decoder from sink, this improves playback quite a lot since the |
| * decoder can continue while the sink blocks for synchronisation. We don't |
| * need a lot of buffers as this consumes a lot of memory and we don't want |
| * too little because else we would be context switching too quickly. */ |
| chain->queue = gst_element_factory_make ("queue", "vqueue"); |
| if (chain->queue == NULL) { |
| post_missing_element_message (playsink, "queue"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "queue"), ("video rendering might be suboptimal")); |
| } else { |
| g_object_set (G_OBJECT (chain->queue), "max-size-buffers", 3, |
| "max-size-bytes", 0, "max-size-time", (gint64) 0, "silent", TRUE, NULL); |
| gst_bin_add (bin, chain->queue); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->queue, "sink", |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } else { |
| head = chain->queue; |
| } |
| prev = chain->queue; |
| } |
| |
| GST_OBJECT_LOCK (playsink); |
| if (playsink->colorbalance_element) { |
| g_signal_handler_disconnect (playsink->colorbalance_element, |
| playsink->colorbalance_value_changed_id); |
| gst_object_unref (playsink->colorbalance_element); |
| playsink->colorbalance_value_changed_id = 0; |
| } |
| playsink->colorbalance_element = find_color_balance_element (chain->sink); |
| if (playsink->colorbalance_element) { |
| playsink->colorbalance_value_changed_id = |
| g_signal_connect (playsink->colorbalance_element, "value-changed", |
| G_CALLBACK (colorbalance_value_changed_cb), playsink); |
| } |
| GST_OBJECT_UNLOCK (playsink); |
| |
| if (!(playsink->flags & GST_PLAY_FLAG_NATIVE_VIDEO) |
| || (!playsink->colorbalance_element |
| && (playsink->flags & GST_PLAY_FLAG_SOFT_COLORBALANCE))) { |
| gboolean use_converters = !(playsink->flags & GST_PLAY_FLAG_NATIVE_VIDEO); |
| gboolean use_balance = !playsink->colorbalance_element |
| && (playsink->flags & GST_PLAY_FLAG_SOFT_COLORBALANCE); |
| |
| GST_DEBUG_OBJECT (playsink, "creating videoconverter"); |
| chain->conv = |
| g_object_new (GST_TYPE_PLAY_SINK_VIDEO_CONVERT, "name", "vconv", |
| "use-converters", use_converters, "use-balance", use_balance, NULL); |
| |
| GST_OBJECT_LOCK (playsink); |
| if (use_balance && GST_PLAY_SINK_VIDEO_CONVERT (chain->conv)->balance) { |
| playsink->colorbalance_element = |
| GST_COLOR_BALANCE (gst_object_ref (GST_PLAY_SINK_VIDEO_CONVERT |
| (chain->conv)->balance)); |
| playsink->colorbalance_value_changed_id = |
| g_signal_connect (playsink->colorbalance_element, "value-changed", |
| G_CALLBACK (colorbalance_value_changed_cb), playsink); |
| } |
| GST_OBJECT_UNLOCK (playsink); |
| |
| gst_bin_add (bin, chain->conv); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->conv, "sink", |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } else { |
| head = chain->conv; |
| } |
| prev = chain->conv; |
| } |
| |
| update_colorbalance (playsink); |
| |
| if (prev) { |
| GST_DEBUG_OBJECT (playsink, "linking to sink"); |
| if (!gst_element_link_pads_full (prev, "src", chain->sink, NULL, |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } |
| |
| pad = gst_element_get_static_pad (head, "sink"); |
| chain->sinkpad = gst_ghost_pad_new ("sink", pad); |
| |
| /* sending audio/video flushes break stream changes when the pipeline |
| * is paused and played again in 0.10 */ |
| #if 0 |
| gst_pad_set_event_function (chain->sinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_video_sink_event)); |
| gst_pad_set_chain_function (chain->sinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_video_sink_chain)); |
| #endif |
| |
| gst_object_unref (pad); |
| gst_element_add_pad (chain->chain.bin, chain->sinkpad); |
| |
| return chain; |
| |
| /* ERRORS */ |
| no_sinks: |
| { |
| if (!elem && !playsink->video_sink) { |
| post_missing_element_message (playsink, "autovideosink"); |
| if (strcmp (DEFAULT_VIDEOSINK, "autovideosink")) { |
| post_missing_element_message (playsink, DEFAULT_VIDEOSINK); |
| GST_ELEMENT_ERROR (playsink, CORE, MISSING_PLUGIN, |
| (_("Both autovideosink and %s elements are missing."), |
| DEFAULT_VIDEOSINK), (NULL)); |
| } else { |
| GST_ELEMENT_ERROR (playsink, CORE, MISSING_PLUGIN, |
| (_("The autovideosink element is missing.")), (NULL)); |
| } |
| } else { |
| if (playsink->video_sink) { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("Configured videosink %s is not working."), |
| GST_ELEMENT_NAME (playsink->video_sink)), (NULL)); |
| } else if (strcmp (DEFAULT_VIDEOSINK, "autovideosink")) { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("Both autovideosink and %s elements are not working."), |
| DEFAULT_VIDEOSINK), (NULL)); |
| } else { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("The autovideosink element is not working.")), (NULL)); |
| } |
| } |
| free_chain ((GstPlayChain *) chain); |
| return NULL; |
| } |
| |
| link_failed: |
| { |
| GST_ELEMENT_ERROR (playsink, CORE, PAD, |
| (NULL), ("Failed to configure the video sink.")); |
| goto cleanup; |
| } |
| filter_with_nonraw: |
| { |
| GST_ELEMENT_ERROR (playsink, CORE, NEGOTIATION, |
| (NULL), ("Cannot apply video-filter on non-raw stream")); |
| goto cleanup; |
| } |
| cleanup: |
| /* checking sink made it READY */ |
| gst_element_set_state (chain->sink, GST_STATE_NULL); |
| /* Remove chain from the bin to allow reuse later */ |
| gst_bin_remove (bin, chain->sink); |
| free_chain ((GstPlayChain *) chain); |
| return NULL; |
| } |
| |
| static gboolean |
| setup_video_chain (GstPlaySink * playsink, gboolean raw, gboolean async) |
| { |
| GstElement *elem; |
| GstPlayVideoChain *chain; |
| GstStateChangeReturn ret; |
| |
| chain = playsink->videochain; |
| |
| /* if we have a filter, and raw-ness changed, we have to force a rebuild */ |
| if (chain->filter && chain->chain.raw != raw) |
| return FALSE; |
| |
| chain->chain.raw = raw; |
| |
| /* if the chain was active we don't do anything */ |
| if (GST_PLAY_CHAIN (chain)->activated) |
| return TRUE; |
| |
| /* try to set the sink element to READY again */ |
| ret = gst_element_set_state (chain->sink, GST_STATE_READY); |
| if (ret == GST_STATE_CHANGE_FAILURE) |
| return FALSE; |
| |
| /* Get the VideoOverlay element */ |
| { |
| GstVideoOverlay *overlay = NULL; |
| |
| GST_OBJECT_LOCK (playsink); |
| if (playsink->overlay_element) |
| gst_object_unref (playsink->overlay_element); |
| playsink->overlay_element = |
| GST_VIDEO_OVERLAY (gst_bin_get_by_interface (GST_BIN (chain->chain.bin), |
| GST_TYPE_VIDEO_OVERLAY)); |
| if (playsink->overlay_element) |
| overlay = GST_VIDEO_OVERLAY (gst_object_ref (playsink->overlay_element)); |
| GST_OBJECT_UNLOCK (playsink); |
| |
| if (overlay) { |
| if (playsink->overlay_handle_set) |
| gst_video_overlay_set_window_handle (overlay, playsink->overlay_handle); |
| if (playsink->overlay_handle_events_set) |
| gst_video_overlay_handle_events (overlay, |
| playsink->overlay_handle_events); |
| if (playsink->overlay_render_rectangle_set) |
| gst_video_overlay_set_render_rectangle (overlay, |
| playsink->overlay_x, playsink->overlay_y, |
| playsink->overlay_width, playsink->overlay_height); |
| gst_object_unref (overlay); |
| } |
| } |
| |
| /* find ts-offset element */ |
| gst_object_replace ((GstObject **) & chain->ts_offset, (GstObject *) |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "ts-offset", |
| G_TYPE_INT64)); |
| |
| /* if we can disable async behaviour of the sink, we can avoid adding a |
| * queue for the audio chain. */ |
| elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "async", |
| G_TYPE_BOOLEAN); |
| if (elem) { |
| GST_DEBUG_OBJECT (playsink, "setting async property to %d on element %s", |
| async, GST_ELEMENT_NAME (elem)); |
| g_object_set (elem, "async", async, NULL); |
| chain->async = async; |
| } else { |
| GST_DEBUG_OBJECT (playsink, "no async property on the sink"); |
| chain->async = TRUE; |
| } |
| |
| /* Make sure the aspect ratio is kept */ |
| elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, |
| "force-aspect-ratio", G_TYPE_BOOLEAN); |
| if (elem) |
| g_object_set (elem, "force-aspect-ratio", playsink->force_aspect_ratio, |
| NULL); |
| |
| GST_OBJECT_LOCK (playsink); |
| if (playsink->colorbalance_element) { |
| g_signal_handler_disconnect (playsink->colorbalance_element, |
| playsink->colorbalance_value_changed_id); |
| playsink->colorbalance_value_changed_id = 0; |
| gst_object_unref (playsink->colorbalance_element); |
| } |
| playsink->colorbalance_element = find_color_balance_element (chain->sink); |
| if (playsink->colorbalance_element) { |
| playsink->colorbalance_value_changed_id = |
| g_signal_connect (playsink->colorbalance_element, "value-changed", |
| G_CALLBACK (colorbalance_value_changed_cb), playsink); |
| } |
| GST_OBJECT_UNLOCK (playsink); |
| |
| if (chain->conv) { |
| gboolean use_balance = !playsink->colorbalance_element |
| && (playsink->flags & GST_PLAY_FLAG_SOFT_COLORBALANCE); |
| |
| g_object_set (chain->conv, "use-balance", use_balance, NULL); |
| |
| GST_OBJECT_LOCK (playsink); |
| if (use_balance && GST_PLAY_SINK_VIDEO_CONVERT (chain->conv)->balance) { |
| playsink->colorbalance_element = |
| GST_COLOR_BALANCE (gst_object_ref (GST_PLAY_SINK_VIDEO_CONVERT |
| (chain->conv)->balance)); |
| playsink->colorbalance_value_changed_id = |
| g_signal_connect (playsink->colorbalance_element, "value-changed", |
| G_CALLBACK (colorbalance_value_changed_cb), playsink); |
| } |
| GST_OBJECT_UNLOCK (playsink); |
| } |
| |
| update_colorbalance (playsink); |
| |
| return TRUE; |
| } |
| |
| static gboolean |
| gst_play_sink_sink_event (GstPad * pad, GstObject * parent, GstEvent * event, |
| const gchar * sink_type, |
| gboolean * sink_ignore_wrong_state, |
| gboolean * sink_custom_flush_finished, gboolean * sink_pending_flush) |
| { |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_object_get_parent (parent)); |
| gboolean ret; |
| const GstStructure *structure = gst_event_get_structure (event); |
| |
| if (GST_EVENT_TYPE (event) == GST_EVENT_CUSTOM_DOWNSTREAM_OOB && structure) { |
| gchar *custom_flush; |
| gchar *custom_flush_finish; |
| |
| custom_flush = g_strdup_printf ("playsink-custom-%s-flush", sink_type); |
| custom_flush_finish = |
| g_strdup_printf ("playsink-custom-%s-flush-finish", sink_type); |
| if (strcmp (gst_structure_get_name (structure), custom_flush) == 0) { |
| GST_DEBUG_OBJECT (pad, |
| "Custom %s flush event received, marking to flush %s", sink_type, |
| sink_type); |
| GST_PLAY_SINK_LOCK (playsink); |
| *sink_ignore_wrong_state = TRUE; |
| *sink_custom_flush_finished = FALSE; |
| GST_PLAY_SINK_UNLOCK (playsink); |
| } else if (strcmp (gst_structure_get_name (structure), |
| custom_flush_finish) == 0) { |
| GST_DEBUG_OBJECT (pad, "Custom %s flush finish event received", |
| sink_type); |
| GST_PLAY_SINK_LOCK (playsink); |
| *sink_pending_flush = TRUE; |
| *sink_custom_flush_finished = TRUE; |
| GST_PLAY_SINK_UNLOCK (playsink); |
| } |
| |
| g_free (custom_flush); |
| g_free (custom_flush_finish); |
| } |
| |
| GST_DEBUG_OBJECT (pad, "Forwarding event %" GST_PTR_FORMAT, event); |
| ret = gst_pad_event_default (pad, parent, gst_event_ref (event)); |
| |
| gst_event_unref (event); |
| gst_object_unref (playsink); |
| return ret; |
| } |
| |
| static GstFlowReturn |
| gst_play_sink_sink_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer, |
| const gchar * sink_type, |
| gboolean * sink_ignore_wrong_state, |
| gboolean * sink_custom_flush_finished, gboolean * sink_pending_flush) |
| { |
| GstBin *tbin = GST_BIN_CAST (gst_pad_get_parent (pad)); |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_pad_get_parent (tbin)); |
| GstFlowReturn ret; |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| |
| if (*sink_pending_flush) { |
| GstEvent *segment_event; |
| GstEvent *event; |
| GstStructure *structure; |
| |
| *sink_pending_flush = FALSE; |
| |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| segment_event = gst_pad_get_sticky_event (pad, GST_EVENT_SEGMENT, 0); |
| |
| /* make the bin drop all cached data. |
| * This event will be dropped on the src pad, if any. */ |
| event = gst_event_new_flush_start (); |
| structure = gst_event_writable_structure (event); |
| gst_structure_id_set (structure, |
| _playsink_reset_segment_event_marker_id, G_TYPE_BOOLEAN, TRUE, NULL); |
| |
| GST_DEBUG_OBJECT (pad, |
| "Pushing %s flush-start event with reset segment marker set: %" |
| GST_PTR_FORMAT, sink_type, event); |
| gst_pad_send_event (pad, event); |
| |
| /* make queue drop all cached data. |
| * This event will be dropped on the src pad. */ |
| event = gst_event_new_flush_stop (TRUE); |
| structure = gst_event_writable_structure (event); |
| gst_structure_id_set (structure, |
| _playsink_reset_segment_event_marker_id, G_TYPE_BOOLEAN, TRUE, NULL); |
| |
| GST_DEBUG_OBJECT (pad, |
| "Pushing %s flush-stop event with reset segment marker set: %" |
| GST_PTR_FORMAT, sink_type, event); |
| gst_pad_send_event (pad, event); |
| |
| /* Re-sync queue segment info after flush-stop. |
| * This event will be dropped on the src pad. */ |
| if (segment_event) { |
| event = gst_event_copy (segment_event); |
| structure = gst_event_writable_structure (event); |
| gst_structure_id_set (structure, |
| _playsink_reset_segment_event_marker_id, G_TYPE_BOOLEAN, TRUE, NULL); |
| |
| GST_DEBUG_OBJECT (playsink, |
| "Pushing segment event with reset " |
| "segment marker set: %" GST_PTR_FORMAT, event); |
| gst_pad_send_event (pad, event); |
| gst_event_unref (segment_event); |
| } |
| } else { |
| GST_PLAY_SINK_UNLOCK (playsink); |
| } |
| |
| ret = gst_proxy_pad_chain_default (pad, parent, buffer); |
| |
| GST_PLAY_SINK_LOCK (playsink); |
| if (ret == GST_FLOW_FLUSHING && *sink_ignore_wrong_state) { |
| GST_DEBUG_OBJECT (pad, "Ignoring wrong state for %s during flush", |
| sink_type); |
| if (*sink_custom_flush_finished) { |
| GST_DEBUG_OBJECT (pad, "Custom flush finished, stop ignoring " |
| "wrong state for %s", sink_type); |
| *sink_ignore_wrong_state = FALSE; |
| } |
| |
| ret = GST_FLOW_OK; |
| } |
| GST_PLAY_SINK_UNLOCK (playsink); |
| |
| gst_object_unref (playsink); |
| gst_object_unref (tbin); |
| return ret; |
| } |
| |
| /* sending audio/video flushes break stream changes when the pipeline |
| * is paused and played again in 0.10 */ |
| #if 0 |
| static gboolean |
| gst_play_sink_video_sink_event (GstPad * pad, GstEvent * event) |
| { |
| GstBin *tbin = GST_BIN_CAST (gst_pad_get_parent (pad)); |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_pad_get_parent (tbin)); |
| gboolean ret; |
| |
| ret = gst_play_sink_sink_event (pad, event, "video", |
| &playsink->video_ignore_wrong_state, |
| &playsink->video_custom_flush_finished, |
| &playsink->video_pending_flush, &playsink->video_segment); |
| |
| gst_object_unref (playsink); |
| gst_object_unref (tbin); |
| return ret; |
| } |
| |
| static GstFlowReturn |
| gst_play_sink_video_sink_chain (GstPad * pad, GstBuffer * buffer) |
| { |
| GstBin *tbin = GST_BIN_CAST (gst_pad_get_parent (pad)); |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_pad_get_parent (tbin)); |
| gboolean ret; |
| |
| ret = gst_play_sink_sink_chain (pad, buffer, "video", |
| &playsink->video_ignore_wrong_state, |
| &playsink->video_custom_flush_finished, |
| &playsink->video_pending_flush, &playsink->video_segment); |
| |
| gst_object_unref (playsink); |
| gst_object_unref (tbin); |
| return ret; |
| } |
| |
| static gboolean |
| gst_play_sink_audio_sink_event (GstPad * pad, GstEvent * event) |
| { |
| GstBin *tbin = GST_BIN_CAST (gst_pad_get_parent (pad)); |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_pad_get_parent (tbin)); |
| gboolean ret; |
| |
| ret = gst_play_sink_sink_event (pad, event, "audio", |
| &playsink->audio_ignore_wrong_state, |
| &playsink->audio_custom_flush_finished, |
| &playsink->audio_pending_flush, &playsink->audio_segment); |
| |
| gst_object_unref (playsink); |
| gst_object_unref (tbin); |
| return ret; |
| } |
| |
| static GstFlowReturn |
| gst_play_sink_audio_sink_chain (GstPad * pad, GstBuffer * buffer) |
| { |
| GstBin *tbin = GST_BIN_CAST (gst_pad_get_parent (pad)); |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_pad_get_parent (tbin)); |
| gboolean ret; |
| |
| ret = gst_play_sink_sink_chain (pad, buffer, "audio", |
| &playsink->audio_ignore_wrong_state, |
| &playsink->audio_custom_flush_finished, |
| &playsink->audio_pending_flush, &playsink->audio_segment); |
| |
| gst_object_unref (playsink); |
| gst_object_unref (tbin); |
| return ret; |
| } |
| #endif |
| |
| static gboolean |
| gst_play_sink_text_sink_event (GstPad * pad, GstObject * parent, |
| GstEvent * event) |
| { |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_object_get_parent (parent)); |
| gboolean ret; |
| |
| ret = gst_play_sink_sink_event (pad, parent, event, "subtitle", |
| &playsink->text_ignore_wrong_state, |
| &playsink->text_custom_flush_finished, &playsink->text_pending_flush); |
| |
| gst_object_unref (playsink); |
| |
| return ret; |
| } |
| |
| static GstFlowReturn |
| gst_play_sink_text_sink_chain (GstPad * pad, GstObject * parent, |
| GstBuffer * buffer) |
| { |
| gboolean ret; |
| GstPlaySink *playsink = GST_PLAY_SINK_CAST (gst_object_get_parent (parent)); |
| |
| ret = gst_play_sink_sink_chain (pad, parent, buffer, "subtitle", |
| &playsink->text_ignore_wrong_state, |
| &playsink->text_custom_flush_finished, &playsink->text_pending_flush); |
| |
| gst_object_unref (playsink); |
| return ret; |
| } |
| |
| static gboolean |
| gst_play_sink_text_src_event (GstPad * pad, GstObject * parent, |
| GstEvent * event) |
| { |
| gboolean ret; |
| const GstStructure *structure; |
| |
| GST_DEBUG_OBJECT (pad, "Got event %" GST_PTR_FORMAT, event); |
| |
| structure = gst_event_get_structure (event); |
| |
| if (structure && |
| gst_structure_id_has_field (structure, |
| _playsink_reset_segment_event_marker_id)) { |
| /* the events marked with a reset segment marker |
| * are sent internally to reset the queue and |
| * must be dropped here */ |
| GST_DEBUG_OBJECT (pad, "Dropping event with reset " |
| "segment marker set: %" GST_PTR_FORMAT, event); |
| ret = TRUE; |
| goto out; |
| } |
| |
| ret = gst_pad_event_default (pad, parent, gst_event_ref (event)); |
| |
| out: |
| gst_event_unref (event); |
| return ret; |
| } |
| |
| /* make an element for playback of video with subtitles embedded. |
| * Only used for *raw* video streams. |
| * |
| * +--------------------------------------------+ |
| * | tbin | |
| * | +--------+ +-----------------+ | |
| * | | queue | | subtitleoverlay | | |
| * video--src sink---video_sink | | |
| * | +--------+ | src--src |
| * text------------------text_sink | | |
| * | +-----------------+ | |
| * +--------------------------------------------+ |
| * |
| */ |
| static GstPlayTextChain * |
| gen_text_chain (GstPlaySink * playsink) |
| { |
| GstPlayTextChain *chain; |
| GstBin *bin; |
| GstElement *elem; |
| GstPad *videosinkpad, *textsinkpad, *srcpad; |
| |
| chain = g_new0 (GstPlayTextChain, 1); |
| chain->chain.playsink = playsink; |
| |
| GST_DEBUG_OBJECT (playsink, "making text chain %p", chain); |
| |
| chain->chain.bin = gst_bin_new ("tbin"); |
| bin = GST_BIN_CAST (chain->chain.bin); |
| gst_object_ref_sink (bin); |
| |
| videosinkpad = textsinkpad = srcpad = NULL; |
| |
| /* first try to hook the text pad to the custom sink */ |
| if (playsink->text_sink) { |
| GST_DEBUG_OBJECT (playsink, "trying configured textsink"); |
| chain->sink = try_element (playsink, playsink->text_sink, FALSE); |
| if (chain->sink) { |
| elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "async", |
| G_TYPE_BOOLEAN); |
| if (elem) { |
| /* make sure the sparse subtitles don't participate in the preroll */ |
| //g_object_set (elem, "async", FALSE, NULL); |
| GST_DEBUG_OBJECT (playsink, "adding custom text sink"); |
| gst_bin_add (bin, chain->sink); |
| /* NOTE streamsynchronizer needs streams decoupled */ |
| /* make a little queue */ |
| chain->queue = gst_element_factory_make ("queue", "subqueue"); |
| if (chain->queue == NULL) { |
| post_missing_element_message (playsink, "queue"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "queue"), ("rendering might be suboptimal")); |
| } else { |
| g_object_set (G_OBJECT (chain->queue), "max-size-buffers", 3, |
| "max-size-bytes", 0, "max-size-time", (gint64) GST_SECOND, |
| "silent", TRUE, NULL); |
| gst_bin_add (bin, chain->queue); |
| } |
| /* we have a custom sink, this will be our textsinkpad */ |
| if (gst_element_link_pads_full (chain->queue, "src", chain->sink, |
| "sink", GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) { |
| /* we're all fine now and we can add the sink to the chain */ |
| GST_DEBUG_OBJECT (playsink, "using custom text sink"); |
| textsinkpad = gst_element_get_static_pad (chain->queue, "sink"); |
| } else { |
| GST_WARNING_OBJECT (playsink, |
| "can't find a sink pad on custom text sink"); |
| gst_bin_remove (bin, chain->sink); |
| gst_bin_remove (bin, chain->queue); |
| chain->sink = NULL; |
| chain->queue = NULL; |
| } |
| /* try to set sync to true but it's no biggie when we can't */ |
| if (chain->sink && (elem = |
| gst_play_sink_find_property_sinks (playsink, chain->sink, |
| "sync", G_TYPE_BOOLEAN))) |
| g_object_set (elem, "sync", TRUE, NULL); |
| |
| if (!textsinkpad) |
| gst_bin_remove (bin, chain->sink); |
| } else { |
| GST_WARNING_OBJECT (playsink, |
| "can't find async property in custom text sink"); |
| } |
| } |
| if (textsinkpad == NULL) { |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Custom text sink element is not usable.")), |
| ("fallback to default subtitleoverlay")); |
| } |
| } |
| |
| if (textsinkpad == NULL) { |
| if (!(playsink->flags & GST_PLAY_FLAG_NATIVE_VIDEO)) { |
| /* make a little queue */ |
| chain->queue = gst_element_factory_make ("queue", "vqueue"); |
| if (chain->queue == NULL) { |
| post_missing_element_message (playsink, "queue"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "queue"), ("video rendering might be suboptimal")); |
| } else { |
| g_object_set (G_OBJECT (chain->queue), "max-size-buffers", 3, |
| "max-size-bytes", 0, "max-size-time", (gint64) 0, |
| "silent", TRUE, NULL); |
| gst_bin_add (bin, chain->queue); |
| videosinkpad = gst_element_get_static_pad (chain->queue, "sink"); |
| } |
| |
| chain->overlay = |
| gst_element_factory_make ("subtitleoverlay", "suboverlay"); |
| if (chain->overlay == NULL) { |
| post_missing_element_message (playsink, "subtitleoverlay"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "subtitleoverlay"), ("subtitle rendering disabled")); |
| } else { |
| GstElement *element; |
| |
| gst_bin_add (bin, chain->overlay); |
| |
| g_object_set (G_OBJECT (chain->overlay), "silent", FALSE, NULL); |
| if (playsink->font_desc) { |
| g_object_set (G_OBJECT (chain->overlay), "font-desc", |
| playsink->font_desc, NULL); |
| } |
| if (playsink->subtitle_encoding) { |
| g_object_set (G_OBJECT (chain->overlay), "subtitle-encoding", |
| playsink->subtitle_encoding, NULL); |
| } |
| |
| gst_element_link_pads_full (chain->queue, "src", chain->overlay, |
| "video_sink", GST_PAD_LINK_CHECK_TEMPLATE_CAPS); |
| |
| /* make another little queue to decouple streams */ |
| element = gst_element_factory_make ("queue", "subqueue"); |
| if (element == NULL) { |
| post_missing_element_message (playsink, "queue"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "queue"), ("rendering might be suboptimal")); |
| } else { |
| g_object_set (G_OBJECT (element), "max-size-buffers", 3, |
| "max-size-bytes", 0, "max-size-time", (gint64) GST_SECOND, |
| "silent", TRUE, NULL); |
| gst_bin_add (bin, element); |
| if (gst_element_link_pads_full (element, "src", chain->overlay, |
| "subtitle_sink", GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) { |
| textsinkpad = gst_element_get_static_pad (element, "sink"); |
| srcpad = gst_element_get_static_pad (chain->overlay, "src"); |
| } else { |
| gst_bin_remove (bin, chain->sink); |
| gst_bin_remove (bin, chain->overlay); |
| chain->sink = NULL; |
| chain->overlay = NULL; |
| gst_object_unref (videosinkpad); |
| videosinkpad = NULL; |
| } |
| } |
| } |
| } |
| } |
| |
| if (videosinkpad == NULL) { |
| /* if we still don't have a videosink, we don't have an overlay. the only |
| * thing we can do is insert an identity and ghost the src |
| * and sink pads. */ |
| chain->identity = gst_element_factory_make ("identity", "tidentity"); |
| if (chain->identity == NULL) { |
| post_missing_element_message (playsink, "identity"); |
| GST_ELEMENT_ERROR (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "identity"), (NULL)); |
| } else { |
| g_object_set (chain->identity, "signal-handoffs", FALSE, NULL); |
| g_object_set (chain->identity, "silent", TRUE, NULL); |
| gst_bin_add (bin, chain->identity); |
| srcpad = gst_element_get_static_pad (chain->identity, "src"); |
| videosinkpad = gst_element_get_static_pad (chain->identity, "sink"); |
| } |
| } |
| |
| /* expose the ghostpads */ |
| if (videosinkpad) { |
| chain->videosinkpad = gst_ghost_pad_new ("sink", videosinkpad); |
| gst_object_unref (videosinkpad); |
| gst_element_add_pad (chain->chain.bin, chain->videosinkpad); |
| } |
| if (textsinkpad) { |
| chain->textsinkpad = gst_ghost_pad_new ("text_sink", textsinkpad); |
| gst_object_unref (textsinkpad); |
| |
| gst_pad_set_event_function (chain->textsinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_text_sink_event)); |
| gst_pad_set_chain_function (chain->textsinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_text_sink_chain)); |
| |
| gst_element_add_pad (chain->chain.bin, chain->textsinkpad); |
| } |
| if (srcpad) { |
| chain->srcpad = gst_ghost_pad_new ("src", srcpad); |
| gst_object_unref (srcpad); |
| |
| gst_pad_set_event_function (chain->srcpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_text_src_event)); |
| |
| gst_element_add_pad (chain->chain.bin, chain->srcpad); |
| } |
| |
| return chain; |
| } |
| |
| static void |
| notify_volume_cb (GObject * object, GParamSpec * pspec, GstPlaySink * playsink) |
| { |
| gdouble vol; |
| |
| g_object_get (object, "volume", &vol, NULL); |
| playsink->volume = vol; |
| |
| g_object_notify (G_OBJECT (playsink), "volume"); |
| } |
| |
| static void |
| notify_mute_cb (GObject * object, GParamSpec * pspec, GstPlaySink * playsink) |
| { |
| gboolean mute; |
| |
| g_object_get (object, "mute", &mute, NULL); |
| playsink->mute = mute; |
| |
| g_object_notify (G_OBJECT (playsink), "mute"); |
| } |
| |
| /* make the chain that contains the elements needed to perform |
| * audio playback. |
| * |
| * We add a tee as the first element so that we can link the visualisation chain |
| * to it when requested. |
| * |
| * +--------------------------------------------------------------+ |
| * | abin | |
| * | +----------+ +--------+ +---------+ +-----------+ | |
| * | | filter | | queue | | convbin | | audiosink | | |
| * | +-sink src-sink src-sink src-sink | | |
| * | | +----------+ +--------+ +---------+ +-----------+ | |
| * sink-+ | |
| * +--------------------------------------------------------------+ |
| */ |
| static GstPlayAudioChain * |
| gen_audio_chain (GstPlaySink * playsink, gboolean raw) |
| { |
| GstPlayAudioChain *chain; |
| GstBin *bin; |
| gboolean have_volume; |
| GstPad *pad; |
| GstElement *head, *prev, *elem = NULL; |
| |
| chain = g_new0 (GstPlayAudioChain, 1); |
| chain->chain.playsink = playsink; |
| chain->chain.raw = raw; |
| |
| GST_DEBUG_OBJECT (playsink, "making audio chain %p", chain); |
| |
| if (playsink->audio_sink) { |
| GST_DEBUG_OBJECT (playsink, "trying configured audiosink %" GST_PTR_FORMAT, |
| playsink->audio_sink); |
| chain->sink = try_element (playsink, playsink->audio_sink, FALSE); |
| } else { |
| /* only try fallback if no specific sink was chosen */ |
| if (chain->sink == NULL) { |
| GST_DEBUG_OBJECT (playsink, "trying autoaudiosink"); |
| elem = gst_element_factory_make ("autoaudiosink", "audiosink"); |
| chain->sink = try_element (playsink, elem, TRUE); |
| } |
| if (chain->sink == NULL) { |
| /* if default sink from config.h is different then try it too */ |
| if (strcmp (DEFAULT_AUDIOSINK, "autoaudiosink")) { |
| GST_DEBUG_OBJECT (playsink, "trying " DEFAULT_AUDIOSINK); |
| elem = gst_element_factory_make (DEFAULT_AUDIOSINK, "audiosink"); |
| chain->sink = try_element (playsink, elem, TRUE); |
| } |
| } |
| if (chain->sink) |
| playsink->audio_sink = gst_object_ref (chain->sink); |
| } |
| if (chain->sink == NULL) |
| goto no_sinks; |
| |
| chain->chain.bin = gst_bin_new ("abin"); |
| bin = GST_BIN_CAST (chain->chain.bin); |
| gst_object_ref_sink (bin); |
| gst_bin_add (bin, chain->sink); |
| |
| head = chain->sink; |
| prev = NULL; |
| |
| /* add the audio filter first, so everything is working with post-filter |
| * samples */ |
| chain->filter = gst_play_sink_get_filter (playsink, |
| GST_PLAY_SINK_TYPE_AUDIO_RAW); |
| if (chain->filter) { |
| if (!raw) { |
| gst_object_unref (chain->filter); |
| chain->filter = NULL; |
| |
| if (playsink->flags & GST_PLAY_FLAG_FORCE_FILTERS) { |
| goto filter_with_nonraw; |
| } else { |
| GST_DEBUG_OBJECT (playsink, |
| "skipping audio filter since we're not raw"); |
| } |
| } else { |
| GST_DEBUG_OBJECT (playsink, "adding audio filter"); |
| chain->filter_conv = |
| gst_element_factory_make ("audioconvert", "filter-convert"); |
| if (!chain->filter_conv) { |
| post_missing_element_message (playsink, "audioconvert"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "audioconvert"), |
| ("audio playback and visualizations might not work")); |
| } else { |
| gst_bin_add (bin, chain->filter_conv); |
| head = prev = chain->filter_conv; |
| } |
| |
| gst_bin_add (bin, chain->filter); |
| /* Bin takes a new reference because we sinked any |
| * floating reference ourselves already */ |
| gst_object_unref (chain->filter); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->filter, NULL, |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) { |
| goto link_failed; |
| } |
| } else { |
| head = chain->filter; |
| } |
| prev = chain->filter; |
| } |
| } |
| |
| /* we have to add a queue when we need to decouple for the video sink in |
| * visualisations and for streamsynchronizer */ |
| GST_DEBUG_OBJECT (playsink, "adding audio queue"); |
| chain->queue = gst_element_factory_make ("queue", "aqueue"); |
| if (chain->queue == NULL) { |
| post_missing_element_message (playsink, "queue"); |
| GST_ELEMENT_WARNING (playsink, CORE, MISSING_PLUGIN, |
| (_("Missing element '%s' - check your GStreamer installation."), |
| "queue"), ("audio playback and visualizations might not work")); |
| } else { |
| g_object_set (chain->queue, "silent", TRUE, NULL); |
| gst_bin_add (bin, chain->queue); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->queue, "sink", |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } else { |
| head = chain->queue; |
| } |
| prev = chain->queue; |
| } |
| |
| /* find ts-offset element */ |
| gst_object_replace ((GstObject **) & chain->ts_offset, (GstObject *) |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "ts-offset", |
| G_TYPE_INT64)); |
| |
| /* check if the sink, or something within the sink, implements the |
| * streamvolume interface. If it does we don't need to add a volume element. */ |
| if (GST_IS_BIN (chain->sink)) |
| elem = |
| gst_bin_get_by_interface (GST_BIN_CAST (chain->sink), |
| GST_TYPE_STREAM_VOLUME); |
| else if (GST_IS_STREAM_VOLUME (chain->sink)) |
| elem = gst_object_ref (chain->sink); |
| else |
| elem = NULL; |
| chain->notify_volume_id = chain->notify_mute_id = 0; |
| if (elem) { |
| chain->volume = elem; |
| |
| chain->notify_volume_id = g_signal_connect (chain->volume, "notify::volume", |
| G_CALLBACK (notify_volume_cb), playsink); |
| |
| GST_DEBUG_OBJECT (playsink, "the sink has a volume property"); |
| have_volume = TRUE; |
| chain->sink_volume = TRUE; |
| chain->notify_mute_id = g_signal_connect (chain->volume, "notify::mute", |
| G_CALLBACK (notify_mute_cb), playsink); |
| /* use the sink to control the volume and mute */ |
| if (playsink->volume_changed) { |
| g_object_set (G_OBJECT (chain->volume), "volume", playsink->volume, NULL); |
| playsink->volume_changed = FALSE; |
| } |
| if (playsink->mute_changed) { |
| g_object_set (chain->volume, "mute", playsink->mute, NULL); |
| playsink->mute_changed = FALSE; |
| } |
| } else { |
| /* no volume, we need to add a volume element when we can */ |
| GST_DEBUG_OBJECT (playsink, "the sink has no volume property"); |
| have_volume = FALSE; |
| chain->sink_volume = FALSE; |
| } |
| |
| if (!(playsink->flags & GST_PLAY_FLAG_NATIVE_AUDIO) || (!have_volume |
| && (playsink->flags & GST_PLAY_FLAG_SOFT_VOLUME))) { |
| gboolean use_converters = !(playsink->flags & GST_PLAY_FLAG_NATIVE_AUDIO); |
| gboolean use_volume = |
| !have_volume && (playsink->flags & GST_PLAY_FLAG_SOFT_VOLUME); |
| GST_DEBUG_OBJECT (playsink, |
| "creating audioconvert with use-converters %d, use-volume %d", |
| use_converters, use_volume); |
| chain->conv = |
| g_object_new (GST_TYPE_PLAY_SINK_AUDIO_CONVERT, "name", "aconv", |
| "use-converters", use_converters, "use-volume", use_volume, NULL); |
| gst_bin_add (bin, chain->conv); |
| if (prev) { |
| if (!gst_element_link_pads_full (prev, "src", chain->conv, "sink", |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } else { |
| head = chain->conv; |
| } |
| prev = chain->conv; |
| |
| if (!have_volume && (playsink->flags & GST_PLAY_FLAG_SOFT_VOLUME)) { |
| GstPlaySinkAudioConvert *conv = |
| GST_PLAY_SINK_AUDIO_CONVERT_CAST (chain->conv); |
| |
| if (conv->volume) { |
| chain->volume = conv->volume; |
| have_volume = TRUE; |
| |
| chain->notify_volume_id = |
| g_signal_connect (chain->volume, "notify::volume", |
| G_CALLBACK (notify_volume_cb), playsink); |
| |
| /* volume also has the mute property */ |
| chain->notify_mute_id = g_signal_connect (chain->volume, "notify::mute", |
| G_CALLBACK (notify_mute_cb), playsink); |
| |
| /* configure with the latest volume and mute */ |
| g_object_set (G_OBJECT (chain->volume), "volume", playsink->volume, |
| NULL); |
| g_object_set (G_OBJECT (chain->volume), "mute", playsink->mute, NULL); |
| } |
| } |
| } |
| |
| if (prev) { |
| /* we only have to link to the previous element if we have something in |
| * front of the sink */ |
| GST_DEBUG_OBJECT (playsink, "linking to sink"); |
| if (!gst_element_link_pads_full (prev, "src", chain->sink, NULL, |
| GST_PAD_LINK_CHECK_TEMPLATE_CAPS)) |
| goto link_failed; |
| } |
| |
| /* post a warning if we have no way to configure the volume */ |
| if (!have_volume) { |
| GST_ELEMENT_WARNING (playsink, STREAM, NOT_IMPLEMENTED, |
| (_("No volume control found")), ("Volume/mute is not available")); |
| } |
| |
| /* and ghost the sinkpad of the headmost element */ |
| GST_DEBUG_OBJECT (playsink, "ghosting sink pad"); |
| pad = gst_element_get_static_pad (head, "sink"); |
| chain->sinkpad = gst_ghost_pad_new ("sink", pad); |
| |
| /* sending audio/video flushes break stream changes when the pipeline |
| * is paused and played again in 0.10 */ |
| #if 0 |
| gst_pad_set_event_function (chain->sinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_audio_sink_event)); |
| gst_pad_set_chain_function (chain->sinkpad, |
| GST_DEBUG_FUNCPTR (gst_play_sink_audio_sink_chain)); |
| #endif |
| |
| gst_object_unref (pad); |
| gst_element_add_pad (chain->chain.bin, chain->sinkpad); |
| |
| return chain; |
| |
| /* ERRORS */ |
| no_sinks: |
| { |
| if (!elem && !playsink->audio_sink) { |
| post_missing_element_message (playsink, "autoaudiosink"); |
| if (strcmp (DEFAULT_AUDIOSINK, "autoaudiosink")) { |
| post_missing_element_message (playsink, DEFAULT_AUDIOSINK); |
| GST_ELEMENT_ERROR (playsink, CORE, MISSING_PLUGIN, |
| (_("Both autoaudiosink and %s elements are missing."), |
| DEFAULT_AUDIOSINK), (NULL)); |
| } else { |
| GST_ELEMENT_ERROR (playsink, CORE, MISSING_PLUGIN, |
| (_("The autoaudiosink element is missing.")), (NULL)); |
| } |
| } else { |
| if (playsink->audio_sink) { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("Configured audiosink %s is not working."), |
| GST_ELEMENT_NAME (playsink->audio_sink)), (NULL)); |
| } else if (strcmp (DEFAULT_AUDIOSINK, "autoaudiosink")) { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("Both autoaudiosink and %s elements are not working."), |
| DEFAULT_AUDIOSINK), (NULL)); |
| } else { |
| GST_ELEMENT_ERROR (playsink, CORE, STATE_CHANGE, |
| (_("The autoaudiosink element is not working.")), (NULL)); |
| } |
| } |
| free_chain ((GstPlayChain *) chain); |
| return NULL; |
| } |
| link_failed: |
| { |
| GST_ELEMENT_ERROR (playsink, CORE, PAD, |
| (NULL), ("Failed to configure the audio sink.")); |
| goto cleanup; |
| } |
| filter_with_nonraw: |
| { |
| GST_ELEMENT_ERROR (playsink, CORE, NEGOTIATION, |
| (NULL), ("Cannot apply video-filter on non-raw stream")); |
| goto cleanup; |
| } |
| cleanup: |
| /* checking sink made it READY */ |
| gst_element_set_state (chain->sink, GST_STATE_NULL); |
| /* Remove chain from the bin to allow reuse later */ |
| gst_bin_remove (bin, chain->sink); |
| free_chain ((GstPlayChain *) chain); |
| return NULL; |
| } |
| |
| static gboolean |
| setup_audio_chain (GstPlaySink * playsink, gboolean raw) |
| { |
| GstElement *elem; |
| GstPlayAudioChain *chain; |
| GstStateChangeReturn ret; |
| GstPlaySinkAudioConvert *conv; |
| |
| chain = playsink->audiochain; |
| conv = GST_PLAY_SINK_AUDIO_CONVERT_CAST (chain->conv); |
| |
| /* if we have a filter, and raw-ness changed, we have to force a rebuild */ |
| if (chain->filter && chain->chain.raw != raw) |
| return FALSE; |
| |
| chain->chain.raw = raw; |
| |
| /* if the chain was active we don't do anything */ |
| if (GST_PLAY_CHAIN (chain)->activated) |
| return TRUE; |
| |
| /* try to set the sink element to READY again */ |
| ret = gst_element_set_state (chain->sink, GST_STATE_READY); |
| if (ret == GST_STATE_CHANGE_FAILURE) |
| return FALSE; |
| |
| /* find ts-offset element */ |
| gst_object_replace ((GstObject **) & chain->ts_offset, (GstObject *) |
| gst_play_sink_find_property_sinks (playsink, chain->sink, "ts-offset", |
| G_TYPE_INT64)); |
| |
| /* Disconnect signals */ |
| disconnect_audio_chain (chain, playsink); |
| |
| /* check if the sink, or something within the sink, implements the |
| * streamvolume interface. If it does we don't need to add a volume element. */ |
| if (GST_IS_BIN (chain->sink)) |
| elem = |
| gst_bin_get_by_interface (GST_BIN_CAST (chain->sink), |
| GST_TYPE_STREAM_VOLUME); |
| else if (GST_IS_STREAM_VOLUME (chain->sink)) |
| elem = gst_object_ref (chain->sink); |
| else |
| elem = NULL; |
| if (elem) { |
| chain->volume = elem; |
| |
| if (playsink->volume_changed) { |
| GST_DEBUG_OBJECT (playsink, "the sink has a volume property, setting %f", |
| playsink->volume); |
| /* use the sink to control the volume */ |
| g_object_set (G_OBJECT (chain->volume), "volume", playsink->volume, NULL); |
| playsink->volume_changed = FALSE; |
| } |
| |
| chain->notify_volume_id = g_signal_connect (chain->volume, "notify::volume", |
| G_CALLBACK (notify_volume_cb), playsink); |
| chain->notify_mute_id = g_signal_connect (chain->volume, "notify::mute", |
| G_CALLBACK (notify_mute_cb), playsink); |
| g_object_set (chain->volume, "mute", playsink->mute, NULL); |
| playsink->mute_changed = FALSE; |
| |
| g_object_set (chain->conv, "use-volume", FALSE, NULL); |
| } else if (conv) { |
| /* no volume, we need to add a volume element when we can */ |
| g_object_set (chain->conv, "use-volume", |
| ! !(playsink->flags & GST_PLAY_FLAG_SOFT_VOLUME), NULL); |
| GST_DEBUG_OBJECT (playsink, "the sink has no volume property"); |
| |
| if (conv->volume && (playsink->flags & GST_PLAY_FLAG_SOFT_VOLUME)) { |
| chain->volume = conv->volume; |
| |
| chain->notify_volume_id = |
| g_signal_connect (chain->volume, "notify::volume", |
| G_CALLBACK (notify_volume_cb), playsink); |
| |
| chain->notify_mute_id = g_signal_connect (chain->volume, "notify::mute", |
| G_CALLBACK (notify_mute_cb), playsink); |
| |
| /* configure with the latest volume and mute */ |
| g_object_set (G_OBJECT (chain->volume), "volume", playsink->volume, NULL); |