Blame tests/icles/audio-trickplay.c

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