check/gst/gstutils.c: New file.

Original commit message from CVS:
2005-09-21  Andy Wingo  <wingo@pobox.com>

* check/gst/gstutils.c: New file.
(test_buffer_probe_n_times): A simple buffer probe test. More to
come, foolios.

* gst/gstutils.c (gst_pad_add_buffer_probe): Connect to
have-data::buffer, not have-data.
(gst_pad_add_event_probe): Likewise for have-data::event.
(gst_pad_add_data_probe): More docs. The part about 'resolving the
peer' isn't quite right yet though.
(gst_pad_remove_buffer_probe, gst_pad_remove_event_probe)
(gst_pad_remove_data_probe): Change to take the guint handler_id
as their arg, not the function+data, which is more glib-like.

* gst/gstpad.c (gst_pad_emit_have_data_signal): Add a detail to
the signal emission to indicate if the data is a buffer or an
event.
(gst_pad_get_type): Initialize buffer and event quarks.
(gst_pad_class_init): have-data is now a detailed signal, yes it
is.
This commit is contained in:
Andy Wingo 2005-09-21 12:21:10 +00:00
parent 9894c90109
commit 5b77a67125
9 changed files with 273 additions and 69 deletions

View file

@ -45,6 +45,7 @@ check_PROGRAMS = \
gst/gstsystemclock \ gst/gstsystemclock \
gst/gststructure \ gst/gststructure \
gst/gsttag \ gst/gsttag \
gst/gstutils \
gst/gstvalue \ gst/gstvalue \
elements/fakesrc \ elements/fakesrc \
elements/identity \ elements/identity \

96
check/gst/gstutils.c Normal file
View file

@ -0,0 +1,96 @@
/* GStreamer
* Copyright (C) <2005> Thomas Vander Stichele <thomas at apestaart dot org>
*
* gstutils.c: Unit test for functions in gstutils
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include <gst/check/gstcheck.h>
#define SPECIAL_POINTER (void*)19283847
static int n_buffer_probes = 0;
static gboolean
buffer_probe (GstPad * pad, GstBuffer * obj, gpointer data)
{
n_buffer_probes++;
g_assert (data == SPECIAL_POINTER);
return TRUE;
}
GST_START_TEST (test_buffer_probe_n_times)
{
GstElement *pipeline, *fakesrc, *fakesink;
GstBus *bus;
GstMessage *message;
GstPad *pad;
guint id;
pipeline = gst_element_factory_make ("pipeline", NULL);
fakesrc = gst_element_factory_make ("fakesrc", NULL);
fakesink = gst_element_factory_make ("fakesink", NULL);
g_object_set (fakesrc, "num-buffers", (int) 10, NULL);
gst_bin_add_many (GST_BIN (pipeline), fakesrc, fakesink, NULL);
gst_element_link (fakesrc, fakesink);
pad = gst_element_get_pad (fakesink, "sink");
id = gst_pad_add_buffer_probe (pad, G_CALLBACK (buffer_probe),
SPECIAL_POINTER);
gst_object_unref (pad);
gst_element_set_state (pipeline, GST_STATE_PLAYING);
bus = gst_element_get_bus (pipeline);
message = gst_bus_poll (bus, GST_MESSAGE_EOS, -1);
gst_message_unref (message);
gst_object_unref (bus);
gst_element_set_state (pipeline, GST_STATE_NULL);
gst_object_unref (pipeline);
g_assert (n_buffer_probes == 10);
} GST_END_TEST;
Suite *
gst_utils_suite (void)
{
Suite *s = suite_create ("GstUtils");
TCase *tc_chain = tcase_create ("general");
suite_add_tcase (s, tc_chain);
tcase_add_test (tc_chain, test_buffer_probe_n_times);
return s;
}
int
main (int argc, char **argv)
{
int nf;
Suite *s = gst_utils_suite ();
SRunner *sr = srunner_create (s);
gst_check_init (&argc, &argv);
srunner_run_all (sr, CK_NORMAL);
nf = srunner_ntests_failed (sr);
srunner_free (sr);
return nf;
}

View file

@ -1306,6 +1306,8 @@ gst_element_request_pad_by_name() or gst_element_request_compatible_pad().
</para> </para>
@pad: @pad:
@handler_id:
<!-- # Unused Parameters # -->
@handler: @handler:
@data: @data:
@ -1316,6 +1318,8 @@ gst_element_request_pad_by_name() or gst_element_request_compatible_pad().
</para> </para>
@pad: @pad:
@handler_id:
<!-- # Unused Parameters # -->
@handler: @handler:
@data: @data:
@ -1326,6 +1330,8 @@ gst_element_request_pad_by_name() or gst_element_request_compatible_pad().
</para> </para>
@pad: @pad:
@handler_id:
<!-- # Unused Parameters # -->
@handler: @handler:
@data: @data:

View file

@ -27,6 +27,9 @@ network connections also need a protocol to do this.
#GstBuffer, #GstCaps, #GstEvent #GstBuffer, #GstCaps, #GstEvent
</para> </para>
<!-- ##### SECTION Stability_Level ##### -->
<!-- ##### ENUM GstDPHeaderFlag ##### --> <!-- ##### ENUM GstDPHeaderFlag ##### -->
<para> <para>

View file

@ -96,6 +96,9 @@ static xmlNodePtr gst_pad_save_thyself (GstObject * object, xmlNodePtr parent);
static GstObjectClass *pad_parent_class = NULL; static GstObjectClass *pad_parent_class = NULL;
static guint gst_pad_signals[PAD_LAST_SIGNAL] = { 0 }; static guint gst_pad_signals[PAD_LAST_SIGNAL] = { 0 };
static GQuark buffer_quark;
static GQuark event_quark;
GType GType
gst_pad_get_type (void) gst_pad_get_type (void)
{ {
@ -111,6 +114,9 @@ gst_pad_get_type (void)
_gst_pad_type = g_type_register_static (GST_TYPE_OBJECT, "GstPad", _gst_pad_type = g_type_register_static (GST_TYPE_OBJECT, "GstPad",
&pad_info, 0); &pad_info, 0);
buffer_quark = g_quark_from_static_string ("buffer");
event_quark = g_quark_from_static_string ("event");
GST_DEBUG_CATEGORY_INIT (debug_dataflow, "GST_DATAFLOW", GST_DEBUG_CATEGORY_INIT (debug_dataflow, "GST_DATAFLOW",
GST_DEBUG_BOLD | GST_DEBUG_FG_GREEN, "dataflow inside pads"); GST_DEBUG_BOLD | GST_DEBUG_FG_GREEN, "dataflow inside pads");
} }
@ -200,7 +206,8 @@ gst_pad_class_init (GstPadClass * klass)
*/ */
gst_pad_signals[PAD_HAVE_DATA] = gst_pad_signals[PAD_HAVE_DATA] =
g_signal_new ("have-data", G_TYPE_FROM_CLASS (klass), g_signal_new ("have-data", G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstPadClass, have_data), G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED,
G_STRUCT_OFFSET (GstPadClass, have_data),
_gst_do_pass_data_accumulator, _gst_do_pass_data_accumulator,
NULL, gst_marshal_BOOLEAN__POINTER, G_TYPE_BOOLEAN, 1, NULL, gst_marshal_BOOLEAN__POINTER, G_TYPE_BOOLEAN, 1,
GST_TYPE_MINI_OBJECT); GST_TYPE_MINI_OBJECT);
@ -2888,6 +2895,7 @@ gst_pad_emit_have_data_signal (GstPad * pad, GstMiniObject * obj)
GValue ret = { 0 }; GValue ret = { 0 };
GValue args[2] = { {0}, {0} }; GValue args[2] = { {0}, {0} };
gboolean res; gboolean res;
GQuark detail;
/* init */ /* init */
g_value_init (&ret, G_TYPE_BOOLEAN); g_value_init (&ret, G_TYPE_BOOLEAN);
@ -2897,8 +2905,13 @@ gst_pad_emit_have_data_signal (GstPad * pad, GstMiniObject * obj)
g_value_init (&args[1], GST_TYPE_MINI_OBJECT); // G_TYPE_POINTER); g_value_init (&args[1], GST_TYPE_MINI_OBJECT); // G_TYPE_POINTER);
gst_value_set_mini_object (&args[1], obj); gst_value_set_mini_object (&args[1], obj);
if (GST_IS_EVENT (obj))
detail = event_quark;
else
detail = buffer_quark;
/* actually emit */ /* actually emit */
g_signal_emitv (args, gst_pad_signals[PAD_HAVE_DATA], 0, &ret); g_signal_emitv (args, gst_pad_signals[PAD_HAVE_DATA], detail, &ret);
res = g_value_get_boolean (&ret); res = g_value_get_boolean (&ret);
/* clean up */ /* clean up */

View file

@ -2431,13 +2431,29 @@ gst_atomic_int_set (gint * atomic_int, gint value)
* @handler: function to call when data is passed over pad * @handler: function to call when data is passed over pad
* @data: data to pass along with the handler * @data: data to pass along with the handler
* *
* Connects a signal handler to the pad's have-data signal, and increases * Adds a "data probe" to a pad. This function will be called whenever data
* the do_{buffer,event}_signals number on the pads so that those * passes through a pad. In this case data means both events and buffers. The
* signals are actually fired. * probe will be called with the data as an argument. Note that the data will
* have a reference count greater than 1, so it will be immutable -- you must
* not change it.
* *
* Returns: The handler id * For source pads, the probe will be called after the blocking function, if any
* (see gst_pad_set_blocked_async()), but before looking up the peer to chain
* to. For sink pads, the probe function will be called before configuring the
* sink with new caps, if any, and before calling the pad's chain function.
*
* Your data probe should return TRUE to let the data continue to flow, or FALSE
* to drop it. Dropping data is rarely useful, but occasionally comes in handy
* with events.
*
* Although probes are implemeted internally by connecting @handler to the
* have-data signal on the pad, if you want to remove a probe it is insufficient
* to only call g_signal_handler_disconnect on the returned handler id. To
* remove a probe, use the appropriate function, such as
* gst_pad_remove_data_probe().
*
* Returns: The handler id.
*/ */
gulong gulong
gst_pad_add_data_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_add_data_probe (GstPad * pad, GCallback handler, gpointer data)
{ {
@ -2464,13 +2480,11 @@ gst_pad_add_data_probe (GstPad * pad, GCallback handler, gpointer data)
* @handler: function to call when data is passed over pad * @handler: function to call when data is passed over pad
* @data: data to pass along with the handler * @data: data to pass along with the handler
* *
* Connects a signal handler to the pad's have-data signal, and increases * Adds a probe that will be called for all events passing through a pad. See
* the do_event_signals number on the pads so that this signal * gst_pad_add_data_probe() for more information.
* is actually fired.
* *
* Returns: The handler id * Returns: The handler id
*/ */
gulong gulong
gst_pad_add_event_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_add_event_probe (GstPad * pad, GCallback handler, gpointer data)
{ {
@ -2480,7 +2494,7 @@ gst_pad_add_event_probe (GstPad * pad, GCallback handler, gpointer data)
g_return_val_if_fail (handler != NULL, 0); g_return_val_if_fail (handler != NULL, 0);
GST_LOCK (pad); GST_LOCK (pad);
sigid = g_signal_connect (pad, "have-data", handler, data); sigid = g_signal_connect (pad, "have-data::event", handler, data);
GST_PAD_DO_EVENT_SIGNALS (pad)++; GST_PAD_DO_EVENT_SIGNALS (pad)++;
GST_DEBUG ("adding event probe to pad %s:%s, now %d probes", GST_DEBUG ("adding event probe to pad %s:%s, now %d probes",
GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad)); GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad));
@ -2495,13 +2509,11 @@ gst_pad_add_event_probe (GstPad * pad, GCallback handler, gpointer data)
* @handler: function to call when data is passed over pad * @handler: function to call when data is passed over pad
* @data: data to pass along with the handler * @data: data to pass along with the handler
* *
* Connects a signal handler to the pad's have-data signal, and increases * Adds a probe that will be called for all buffers passing through a pad. See
* the do_buffer_signals number on the pads so that this signal * gst_pad_add_data_probe() for more information.
* is actually fired.
* *
* Returns: The handler id * Returns: The handler id
*/ */
gulong gulong
gst_pad_add_buffer_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_add_buffer_probe (GstPad * pad, GCallback handler, gpointer data)
{ {
@ -2511,7 +2523,7 @@ gst_pad_add_buffer_probe (GstPad * pad, GCallback handler, gpointer data)
g_return_val_if_fail (handler != NULL, 0); g_return_val_if_fail (handler != NULL, 0);
GST_LOCK (pad); GST_LOCK (pad);
sigid = g_signal_connect (pad, "have-data", handler, data); sigid = g_signal_connect (pad, "have-data::buffer", handler, data);
GST_PAD_DO_BUFFER_SIGNALS (pad)++; GST_PAD_DO_BUFFER_SIGNALS (pad)++;
GST_DEBUG ("adding buffer probe to pad %s:%s, now %d probes", GST_DEBUG ("adding buffer probe to pad %s:%s, now %d probes",
GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_BUFFER_SIGNALS (pad)); GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_BUFFER_SIGNALS (pad));
@ -2523,29 +2535,23 @@ gst_pad_add_buffer_probe (GstPad * pad, GCallback handler, gpointer data)
/** /**
* gst_pad_remove_data_probe: * gst_pad_remove_data_probe:
* @pad: pad to remove the data probe handler from * @pad: pad to remove the data probe handler from
* @handler: function that was assigned to the signal * @handler_id: handler id returned from gst_pad_add_data_probe
* @data: data that was assigned to the signal handler
* *
* Unconnects a signal handler to the pad's have-data signal, and decreases * Removes a data probe from @pad.
* the do_{buffer,event}_signals number on the pads so that those
* signals are actually no more fired if no signals are connected.
*/ */
void void
gst_pad_remove_data_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_remove_data_probe (GstPad * pad, guint handler_id)
{ {
guint count;
g_return_if_fail (GST_IS_PAD (pad)); g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (handler != NULL); g_return_if_fail (handler_id > 0);
GST_LOCK (pad); GST_LOCK (pad);
count = g_signal_handlers_disconnect_by_func (pad, handler, data); g_signal_handler_disconnect (pad, handler_id);
GST_PAD_DO_BUFFER_SIGNALS (pad) -= count; GST_PAD_DO_BUFFER_SIGNALS (pad)--;
GST_PAD_DO_EVENT_SIGNALS (pad) -= count; GST_PAD_DO_EVENT_SIGNALS (pad)--;
GST_DEBUG GST_DEBUG
("removing %d data probes from pad %s:%s, now %d event, %d buffer probes", ("removed data probe from pad %s:%s, now %d event, %d buffer probes",
count, GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad), GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad),
GST_PAD_DO_BUFFER_SIGNALS (pad)); GST_PAD_DO_BUFFER_SIGNALS (pad));
GST_UNLOCK (pad); GST_UNLOCK (pad);
} }
@ -2553,54 +2559,42 @@ gst_pad_remove_data_probe (GstPad * pad, GCallback handler, gpointer data)
/** /**
* gst_pad_remove_event_probe: * gst_pad_remove_event_probe:
* @pad: pad to remove the event probe handler from * @pad: pad to remove the event probe handler from
* @handler: function that was assigned to the signal * @handler_id: handler id returned from gst_pad_add_event_probe
* @data: data that was assigned to the signal handler
* *
* Unconnects a signal handler to the pad's have-data signal, and decreases * Removes an event probe from @pad.
* the do_event_signals number on the pads so that this signal is
* actually no more fired if no signals are connected.
*/ */
void void
gst_pad_remove_event_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_remove_event_probe (GstPad * pad, guint handler_id)
{ {
guint count;
g_return_if_fail (GST_IS_PAD (pad)); g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (handler != NULL); g_return_if_fail (handler_id > 0);
GST_LOCK (pad); GST_LOCK (pad);
count = g_signal_handlers_disconnect_by_func (pad, handler, data); g_signal_handler_disconnect (pad, handler_id);
GST_PAD_DO_EVENT_SIGNALS (pad) -= count; GST_PAD_DO_EVENT_SIGNALS (pad)--;
GST_DEBUG ("removing %d event probes from pad %s:%s, now %d event probes", GST_DEBUG ("removed event probe from pad %s:%s, now %d event probes",
count, GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad)); GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_EVENT_SIGNALS (pad));
GST_UNLOCK (pad); GST_UNLOCK (pad);
} }
/** /**
* gst_pad_remove_buffer_probe: * gst_pad_remove_buffer_probe:
* @pad: pad to remove the buffer probe handler from * @pad: pad to remove the buffer probe handler from
* @handler: function that was assigned to the signal * @handler_id: handler id returned from gst_pad_add_buffer_probe
* @data: data that was assigned to the signal handler
* *
* Unconnects a signal handler to the pad's have-data signal, and decreases * Removes a buffer probe from @pad.
* the emit_buffer_signals number on the pads so that this signal is
* actually no more fired if no signals are connected.
*/ */
void void
gst_pad_remove_buffer_probe (GstPad * pad, GCallback handler, gpointer data) gst_pad_remove_buffer_probe (GstPad * pad, guint handler_id)
{ {
guint count;
g_return_if_fail (GST_IS_PAD (pad)); g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (handler != NULL); g_return_if_fail (handler_id > 0);
GST_LOCK (pad); GST_LOCK (pad);
count = g_signal_handlers_disconnect_by_func (pad, handler, data); g_signal_handler_disconnect (pad, handler_id);
GST_PAD_DO_BUFFER_SIGNALS (pad) -= count; GST_PAD_DO_BUFFER_SIGNALS (pad)--;
GST_DEBUG ("removing %d buffer probes from pad %s:%s, now %d buffer probes", GST_DEBUG ("removed buffer probe from pad %s:%s, now %d buffer probes",
count, GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_BUFFER_SIGNALS (pad)); GST_DEBUG_PAD_NAME (pad), GST_PAD_DO_BUFFER_SIGNALS (pad));
GST_UNLOCK (pad); GST_UNLOCK (pad);
} }

View file

@ -366,21 +366,15 @@ void gst_atomic_int_set (gint * atomic_int, gint
gulong gst_pad_add_data_probe (GstPad * pad, gulong gst_pad_add_data_probe (GstPad * pad,
GCallback handler, GCallback handler,
gpointer data); gpointer data);
void gst_pad_remove_data_probe (GstPad * pad, void gst_pad_remove_data_probe (GstPad * pad, guint handler_id);
GCallback handler,
gpointer data);
gulong gst_pad_add_event_probe (GstPad * pad, gulong gst_pad_add_event_probe (GstPad * pad,
GCallback handler, GCallback handler,
gpointer data); gpointer data);
void gst_pad_remove_event_probe (GstPad * pad, void gst_pad_remove_event_probe (GstPad * pad, guint handler_id);
GCallback handler,
gpointer data);
gulong gst_pad_add_buffer_probe (GstPad * pad, gulong gst_pad_add_buffer_probe (GstPad * pad,
GCallback handler, GCallback handler,
gpointer data); gpointer data);
void gst_pad_remove_buffer_probe (GstPad * pad, void gst_pad_remove_buffer_probe (GstPad * pad, guint handler_id);
GCallback handler,
gpointer data);
/* tag emission utility functions */ /* tag emission utility functions */
void gst_element_found_tags_for_pad (GstElement * element, void gst_element_found_tags_for_pad (GstElement * element,

View file

@ -45,6 +45,7 @@ check_PROGRAMS = \
gst/gstsystemclock \ gst/gstsystemclock \
gst/gststructure \ gst/gststructure \
gst/gsttag \ gst/gsttag \
gst/gstutils \
gst/gstvalue \ gst/gstvalue \
elements/fakesrc \ elements/fakesrc \
elements/identity \ elements/identity \

View file

@ -0,0 +1,96 @@
/* GStreamer
* Copyright (C) <2005> Thomas Vander Stichele <thomas at apestaart dot org>
*
* gstutils.c: Unit test for functions in gstutils
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include <gst/check/gstcheck.h>
#define SPECIAL_POINTER (void*)19283847
static int n_buffer_probes = 0;
static gboolean
buffer_probe (GstPad * pad, GstBuffer * obj, gpointer data)
{
n_buffer_probes++;
g_assert (data == SPECIAL_POINTER);
return TRUE;
}
GST_START_TEST (test_buffer_probe_n_times)
{
GstElement *pipeline, *fakesrc, *fakesink;
GstBus *bus;
GstMessage *message;
GstPad *pad;
guint id;
pipeline = gst_element_factory_make ("pipeline", NULL);
fakesrc = gst_element_factory_make ("fakesrc", NULL);
fakesink = gst_element_factory_make ("fakesink", NULL);
g_object_set (fakesrc, "num-buffers", (int) 10, NULL);
gst_bin_add_many (GST_BIN (pipeline), fakesrc, fakesink, NULL);
gst_element_link (fakesrc, fakesink);
pad = gst_element_get_pad (fakesink, "sink");
id = gst_pad_add_buffer_probe (pad, G_CALLBACK (buffer_probe),
SPECIAL_POINTER);
gst_object_unref (pad);
gst_element_set_state (pipeline, GST_STATE_PLAYING);
bus = gst_element_get_bus (pipeline);
message = gst_bus_poll (bus, GST_MESSAGE_EOS, -1);
gst_message_unref (message);
gst_object_unref (bus);
gst_element_set_state (pipeline, GST_STATE_NULL);
gst_object_unref (pipeline);
g_assert (n_buffer_probes == 10);
} GST_END_TEST;
Suite *
gst_utils_suite (void)
{
Suite *s = suite_create ("GstUtils");
TCase *tc_chain = tcase_create ("general");
suite_add_tcase (s, tc_chain);
tcase_add_test (tc_chain, test_buffer_probe_n_times);
return s;
}
int
main (int argc, char **argv)
{
int nf;
Suite *s = gst_utils_suite ();
SRunner *sr = srunner_create (s);
gst_check_init (&argc, &argv);
srunner_run_all (sr, CK_NORMAL);
nf = srunner_ntests_failed (sr);
srunner_free (sr);
return nf;
}