Remove dummy plugin_init. Remove some undefined entries from doc- section file. Add...
[platform/upstream/gstreamer.git] / gst / debug / testplugin.c
1 /* GStreamer
2  * Copyright (C) 2004 Benjamin Otte <otte@gnome.org>
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., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  */
19
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #endif
23
24 #include <gst/gst.h>
25 #include <gst/base/gstbasesink.h>
26 #include "tests.h"
27
28 GST_DEBUG_CATEGORY_STATIC (gst_test_debug);
29 #define GST_CAT_DEFAULT gst_test_debug
30
31 /* This plugin does all the tests registered in the tests.h file
32  */
33
34 #define GST_TYPE_TEST \
35   (gst_test_get_type())
36 #define GST_TEST(obj) \
37   (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_TEST,GstTest))
38 #define GST_TEST_CLASS(klass) \
39   (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_TEST,GstTestClass))
40 #define GST_TEST_GET_CLASS(obj) \
41   (G_TYPE_INSTANCE_GET_CLASS ((obj),GST_TYPE_TEST,GstTestClass))
42 #define GST_IS_TEST(obj) \
43   (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_TEST))
44 #define GST_IS_TEST_CLASS(klass) \
45   (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_TEST))
46
47 typedef struct _GstTest GstTest;
48 typedef struct _GstTestClass GstTestClass;
49
50 struct _GstTest
51 {
52   GstBaseSink basesink;
53
54   gpointer tests[TESTS_COUNT];
55   GValue values[TESTS_COUNT];
56 };
57
58 struct _GstTestClass
59 {
60   GstBaseSinkClass parent_class;
61
62   gchar *param_names[2 * TESTS_COUNT];
63 };
64
65 static void gst_test_finalize (GstTest * test);
66
67 static gboolean gst_test_start (GstBaseSink * trans);
68 static gboolean gst_test_stop (GstBaseSink * trans);
69 static gboolean gst_test_sink_event (GstBaseSink * basesink, GstEvent * event);
70 static GstFlowReturn gst_test_render_buffer (GstBaseSink * basesink,
71     GstBuffer * buf);
72
73 static void gst_test_set_property (GObject * object, guint prop_id,
74     const GValue * value, GParamSpec * pspec);
75 static void gst_test_get_property (GObject * object, guint prop_id,
76     GValue * value, GParamSpec * pspec);
77
78 static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
79     GST_PAD_SINK,
80     GST_PAD_ALWAYS,
81     GST_STATIC_CAPS_ANY);
82
83
84 static const GstElementDetails details = GST_ELEMENT_DETAILS ("Test plugin",
85     "Testing",
86     "perform a number of tests",
87     "Benjamin Otte <otte@gnome>");
88
89 #define DEBUG_INIT(bla) \
90   GST_DEBUG_CATEGORY_INIT (gst_test_debug, "testsink", 0, \
91       "debugging category for testsink element");
92
93 GST_BOILERPLATE_FULL (GstTest, gst_test, GstBaseSink, GST_TYPE_BASE_SINK,
94     DEBUG_INIT);
95
96
97 static void
98 gst_test_base_init (gpointer g_class)
99 {
100   GstElementClass *gstelement_class = GST_ELEMENT_CLASS (g_class);
101
102   gst_element_class_add_pad_template (gstelement_class,
103       gst_static_pad_template_get (&sinktemplate));
104
105   gst_element_class_set_details (gstelement_class, &details);
106 }
107
108 static void
109 gst_test_class_init (GstTestClass * klass)
110 {
111   GstBaseSinkClass *basesink_class = GST_BASE_SINK_CLASS (klass);
112   GObjectClass *object_class = G_OBJECT_CLASS (klass);
113   guint i;
114
115   object_class->set_property = GST_DEBUG_FUNCPTR (gst_test_set_property);
116   object_class->get_property = GST_DEBUG_FUNCPTR (gst_test_get_property);
117
118   object_class->finalize = (GObjectFinalizeFunc) gst_test_finalize;
119
120   for (i = 0; i < TESTS_COUNT; i++) {
121     GParamSpec *spec;
122
123     spec = tests[i].get_spec (&tests[i], FALSE);
124     klass->param_names[2 * i] = g_strdup (g_param_spec_get_name (spec));
125     g_object_class_install_property (object_class, 2 * i + 1, spec);
126     spec = tests[i].get_spec (&tests[i], TRUE);
127     klass->param_names[2 * i + 1] = g_strdup (g_param_spec_get_name (spec));
128     g_object_class_install_property (object_class, 2 * i + 2, spec);
129   }
130
131   basesink_class->preroll = GST_DEBUG_FUNCPTR (gst_test_render_buffer);
132   basesink_class->render = GST_DEBUG_FUNCPTR (gst_test_render_buffer);
133   basesink_class->event = GST_DEBUG_FUNCPTR (gst_test_sink_event);
134   basesink_class->start = GST_DEBUG_FUNCPTR (gst_test_start);
135   basesink_class->stop = GST_DEBUG_FUNCPTR (gst_test_stop);
136 }
137
138 static void
139 gst_test_init (GstTest * test, GstTestClass * g_class)
140 {
141   GstTestClass *klass;
142   guint i;
143
144   klass = GST_TEST_GET_CLASS (test);
145   for (i = 0; i < TESTS_COUNT; i++) {
146     GParamSpec *spec = g_object_class_find_property (G_OBJECT_CLASS (klass),
147         klass->param_names[2 * i + 1]);
148
149     g_value_init (&test->values[i], G_PARAM_SPEC_VALUE_TYPE (spec));
150   }
151 }
152
153 static void
154 gst_test_finalize (GstTest * test)
155 {
156   guint i;
157
158   for (i = 0; i < TESTS_COUNT; i++) {
159     g_value_unset (&test->values[i]);
160   }
161
162   G_OBJECT_CLASS (parent_class)->finalize ((GObject *) test);
163 }
164
165 static void
166 tests_unset (GstTest * test)
167 {
168   guint i;
169
170   for (i = 0; i < TESTS_COUNT; i++) {
171     if (test->tests[i]) {
172       tests[i].free (test->tests[i]);
173       test->tests[i] = NULL;
174     }
175   }
176 }
177
178 static void
179 tests_set (GstTest * test)
180 {
181   guint i;
182
183   for (i = 0; i < TESTS_COUNT; i++) {
184     g_assert (test->tests[i] == NULL);
185     test->tests[i] = tests[i].new (&tests[i]);
186   }
187 }
188
189 static gboolean
190 gst_test_sink_event (GstBaseSink * basesink, GstEvent * event)
191 {
192   GstTestClass *klass = GST_TEST_GET_CLASS (basesink);
193   GstTest *test = GST_TEST (basesink);
194   gboolean ret = FALSE;
195
196   switch (GST_EVENT_TYPE (event)) {
197 /*
198     case GST_EVENT_NEWSEGMENT:
199       if (GST_EVENT_DISCONT_NEW_MEDIA (event)) {
200         tests_unset (test);
201         tests_set (test);
202       }
203       break;
204 */
205     case GST_EVENT_EOS:{
206       gint i;
207
208       g_object_freeze_notify (G_OBJECT (test));
209       for (i = 0; i < TESTS_COUNT; i++) {
210         if (test->tests[i]) {
211           if (!tests[i].finish (test->tests[i], &test->values[i])) {
212             GValue v = { 0, };
213             gchar *real, *expected;
214
215             expected = gst_value_serialize (&test->values[i]);
216             g_value_init (&v, G_VALUE_TYPE (&test->values[i]));
217             g_object_get_property (G_OBJECT (test), klass->param_names[2 * i],
218                 &v);
219             real = gst_value_serialize (&v);
220             g_value_unset (&v);
221             GST_ELEMENT_ERROR (test, STREAM, FORMAT, (NULL),
222                 ("test %s returned value \"%s\" and not expected value \"%s\"",
223                     klass->param_names[2 * i], real, expected));
224             g_free (real);
225             g_free (expected);
226           }
227           g_object_notify (G_OBJECT (test), klass->param_names[2 * i]);
228         }
229       }
230       g_object_thaw_notify (G_OBJECT (test));
231       ret = TRUE;
232       break;
233     }
234     default:
235       break;
236   }
237
238   return ret;
239 }
240
241 static GstFlowReturn
242 gst_test_render_buffer (GstBaseSink * basesink, GstBuffer * buf)
243 {
244   GstTest *test = GST_TEST (basesink);
245   guint i;
246
247   for (i = 0; i < TESTS_COUNT; i++) {
248     if (test->tests[i]) {
249       tests[i].add (test->tests[i], buf);
250     }
251   }
252   return GST_FLOW_OK;
253 }
254
255 static gboolean
256 gst_test_start (GstBaseSink * sink)
257 {
258   GstTest *test = GST_TEST (sink);
259
260   tests_set (test);
261   return TRUE;
262 }
263
264 static gboolean
265 gst_test_stop (GstBaseSink * sink)
266 {
267   GstTest *test = GST_TEST (sink);
268
269   tests_unset (test);
270   return TRUE;
271 }
272
273 static void
274 gst_test_set_property (GObject * object, guint prop_id,
275     const GValue * value, GParamSpec * pspec)
276 {
277   GstTest *test = GST_TEST (object);
278
279   if (prop_id == 0 || prop_id > 2 * TESTS_COUNT) {
280     G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
281     return;
282   }
283
284   if (prop_id % 2) {
285     /* real values can't be set */
286     G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
287   } else {
288     /* expected values */
289     GST_OBJECT_LOCK (test);
290     g_value_copy (value, &test->values[prop_id / 2 - 1]);
291     GST_OBJECT_UNLOCK (test);
292   }
293 }
294
295 static void
296 gst_test_get_property (GObject * object, guint prop_id, GValue * value,
297     GParamSpec * pspec)
298 {
299   GstTest *test = GST_TEST (object);
300   guint id = (prop_id - 1) / 2;
301
302   if (prop_id == 0 || prop_id > 2 * TESTS_COUNT) {
303     G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
304     return;
305   }
306
307   GST_OBJECT_LOCK (test);
308
309   if (prop_id % 2) {
310     /* real values */
311     tests[id].get_value (test->tests[id], value);
312   } else {
313     /* expected values */
314     g_value_copy (&test->values[id], value);
315   }
316
317   GST_OBJECT_UNLOCK (test);
318 }