/* GStreamer unit tests for playbin complex stream support * * Copyright (C) 2006 Tim-Philipp Müller * Copyright (C) 2010 Sebastian Dröge * * 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 #endif #include #include #include #include #include #ifndef GST_DISABLE_REGISTRY #define NBUFFERS 100 static GType gst_caps_src_get_type (void); static GType gst_codec_demuxer_get_type (void); static GType gst_codec_sink_get_type (void); static GType gst_audio_codec_sink_get_type (void); static GType gst_video_codec_sink_get_type (void); static GType gst_video_decoder1_get_type (void); static GType gst_video_decoder2_get_type (void); static GType gst_video_sink1_get_type (void); static GType gst_video_sink2_get_type (void); typedef struct _GstCapsSrc GstCapsSrc; typedef GstPushSrcClass GstCapsSrcClass; struct _GstCapsSrc { GstPushSrc parent; GstCaps *caps; gchar *uri; gint nbuffers; }; static GstURIType gst_caps_src_uri_get_type (GType type) { return GST_URI_SRC; } static const gchar *const * gst_caps_src_uri_get_protocols (GType type) { static const gchar *protocols[] = { "caps", NULL }; return protocols; } static gchar * gst_caps_src_uri_get_uri (GstURIHandler * handler) { GstCapsSrc *src = (GstCapsSrc *) handler; return g_strdup (src->uri); } static gboolean gst_caps_src_uri_set_uri (GstURIHandler * handler, const gchar * uri, GError ** error) { GstCapsSrc *src = (GstCapsSrc *) handler; if (uri == NULL || !g_str_has_prefix (uri, "caps:")) return FALSE; g_free (src->uri); src->uri = g_strdup (uri); if (src->caps) gst_caps_unref (src->caps); src->caps = NULL; return TRUE; } static void gst_caps_src_uri_handler_init (gpointer g_iface, gpointer iface_data) { GstURIHandlerInterface *iface = (GstURIHandlerInterface *) g_iface; iface->get_type = gst_caps_src_uri_get_type; iface->get_protocols = gst_caps_src_uri_get_protocols; iface->get_uri = gst_caps_src_uri_get_uri; iface->set_uri = gst_caps_src_uri_set_uri; } G_DEFINE_TYPE_WITH_CODE (GstCapsSrc, gst_caps_src, GST_TYPE_PUSH_SRC, G_IMPLEMENT_INTERFACE (GST_TYPE_URI_HANDLER, gst_caps_src_uri_handler_init)); static void gst_caps_src_finalize (GObject * object) { GstCapsSrc *src = (GstCapsSrc *) object; if (src->caps) gst_caps_unref (src->caps); src->caps = NULL; g_free (src->uri); G_OBJECT_CLASS (gst_caps_src_parent_class)->finalize (object); } static GstFlowReturn gst_caps_src_create (GstPushSrc * psrc, GstBuffer ** p_buf) { GstCapsSrc *src = (GstCapsSrc *) psrc; GstBuffer *buf; if (src->nbuffers >= NBUFFERS) { return GST_FLOW_EOS; } if (!src->caps) { if (!src->uri) { return GST_FLOW_ERROR; } src->caps = gst_caps_from_string (src->uri + sizeof ("caps")); if (!src->caps) { return GST_FLOW_ERROR; } } if (src->nbuffers == 0) { gst_pad_push_event (GST_BASE_SRC_PAD (psrc), gst_event_new_stream_start ("test")); gst_pad_set_caps (GST_BASE_SRC_PAD (psrc), src->caps); } buf = gst_buffer_new_wrapped (g_malloc0 (4), 4); GST_BUFFER_TIMESTAMP (buf) = gst_util_uint64_scale (src->nbuffers, GST_SECOND, 25); src->nbuffers++; *p_buf = buf; return GST_FLOW_OK; } static void gst_caps_src_class_init (GstCapsSrcClass * klass) { GObjectClass *gobject_class = (GObjectClass *) klass; GstElementClass *element_class = GST_ELEMENT_CLASS (klass); GstPushSrcClass *pushsrc_class = (GstPushSrcClass *) klass; static GstStaticPadTemplate src_templ = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS_ANY); gst_element_class_add_static_pad_template (element_class, &src_templ); gst_element_class_set_metadata (element_class, "CapsSource", "Source/Generic", "yep", "me"); gobject_class->finalize = gst_caps_src_finalize; pushsrc_class->create = gst_caps_src_create; } static void gst_caps_src_init (GstCapsSrc * src) { gst_base_src_set_format (GST_BASE_SRC (src), GST_FORMAT_TIME); } typedef struct _GstCodecSink GstCodecSink; typedef GstBaseSinkClass GstCodecSinkClass; struct _GstCodecSink { GstBaseSink parent; gboolean audio; gboolean raw; gint n_raw, n_compressed; }; G_DEFINE_TYPE (GstCodecSink, gst_codec_sink, GST_TYPE_BASE_SINK); static gboolean gst_codec_sink_start (GstBaseSink * bsink) { GstCodecSink *sink = (GstCodecSink *) bsink; sink->n_raw = 0; sink->n_compressed = 0; return TRUE; } static GstFlowReturn gst_codec_sink_render (GstBaseSink * bsink, GstBuffer * buffer) { GstCodecSink *sink = (GstCodecSink *) bsink; if (sink->raw) sink->n_raw++; else sink->n_compressed++; return GST_FLOW_OK; } static void gst_codec_sink_class_init (GstCodecSinkClass * klass) { GstBaseSinkClass *basesink_class = (GstBaseSinkClass *) klass; basesink_class->start = gst_codec_sink_start; basesink_class->render = gst_codec_sink_render; } static void gst_codec_sink_init (GstCodecSink * sink) { } typedef GstCodecSink GstAudioCodecSink; typedef GstCodecSinkClass GstAudioCodecSinkClass; G_DEFINE_TYPE_WITH_CODE (GstAudioCodecSink, gst_audio_codec_sink, gst_codec_sink_get_type (), G_IMPLEMENT_INTERFACE (GST_TYPE_STREAM_VOLUME, NULL)); static void gst_audio_codec_sink_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec) { switch (prop_id) { case 1: case 2: break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gst_audio_codec_sink_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec) { switch (prop_id) { case 1: g_value_set_double (value, 1.0); break; case 2: g_value_set_boolean (value, FALSE); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean gst_audio_codec_sink_set_caps (GstBaseSink * bsink, GstCaps * caps) { GstAudioCodecSink *sink = (GstAudioCodecSink *) bsink; GstStructure *s; s = gst_caps_get_structure (caps, 0); if (gst_structure_has_name (s, "audio/x-raw")) { sink->raw = TRUE; } else if (gst_structure_has_name (s, "audio/x-compressed")) { sink->raw = FALSE; } else { fail_unless (gst_structure_has_name (s, "audio/x-raw") || gst_structure_has_name (s, "audio/x-compressed")); return FALSE; } return TRUE; } static void gst_audio_codec_sink_class_init (GstAudioCodecSinkClass * klass) { GObjectClass *gobject_class = (GObjectClass *) klass; GstElementClass *element_class = GST_ELEMENT_CLASS (klass); GstBaseSinkClass *basesink_class = (GstBaseSinkClass *) klass; static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("audio/x-raw; audio/x-compressed") ); gobject_class->set_property = gst_audio_codec_sink_set_property; gobject_class->get_property = gst_audio_codec_sink_get_property; g_object_class_install_property (gobject_class, 1, g_param_spec_double ("volume", "Volume", "Linear volume of this stream, 1.0=100%", 0.0, 10.0, 1.0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, 2, g_param_spec_boolean ("mute", "Mute", "Mute", FALSE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_set_metadata (element_class, "AudioCodecSink", "Sink/Audio", "yep", "me"); basesink_class->set_caps = gst_audio_codec_sink_set_caps; } static void gst_audio_codec_sink_init (GstAudioCodecSink * sink) { sink->audio = TRUE; } typedef GstCodecSink GstVideoCodecSink; typedef GstCodecSinkClass GstVideoCodecSinkClass; G_DEFINE_TYPE (GstVideoCodecSink, gst_video_codec_sink, gst_codec_sink_get_type ()); static gboolean gst_video_codec_sink_set_caps (GstBaseSink * bsink, GstCaps * caps) { GstVideoCodecSink *sink = (GstVideoCodecSink *) bsink; GstStructure *s; s = gst_caps_get_structure (caps, 0); if (gst_structure_has_name (s, "video/x-raw")) { sink->raw = TRUE; } else if (gst_structure_has_name (s, "video/x-compressed")) { sink->raw = FALSE; } else { fail_unless (gst_structure_has_name (s, "video/x-raw") || gst_structure_has_name (s, "video/x-compressed")); return FALSE; } return TRUE; } static void gst_video_codec_sink_class_init (GstVideoCodecSinkClass * klass) { GstBaseSinkClass *basesink_class = (GstBaseSinkClass *) klass; static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-raw; video/x-compressed") ); GstElementClass *element_class = GST_ELEMENT_CLASS (klass); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_set_metadata (element_class, "VideoCodecSink", "Sink/Video", "yep", "me"); basesink_class->set_caps = gst_video_codec_sink_set_caps; } static void gst_video_codec_sink_init (GstVideoCodecSink * sink) { sink->audio = FALSE; } typedef struct _GstCodecDemuxer GstCodecDemuxer; typedef GstElementClass GstCodecDemuxerClass; struct _GstCodecDemuxer { GstElement parent; GstPad *sinkpad; GstPad *srcpad0, *srcpad1; GstEvent *newseg_event; }; #define STREAM_TYPES "{ " \ "none, " \ "raw-audio, " \ "compressed-audio, " \ "raw-video, " \ "compressed-video " \ "}" static GstStaticPadTemplate cd_sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("application/x-container," " stream0 = (string)" STREAM_TYPES " ," " stream1 = (string)" STREAM_TYPES) ); static GstStaticPadTemplate cd_src_templ = GST_STATIC_PAD_TEMPLATE ("src_%u", GST_PAD_SRC, GST_PAD_SOMETIMES, GST_STATIC_CAPS ("audio/x-raw; audio/x-compressed; " "video/x-raw; video/x-compressed") ); G_DEFINE_TYPE (GstCodecDemuxer, gst_codec_demuxer, GST_TYPE_ELEMENT); static void gst_codec_demuxer_finalize (GObject * object) { GstCodecDemuxer *demux = (GstCodecDemuxer *) object; if (demux->newseg_event) gst_event_unref (demux->newseg_event); demux->newseg_event = NULL; G_OBJECT_CLASS (gst_codec_demuxer_parent_class)->finalize (object); } static void gst_codec_demuxer_class_init (GstCodecDemuxerClass * klass) { GObjectClass *gobject_class = (GObjectClass *) klass; GstElementClass *element_class = GST_ELEMENT_CLASS (klass); gobject_class->finalize = gst_codec_demuxer_finalize; gst_element_class_add_static_pad_template (element_class, &cd_sink_templ); gst_element_class_add_static_pad_template (element_class, &cd_src_templ); gst_element_class_set_metadata (element_class, "CodecDemuxer", "Codec/Demuxer", "yep", "me"); } static GstFlowReturn gst_codec_demuxer_chain (GstPad * pad, GstObject * parent, GstBuffer * buf) { GstCodecDemuxer *demux = (GstCodecDemuxer *) parent; GstFlowReturn ret0 = GST_FLOW_OK, ret1 = GST_FLOW_OK; if (demux->newseg_event) { if (demux->srcpad0) gst_pad_push_event (demux->srcpad0, gst_event_ref (demux->newseg_event)); if (demux->srcpad1) gst_pad_push_event (demux->srcpad1, gst_event_ref (demux->newseg_event)); gst_event_unref (demux->newseg_event); demux->newseg_event = NULL; } if (demux->srcpad0) { GstBuffer *outbuf = gst_buffer_copy (buf); ret0 = gst_pad_push (demux->srcpad0, outbuf); } if (demux->srcpad1) { GstBuffer *outbuf = gst_buffer_copy (buf); ret1 = gst_pad_push (demux->srcpad1, outbuf); } gst_buffer_unref (buf); if (ret0 == GST_FLOW_NOT_LINKED && ret1 == GST_FLOW_NOT_LINKED) return GST_FLOW_NOT_LINKED; if (ret0 == GST_FLOW_OK && ret1 == GST_FLOW_OK) return GST_FLOW_OK; return MIN (ret0, ret1); } static void gst_codec_demuxer_setup_pad (GstCodecDemuxer * demux, GstPad ** pad, const gchar * streaminfo) { if (g_str_equal (streaminfo, "none")) { if (*pad) { gst_pad_set_active (*pad, FALSE); gst_element_remove_pad (GST_ELEMENT (demux), *pad); *pad = NULL; } } else { GstCaps *caps; if (!*pad) { GstPadTemplate *templ; templ = gst_element_class_get_pad_template (GST_ELEMENT_GET_CLASS (demux), "src_%u"); if (pad == &demux->srcpad0) *pad = gst_pad_new_from_template (templ, "src_0"); else *pad = gst_pad_new_from_template (templ, "src_1"); gst_pad_set_active (*pad, TRUE); gst_pad_use_fixed_caps (*pad); gst_element_add_pad (GST_ELEMENT (demux), *pad); } if (g_str_equal (streaminfo, "raw-video")) { caps = gst_caps_new_simple ("video/x-raw", "format", G_TYPE_STRING, "I420", "width", G_TYPE_INT, 320, "height", G_TYPE_INT, 240, "framerate", GST_TYPE_FRACTION, 25, 1, "pixel-aspect-ratio", GST_TYPE_FRACTION, 1, 1, NULL); } else if (g_str_equal (streaminfo, "compressed-video")) { caps = gst_caps_new_empty_simple ("video/x-compressed"); } else if (g_str_equal (streaminfo, "raw-audio")) { caps = gst_caps_new_simple ("audio/x-raw", "format", G_TYPE_STRING, GST_AUDIO_NE (S16), "layout", G_TYPE_STRING, "interleaved", "rate", G_TYPE_INT, 48000, "channels", G_TYPE_INT, 2, NULL); } else { caps = gst_caps_new_empty_simple ("audio/x-compressed"); } gst_pad_push_event (*pad, gst_event_new_stream_start ("test")); gst_pad_set_caps (*pad, caps); gst_caps_unref (caps); } } static gboolean gst_codec_demuxer_setcaps (GstPad * pad, GstObject * parent, GstCaps * caps) { GstCodecDemuxer *demux = (GstCodecDemuxer *) parent; GstStructure *s; const gchar *streaminfo; s = gst_caps_get_structure (caps, 0); streaminfo = gst_structure_get_string (s, "stream0"); gst_codec_demuxer_setup_pad (demux, &demux->srcpad0, streaminfo); streaminfo = gst_structure_get_string (s, "stream1"); gst_codec_demuxer_setup_pad (demux, &demux->srcpad1, streaminfo); return TRUE; } static gboolean gst_codec_demuxer_event (GstPad * pad, GstObject * parent, GstEvent * event) { GstCodecDemuxer *demux = (GstCodecDemuxer *) parent; gboolean ret = TRUE; /* The single newsegment event is pushed when the pads are created */ switch (GST_EVENT_TYPE (event)) { case GST_EVENT_CAPS: { GstCaps *caps; gst_event_parse_caps (event, &caps); ret = gst_codec_demuxer_setcaps (pad, parent, caps); break; } case GST_EVENT_SEGMENT: gst_event_replace (&demux->newseg_event, event); break; default: if (demux->srcpad0) ret = ret && gst_pad_push_event (demux->srcpad0, gst_event_ref (event)); if (demux->srcpad1) ret = ret && gst_pad_push_event (demux->srcpad1, gst_event_ref (event)); break; } gst_event_unref (event); return ret; } static void gst_codec_demuxer_init (GstCodecDemuxer * demux) { demux->sinkpad = gst_pad_new_from_static_template (&cd_sink_templ, "sink"); gst_pad_set_chain_function (demux->sinkpad, GST_DEBUG_FUNCPTR (gst_codec_demuxer_chain)); gst_pad_set_event_function (demux->sinkpad, GST_DEBUG_FUNCPTR (gst_codec_demuxer_event)); gst_element_add_pad (GST_ELEMENT (demux), demux->sinkpad); } typedef struct _GstVideoDecoder1 GstVideoDecoder1; typedef GstVideoDecoderClass GstVideoDecoder1Class; typedef struct _GstVideoDecoder2 GstVideoDecoder2; typedef GstVideoDecoderClass GstVideoDecoder2Class; typedef struct _GstVideoSink1 GstVideoSink1; typedef GstBaseSinkClass GstVideoSink1Class; typedef struct _GstVideoSink2 GstVideoSink2; typedef GstBaseSinkClass GstVideoSink2Class; struct _GstVideoDecoder1 { GstVideoDecoder parent; }; struct _GstVideoDecoder2 { GstVideoDecoder parent; }; struct _GstVideoSink1 { GstBaseSink parent; }; struct _GstVideoSink2 { GstBaseSink parent; }; #define GST_CAPS_FEATURE_MEMORY_FAKE "memory:FakeMem" G_DEFINE_TYPE (GstVideoDecoder1, gst_video_decoder1, GST_TYPE_VIDEO_DECODER); G_DEFINE_TYPE (GstVideoDecoder2, gst_video_decoder2, GST_TYPE_VIDEO_DECODER); G_DEFINE_TYPE (GstVideoSink1, gst_video_sink1, GST_TYPE_BASE_SINK); G_DEFINE_TYPE (GstVideoSink2, gst_video_sink2, GST_TYPE_BASE_SINK); static void gst_video_sink1_class_init (GstVideoSink1Class * klass) { GstElementClass *element_class = GST_ELEMENT_CLASS (klass); static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE_WITH_FEATURES (GST_CAPS_FEATURE_MEMORY_FAKE, GST_VIDEO_FORMATS_ALL) ";" GST_VIDEO_CAPS_MAKE (GST_VIDEO_FORMATS_ALL)) ); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_set_static_metadata (element_class, "Fake Video Sink1", "Sink/Video", "fake sink1", "Sreerenj Balachandran "); } static void gst_video_sink1_init (GstVideoSink1 * sink) { } static void gst_video_sink2_class_init (GstVideoSink2Class * klass) { GstElementClass *element_class = GST_ELEMENT_CLASS (klass); static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE (GST_VIDEO_FORMATS_ALL))); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_set_static_metadata (element_class, "Fake Video Sink2", "Sink/Video", "fake sink2", "Sreerenj Balachandran "); } static void gst_video_sink2_init (GstVideoSink2 * sink) { } static GstFlowReturn fake_theora_dec_handle_frame (GstVideoDecoder * bdec, GstVideoCodecFrame * frame) { /* the width and height are hard-coded for ../../files/theora-vorbis.ogg */ gst_video_codec_state_unref (gst_video_decoder_set_output_state (bdec, GST_VIDEO_FORMAT_NV12, 320, 240, NULL)); gst_video_decoder_allocate_output_frame (bdec, frame); return gst_video_decoder_finish_frame (bdec, frame); } static void gst_video_decoder1_class_init (GstVideoDecoder1Class * klass) { GstElementClass *element_class = GST_ELEMENT_CLASS (klass); GstVideoDecoderClass *vd_class = GST_VIDEO_DECODER_CLASS (klass); static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-theora")); static GstStaticPadTemplate src_templ = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE_WITH_FEATURES (GST_CAPS_FEATURE_MEMORY_FAKE, GST_VIDEO_FORMATS_ALL) ";" GST_VIDEO_CAPS_MAKE (GST_VIDEO_FORMATS_ALL))); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_add_static_pad_template (element_class, &src_templ); gst_element_class_set_static_metadata (element_class, "Fake theora video decoder1", "Codec/Decoder/Video", "decode theora stream", "Sreerenj Balachandran "); vd_class->handle_frame = fake_theora_dec_handle_frame; } static void gst_video_decoder1_init (GstVideoDecoder1 * dec1) { } static void gst_video_decoder2_class_init (GstVideoDecoder2Class * klass) { GstElementClass *element_class = GST_ELEMENT_CLASS (klass); GstVideoDecoderClass *vd_class = GST_VIDEO_DECODER_CLASS (klass); static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-theora")); static GstStaticPadTemplate src_templ = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE (GST_VIDEO_FORMATS_ALL))); gst_element_class_add_static_pad_template (element_class, &sink_templ); gst_element_class_add_static_pad_template (element_class, &src_templ); gst_element_class_set_static_metadata (element_class, "Fake theora video decoder2", "Codec/Decoder/Video", "decode theora stream", "Sreerenj Balachandran "); vd_class->handle_frame = fake_theora_dec_handle_frame; } static void gst_video_decoder2_init (GstVideoDecoder2 * dec2) { } /**** * Start of the tests ***/ static void pipeline_deep_notify_caps_cb (GstObject * pipeline, GstObject * object, GParamSpec * pspec, GstElement ** dec) { GstObject *pad_parent; if (!GST_IS_PAD (object)) return; pad_parent = gst_object_get_parent (object); if (GST_IS_ELEMENT (pad_parent) && *dec == NULL) { GstElementFactory *factory; GstElement *element; const gchar *klass; element = GST_ELEMENT_CAST (pad_parent); if ((factory = gst_element_get_factory (element)) && (klass = gst_element_factory_get_metadata (factory, GST_ELEMENT_METADATA_KLASS)) && strstr (klass, "Video") && strstr (klass, "Decoder")) { *dec = gst_object_ref (element); } } if (pad_parent) gst_object_unref (pad_parent); } static GstElement * create_playbin (const gchar * uri, gboolean set_sink) { GstElement *playbin, *sink; playbin = gst_element_factory_make ("playbin", "playbin"); fail_unless (playbin != NULL, "Failed to create playbin element"); if (set_sink) { sink = gst_element_factory_make ("videocodecsink", NULL); fail_unless (sink != NULL, "Failed to create videocodecsink element"); /* Set sync to FALSE to prevent buffers from being dropped because * they're too late */ g_object_set (sink, "sync", FALSE, NULL); g_object_set (playbin, "video-sink", sink, NULL); sink = gst_element_factory_make ("audiocodecsink", NULL); fail_unless (sink != NULL, "Failed to create audiocodecsink"); /* Set sync to FALSE to prevent buffers from being dropped because * they're too late */ g_object_set (sink, "sync", FALSE, NULL); g_object_set (playbin, "audio-sink", sink, NULL); } g_object_set (playbin, "uri", uri, NULL); return playbin; } GST_START_TEST (test_autoplug_decoder_sink_combination) { GstElement *playbin; GstElement *decoder = NULL; GstElement *sink, *asink; gchar *path, *uri; fail_unless (gst_element_register (NULL, "faketheoradec1", GST_RANK_PRIMARY + 1, gst_video_decoder1_get_type ())); fail_unless (gst_element_register (NULL, "faketheoradec2", GST_RANK_PRIMARY + 1, gst_video_decoder2_get_type ())); fail_unless (gst_element_register (NULL, "fakevideosink1", GST_RANK_PRIMARY + 1, gst_video_sink1_get_type ())); fail_unless (gst_element_register (NULL, "fakevideosink2", GST_RANK_PRIMARY + 1, gst_video_sink2_get_type ())); path = g_build_filename (GST_TEST_FILES_PATH, "theora-vorbis.ogg", NULL); uri = gst_filename_to_uri (path, NULL); g_free (path); playbin = create_playbin (uri, FALSE); g_free (uri); asink = gst_element_factory_make ("fakesink", NULL); g_object_set (asink, "sync", TRUE, NULL); g_object_set (playbin, "audio-sink", asink, NULL); g_signal_connect (playbin, "deep-notify::caps", G_CALLBACK (pipeline_deep_notify_caps_cb), &decoder); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); fail_unless_equals_int (gst_element_get_state (playbin, NULL, NULL, -1), GST_STATE_CHANGE_SUCCESS); /* there shouldn't be any errors */ fail_if (gst_bus_poll (GST_ELEMENT_BUS (playbin), GST_MESSAGE_ERROR, 0) != NULL); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_sink1_get_type ()); gst_object_unref (sink); } fail_unless (decoder != NULL); { fail_unless (G_TYPE_FROM_INSTANCE (decoder) == gst_video_decoder1_get_type ()); gst_object_unref (decoder); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_single_video_stream_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:video/x-raw, " "format=(string)I420, " "width=(int)320, " "height=(int)240, " "framerate=(fraction)0/1, " "pixel-aspect-ratio=(fraction)1/1", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_single_video_stream_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:video/x-compressed", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_single_video_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-video, " "stream1=(string)none", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_single_video_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-video, " "stream1=(string)none", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_single_audio_stream_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:audio/x-raw," " format=(string)" GST_AUDIO_NE (S16) ", " "layout=(string)interleaved, " " rate=(int)48000, " " channels=(int)2", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_single_audio_stream_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:audio/x-compressed", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_single_audio_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)none", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_single_audio_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-audio, " "stream1=(string)none", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_audio_video_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)raw-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_audio_video_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-audio, " "stream1=(string)compressed-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->raw == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, NBUFFERS); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_compressed_video_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-video, " "stream1=(string)compressed-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw + csink->n_compressed, NBUFFERS); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_compressed_audio_stream_demuxer_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)compressed-audio", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw + csink->n_compressed, NBUFFERS); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_raw_audio_stream_adder_manual_sink) { GstMessage *msg; GstElement *adder; GstElement *playbin_combiner; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean adder_used = FALSE; gboolean done = FALSE; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)raw-audio", TRUE); /* before playback starts, and with no custom combiner, these should all be NULL */ g_object_get (G_OBJECT (playbin), "audio-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == NULL); g_object_get (G_OBJECT (playbin), "text-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == NULL); g_object_get (G_OBJECT (playbin), "video-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == NULL); /* set audio combiner */ adder = gst_element_factory_make ("adder", NULL); fail_unless (adder != NULL); g_object_set (G_OBJECT (playbin), "audio-stream-combiner", adder, NULL); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); /* audio combiner should still be there */ g_object_get (G_OBJECT (playbin), "audio-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == adder); gst_object_unref (playbin_combiner); /* text and video combiners should still be NULL */ g_object_get (G_OBJECT (playbin), "text-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == NULL); g_object_get (G_OBJECT (playbin), "video-stream-combiner", &playbin_combiner, NULL); fail_unless (playbin_combiner == NULL); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; case GST_MESSAGE_STATE_CHANGED: if (GST_MESSAGE_SRC (msg) == GST_OBJECT (adder)) { GstState state; gst_message_parse_state_changed (msg, &state, NULL, NULL); if (state == GST_STATE_PAUSED) adder_used = TRUE; } default: break; } gst_message_unref (msg); } gst_object_unref (bus); fail_unless (adder_used); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, NBUFFERS); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; #if 0 typedef struct { GstElement *playbin; GstElement *sink; } SwitchBufferProbeCtx; static gboolean switch_video_buffer_probe (GstPad * pad, GstBuffer * buffer, SwitchBufferProbeCtx * ctx) { GstElement *playbin = ctx->playbin; GstVideoCodecSink *sink = (GstVideoCodecSink *) ctx->sink; if (sink->n_raw + sink->n_compressed == NBUFFERS / 2) { gint cur_video; g_object_get (G_OBJECT (playbin), "current-video", &cur_video, NULL); cur_video = (cur_video == 0) ? 1 : 0; g_object_set (G_OBJECT (playbin), "current-video", cur_video, NULL); } return TRUE; } GST_START_TEST (test_raw_compressed_video_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-video, " "stream1=(string)compressed-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_video_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->n_raw > 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_raw_video_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-video, " "stream1=(string)raw-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_video_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->n_raw > 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_raw_video_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-video, " "stream1=(string)raw-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_video_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless (csink->n_raw > 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_compressed_video_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-video, " "stream1=(string)compressed-video", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_video_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; static gboolean switch_audio_buffer_probe (GstPad * pad, GstBuffer * buffer, SwitchBufferProbeCtx * ctx) { GstElement *playbin = ctx->playbin; GstAudioCodecSink *sink = (GstAudioCodecSink *) ctx->sink; if (sink->n_raw + sink->n_compressed == NBUFFERS / 3) { gint cur_audio; g_object_get (G_OBJECT (playbin), "current-audio", &cur_audio, NULL); cur_audio = (cur_audio == 0) ? 1 : 0; g_object_set (G_OBJECT (playbin), "current-audio", cur_audio, NULL); } return TRUE; } GST_START_TEST (test_raw_compressed_audio_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)compressed-audio", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_audio_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->n_raw > 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_raw_audio_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-audio, " "stream1=(string)raw-audio", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_audio_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->n_raw > 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_raw_raw_audio_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)raw-audio, " "stream1=(string)raw-audio", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_audio_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless (csink->n_raw > 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; GST_START_TEST (test_compressed_compressed_audio_stream_demuxer_switch_manual_sink) { GstMessage *msg; GstElement *playbin; GstElement *sink; GstBus *bus; gboolean done = FALSE; GstPad *pad; SwitchBufferProbeCtx switch_ctx; fail_unless (gst_element_register (NULL, "capssrc", GST_RANK_PRIMARY, gst_caps_src_get_type ())); fail_unless (gst_element_register (NULL, "codecdemuxer", GST_RANK_PRIMARY + 100, gst_codec_demuxer_get_type ())); fail_unless (gst_element_register (NULL, "audiocodecsink", GST_RANK_PRIMARY + 100, gst_audio_codec_sink_get_type ())); fail_unless (gst_element_register (NULL, "videocodecsink", GST_RANK_PRIMARY + 100, gst_video_codec_sink_get_type ())); playbin = create_playbin ("caps:application/x-container, " "stream0=(string)compressed-audio, " "stream1=(string)compressed-audio", TRUE); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_READY), GST_STATE_CHANGE_SUCCESS); g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); pad = gst_element_get_static_pad (sink, "sink"); fail_unless (pad != NULL); switch_ctx.playbin = playbin; switch_ctx.sink = sink; gst_pad_add_buffer_probe (pad, G_CALLBACK (switch_audio_buffer_probe), &switch_ctx); gst_object_unref (pad); gst_object_unref (sink); fail_unless_equals_int (gst_element_set_state (playbin, GST_STATE_PLAYING), GST_STATE_CHANGE_ASYNC); bus = gst_element_get_bus (playbin); while (!done) { msg = gst_bus_poll (bus, GST_MESSAGE_ANY, -1); switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_EOS: done = TRUE; break; case GST_MESSAGE_ERROR: fail_if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR); break; default: break; } gst_message_unref (msg); } gst_object_unref (bus); g_object_get (G_OBJECT (playbin), "video-sink", &sink, NULL); fail_unless (sink != NULL); { GstVideoCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_video_codec_sink_get_type ()); csink = (GstVideoCodecSink *) sink; fail_unless (csink->audio == FALSE); fail_unless_equals_int (csink->n_raw, 0); fail_unless_equals_int (csink->n_compressed, 0); gst_object_unref (sink); } g_object_get (G_OBJECT (playbin), "audio-sink", &sink, NULL); fail_unless (sink != NULL); { GstAudioCodecSink *csink; fail_unless (G_TYPE_FROM_INSTANCE (sink) == gst_audio_codec_sink_get_type ()); csink = (GstAudioCodecSink *) sink; fail_unless (csink->audio == TRUE); fail_unless_equals_int (csink->n_raw, 0); fail_unless (csink->n_compressed > 0); gst_object_unref (sink); } gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); } GST_END_TEST; #endif #endif static Suite * playbin_complex_suite (void) { Suite *s = suite_create ("playbin_complex"); TCase *tc_chain = tcase_create ("general"); suite_add_tcase (s, tc_chain); #ifndef GST_DISABLE_REGISTRY tcase_add_test (tc_chain, test_raw_single_video_stream_manual_sink); tcase_add_test (tc_chain, test_raw_single_audio_stream_manual_sink); tcase_add_test (tc_chain, test_compressed_single_video_stream_manual_sink); tcase_add_test (tc_chain, test_compressed_single_audio_stream_manual_sink); tcase_add_test (tc_chain, test_raw_single_video_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_raw_single_audio_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_compressed_single_video_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_compressed_single_audio_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_raw_audio_video_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_compressed_audio_video_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_raw_compressed_audio_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_raw_compressed_video_stream_demuxer_manual_sink); tcase_add_test (tc_chain, test_raw_raw_audio_stream_adder_manual_sink); tcase_add_test (tc_chain, test_autoplug_decoder_sink_combination); /* These tests need something like the stream-activate event * and are racy otherwise */ #if 0 tcase_add_test (tc_chain, test_raw_raw_audio_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_raw_compressed_audio_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_compressed_raw_audio_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_compressed_compressed_audio_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_raw_raw_video_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_raw_compressed_video_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_compressed_raw_video_stream_demuxer_switch_manual_sink); tcase_add_test (tc_chain, test_compressed_compressed_video_stream_demuxer_switch_manual_sink); #endif #endif return s; } GST_CHECK_MAIN (playbin_complex);