/* A generic test engine for elements based upon GstAdaptiveDemux * * Copyright (c) <2015> YouView TV Ltd * * 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. */ #include #include #include "adaptive_demux_engine.h" typedef struct _GstAdaptiveDemuxTestEnginePrivate { GstAdaptiveDemuxTestEngine engine; const GstAdaptiveDemuxTestCallbacks *callbacks; gpointer user_data; guint clock_update_id; } GstAdaptiveDemuxTestEnginePrivate; #define GST_TEST_GET_LOCK(d) (&(((GstAdaptiveDemuxTestEnginePrivate*)(d))->engine.lock)) #define GST_TEST_LOCK(d) g_mutex_lock (GST_TEST_GET_LOCK (d)) #define GST_TEST_UNLOCK(d) g_mutex_unlock (GST_TEST_GET_LOCK (d)) static void adaptive_demux_engine_stream_state_finalize (gpointer data) { GstAdaptiveDemuxTestOutputStream *stream = (GstAdaptiveDemuxTestOutputStream *) data; if (stream->appsink) gst_object_unref (stream->appsink); if (stream->pad) gst_object_unref (stream->pad); if (stream->internal_pad) gst_object_unref (stream->internal_pad); g_slice_free (GstAdaptiveDemuxTestOutputStream, stream); } /* get the testOutput entry in testData corresponding to the given AppSink */ static GstAdaptiveDemuxTestOutputStream * getTestOutputDataByAppsink (GstAdaptiveDemuxTestEnginePrivate * priv, GstAppSink * appsink) { guint i; for (i = 0; i < priv->engine.output_streams->len; ++i) { GstAdaptiveDemuxTestOutputStream *state; state = g_ptr_array_index (priv->engine.output_streams, i); if (state->appsink == appsink) { return state; } } ck_abort_msg ("cannot find appsink %p in the output data", appsink); return NULL; } /* get the output stream entry in corresponding to the given Pad */ static GstAdaptiveDemuxTestOutputStream * getTestOutputDataByPad (GstAdaptiveDemuxTestEnginePrivate * priv, GstPad * pad, gboolean abort_if_not_found) { guint i; for (i = 0; i < priv->engine.output_streams->len; ++i) { GstAdaptiveDemuxTestOutputStream *stream; stream = g_ptr_array_index (priv->engine.output_streams, i); if (stream->internal_pad == pad || stream->pad == pad) { return stream; } } if (abort_if_not_found) ck_abort_msg ("cannot find pad %p in the output data", pad); return NULL; } /* callback called when AppSink receives data */ static GstFlowReturn on_appSinkNewSample (GstAppSink * appsink, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstAdaptiveDemuxTestEngine *engine; GstAdaptiveDemuxTestOutputStream *testOutputStream = NULL; GstSample *sample; GstBuffer *buffer; gboolean ret = TRUE; fail_unless (priv != NULL); GST_TEST_LOCK (priv); engine = &priv->engine; testOutputStream = getTestOutputDataByAppsink (priv, appsink); sample = gst_app_sink_pull_sample (appsink); fail_unless (sample != NULL); buffer = gst_sample_get_buffer (sample); fail_unless (buffer != NULL); /* call the test callback, if registered */ if (priv->callbacks->appsink_received_data) ret = priv->callbacks->appsink_received_data (engine, testOutputStream, buffer, priv->user_data); testOutputStream->segment_received_size += gst_buffer_get_size (buffer); gst_sample_unref (sample); GST_TEST_UNLOCK (priv); if (!ret) return GST_FLOW_EOS; return GST_FLOW_OK; } /* callback called when AppSink receives eos */ static void on_appSinkEOS (GstAppSink * appsink, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstAdaptiveDemuxTestOutputStream *testOutputStream = NULL; fail_unless (priv != NULL); GST_TEST_LOCK (priv); testOutputStream = getTestOutputDataByAppsink (priv, appsink); testOutputStream->total_received_size += testOutputStream->segment_received_size; testOutputStream->segment_received_size = 0; if (priv->callbacks->appsink_eos) priv->callbacks->appsink_eos (&priv->engine, testOutputStream, priv->user_data); GST_TEST_UNLOCK (priv); } static GstPadProbeReturn on_appsink_event (GstPad * pad, GstPadProbeInfo * info, gpointer data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstEvent *event; event = GST_PAD_PROBE_INFO_EVENT (info); GST_DEBUG ("Received event %" GST_PTR_FORMAT " on pad %" GST_PTR_FORMAT, event, pad); if (priv->callbacks->appsink_event) { GstPad *stream_pad = gst_pad_get_peer (pad); fail_unless (stream_pad != NULL); GST_TEST_LOCK (priv); stream = getTestOutputDataByPad (priv, stream_pad, TRUE); GST_TEST_UNLOCK (priv); gst_object_unref (stream_pad); priv->callbacks->appsink_event (&priv->engine, stream, event, priv->user_data); } return GST_PAD_PROBE_OK; } /* callback called when demux sends data to AppSink */ static GstPadProbeReturn on_demux_sent_data (GstPad * pad, GstPadProbeInfo * info, gpointer data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstBuffer *buffer; buffer = GST_PAD_PROBE_INFO_BUFFER (info); GST_TEST_LOCK (priv); stream = getTestOutputDataByPad (priv, pad, TRUE); GST_TEST_UNLOCK (priv); if (priv->callbacks->demux_sent_data) { (*priv->callbacks->demux_sent_data) (&priv->engine, stream, buffer, priv->user_data); } return GST_PAD_PROBE_OK; } /* callback called when dash sends event to AppSink */ static GstPadProbeReturn on_demux_sent_event (GstPad * pad, GstPadProbeInfo * info, gpointer data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstEvent *event; event = GST_PAD_PROBE_INFO_EVENT (info); GST_TEST_LOCK (&priv->engine); if (priv->callbacks->demux_sent_event) { stream = getTestOutputDataByPad (priv, pad, TRUE); (*priv->callbacks->demux_sent_event) (&priv->engine, stream, event, priv->user_data); } GST_TEST_UNLOCK (&priv->engine); return GST_PAD_PROBE_OK; } /* callback called when demux receives events from GstFakeSoupHTTPSrc */ static GstPadProbeReturn on_demuxReceivesEvent (GstPad * pad, GstPadProbeInfo * info, gpointer data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstEvent *event; const GstSegment *segment; event = GST_PAD_PROBE_INFO_EVENT (info); GST_DEBUG ("Received event %" GST_PTR_FORMAT " on pad %" GST_PTR_FORMAT, event, pad); if (GST_EVENT_TYPE (event) == GST_EVENT_SEGMENT) { /* a new segment will start arriving * update segment_start used by pattern validation */ gst_event_parse_segment (event, &segment); GST_TEST_LOCK (priv); stream = getTestOutputDataByPad (priv, pad, TRUE); stream->total_received_size += stream->segment_received_size; stream->segment_received_size = 0; stream->segment_start = segment->start; GST_TEST_UNLOCK (priv); } return GST_PAD_PROBE_OK; } static void on_demuxElementAdded (GstBin * demux, GstElement * element, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstPad *internal_pad; gchar *srcbin_name; gint i; srcbin_name = GST_ELEMENT_NAME (element); GST_TEST_LOCK (priv); for (i = 0; i < priv->engine.output_streams->len; i++) { stream = g_ptr_array_index (priv->engine.output_streams, i); if (strstr (srcbin_name, GST_PAD_NAME (stream->pad)) != NULL) break; } fail_unless (stream != NULL); /* keep the reference to the internal_pad. * We will need it to identify the stream in the on_demuxReceivesEvent callback */ if (stream->internal_pad) { gst_pad_remove_probe (stream->internal_pad, stream->internal_pad_probe); gst_object_unref (stream->internal_pad); } internal_pad = gst_element_get_static_pad (element, "src"); stream->internal_pad_probe = gst_pad_add_probe (internal_pad, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM, (GstPadProbeCallback) on_demuxReceivesEvent, priv, NULL); stream->internal_pad = internal_pad; GST_TEST_UNLOCK (priv); } /* callback called when demux creates a src pad. * We will create an AppSink to get the data */ static void on_demuxNewPad (GstElement * demux, GstPad * pad, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstElement *pipeline; GstElement *sink; gboolean ret; gchar *name; GstPad *appsink_pad; GstAppSinkCallbacks appSinkCallbacks; GstAdaptiveDemuxTestOutputStream *stream; GObjectClass *gobject_class; fail_unless (priv != NULL); name = gst_pad_get_name (pad); stream = g_slice_new0 (GstAdaptiveDemuxTestOutputStream); GST_DEBUG ("created pad %p", pad); sink = gst_element_factory_make ("appsink", name); g_free (name); fail_unless (sink != NULL); GST_TEST_LOCK (priv); /* register the AppSink pointer in the test output data */ gst_object_ref (sink); stream->appsink = GST_APP_SINK (sink); appSinkCallbacks.eos = on_appSinkEOS; appSinkCallbacks.new_preroll = NULL; appSinkCallbacks.new_sample = on_appSinkNewSample; gst_app_sink_set_callbacks (GST_APP_SINK (sink), &appSinkCallbacks, priv, NULL); appsink_pad = gst_element_get_static_pad (sink, "sink"); gst_pad_add_probe (appsink_pad, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM | GST_PAD_PROBE_TYPE_EVENT_FLUSH, (GstPadProbeCallback) on_appsink_event, priv, NULL); gst_object_unref (appsink_pad); gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_BUFFER, (GstPadProbeCallback) on_demux_sent_data, priv, NULL); gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM | GST_PAD_PROBE_TYPE_EVENT_FLUSH, (GstPadProbeCallback) on_demux_sent_event, priv, NULL); gobject_class = G_OBJECT_GET_CLASS (sink); if (g_object_class_find_property (gobject_class, "sync")) { GST_DEBUG ("Setting sync=FALSE on AppSink"); g_object_set (G_OBJECT (sink), "sync", FALSE, NULL); } stream->pad = gst_object_ref (pad); g_ptr_array_add (priv->engine.output_streams, stream); GST_TEST_UNLOCK (priv); pipeline = GST_ELEMENT (gst_element_get_parent (demux)); fail_unless (pipeline != NULL); ret = gst_bin_add (GST_BIN (pipeline), sink); fail_unless_equals_int (ret, TRUE); gst_object_unref (pipeline); ret = gst_element_link (demux, sink); fail_unless_equals_int (ret, TRUE); ret = gst_element_sync_state_with_parent (sink); fail_unless_equals_int (ret, TRUE); GST_TEST_LOCK (priv); if (priv->callbacks->demux_pad_added) { priv->callbacks->demux_pad_added (&priv->engine, stream, priv->user_data); } GST_TEST_UNLOCK (priv); } /* callback called when demux removes a src pad. * We remove the AppSink associated with this pad */ static void on_demuxPadRemoved (GstElement * demux, GstPad * pad, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstAdaptiveDemuxTestOutputStream *stream = NULL; GstStateChangeReturn ret; GstState currentState, pending; GstElement *appSink; fail_unless (priv != NULL); GST_DEBUG ("Pad removed: %" GST_PTR_FORMAT, pad); GST_TEST_LOCK (priv); stream = getTestOutputDataByPad (priv, pad, TRUE); if (priv->callbacks->demux_pad_removed) { priv->callbacks->demux_pad_removed (&priv->engine, stream, priv->user_data); } fail_unless (stream->appsink != NULL); fail_unless (stream->internal_pad != NULL); gst_object_unref (stream->internal_pad); stream->internal_pad = NULL; appSink = GST_ELEMENT (stream->appsink); ret = gst_element_get_state (appSink, ¤tState, &pending, 0); if ((ret == GST_STATE_CHANGE_SUCCESS && currentState == GST_STATE_PLAYING) || (ret == GST_STATE_CHANGE_ASYNC && pending == GST_STATE_PLAYING)) { GST_DEBUG ("Changing AppSink element to PAUSED"); gst_element_set_state (appSink, GST_STATE_PAUSED); } GST_TEST_UNLOCK (priv); } /* callback called when main_loop detects an error message * We will signal main loop to quit */ static void on_ErrorMessageOnBus (GstBus * bus, GstMessage * msg, gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GError *err = NULL; gchar *dbg_info = NULL; gst_message_parse_error (msg, &err, &dbg_info); GST_DEBUG ("ERROR from element %s: '%s'. Debugging info: %s", GST_OBJECT_NAME (msg->src), err->message, (dbg_info) ? dbg_info : "none"); g_error_free (err); g_free (dbg_info); GST_TEST_LOCK (priv); fail_unless (priv->callbacks->bus_error_message, "unexpected error detected on bus"); priv->callbacks->bus_error_message (&priv->engine, msg, priv->user_data); GST_TEST_UNLOCK (priv); } static gboolean gst_adaptive_demux_update_test_clock (gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstClockID id; GstClockTime next_entry; GstTestClock *clock = GST_TEST_CLOCK (priv->engine.clock); fail_unless (clock != NULL); next_entry = gst_test_clock_get_next_entry_time (clock); if (next_entry != GST_CLOCK_TIME_NONE) { /* tests that do not want the manifest to update will set the update period * to a big value, eg 500s. The manifest update task will register an alarm * for that value. * We do not want the clock to jump to that. If it does, the manifest update * task will keep scheduling and use all the cpu power, starving the other * threads. * Usually the test require the clock to update with approx 3s, so we will * allow only updates smaller than 100s */ GstClockTime curr_time = gst_clock_get_time (GST_CLOCK (clock)); if (next_entry - curr_time < 100 * GST_SECOND) { gst_test_clock_set_time (clock, next_entry); id = gst_test_clock_process_next_clock_id (clock); fail_unless (id != NULL); gst_clock_id_unref (id); } } return TRUE; } static gboolean start_pipeline_playing (gpointer user_data) { GstAdaptiveDemuxTestEnginePrivate *priv = (GstAdaptiveDemuxTestEnginePrivate *) user_data; GstStateChangeReturn stateChange; GST_DEBUG ("Moving pipeline to PLAYING state"); stateChange = gst_element_set_state (priv->engine.pipeline, GST_STATE_PLAYING); fail_unless (stateChange != GST_STATE_CHANGE_FAILURE); GST_DEBUG ("PLAYING stateChange = %d", stateChange); return FALSE; } /* * Create a demux element, run a test using the input data and check * the output data */ void gst_adaptive_demux_test_run (const gchar * element_name, const gchar * manifest_uri, const GstAdaptiveDemuxTestCallbacks * callbacks, gpointer user_data) { GstBus *bus; GstElement *demux; GstElement *manifest_source; gboolean ret; GstStateChangeReturn stateChange; GstAdaptiveDemuxTestEnginePrivate *priv; priv = g_slice_new0 (GstAdaptiveDemuxTestEnginePrivate); priv->engine.output_streams = g_ptr_array_new_with_free_func (adaptive_demux_engine_stream_state_finalize); g_mutex_init (&priv->engine.lock); priv->callbacks = callbacks; priv->user_data = user_data; priv->engine.loop = g_main_loop_new (NULL, TRUE); fail_unless (priv->engine.loop != NULL); GST_TEST_LOCK (priv); priv->engine.pipeline = gst_pipeline_new ("pipeline"); fail_unless (priv->engine.pipeline != NULL); GST_DEBUG ("created pipeline %" GST_PTR_FORMAT, priv->engine.pipeline); /* register a callback to listen for error messages */ bus = gst_pipeline_get_bus (GST_PIPELINE (priv->engine.pipeline)); gst_bus_add_signal_watch_full (bus, G_PRIORITY_HIGH); g_signal_connect (bus, "message::error", G_CALLBACK (on_ErrorMessageOnBus), priv); manifest_source = gst_element_make_from_uri (GST_URI_SRC, manifest_uri, NULL, NULL); fail_unless (manifest_source != NULL); priv->engine.manifest_source = manifest_source; demux = gst_check_setup_element (element_name); fail_unless (demux != NULL); priv->engine.demux = demux; GST_DEBUG ("created demux %" GST_PTR_FORMAT, demux); g_signal_connect (demux, "element-added", G_CALLBACK (on_demuxElementAdded), priv); g_signal_connect (demux, "pad-added", G_CALLBACK (on_demuxNewPad), priv); g_signal_connect (demux, "pad-removed", G_CALLBACK (on_demuxPadRemoved), priv); gst_bin_add_many (GST_BIN (priv->engine.pipeline), manifest_source, demux, NULL); ASSERT_OBJECT_REFCOUNT (manifest_source, element_name, 1); ASSERT_OBJECT_REFCOUNT (demux, element_name, 1); ret = gst_element_link (manifest_source, demux); fail_unless_equals_int (ret, TRUE); priv->engine.clock = gst_system_clock_obtain (); if (GST_IS_TEST_CLOCK (priv->engine.clock)) { /* * live tests will want to manipulate the clock, so they will register a * gst_test_clock as the system clock. * The on demand tests do not care about the clock, so they will let the * system clock to the default one. * If a gst_test_clock was installed as system clock, we register a * periodic callback to update its value. */ priv->clock_update_id = g_timeout_add (100, gst_adaptive_demux_update_test_clock, priv); } /* call a test callback before we start the pipeline */ if (callbacks->pre_test) (*callbacks->pre_test) (&priv->engine, priv->user_data); GST_TEST_UNLOCK (priv); GST_DEBUG ("Starting pipeline"); stateChange = gst_element_set_state (priv->engine.pipeline, GST_STATE_PAUSED); fail_unless (stateChange != GST_STATE_CHANGE_FAILURE); /* wait for completion of the move to PAUSED */ stateChange = gst_element_get_state (priv->engine.pipeline, NULL, NULL, GST_CLOCK_TIME_NONE); fail_unless (stateChange != GST_STATE_CHANGE_FAILURE); g_idle_add ((GSourceFunc) start_pipeline_playing, priv); /* block until a callback calls g_main_loop_quit (engine.loop) */ GST_DEBUG ("main thread waiting for streams to finish"); g_main_loop_run (priv->engine.loop); GST_DEBUG ("main thread finished. Stopping pipeline"); /* no need to use gst_element_get_state as the move the GST_STATE_NULL is always synchronous */ stateChange = gst_element_set_state (priv->engine.pipeline, GST_STATE_NULL); fail_unless (stateChange != GST_STATE_CHANGE_FAILURE); GST_TEST_LOCK (priv); /* call a test callback after the stop of the pipeline */ if (callbacks->post_test) (*callbacks->post_test) (&priv->engine, priv->user_data); g_signal_handlers_disconnect_by_func (bus, G_CALLBACK (on_ErrorMessageOnBus), priv); gst_bus_remove_signal_watch (bus); g_signal_handlers_disconnect_by_func (demux, G_CALLBACK (on_demuxNewPad), priv); g_signal_handlers_disconnect_by_func (demux, G_CALLBACK (on_demuxPadRemoved), priv); GST_DEBUG ("main thread pipeline stopped"); if (priv->clock_update_id != 0) g_source_remove (priv->clock_update_id); gst_object_unref (priv->engine.clock); gst_object_unref (priv->engine.pipeline); priv->engine.pipeline = NULL; g_main_loop_unref (priv->engine.loop); g_ptr_array_unref (priv->engine.output_streams); gst_object_unref (bus); GST_TEST_UNLOCK (priv); g_mutex_clear (&priv->engine.lock); g_slice_free (GstAdaptiveDemuxTestEnginePrivate, priv); }