Blame tests/check/elements/funnel.c

Packit f546b1
/* GStreamer unit tests for the funnel
Packit f546b1
 *
Packit f546b1
 * Copyright (C) 2008 Collabora, Nokia
Packit f546b1
 * @author: Olivier Crete <olivier.crete@collabora.co.uk>
Packit f546b1
 *
Packit f546b1
 * This library is free software; you can redistribute it and/or
Packit f546b1
 * modify it under the terms of the GNU Lesser General Public
Packit f546b1
 * License as published by the Free Software Foundation; either
Packit f546b1
 * version 2.1 of the License, or (at your option) any later version.
Packit f546b1
 *
Packit f546b1
 * This library is distributed in the hope that it will be useful,
Packit f546b1
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit f546b1
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit f546b1
 * Lesser General Public License for more details.
Packit f546b1
 *
Packit f546b1
 * You should have received a copy of the GNU Lesser General Public
Packit f546b1
 * License along with this library; if not, write to the Free Software
Packit f546b1
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301 USA
Packit f546b1
*/
Packit f546b1
Packit f546b1
Packit f546b1
#ifdef HAVE_CONFIG_H
Packit f546b1
# include <config.h>
Packit f546b1
#endif
Packit f546b1
Packit f546b1
#include <gst/check/gstharness.h>
Packit f546b1
#include <gst/check/gstcheck.h>
Packit f546b1
Packit f546b1
struct TestData
Packit f546b1
{
Packit f546b1
  GstElement *funnel;
Packit f546b1
  GstPad *funnelsrc, *funnelsink11, *funnelsink22;
Packit f546b1
  GstPad *mysink, *mysrc1, *mysrc2;
Packit f546b1
  GstCaps *mycaps;
Packit f546b1
};
Packit f546b1
Packit f546b1
static void
Packit f546b1
setup_test_objects (struct TestData *td, GstPadChainFunction chain_func)
Packit f546b1
{
Packit f546b1
  td->mycaps = gst_caps_new_empty_simple ("test/test");
Packit f546b1
Packit f546b1
  td->funnel = gst_element_factory_make ("funnel", NULL);
Packit f546b1
Packit f546b1
  td->funnelsrc = gst_element_get_static_pad (td->funnel, "src");
Packit f546b1
  fail_unless (td->funnelsrc != NULL);
Packit f546b1
Packit f546b1
  td->funnelsink11 = gst_element_get_request_pad (td->funnel, "sink_11");
Packit f546b1
  fail_unless (td->funnelsink11 != NULL);
Packit f546b1
  fail_unless (!strcmp (GST_OBJECT_NAME (td->funnelsink11), "sink_11"));
Packit f546b1
Packit f546b1
  td->funnelsink22 = gst_element_get_request_pad (td->funnel, "sink_22");
Packit f546b1
  fail_unless (td->funnelsink22 != NULL);
Packit f546b1
  fail_unless (!strcmp (GST_OBJECT_NAME (td->funnelsink22), "sink_22"));
Packit f546b1
Packit f546b1
  fail_unless (gst_element_set_state (td->funnel, GST_STATE_PLAYING) ==
Packit f546b1
      GST_STATE_CHANGE_SUCCESS);
Packit f546b1
Packit f546b1
  td->mysink = gst_pad_new ("sink", GST_PAD_SINK);
Packit f546b1
  gst_pad_set_chain_function (td->mysink, chain_func);
Packit f546b1
  gst_pad_set_active (td->mysink, TRUE);
Packit f546b1
Packit f546b1
  td->mysrc1 = gst_pad_new ("src1", GST_PAD_SRC);
Packit f546b1
  gst_pad_set_active (td->mysrc1, TRUE);
Packit f546b1
  gst_check_setup_events_with_stream_id (td->mysrc1, td->funnel, td->mycaps,
Packit f546b1
      GST_FORMAT_BYTES, "test1");
Packit f546b1
Packit f546b1
  td->mysrc2 = gst_pad_new ("src2", GST_PAD_SRC);
Packit f546b1
  gst_pad_set_active (td->mysrc2, TRUE);
Packit f546b1
  gst_check_setup_events_with_stream_id (td->mysrc2, td->funnel, td->mycaps,
Packit f546b1
      GST_FORMAT_BYTES, "test2");
Packit f546b1
Packit f546b1
  fail_unless (GST_PAD_LINK_SUCCESSFUL (gst_pad_link (td->funnelsrc,
Packit f546b1
              td->mysink)));
Packit f546b1
Packit f546b1
  fail_unless (GST_PAD_LINK_SUCCESSFUL (gst_pad_link (td->mysrc1,
Packit f546b1
              td->funnelsink11)));
Packit f546b1
Packit f546b1
  fail_unless (GST_PAD_LINK_SUCCESSFUL (gst_pad_link (td->mysrc2,
Packit f546b1
              td->funnelsink22)));
Packit f546b1
Packit f546b1
}
Packit f546b1
Packit f546b1
static void
Packit f546b1
release_test_objects (struct TestData *td)
Packit f546b1
{
Packit f546b1
  gst_pad_set_active (td->mysink, FALSE);
Packit f546b1
  gst_pad_set_active (td->mysrc1, FALSE);
Packit f546b1
  gst_pad_set_active (td->mysrc1, FALSE);
Packit f546b1
Packit f546b1
  gst_object_unref (td->mysink);
Packit f546b1
  gst_object_unref (td->mysrc1);
Packit f546b1
  gst_object_unref (td->mysrc2);
Packit f546b1
Packit f546b1
  fail_unless (gst_element_set_state (td->funnel, GST_STATE_NULL) ==
Packit f546b1
      GST_STATE_CHANGE_SUCCESS);
Packit f546b1
Packit f546b1
  gst_object_unref (td->funnelsrc);
Packit f546b1
  gst_element_release_request_pad (td->funnel, td->funnelsink11);
Packit f546b1
  gst_object_unref (td->funnelsink11);
Packit f546b1
  gst_element_release_request_pad (td->funnel, td->funnelsink22);
Packit f546b1
  gst_object_unref (td->funnelsink22);
Packit f546b1
Packit f546b1
  gst_caps_unref (td->mycaps);
Packit f546b1
  gst_object_unref (td->funnel);
Packit f546b1
}
Packit f546b1
Packit f546b1
static gint bufcount = 0;
Packit f546b1
static gint alloccount = 0;
Packit f546b1
Packit f546b1
static GstFlowReturn
Packit f546b1
chain_ok (GstPad * pad, GstObject * parent, GstBuffer * buffer)
Packit f546b1
{
Packit f546b1
  bufcount++;
Packit f546b1
Packit f546b1
  gst_buffer_unref (buffer);
Packit f546b1
Packit f546b1
  return GST_FLOW_OK;
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_START_TEST (test_funnel_simple)
Packit f546b1
{
Packit f546b1
  struct TestData td;
Packit f546b1
Packit f546b1
  setup_test_objects (&td, chain_ok);
Packit f546b1
Packit f546b1
  bufcount = 0;
Packit f546b1
  alloccount = 0;
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
Packit f546b1
  fail_unless (bufcount == 2);
Packit f546b1
Packit f546b1
  release_test_objects (&td);
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_END_TEST;
Packit f546b1
Packit f546b1
guint num_eos = 0;
Packit f546b1
Packit f546b1
static gboolean
Packit f546b1
eos_event_func (GstPad * pad, GstObject * parent, GstEvent * event)
Packit f546b1
{
Packit f546b1
  if (GST_EVENT_TYPE (event) == GST_EVENT_EOS)
Packit f546b1
    ++num_eos;
Packit f546b1
Packit f546b1
  return gst_pad_event_default (pad, parent, event);
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_START_TEST (test_funnel_eos)
Packit f546b1
{
Packit f546b1
  struct TestData td;
Packit f546b1
  GstSegment segment;
Packit f546b1
Packit f546b1
  setup_test_objects (&td, chain_ok);
Packit f546b1
Packit f546b1
  num_eos = 0;
Packit f546b1
  bufcount = 0;
Packit f546b1
Packit f546b1
  gst_pad_set_event_function (td.mysink, eos_event_func);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
Packit f546b1
  fail_unless (bufcount == 2);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_eos ()));
Packit f546b1
  fail_unless (num_eos == 0);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_EOS);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
Packit f546b1
  fail_unless (bufcount == 3);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_eos ()));
Packit f546b1
  fail_unless (num_eos == 1);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_EOS);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_EOS);
Packit f546b1
Packit f546b1
  fail_unless (bufcount == 3);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_flush_start ()));
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_flush_stop (TRUE)));
Packit f546b1
Packit f546b1
  gst_segment_init (&segment, GST_FORMAT_BYTES);
Packit f546b1
  gst_pad_push_event (td.mysrc1, gst_event_new_segment (&segment));
Packit f546b1
  gst_pad_push_event (td.mysrc2, gst_event_new_segment (&segment));
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_EOS);
Packit f546b1
Packit f546b1
  fail_unless (bufcount == 4);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_unlink (td.mysrc1, td.funnelsink11));
Packit f546b1
  gst_element_release_request_pad (td.funnel, td.funnelsink11);
Packit f546b1
  gst_object_unref (td.funnelsink11);
Packit f546b1
  fail_unless (num_eos == 2);
Packit f546b1
Packit f546b1
  td.funnelsink11 = gst_element_get_request_pad (td.funnel, "sink_11");
Packit f546b1
  fail_unless (td.funnelsink11 != NULL);
Packit f546b1
  fail_unless (!strcmp (GST_OBJECT_NAME (td.funnelsink11), "sink_11"));
Packit f546b1
Packit f546b1
  fail_unless (GST_PAD_LINK_SUCCESSFUL (gst_pad_link (td.mysrc1,
Packit f546b1
              td.funnelsink11)));
Packit f546b1
Packit f546b1
  /* This will fail because everything is EOS already */
Packit f546b1
  fail_if (gst_pad_push_event (td.mysrc1, gst_event_new_eos ()));
Packit f546b1
  fail_unless (num_eos == 2);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_unlink (td.mysrc1, td.funnelsink11));
Packit f546b1
  gst_element_release_request_pad (td.funnel, td.funnelsink11);
Packit f546b1
  gst_object_unref (td.funnelsink11);
Packit f546b1
  fail_unless (num_eos == 2);
Packit f546b1
Packit f546b1
  /* send only eos to check, it handles empty streams */
Packit f546b1
  td.funnelsink11 = gst_element_get_request_pad (td.funnel, "sink_11");
Packit f546b1
  fail_unless (td.funnelsink11 != NULL);
Packit f546b1
  fail_unless (!strcmp (GST_OBJECT_NAME (td.funnelsink11), "sink_11"));
Packit f546b1
Packit f546b1
  fail_unless (GST_PAD_LINK_SUCCESSFUL (gst_pad_link (td.mysrc1,
Packit f546b1
              td.funnelsink11)));
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_flush_start ()));
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_flush_stop (TRUE)));
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_flush_start ()));
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_flush_stop (TRUE)));
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_eos ()));
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_eos ()));
Packit f546b1
  fail_unless (num_eos == 3);
Packit f546b1
Packit f546b1
  fail_unless (gst_pad_unlink (td.mysrc1, td.funnelsink11));
Packit f546b1
  gst_element_release_request_pad (td.funnel, td.funnelsink11);
Packit f546b1
  gst_object_unref (td.funnelsink11);
Packit f546b1
  fail_unless (num_eos == 3);
Packit f546b1
Packit f546b1
  td.funnelsink11 = gst_element_get_request_pad (td.funnel, "sink_11");
Packit f546b1
  fail_unless (td.funnelsink11 != NULL);
Packit f546b1
  fail_unless (!strcmp (GST_OBJECT_NAME (td.funnelsink11), "sink_11"));
Packit f546b1
Packit f546b1
  release_test_objects (&td);
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_END_TEST;
Packit f546b1
Packit f546b1
guint nb_stream_start_event = 0;
Packit f546b1
guint nb_caps_event = 0;
Packit f546b1
guint nb_segment_event = 0;
Packit f546b1
guint nb_gap_event = 0;
Packit f546b1
Packit f546b1
static GstPadProbeReturn
Packit f546b1
event_counter (GstObject * pad, GstPadProbeInfo * info, gpointer user_data)
Packit f546b1
{
Packit f546b1
  GstEvent *event = GST_PAD_PROBE_INFO_EVENT (info);
Packit f546b1
Packit f546b1
  fail_unless (event != NULL);
Packit f546b1
  fail_unless (GST_IS_EVENT (event));
Packit f546b1
Packit f546b1
  switch (GST_EVENT_TYPE (event)) {
Packit f546b1
    case GST_EVENT_STREAM_START:
Packit f546b1
      ++nb_stream_start_event;
Packit f546b1
      break;
Packit f546b1
    case GST_EVENT_CAPS:
Packit f546b1
      ++nb_caps_event;
Packit f546b1
      break;
Packit f546b1
    case GST_EVENT_SEGMENT:
Packit f546b1
      ++nb_segment_event;
Packit f546b1
      break;
Packit f546b1
    case GST_EVENT_GAP:
Packit f546b1
      ++nb_gap_event;
Packit f546b1
      break;
Packit f546b1
    default:
Packit f546b1
      break;
Packit f546b1
  }
Packit f546b1
Packit f546b1
  return GST_PAD_PROBE_OK;
Packit f546b1
}
Packit f546b1
Packit f546b1
/*
Packit f546b1
 * Push GAP events into funnel to forward sticky events.
Packit f546b1
 * Funnel element shoud also treat GAP events likes buffers.
Packit f546b1
 * For example, funnel can be used for internal subtitle with streamiddemux. 
Packit f546b1
 *  +--------------------------------------------------------------------------+
Packit f546b1
 *  | playbin                               +--------------------------------+ |
Packit f546b1
 *  | +--------------+  +----------------+  | +------------+     playsink    | |
Packit f546b1
 *  | | uridecodebin |  | input-selector |  | | video-sink |                 | |
Packit f546b1
 *  | |              |  +----------------+  | +------------+                 | |
Packit f546b1
 *  | |              |                      |                                | |
Packit f546b1
 *  | |              |  +----------------+  | +------------+                 | |
Packit f546b1
 *  | |              |  | input-selector |  | | audio-sink |                 | |
Packit f546b1
 *  | |              |  +----------------+  | +------------+                 | |
Packit f546b1
 *  | |              |                      |                                | |
Packit f546b1
 *  | |              |  +----------------+  | +---------------+ +----------+ | |
Packit f546b1
 *  | |              |  | funnel         |  | | streamiddemux | | appsink0 | | |
Packit f546b1
 *  | +--------------+  +----------------+  | +---------------+ +----------+ | |
Packit f546b1
 *  |                                       |                   +----------+ | |
Packit f546b1
 *  |                                       |                   | appsinkn | | |
Packit f546b1
 *  |                                       |                   +----------+ | |
Packit f546b1
 *  |                                       +--------------------------------+ |
Packit f546b1
 *  +--------------------------------------------------------------------------+
Packit f546b1
 * If no data was received in funnel and then sticky events can be pending continuously. 
Packit f546b1
 * And streamiddemux only receive gap events continuously. 
Packit f546b1
 * Thus, pipeline can not be constructed completely.
Packit f546b1
 * For support it, need to handle GAP events likes buffers.
Packit f546b1
 */
Packit f546b1
GST_START_TEST (test_funnel_gap_event)
Packit f546b1
{
Packit f546b1
  struct TestData td;
Packit f546b1
  guint probe = 0;
Packit f546b1
Packit f546b1
  setup_test_objects (&td, chain_ok);
Packit f546b1
Packit f546b1
  nb_stream_start_event = 0;
Packit f546b1
  nb_caps_event = 0;
Packit f546b1
  nb_segment_event = 0;
Packit f546b1
  nb_gap_event = 0;
Packit f546b1
  bufcount = 0;
Packit f546b1
Packit f546b1
  probe = gst_pad_add_probe (td.mysink, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM,
Packit f546b1
      (GstPadProbeCallback) event_counter, NULL, NULL);
Packit f546b1
Packit f546b1
  /* push a gap event to srcpad1 to push sticky events */
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_gap (0,
Packit f546b1
              GST_SECOND)));
Packit f546b1
Packit f546b1
  fail_unless (nb_stream_start_event == 1);
Packit f546b1
  fail_unless (nb_caps_event == 1);
Packit f546b1
  fail_unless (nb_segment_event == 1);
Packit f546b1
  fail_unless (nb_gap_event == 1);
Packit f546b1
Packit f546b1
  /* push a gap event to srcpad2 to push sticky events */
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_gap (0,
Packit f546b1
              GST_SECOND)));
Packit f546b1
Packit f546b1
  fail_unless (nb_stream_start_event == 2);
Packit f546b1
  fail_unless (nb_caps_event == 2);
Packit f546b1
  fail_unless (nb_segment_event == 2);
Packit f546b1
  fail_unless (nb_gap_event == 2);
Packit f546b1
Packit f546b1
  /* push a gap event to srcpad2 */
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc2, gst_event_new_gap (0,
Packit f546b1
              GST_SECOND)));
Packit f546b1
Packit f546b1
  fail_unless (nb_stream_start_event == 2);
Packit f546b1
  fail_unless (nb_caps_event == 2);
Packit f546b1
  fail_unless (nb_segment_event == 2);
Packit f546b1
  fail_unless (nb_gap_event == 3);
Packit f546b1
Packit f546b1
  /* push a gap event to srcpad1 */
Packit f546b1
  fail_unless (gst_pad_push_event (td.mysrc1, gst_event_new_gap (0,
Packit f546b1
              GST_SECOND)));
Packit f546b1
Packit f546b1
  fail_unless (nb_stream_start_event == 3);
Packit f546b1
  fail_unless (nb_caps_event == 3);
Packit f546b1
  fail_unless (nb_segment_event == 3);
Packit f546b1
  fail_unless (nb_gap_event == 4);
Packit f546b1
Packit f546b1
  /* push buffer */
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc1, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
  fail_unless (gst_pad_push (td.mysrc2, gst_buffer_new ()) == GST_FLOW_OK);
Packit f546b1
Packit f546b1
  fail_unless (nb_stream_start_event == 4);
Packit f546b1
  fail_unless (nb_caps_event == 4);
Packit f546b1
  fail_unless (nb_segment_event == 4);
Packit f546b1
  fail_unless (nb_gap_event == 4);
Packit f546b1
  fail_unless (bufcount == 2);
Packit f546b1
Packit f546b1
  gst_pad_remove_probe (td.mysink, probe);
Packit f546b1
Packit f546b1
  release_test_objects (&td);
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_END_TEST;
Packit f546b1
Packit f546b1
GST_START_TEST (test_funnel_stress)
Packit f546b1
{
Packit f546b1
  GstHarness *h0 = gst_harness_new_with_padnames ("funnel", "sink_0", "src");
Packit f546b1
  GstHarness *h1 = gst_harness_new_with_element (h0->element, "sink_1", NULL);
Packit f546b1
  GstHarnessThread *req, *push0, *push1;
Packit f546b1
  GstPadTemplate *templ =
Packit f546b1
      gst_element_class_get_pad_template (GST_ELEMENT_GET_CLASS (h0->element),
Packit f546b1
      "sink_%u");
Packit f546b1
  GstCaps *caps = gst_caps_from_string ("testcaps");
Packit f546b1
  GstBuffer *buf = gst_buffer_new ();
Packit f546b1
  GstSegment segment;
Packit f546b1
Packit f546b1
  gst_segment_init (&segment, GST_FORMAT_TIME);
Packit f546b1
Packit f546b1
  req = gst_harness_stress_requestpad_start (h0, templ, NULL, NULL, TRUE);
Packit f546b1
  push0 = gst_harness_stress_push_buffer_start (h0, caps, &segment, buf);
Packit f546b1
  push1 = gst_harness_stress_push_buffer_start (h1, caps, &segment, buf);
Packit f546b1
Packit f546b1
  gst_caps_unref (caps);
Packit f546b1
  gst_buffer_unref (buf);
Packit f546b1
Packit f546b1
  /* test-length */
Packit f546b1
  g_usleep (G_USEC_PER_SEC * 1);
Packit f546b1
Packit f546b1
  gst_harness_stress_thread_stop (push1);
Packit f546b1
  gst_harness_stress_thread_stop (push0);
Packit f546b1
  gst_harness_stress_thread_stop (req);
Packit f546b1
Packit f546b1
  gst_harness_teardown (h1);
Packit f546b1
  gst_harness_teardown (h0);
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_END_TEST;
Packit f546b1
Packit f546b1
Packit f546b1
static Suite *
Packit f546b1
funnel_suite (void)
Packit f546b1
{
Packit f546b1
  Suite *s = suite_create ("funnel");
Packit f546b1
  TCase *tc_chain;
Packit f546b1
Packit f546b1
  tc_chain = tcase_create ("funnel simple");
Packit f546b1
  tcase_add_test (tc_chain, test_funnel_simple);
Packit f546b1
  tcase_add_test (tc_chain, test_funnel_eos);
Packit f546b1
  tcase_add_test (tc_chain, test_funnel_gap_event);
Packit f546b1
  tcase_add_test (tc_chain, test_funnel_stress);
Packit f546b1
  suite_add_tcase (s, tc_chain);
Packit f546b1
Packit f546b1
  return s;
Packit f546b1
}
Packit f546b1
Packit f546b1
GST_CHECK_MAIN (funnel);