controller: no need to explicitely add controlled properties anymore
[platform/upstream/gstreamer.git] / tests / icles / audio-trickplay.c
1 /*
2  * audio-trickplay.c
3  *
4  * Builds a pipeline with two audiotestsources mixed with adder. Assigns
5  * controller patterns to the audio generators and test various trick modes.
6  *
7  * There are currently several issues:
8  * - adder only work with flushing seeks
9  * - there is a gap of almost 4 seconds before backwards playback
10  *   - it is "waiting for free space"
11  *   - using sync=false on the sink does not help (but has some other weird effects)
12  *   - using fakesink shows same behaviour
13  *
14  * GST_DEBUG_NO_COLOR=1 GST_DEBUG="*:2,default:3,*sink*:4,*ring*:4,*pulse*:5" ./audio-trickplay 2>log.txt
15  * GST_DEBUG_NO_COLOR=1 GST_DEBUG="*:2,default:3,*sink*:4,*ring*:4,*pulse*:5" ./audio-trickplay -a -f 2>log-af.txt
16  */
17
18 #include <string.h>
19 #include <gst/gst.h>
20 #include <gst/controller/gstinterpolationcontrolsource.h>
21
22 static void
23 check_position (GstElement * elem, GstQuery * pos, const gchar * info)
24 {
25   if (gst_element_query (elem, pos)) {
26     gint64 play_pos;
27     gst_query_parse_position (pos, NULL, &play_pos);
28     GST_INFO ("pos : %" GST_TIME_FORMAT " %s", GST_TIME_ARGS (play_pos), info);
29   } else {
30     GST_WARNING ("position query failed");
31   }
32 }
33
34 static GstPadProbeReturn
35 print_buffer_ts (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
36 {
37   GstBuffer *buffer = GST_PAD_PROBE_INFO_BUFFER (info);
38
39   GST_DEBUG_OBJECT (pad, "  ts: %" GST_TIME_FORMAT,
40       GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)));
41
42   return GST_PAD_PROBE_OK;
43 }
44
45 gint
46 main (gint argc, gchar ** argv)
47 {
48   gint res = 1;
49   GstElement *src, *mix = NULL, *sink;
50   GstElement *bin;
51   GstInterpolationControlSource *csource1, *csource2;
52   GstClock *clock;
53   GstClockID clock_id;
54   GstClockReturn wait_ret;
55   GValue vol = { 0, };
56   GstEvent *pos_seek, *rate_seek1, *rate_seek2;
57   GstQuery *pos;
58   GstSeekFlags flags;
59   GstPad *src_pad;
60   /* options */
61   gboolean use_adder = FALSE;
62   gboolean use_flush = FALSE;
63   gboolean be_quiet = FALSE;
64
65   gst_init (&argc, &argv);
66
67   if (argc) {
68     gint arg;
69     for (arg = 0; arg < argc; arg++) {
70       if (!strcmp (argv[arg], "-a"))
71         use_adder = TRUE;
72       else if (!strcmp (argv[arg], "-f"))
73         use_flush = TRUE;
74       else if (!strcmp (argv[arg], "-q"))
75         be_quiet = TRUE;
76     }
77   }
78
79   /* build pipeline */
80   bin = gst_pipeline_new ("pipeline");
81   clock = gst_pipeline_get_clock (GST_PIPELINE (bin));
82   src = gst_element_factory_make ("audiotestsrc", NULL);
83   if (!src) {
84     GST_WARNING ("need audiotestsrc from gst-plugins-base");
85     goto Error;
86   }
87   if (use_adder) {
88     mix = gst_element_factory_make ("adder", NULL);
89     if (!mix) {
90       GST_WARNING ("need adder from gst-plugins-base");
91       goto Error;
92     }
93   }
94   sink = gst_element_factory_make ((be_quiet ? "fakesink" : "autoaudiosink"),
95       NULL);
96   if (!sink) {
97     GST_WARNING ("need autoaudiosink from gst-plugins-base");
98     goto Error;
99   }
100
101   if (use_adder) {
102     gst_bin_add_many (GST_BIN (bin), src, mix, sink, NULL);
103     if (!gst_element_link_many (src, mix, sink, NULL)) {
104       GST_WARNING ("can't link elements");
105       goto Error;
106     }
107   } else {
108     gst_bin_add_many (GST_BIN (bin), src, sink, NULL);
109     if (!gst_element_link_many (src, sink, NULL)) {
110       GST_WARNING ("can't link elements");
111       goto Error;
112     }
113   }
114
115   /* use 10 buffers per second */
116   g_object_set (src, "samplesperbuffer", 44100 / 10, NULL);
117
118   if (be_quiet) {
119     g_object_set (sink, "sync", TRUE, NULL);
120   }
121
122   src_pad = gst_element_get_static_pad (src, "src");
123   gst_pad_add_probe (src_pad, GST_PAD_PROBE_TYPE_BUFFER, print_buffer_ts, NULL,
124       NULL);
125   gst_object_unref (src_pad);
126
127   csource1 = gst_interpolation_control_source_new ();
128   csource2 = gst_interpolation_control_source_new ();
129
130   gst_object_set_control_source (GST_OBJECT (src), "volume",
131       GST_CONTROL_SOURCE (csource1));
132   gst_object_set_control_source (GST_OBJECT (src), "freq",
133       GST_CONTROL_SOURCE (csource2));
134
135   /* Set interpolation mode */
136
137   gst_interpolation_control_source_set_interpolation_mode (csource1,
138       GST_INTERPOLATE_LINEAR);
139   gst_interpolation_control_source_set_interpolation_mode (csource2,
140       GST_INTERPOLATE_LINEAR);
141
142   /* set control values */
143   g_value_init (&vol, G_TYPE_DOUBLE);
144   g_value_set_double (&vol, 0.0);
145   gst_interpolation_control_source_set (csource1, 0 * GST_SECOND, &vol);
146   g_value_set_double (&vol, 1.0);
147   gst_interpolation_control_source_set (csource1, 5 * GST_SECOND, &vol);
148
149   g_object_unref (csource1);
150
151   g_value_set_double (&vol, 220.0);
152   gst_interpolation_control_source_set (csource2, 0 * GST_SECOND, &vol);
153   g_value_set_double (&vol, 3520.0);
154   gst_interpolation_control_source_set (csource2, 2 * GST_SECOND, &vol);
155   g_value_set_double (&vol, 440.0);
156   gst_interpolation_control_source_set (csource2, 6 * GST_SECOND, &vol);
157
158   g_object_unref (csource2);
159
160   /* prepare events */
161   flags = use_flush ? GST_SEEK_FLAG_FLUSH : GST_SEEK_FLAG_NONE;
162   pos_seek = gst_event_new_seek (1.0, GST_FORMAT_TIME, flags,
163       GST_SEEK_TYPE_SET, 3 * GST_SECOND,
164       GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
165   rate_seek1 = gst_event_new_seek (0.5, GST_FORMAT_TIME, flags,
166       GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE,
167       GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
168   rate_seek2 = gst_event_new_seek (-1.0, GST_FORMAT_TIME, flags,
169       GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE,
170       GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE);
171
172   /* prepare queries */
173   pos = gst_query_new_position (GST_FORMAT_TIME);
174
175
176   /* run the show */
177   if (gst_element_set_state (bin, GST_STATE_PAUSED) != GST_STATE_CHANGE_FAILURE) {
178
179     /* run for 5 seconds */
180     clock_id =
181         gst_clock_new_single_shot_id (clock,
182         gst_clock_get_time (clock) + (5 * GST_SECOND));
183
184     if (gst_element_set_state (bin,
185             GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE) {
186       check_position (bin, pos, "start");
187       if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
188         GST_WARNING ("clock_id_wait returned: %d", wait_ret);
189       }
190     }
191     gst_clock_id_unref (clock_id);
192
193     check_position (bin, pos, "before seek to new pos");
194
195     /* seek to 3:00 sec (back 2 sec) */
196     if (!gst_element_send_event (sink, pos_seek)) {
197       GST_WARNING ("element failed to seek to new position");
198     }
199
200     check_position (bin, pos, "after seek to new pos");
201
202     /* run for 2 seconds */
203     clock_id =
204         gst_clock_new_single_shot_id (clock,
205         gst_clock_get_time (clock) + (2 * GST_SECOND));
206     if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
207       GST_WARNING ("clock_id_wait returned: %d", wait_ret);
208     }
209     gst_clock_id_unref (clock_id);
210
211     check_position (bin, pos, "before slow down rate change");
212
213     /* change playback rate to 0.5 */
214     if (!gst_element_send_event (sink, rate_seek1)) {
215       GST_WARNING ("element failed to change playback rate");
216     }
217
218     check_position (bin, pos, "after slow down rate change");
219
220     /* run for 4 seconds */
221     clock_id =
222         gst_clock_new_single_shot_id (clock,
223         gst_clock_get_time (clock) + (4 * GST_SECOND));
224     if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
225       GST_WARNING ("clock_id_wait returned: %d", wait_ret);
226     }
227     gst_clock_id_unref (clock_id);
228
229     check_position (bin, pos, "before reverse rate change");
230
231     /* change playback rate to -1.0  */
232     if (!gst_element_send_event (sink, rate_seek2)) {
233       GST_WARNING ("element failed to change playback rate");
234     }
235
236     check_position (bin, pos, "after reverse rate change");
237
238     /* run for 7 seconds */
239     clock_id =
240         gst_clock_new_single_shot_id (clock,
241         gst_clock_get_time (clock) + (7 * GST_SECOND));
242     if ((wait_ret = gst_clock_id_wait (clock_id, NULL)) != GST_CLOCK_OK) {
243       GST_WARNING ("clock_id_wait returned: %d", wait_ret);
244     }
245     gst_clock_id_unref (clock_id);
246
247     check_position (bin, pos, "done");
248
249     gst_element_set_state (bin, GST_STATE_NULL);
250   }
251
252   /* cleanup */
253   gst_query_unref (pos);
254   gst_object_unref (G_OBJECT (clock));
255   gst_object_unref (G_OBJECT (bin));
256   res = 0;
257 Error:
258   return (res);
259 }