Blame libs/gst/check/gstconsistencychecker.c

Packit Service 963350
/* GStreamer
Packit Service 963350
 *
Packit Service 963350
 * unit testing helper lib
Packit Service 963350
 *
Packit Service 963350
 * Copyright (C) 2009 Edward Hervey <bilboed@bilboed.com>
Packit Service 963350
 * Copyright (C) 2012 Stefan Sauer <ensonic@users.sf.net>
Packit Service 963350
 *
Packit Service 963350
 * This library is free software; you can redistribute it and/or
Packit Service 963350
 * modify it under the terms of the GNU Library General Public
Packit Service 963350
 * License as published by the Free Software Foundation; either
Packit Service 963350
 * version 2 of the License, or (at your option) any later version.
Packit Service 963350
 *
Packit Service 963350
 * This library is distributed in the hope that it will be useful,
Packit Service 963350
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 963350
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 963350
 * Library General Public License for more details.
Packit Service 963350
 *
Packit Service 963350
 * You should have received a copy of the GNU Library General Public
Packit Service 963350
 * License along with this library; if not, write to the
Packit Service 963350
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit Service 963350
 * Boston, MA 02110-1301, USA.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * SECTION:gstcheckconsistencychecker
Packit Service 963350
 * @title: GstStreamConsistencyChecker
Packit Service 963350
 * @short_description: Data flow consistency checker for GStreamer unit tests.
Packit Service 963350
 *
Packit Service 963350
 * These macros and functions are for internal use of the unit tests found
Packit Service 963350
 * inside the 'check' directories of various GStreamer packages.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#include "gstconsistencychecker.h"
Packit Service 963350
Packit Service 963350
struct _GstStreamConsistency
Packit Service 963350
{
Packit Service 963350
  /* FIXME: do we want to track some states per pad? */
Packit Service 963350
  volatile gboolean flushing;
Packit Service 963350
  volatile gboolean segment;
Packit Service 963350
  volatile gboolean eos;
Packit Service 963350
  volatile gboolean expect_flush;
Packit Service 963350
  volatile gboolean saw_serialized_event;
Packit Service 963350
  volatile gboolean saw_stream_start;
Packit Service 963350
  GstObject *parent;
Packit Service 963350
  GList *pads;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
typedef struct _GstStreamConsistencyProbe
Packit Service 963350
{
Packit Service 963350
  GstPad *pad;
Packit Service 963350
  gulong probeid;
Packit Service 963350
} GstStreamConsistencyProbe;
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
source_pad_data_cb (GstPad * pad, GstPadProbeInfo * info,
Packit Service 963350
    GstStreamConsistency * consist)
Packit Service 963350
{
Packit Service 963350
  GstMiniObject *data = GST_PAD_PROBE_INFO_DATA (info);
Packit Service 963350
Packit Service 963350
  GST_DEBUG_OBJECT (pad, "%p: %d %d %d %d", consist, consist->flushing,
Packit Service 963350
      consist->segment, consist->eos, consist->expect_flush);
Packit Service 963350
Packit Service 963350
  if (GST_IS_BUFFER (data)) {
Packit Service 963350
    GST_DEBUG_OBJECT (pad,
Packit Service 963350
        "Buffer pts %" GST_TIME_FORMAT ", dts %" GST_TIME_FORMAT,
Packit Service 963350
        GST_TIME_ARGS (GST_BUFFER_PTS (GST_BUFFER_CAST (data))),
Packit Service 963350
        GST_TIME_ARGS (GST_BUFFER_DTS (GST_BUFFER_CAST (data))));
Packit Service 963350
    /* If an EOS went through, a buffer would be invalid */
Packit Service 963350
    fail_if (consist->eos, "Buffer received after EOS on pad %s:%s",
Packit Service 963350
        GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
    /* Buffers need to be preceded by a segment event */
Packit Service 963350
    fail_unless (consist->segment, "Buffer received without segment "
Packit Service 963350
        "on pad %s:%s", GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
  } else if (GST_IS_EVENT (data)) {
Packit Service 963350
    GstEvent *event = (GstEvent *) data;
Packit Service 963350
Packit Service 963350
    GST_DEBUG_OBJECT (pad, "Event : %s", GST_EVENT_TYPE_NAME (event));
Packit Service 963350
    switch (GST_EVENT_TYPE (event)) {
Packit Service 963350
      case GST_EVENT_FLUSH_START:
Packit Service 963350
        /* getting two flush_start in a row seems to be okay
Packit Service 963350
           fail_if (consist->flushing, "Received another FLUSH_START");
Packit Service 963350
         */
Packit Service 963350
        consist->flushing = TRUE;
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_FLUSH_STOP:
Packit Service 963350
        /* Receiving a flush-stop is only valid after receiving a flush-start */
Packit Service 963350
        fail_unless (consist->flushing,
Packit Service 963350
            "Received a FLUSH_STOP without a FLUSH_START on pad %s:%s",
Packit Service 963350
            GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        fail_if (consist->eos, "Received a FLUSH_STOP after an EOS on "
Packit Service 963350
            "pad %s:%s", GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        consist->flushing = consist->expect_flush = FALSE;
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_STREAM_START:
Packit Service 963350
        fail_if (consist->saw_serialized_event && !consist->saw_stream_start,
Packit Service 963350
            "Got a STREAM_START event after a serialized event on pad %s:%s",
Packit Service 963350
            GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        consist->saw_stream_start = TRUE;
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_CAPS:
Packit Service 963350
        /* ok to have these before segment event */
Packit Service 963350
        /* FIXME check order more precisely, if so spec'ed somehow ? */
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_SEGMENT:
Packit Service 963350
        fail_if ((consist->expect_flush && consist->flushing),
Packit Service 963350
            "Received SEGMENT while in a flushing seek on pad %s:%s",
Packit Service 963350
            GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        consist->segment = TRUE;
Packit Service 963350
        consist->eos = FALSE;
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_EOS:
Packit Service 963350
        /* FIXME : not 100% sure about whether two eos in a row is valid */
Packit Service 963350
        fail_if (consist->eos, "Received EOS just after another EOS on "
Packit Service 963350
            "pad %s:%s", GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        consist->eos = TRUE;
Packit Service 963350
        consist->segment = FALSE;
Packit Service 963350
        break;
Packit Service 963350
      case GST_EVENT_TAG:
Packit Service 963350
        GST_DEBUG_OBJECT (pad, "tag %" GST_PTR_FORMAT,
Packit Service 963350
            gst_event_get_structure (event));
Packit Service 963350
        /* fall through */
Packit Service 963350
      default:
Packit Service 963350
        if (GST_EVENT_IS_SERIALIZED (event) && GST_EVENT_IS_DOWNSTREAM (event)) {
Packit Service 963350
          fail_if (consist->eos, "Event received after EOS");
Packit Service 963350
          fail_unless (consist->segment, "Event %s received before segment "
Packit Service 963350
              "on pad %s:%s", GST_EVENT_TYPE_NAME (event),
Packit Service 963350
              GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        }
Packit Service 963350
        /* FIXME : Figure out what to do for other events */
Packit Service 963350
        break;
Packit Service 963350
    }
Packit Service 963350
    if (GST_EVENT_IS_SERIALIZED (event)) {
Packit Service 963350
      fail_if (!consist->saw_stream_start
Packit Service 963350
          && GST_EVENT_TYPE (event) != GST_EVENT_STREAM_START,
Packit Service 963350
          "Got a serialized event (%s) before a STREAM_START on pad %s:%s",
Packit Service 963350
          GST_EVENT_TYPE_NAME (event), GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
      consist->saw_serialized_event = TRUE;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
sink_pad_data_cb (GstPad * pad, GstPadProbeInfo * info,
Packit Service 963350
    GstStreamConsistency * consist)
Packit Service 963350
{
Packit Service 963350
  GstMiniObject *data = GST_PAD_PROBE_INFO_DATA (info);
Packit Service 963350
Packit Service 963350
  GST_DEBUG_OBJECT (pad, "%p: %d %d %d %d", consist, consist->flushing,
Packit Service 963350
      consist->segment, consist->eos, consist->expect_flush);
Packit Service 963350
Packit Service 963350
  if (GST_IS_BUFFER (data)) {
Packit Service 963350
    GST_DEBUG_OBJECT (pad, "Buffer %" GST_TIME_FORMAT,
Packit Service 963350
        GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (GST_BUFFER (data))));
Packit Service 963350
    /* If an EOS went through, a buffer would be invalid */
Packit Service 963350
    fail_if (consist->eos, "Buffer received after EOS on pad %s:%s",
Packit Service 963350
        GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
    /* Buffers need to be preceded by a segment event */
Packit Service 963350
    fail_unless (consist->segment, "Buffer received without segment "
Packit Service 963350
        "on pad %s:%s", GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
  } else if (GST_IS_EVENT (data)) {
Packit Service 963350
    GstEvent *event = (GstEvent *) data;
Packit Service 963350
Packit Service 963350
    GST_DEBUG_OBJECT (pad, "%s", GST_EVENT_TYPE_NAME (event));
Packit Service 963350
    switch (GST_EVENT_TYPE (event)) {
Packit Service 963350
      case GST_EVENT_SEEK:
Packit Service 963350
      {
Packit Service 963350
        GstSeekFlags flags;
Packit Service 963350
Packit Service 963350
        gst_event_parse_seek (event, NULL, NULL, &flags, NULL, NULL, NULL,
Packit Service 963350
            NULL);
Packit Service 963350
        consist->expect_flush =
Packit Service 963350
            ((flags & GST_SEEK_FLAG_FLUSH) == GST_SEEK_FLAG_FLUSH);
Packit Service 963350
        break;
Packit Service 963350
      }
Packit Service 963350
      case GST_EVENT_SEGMENT:
Packit Service 963350
        fail_if ((consist->expect_flush && consist->flushing),
Packit Service 963350
            "Received SEGMENT while in a flushing seek on pad %s:%s",
Packit Service 963350
            GST_DEBUG_PAD_NAME (pad));
Packit Service 963350
        consist->segment = TRUE;
Packit Service 963350
        consist->eos = FALSE;
Packit Service 963350
        break;
Packit Service 963350
      default:
Packit Service 963350
        /* FIXME : Figure out what to do for other events */
Packit Service 963350
        break;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
add_pad (GstStreamConsistency * consist, GstPad * pad)
Packit Service 963350
{
Packit Service 963350
  GstStreamConsistencyProbe *p;
Packit Service 963350
  GstPadDirection dir;
Packit Service 963350
Packit Service 963350
  p = g_new0 (GstStreamConsistencyProbe, 1);
Packit Service 963350
  p->pad = g_object_ref (pad);
Packit Service 963350
  dir = gst_pad_get_direction (pad);
Packit Service 963350
  if (dir == GST_PAD_SRC) {
Packit Service 963350
Packit Service 963350
    p->probeid =
Packit Service 963350
        gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_DATA_DOWNSTREAM,
Packit Service 963350
        (GstPadProbeCallback) source_pad_data_cb, consist, NULL);
Packit Service 963350
Packit Service 963350
  } else if (dir == GST_PAD_SINK) {
Packit Service 963350
    p->probeid =
Packit Service 963350
        gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_DATA_DOWNSTREAM,
Packit Service 963350
        (GstPadProbeCallback) sink_pad_data_cb, consist, NULL);
Packit Service 963350
  }
Packit Service 963350
  consist->pads = g_list_prepend (consist->pads, p);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_consistency_checker_new: (skip)
Packit Service 963350
 * @pad: The #GstPad on which the dataflow will be checked.
Packit Service 963350
 *
Packit Service 963350
 * Sets up a data probe on the given pad which will raise assertions if the
Packit Service 963350
 * data flow is inconsistent.
Packit Service 963350
 *
Packit Service 963350
 * Returns: A #GstStreamConsistency structure used to track data flow.
Packit Service 963350
 */
Packit Service 963350
GstStreamConsistency *
Packit Service 963350
gst_consistency_checker_new (GstPad * pad)
Packit Service 963350
{
Packit Service 963350
  GstStreamConsistency *consist;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (pad != NULL, NULL);
Packit Service 963350
Packit Service 963350
  consist = g_new0 (GstStreamConsistency, 1);
Packit Service 963350
Packit Service 963350
  if (!consist->pads) {
Packit Service 963350
    consist->parent = GST_OBJECT_PARENT (pad);
Packit Service 963350
  }
Packit Service 963350
  add_pad (consist, pad);
Packit Service 963350
  return consist;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_consistency_checker_add_pad:
Packit Service 963350
 * @consist: The #GstStreamConsistency handle
Packit Service 963350
 * @pad: The #GstPad on which the dataflow will be checked.
Packit Service 963350
 *
Packit Service 963350
 * Sets up a data probe on the given pad which will raise assertions if the
Packit Service 963350
 * data flow is inconsistent.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the pad was added
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_consistency_checker_add_pad (GstStreamConsistency * consist, GstPad * pad)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (consist != NULL, FALSE);
Packit Service 963350
  g_return_val_if_fail (pad != NULL, FALSE);
Packit Service 963350
  g_return_val_if_fail (GST_OBJECT_PARENT (pad) == consist->parent, FALSE);
Packit Service 963350
Packit Service 963350
  add_pad (consist, pad);
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_consistency_checker_reset:
Packit Service 963350
 * @consist: The #GstStreamConsistency to reset.
Packit Service 963350
 *
Packit Service 963350
 * Reset the stream checker's internal variables.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
gst_consistency_checker_reset (GstStreamConsistency * consist)
Packit Service 963350
{
Packit Service 963350
  consist->flushing = FALSE;
Packit Service 963350
  consist->segment = FALSE;
Packit Service 963350
  consist->eos = FALSE;
Packit Service 963350
  consist->expect_flush = FALSE;
Packit Service 963350
  consist->saw_serialized_event = FALSE;
Packit Service 963350
  consist->saw_stream_start = FALSE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_consistency_checker_free:
Packit Service 963350
 * @consist: The #GstStreamConsistency to free.
Packit Service 963350
 *
Packit Service 963350
 * Frees the allocated data and probes associated with @consist.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
gst_consistency_checker_free (GstStreamConsistency * consist)
Packit Service 963350
{
Packit Service 963350
  GList *node;
Packit Service 963350
  GstStreamConsistencyProbe *p;
Packit Service 963350
Packit Service 963350
  /* Remove the data probes */
Packit Service 963350
  for (node = consist->pads; node; node = g_list_next (node)) {
Packit Service 963350
    p = (GstStreamConsistencyProbe *) node->data;
Packit Service 963350
    gst_pad_remove_probe (p->pad, p->probeid);
Packit Service 963350
    gst_object_unref (p->pad);
Packit Service 963350
    g_free (p);
Packit Service 963350
  }
Packit Service 963350
  g_list_free (consist->pads);
Packit Service 963350
  g_free (consist);
Packit Service 963350
}