1 /* GStreamer
2 * Copyright (C) 2010 Sebastian Dröge <sebastian.droege@collabora.co.uk>
3 *
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
8 *
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
13 *
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
17 * Boston, MA 02110-1301, USA.
18 */
19
20 #include <string.h>
21
22 #include <gst/check/gstcheck.h>
23 #include <gst/video/video.h>
24 #include <gst/app/gstappsrc.h>
25
26 static gboolean
bus_handler(GstBus * bus,GstMessage * message,gpointer data)27 bus_handler (GstBus * bus, GstMessage * message, gpointer data)
28 {
29 GMainLoop *loop = (GMainLoop *) data;
30
31 switch (message->type) {
32 case GST_MESSAGE_EOS:
33 g_main_loop_quit (loop);
34 break;
35 case GST_MESSAGE_WARNING:
36 case GST_MESSAGE_ERROR:{
37 GError *gerror;
38 gchar *debug;
39
40 if (message->type == GST_MESSAGE_WARNING)
41 gst_message_parse_warning (message, &gerror, &debug);
42 else
43 gst_message_parse_error (message, &gerror, &debug);
44 gst_object_default_error (GST_MESSAGE_SRC (message), gerror, debug);
45 gst_message_unref (message);
46 g_error_free (gerror);
47 g_free (debug);
48 g_main_loop_quit (loop);
49 break;
50 }
51 default:
52 break;
53 }
54
55 return TRUE;
56 }
57
58 typedef struct
59 {
60 GstClockTime ts;
61 GstClockTime duration;
62 const gchar buf[];
63 } TestBuffer;
64
65 static const TestBuffer buf0 = {
66 0,
67 0,
68 {"[Script Info]\n"
69 "; This is a Sub Station Alpha v4 script.\n"
70 "; For Sub Station Alpha info and downloads,\n"
71 "; go to http://www.eswat.demon.co.uk/\n"
72 "Title: Some Test\n"
73 "Script Updated By: version 2.8.01\n"
74 "ScriptType: v4.00\n"
75 "Collisions: Normal\n"
76 "PlayResY: 200\n"
77 "PlayDepth: 0\n"
78 "Timer: 100,0000\n"
79 " \n"
80 "[V4 Styles]\n"
81 "Format: Name, Fontname, Fontsize, PrimaryColour, SecondaryColour, TertiaryColour, BackColour, Bold, Italic, BorderStyle, Outline, Shadow, \n"
82 " Alignment, MarginL, MarginR, MarginV, AlphaLevel, Encoding\n"
83 "Style: DefaultVCD, Arial,28,11861244,11861244,11861244,-2147483640,-1,0,1,1,2,2,30,30,30,0,0\n"
84 " \n"
85 "[Events]\n"
86 "Format: Marked, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text"}
87 };
88
89 static const TestBuffer buf1 = {
90 40 * GST_MSECOND,
91 60 * GST_MSECOND,
92 {"1,,DefaultVCD, NTP,0000,0000,0000,,Some Test Blabla"}
93 };
94
95 static void
sink_handoff_cb_xRGB(GstElement * object,GstBuffer * buffer,GstPad * pad,gpointer user_data)96 sink_handoff_cb_xRGB (GstElement * object, GstBuffer * buffer, GstPad * pad,
97 gpointer user_data)
98 {
99 guint *sink_pos = (guint *) user_data;
100 gboolean contains_text = (*sink_pos == 1 || *sink_pos == 2);
101 guint i, j;
102 GstMapInfo map;
103 gboolean all_red = TRUE;
104
105 gst_buffer_map (buffer, &map, GST_MAP_READ);
106
107 fail_unless_equals_int (map.size, 640 * 480 * 4);
108
109 for (i = 0; i < 640; i++) {
110 for (j = 0; j < 480; j++) {
111 all_red = all_red && (map.data[i * 480 * 4 + j * 4 + 1] == 255 &&
112 map.data[i * 480 * 4 + j * 4 + 2] == 0 &&
113 map.data[i * 480 * 4 + j * 4 + 3] == 0);
114 }
115 }
116 gst_buffer_unmap (buffer, &map);
117
118 fail_unless (contains_text != all_red,
119 "Frame %d is incorrect (all red %d, contains text %d)", *sink_pos,
120 all_red, contains_text);
121 *sink_pos = *sink_pos + 1;
122 }
123
124 static void
sink_handoff_cb_I420(GstElement * object,GstBuffer * buffer,GstPad * pad,gpointer user_data)125 sink_handoff_cb_I420 (GstElement * object, GstBuffer * buffer, GstPad * pad,
126 gpointer user_data)
127 {
128 guint *sink_pos = (guint *) user_data;
129 gboolean contains_text = (*sink_pos == 1 || *sink_pos == 2);
130 guint c, i, j;
131 gboolean all_red = TRUE;
132 guint8 *comp;
133 gint comp_stride, comp_width, comp_height;
134 const guint8 color[] = { 81, 90, 240 };
135 GstVideoInfo info;
136 GstVideoFrame frame;
137
138 gst_video_info_init (&info);
139 gst_video_info_set_format (&info, GST_VIDEO_FORMAT_I420, 640, 480);
140
141 gst_video_frame_map (&frame, &info, buffer, GST_MAP_READ);
142
143 for (c = 0; c < 3; c++) {
144 comp = GST_VIDEO_FRAME_COMP_DATA (&frame, c);
145 comp_stride = GST_VIDEO_FRAME_COMP_STRIDE (&frame, c);
146 comp_width = GST_VIDEO_FRAME_COMP_WIDTH (&frame, c);
147 comp_height = GST_VIDEO_FRAME_COMP_HEIGHT (&frame, c);
148
149 for (i = 0; i < comp_height; i++) {
150 for (j = 0; j < comp_width; j++) {
151 all_red = all_red && (comp[i * comp_stride + j] == color[c]);
152 }
153 }
154 }
155 gst_video_frame_unmap (&frame);
156
157 fail_unless (contains_text != all_red,
158 "Frame %d is incorrect (all red %d, contains text %d)", *sink_pos,
159 all_red, contains_text);
160 *sink_pos = *sink_pos + 1;
161 }
162
163 static gulong probe_id = 0;
164
165 static GstPadProbeReturn
src_buffer_probe_cb(GstPad * pad,GstPadProbeInfo * info,gpointer user_data)166 src_buffer_probe_cb (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
167 {
168 GstBuffer *buffer = GST_PAD_PROBE_INFO_BUFFER (info);
169 GstPad *otherpad = GST_PAD (user_data);
170
171 if (GST_BUFFER_TIMESTAMP (buffer) == buf1.ts)
172 gst_pad_remove_probe (otherpad, probe_id);
173
174 return GST_PAD_PROBE_OK;
175 }
176
177 #define CREATE_BASIC_TEST(format) \
178 GST_START_TEST (test_assrender_basic_##format) \
179 { \
180 GstElement *pipeline; \
181 GstElement *appsrc, *videotestsrc, *capsfilter, *assrender, *fakesink; \
182 guint sink_pos = 0; \
183 GstCaps *video_caps; \
184 GstCaps *text_caps; \
185 GstBuffer *buf; \
186 GstBus *bus; \
187 GMainLoop *loop; \
188 GstPad *pad, *blocked_pad; \
189 guint bus_watch = 0; \
190 GstVideoInfo info; \
191 \
192 pipeline = gst_pipeline_new ("pipeline"); \
193 fail_unless (pipeline != NULL); \
194 \
195 capsfilter = gst_element_factory_make ("capsfilter", NULL); \
196 fail_unless (capsfilter != NULL); \
197 gst_video_info_init (&info); \
198 gst_video_info_set_format (&info, GST_VIDEO_FORMAT_##format, 640, 480); \
199 info.fps_n = 25; \
200 info.fps_d = 1; \
201 video_caps = gst_video_info_to_caps (&info); \
202 g_object_set (capsfilter, "caps", video_caps, NULL); \
203 gst_caps_unref (video_caps); \
204 blocked_pad = gst_element_get_static_pad (capsfilter, "src"); \
205 gst_pad_add_probe (blocked_pad, GST_PAD_PROBE_TYPE_BLOCK_DOWNSTREAM, NULL, NULL, NULL); \
206 \
207 appsrc = gst_element_factory_make ("appsrc", NULL); \
208 fail_unless (appsrc != NULL); \
209 buf = gst_buffer_new_and_alloc (strlen (buf0.buf) + 1); \
210 gst_buffer_fill (buf, 0, buf0.buf, strlen (buf0.buf) + 1); \
211 GST_BUFFER_TIMESTAMP (buf) = buf0.ts; \
212 GST_BUFFER_DURATION (buf) = buf0.duration; \
213 text_caps = \
214 gst_caps_new_simple ("application/x-ssa", "codec_data", GST_TYPE_BUFFER, \
215 buf, NULL); \
216 gst_buffer_unref (buf); \
217 gst_app_src_set_caps (GST_APP_SRC (appsrc), text_caps); \
218 g_object_set (appsrc, "format", GST_FORMAT_TIME, NULL); \
219 pad = gst_element_get_static_pad (appsrc, "src"); \
220 probe_id = gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_BUFFER, src_buffer_probe_cb, \
221 gst_object_ref (blocked_pad), (GDestroyNotify) gst_object_unref); \
222 gst_object_unref (blocked_pad); \
223 gst_object_unref (pad); \
224 \
225 videotestsrc = gst_element_factory_make ("videotestsrc", NULL); \
226 fail_unless (videotestsrc != NULL); \
227 g_object_set (videotestsrc, "num-buffers", 5, "pattern", 4, NULL); \
228 \
229 assrender = gst_element_factory_make ("assrender", NULL); \
230 fail_unless (assrender != NULL); \
231 \
232 fakesink = gst_element_factory_make ("fakesink", NULL); \
233 fail_unless (fakesink != NULL); \
234 g_object_set (fakesink, "signal-handoffs", TRUE, "async", FALSE, NULL); \
235 g_signal_connect (fakesink, "handoff", G_CALLBACK (sink_handoff_cb_##format), \
236 &sink_pos); \
237 \
238 gst_bin_add_many (GST_BIN (pipeline), appsrc, videotestsrc, capsfilter, \
239 assrender, fakesink, NULL); \
240 \
241 fail_unless (gst_element_link_pads (appsrc, "src", assrender, "text_sink")); \
242 fail_unless (gst_element_link_pads (videotestsrc, "src", capsfilter, "sink")); \
243 fail_unless (gst_element_link_pads (capsfilter, "src", assrender, \
244 "video_sink")); \
245 fail_unless (gst_element_link_pads (assrender, "src", fakesink, "sink")); \
246 \
247 loop = g_main_loop_new (NULL, TRUE); \
248 fail_unless (loop != NULL); \
249 \
250 bus = gst_element_get_bus (pipeline); \
251 fail_unless (bus != NULL); \
252 bus_watch = gst_bus_add_watch (bus, bus_handler, loop); \
253 gst_object_unref (bus); \
254 \
255 fail_unless_equals_int (gst_element_set_state (pipeline, GST_STATE_PLAYING), \
256 GST_STATE_CHANGE_SUCCESS); \
257 \
258 buf = gst_buffer_new_and_alloc (strlen (buf1.buf) + 1); \
259 gst_buffer_fill (buf, 0, buf1.buf, strlen (buf1.buf) + 1); \
260 GST_BUFFER_TIMESTAMP (buf) = buf1.ts; \
261 GST_BUFFER_DURATION (buf) = buf1.duration; \
262 gst_app_src_push_buffer (GST_APP_SRC (appsrc), buf); \
263 gst_caps_unref (text_caps); \
264 gst_app_src_end_of_stream (GST_APP_SRC (appsrc)); \
265 \
266 g_main_loop_run (loop); \
267 \
268 gst_element_set_state (pipeline, GST_STATE_NULL); \
269 \
270 fail_unless_equals_int (sink_pos, 5); \
271 \
272 g_object_unref (pipeline); \
273 g_main_loop_unref (loop); \
274 g_source_remove (bus_watch); \
275 } \
276 \
277 GST_END_TEST;
278
279 CREATE_BASIC_TEST (xRGB);
280 CREATE_BASIC_TEST (I420);
281
282 static Suite *
assrender_suite(void)283 assrender_suite (void)
284 {
285 Suite *s = suite_create ("assrender");
286 TCase *tc_chain = tcase_create ("linear");
287
288 /* time out after 120s, not the default 3 */
289 tcase_set_timeout (tc_chain, 120);
290
291 suite_add_tcase (s, tc_chain);
292 tcase_add_test (tc_chain, test_assrender_basic_xRGB);
293 tcase_add_test (tc_chain, test_assrender_basic_I420);
294
295 return s;
296 }
297
298 GST_CHECK_MAIN (assrender);
299