Added gst-plugins-base-subtitles0.10-0.10.34 for Meego Harmattan 1.2
[mafwsubrenderer] / gst-plugins-base-subtitles0.10 / tests / icles / audio-trickplay.c
diff --git a/gst-plugins-base-subtitles0.10/tests/icles/audio-trickplay.c b/gst-plugins-base-subtitles0.10/tests/icles/audio-trickplay.c
new file mode 100644 (file)
index 0000000..7da0be4
--- /dev/null
@@ -0,0 +1,264 @@
+/*
+ * audio-trickplay.c
+ *
+ * Builds a pipeline with two audiotestsources mixed with adder. Assigns
+ * controller patterns to the audio generators and test various trick modes.
+ *
+ * There are currently several issues:
+ * - adder only work with flushing seeks
+ * - there is a gap of almost 4 seconds before backwards playback
+ *   - it is "waiting for free space"
+ *   - using sync=false on the sink does not help (but has some other weird effects)
+ *   - using fakesink shows same behaviour
+ *
+ * GST_DEBUG_NO_COLOR=1 GST_DEBUG="*:2,default:3,*sink*:4,*ring*:4,*pulse*:5" ./audio-trickplay 2>log.txt
+ * GST_DEBUG_NO_COLOR=1 GST_DEBUG="*:2,default:3,*sink*:4,*ring*:4,*pulse*:5" ./audio-trickplay -a -f 2>log-af.txt
+ */
+
+#include <gst/gst.h>
+#include <gst/controller/gstcontroller.h>
+#include <gst/controller/gstinterpolationcontrolsource.h>
+
+static void
+check_position (GstElement * elem, GstQuery * pos, const gchar * info)
+{
+  if (gst_element_query (elem, pos)) {
+    gint64 play_pos;
+    gst_query_parse_position (pos, NULL, &play_pos);
+    GST_INFO ("pos : %" GST_TIME_FORMAT " %s", GST_TIME_ARGS (play_pos), info);
+  } else {
+    GST_WARNING ("position query failed");
+  }
+}
+
+static gboolean
+print_buffer_ts (GstPad * pad, GstBuffer * buffer, gpointer user_data)
+{
+  GST_DEBUG_OBJECT (pad, "  ts: %" GST_TIME_FORMAT,
+      GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)));
+  return TRUE;
+}
+
+gint
+main (gint argc, gchar ** argv)
+{
+  gint res = 1;
+  GstElement *src, *mix = NULL, *sink;
+  GstElement *bin;
+  GstController *ctrl;
+  GstInterpolationControlSource *csource1, *csource2;
+  GstClock *clock;
+  GstClockID clock_id;
+  GstClockReturn wait_ret;
+  GValue vol = { 0, };
+  GstEvent *pos_seek, *rate_seek1, *rate_seek2;
+  GstQuery *pos;
+  GstSeekFlags flags;
+  GstPad *src_pad;
+  /* options */
+  gboolean use_adder = FALSE;
+  gboolean use_flush = FALSE;
+  gboolean be_quiet = FALSE;
+
+  gst_init (&argc, &argv);
+  gst_controller_init (&argc, &argv);
+
+  if (argc) {
+    gint arg;
+    for (arg = 0; arg < argc; arg++) {
+      if (!strcmp (argv[arg], "-a"))
+        use_adder = TRUE;
+      else if (!strcmp (argv[arg], "-f"))
+        use_flush = TRUE;
+      else if (!strcmp (argv[arg], "-q"))
+        be_quiet = TRUE;
+    }
+  }
+
+  /* build pipeline */
+  bin = gst_pipeline_new ("pipeline");
+  clock = gst_pipeline_get_clock (GST_PIPELINE (bin));
+  src = gst_element_factory_make ("audiotestsrc", NULL);
+  if (!src) {
+    GST_WARNING ("need audiotestsrc from gst-plugins-base");
+    goto Error;
+  }
+  if (use_adder) {
+    mix = gst_element_factory_make ("adder", NULL);
+    if (!mix) {
+      GST_WARNING ("need adder from gst-plugins-base");
+      goto Error;
+    }
+  }
+  sink = gst_element_factory_make ((be_quiet ? "fakesink" : "autoaudiosink"),
+      NULL);
+  if (!sink) {
+    GST_WARNING ("need autoaudiosink from gst-plugins-base");
+    goto Error;
+  }
+
+  if (use_adder) {
+    gst_bin_add_many (GST_BIN (bin), src, mix, sink, NULL);
+    if (!gst_element_link_many (src, mix, sink, NULL)) {
+      GST_WARNING ("can't link elements");
+      goto Error;
+    }
+  } else {
+    gst_bin_add_many (GST_BIN (bin), src, sink, NULL);
+    if (!gst_element_link_many (src, sink, NULL)) {
+      GST_WARNING ("can't link elements");
+      goto Error;
+    }
+  }
+
+  /* use 10 buffers per second */
+  g_object_set (src, "samplesperbuffer", 44100 / 10, NULL);
+
+  if (be_quiet) {
+    g_object_set (sink, "sync", TRUE, NULL);
+  }
+
+  src_pad = gst_element_get_static_pad (src, "src");
+  gst_pad_add_buffer_probe (src_pad, G_CALLBACK (print_buffer_ts), NULL);
+  gst_object_unref (src_pad);
+
+  /* add a controller to the source */
+  if (!(ctrl = gst_controller_new (G_OBJECT (src), "freq", "volume", NULL))) {
+    GST_WARNING ("can't control source element");
+    goto Error;
+  }
+
+  csource1 = gst_interpolation_control_source_new ();
+  csource2 = gst_interpolation_control_source_new ();
+
+  gst_controller_set_control_source (ctrl, "volume",
+      GST_CONTROL_SOURCE (csource1));
+  gst_controller_set_control_source (ctrl, "freq",
+      GST_CONTROL_SOURCE (csource2));
+
+  /* Set interpolation mode */
+
+  gst_interpolation_control_source_set_interpolation_mode (csource1,
+      GST_INTERPOLATE_LINEAR);
+  gst_interpolation_control_source_set_interpolation_mode (csource2,
+      GST_INTERPOLATE_LINEAR);
+
+  /* set control values */
+  g_value_init (&vol, G_TYPE_DOUBLE);
+  g_value_set_double (&vol, 0.0);
+  gst_interpolation_control_source_set (csource1, 0 * GST_SECOND, &vol);
+  g_value_set_double (&vol, 1.0);
+  gst_interpolation_control_source_set (csource1, 5 * GST_SECOND, &vol);
+
+  g_object_unref (csource1);
+
+  g_value_set_double (&vol, 220.0);
+  gst_interpolation_control_source_set (csource2, 0 * GST_SECOND, &vol);
+  g_value_set_double (&vol, 3520.0);
+  gst_interpolation_control_source_set (csource2, 2 * GST_SECOND, &vol);
+  g_value_set_double (&vol, 440.0);
+  gst_interpolation_control_source_set (csource2, 6 * GST_SECOND, &vol);
+
+  g_object_unref (csource2);
+
+  /* prepare events */
+  flags = use_flush ? GST_SEEK_FLAG_FLUSH : GST_SEEK_FLAG_NONE;
+  pos_seek = gst_event_new_seek (1.0, GST_FORMAT_TIME, flags,
+      GST_SEEK_TYPE_SET, 3 * GST_SECOND,
+      GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
+  rate_seek1 = gst_event_new_seek (0.5, GST_FORMAT_TIME, flags,
+      GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE,
+      GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
+  rate_seek2 = gst_event_new_seek (-1.0, GST_FORMAT_TIME, flags,
+      GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE,
+      GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
+
+  /* prepare queries */
+  pos = gst_query_new_position (GST_FORMAT_TIME);
+
+
+  /* run the show */
+  if (gst_element_set_state (bin, GST_STATE_PAUSED) != GST_STATE_CHANGE_FAILURE) {
+
+    /* run for 5 seconds */
+    clock_id =
+        gst_clock_new_single_shot_id (clock,
+        gst_clock_get_time (clock) + (5 * GST_SECOND));
+
+    if (gst_element_set_state (bin,
+            GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE) {
+      check_position (bin, pos, "start");
+      if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
+        GST_WARNING ("clock_id_wait returned: %d", wait_ret);
+      }
+    }
+    gst_clock_id_unref (clock_id);
+
+    check_position (bin, pos, "before seek to new pos");
+
+    /* seek to 3:00 sec (back 2 sec) */
+    if (!gst_element_send_event (sink, pos_seek)) {
+      GST_WARNING ("element failed to seek to new position");
+    }
+
+    check_position (bin, pos, "after seek to new pos");
+
+    /* run for 2 seconds */
+    clock_id =
+        gst_clock_new_single_shot_id (clock,
+        gst_clock_get_time (clock) + (2 * GST_SECOND));
+    if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
+      GST_WARNING ("clock_id_wait returned: %d", wait_ret);
+    }
+    gst_clock_id_unref (clock_id);
+
+    check_position (bin, pos, "before slow down rate change");
+
+    /* change playback rate to 0.5 */
+    if (!gst_element_send_event (sink, rate_seek1)) {
+      GST_WARNING ("element failed to change playback rate");
+    }
+
+    check_position (bin, pos, "after slow down rate change");
+
+    /* run for 4 seconds */
+    clock_id =
+        gst_clock_new_single_shot_id (clock,
+        gst_clock_get_time (clock) + (4 * GST_SECOND));
+    if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
+      GST_WARNING ("clock_id_wait returned: %d", wait_ret);
+    }
+    gst_clock_id_unref (clock_id);
+
+    check_position (bin, pos, "before reverse rate change");
+
+    /* change playback rate to -1.0  */
+    if (!gst_element_send_event (sink, rate_seek2)) {
+      GST_WARNING ("element failed to change playback rate");
+    }
+
+    check_position (bin, pos, "after reverse rate change");
+
+    /* run for 7 seconds */
+    clock_id =
+        gst_clock_new_single_shot_id (clock,
+        gst_clock_get_time (clock) + (7 * GST_SECOND));
+    if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
+      GST_WARNING ("clock_id_wait returned: %d", wait_ret);
+    }
+    gst_clock_id_unref (clock_id);
+
+    check_position (bin, pos, "done");
+
+    gst_element_set_state (bin, GST_STATE_NULL);
+  }
+
+  /* cleanup */
+  gst_query_unref (pos);
+  g_object_unref (G_OBJECT (ctrl));
+  gst_object_unref (G_OBJECT (clock));
+  gst_object_unref (G_OBJECT (bin));
+  res = 0;
+Error:
+  return (res);
+}