blob: 564e1dbaa394c2636455603a72cc2d9b278e16c8 [file] [log] [blame]
/* GStreamer unit tests for the deinterlace element
* Copyright (C) 2010 Thiago Santos <thiago.sousa.santos@collabora.co.uk>
*
* 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., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <stdio.h>
#include <gst/check/gstcheck.h>
#include <gst/video/video.h>
static gboolean
gst_caps_is_interlaced (GstCaps * caps)
{
GstVideoInfo info;
fail_unless (gst_caps_is_fixed (caps));
fail_unless (gst_video_info_from_caps (&info, caps));
return GST_VIDEO_INFO_IS_INTERLACED (&info);
}
GST_START_TEST (test_create_and_unref)
{
GstElement *deinterlace;
deinterlace = gst_element_factory_make ("deinterlace", NULL);
fail_unless (deinterlace != NULL);
gst_element_set_state (deinterlace, GST_STATE_NULL);
gst_object_unref (deinterlace);
}
GST_END_TEST;
#define CAPS_VIDEO_COMMON \
"width=(int)800, height=(int)600, framerate=(fraction)15/1"
#define CAPS_IMAGE_COMMON \
"width=(int)3200, height=(int)3400, framerate=(fraction)0/1"
#define CAPS_YUY2 \
"video/x-raw, " \
CAPS_VIDEO_COMMON ", " \
"format=(string)YUY2"
#define CAPS_YUY2_INTERLACED \
CAPS_YUY2 ", " \
"interlace-mode=interleaved"
#define CAPS_YVYU \
"video/x-raw, " \
CAPS_VIDEO_COMMON ", " \
"format=(string)YVYU"
#define CAPS_YVYU_INTERLACED \
CAPS_YVYU ", " \
"interlace-mode=interleaved"
#define CAPS_YUY2_IMAGE \
"video/x-raw, " \
CAPS_IMAGE_COMMON ", " \
"format=(string)YUY2"
#define CAPS_YUY2_INTERLACED_IMAGE \
CAPS_YUY2_IMAGE ", " \
"interlace-mode=interleaved"
#define CAPS_YVYU_IMAGE \
"video/x-raw, " \
CAPS_IMAGE_COMMON ", " \
"format=(string)YVYU"
#define CAPS_YVYU_INTERLACED_IMAGE \
CAPS_YVYU_IMAGE ", " \
"interlace-mode=interleaved"
static GstElement *deinterlace;
static GstPad *srcpad;
static GstPad *sinkpad;
static GstElement *pipeline;
/* sets up deinterlace and shortcut pointers to its pads */
static void
setup_deinterlace (void)
{
deinterlace = gst_element_factory_make ("deinterlace", NULL);
fail_unless (deinterlace != NULL);
sinkpad = gst_element_get_static_pad (deinterlace, "sink");
fail_unless (sinkpad != NULL);
srcpad = gst_element_get_static_pad (deinterlace, "src");
fail_unless (srcpad != NULL);
}
/* sets up a basic test pipeline containing:
*
* videotestsrc ! capsfilter ! deinterlace ! fakesink
*
* The parameters set the capsfilter caps and the num-buffers
* property of videotestsrc
*
* It is useful for adding buffer probes to deinterlace pads
* and validating inputs/outputs
*/
static void
setup_test_pipeline (gint mode, GstCaps * infiltercaps, GstCaps * outfiltercaps,
gint numbuffers)
{
GstElement *src;
GstElement *infilter;
GstElement *outfilter;
GstElement *sink;
setup_deinterlace ();
pipeline = gst_pipeline_new ("pipeline");
src = gst_element_factory_make ("videotestsrc", NULL);
infilter = gst_element_factory_make ("capsfilter", "infilter");
outfilter = gst_element_factory_make ("capsfilter", "outfilter");
sink = gst_element_factory_make ("fakesink", NULL);
fail_if (src == NULL);
fail_if (infilter == NULL);
fail_if (outfilter == NULL);
fail_if (sink == NULL);
fail_unless (gst_bin_add (GST_BIN (pipeline), src));
fail_unless (gst_bin_add (GST_BIN (pipeline), infilter));
fail_unless (gst_bin_add (GST_BIN (pipeline), deinterlace));
fail_unless (gst_bin_add (GST_BIN (pipeline), outfilter));
fail_unless (gst_bin_add (GST_BIN (pipeline), sink));
/* set the properties */
g_object_set (deinterlace, "mode", mode, NULL);
if (numbuffers > 0)
g_object_set (src, "num-buffers", numbuffers, NULL);
if (infiltercaps)
g_object_set (infilter, "caps", infiltercaps, NULL);
if (outfiltercaps)
g_object_set (outfilter, "caps", outfiltercaps, NULL);
fail_unless (gst_element_link_many (src, infilter, deinterlace, outfilter,
sink, NULL));
if (infiltercaps)
gst_caps_unref (infiltercaps);
if (outfiltercaps)
gst_caps_unref (outfiltercaps);
}
/*
* Checks if 2 buffers are equal
*
* Equals means same data
*/
static gboolean
test_buffer_equals (GstBuffer * buf_a, GstBuffer * buf_b)
{
GstMapInfo m1, m2;
gboolean res = FALSE;
gst_buffer_map (buf_a, &m1, GST_MAP_READ);
gst_buffer_map (buf_b, &m2, GST_MAP_READ);
if (m1.size == m2.size) {
res = memcmp (m1.data, m2.data, m1.size) == 0;
}
gst_buffer_unmap (buf_a, &m1);
gst_buffer_unmap (buf_b, &m2);
return res;
}
static GstPadProbeReturn
sinkpad_enqueue_buffer (GstPad * pad, GstPadProbeInfo * info, gpointer data)
{
GQueue *queue = (GQueue *) data;
GstBuffer *buf = GST_PAD_PROBE_INFO_BUFFER (info);
/* enqueue a copy for being compared later */
g_queue_push_tail (queue, gst_buffer_copy (buf));
return GST_PAD_PROBE_OK;
}
/*
* pad buffer probe that compares the buffer with the top one
* in the GQueue passed as the user data
*/
static GstPadProbeReturn
srcpad_dequeue_and_compare_buffer (GstPad * pad, GstPadProbeInfo * info,
gpointer data)
{
GQueue *queue = (GQueue *) data;
GstBuffer *buf = GST_PAD_PROBE_INFO_BUFFER (info);
GstBuffer *queue_buf;
queue_buf = (GstBuffer *) g_queue_pop_head (queue);
fail_if (queue_buf == NULL);
fail_unless (test_buffer_equals (buf, queue_buf));
gst_buffer_unref (queue_buf);
return GST_PAD_PROBE_OK;
}
/*
* Utility function that sets up a pipeline with deinterlace for
* validanting that it operates in passthrough mode when receiving
* data with 'infiltercaps' as the input caps and operating in 'mode' mode
*/
static void
deinterlace_check_passthrough (gint mode, const gchar * infiltercaps)
{
GstMessage *msg;
GQueue *queue;
GstCaps *incaps = NULL;
if (infiltercaps)
incaps = gst_caps_from_string (infiltercaps);
setup_test_pipeline (mode, incaps, NULL, 20);
queue = g_queue_new ();
/* set up probes for testing */
gst_pad_add_probe (sinkpad, GST_PAD_PROBE_TYPE_BUFFER, sinkpad_enqueue_buffer,
queue, NULL);
gst_pad_add_probe (srcpad, GST_PAD_PROBE_TYPE_BUFFER,
srcpad_dequeue_and_compare_buffer, queue, NULL);
fail_unless (gst_element_set_state (pipeline, GST_STATE_PLAYING) !=
GST_STATE_CHANGE_FAILURE);
msg = gst_bus_poll (GST_ELEMENT_BUS (pipeline),
GST_MESSAGE_ERROR | GST_MESSAGE_EOS, -1);
if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR) {
GST_ERROR ("ERROR: %" GST_PTR_FORMAT, msg);
fail ("Unexpected error message");
}
gst_message_unref (msg);
/* queue should be empty */
fail_unless (g_queue_is_empty (queue));
fail_unless (gst_element_set_state (pipeline, GST_STATE_NULL) ==
GST_STATE_CHANGE_SUCCESS);
gst_object_unref (pipeline);
gst_object_unref (sinkpad);
gst_object_unref (srcpad);
g_queue_free (queue);
}
/*
* Sets the caps on deinterlace sinkpad and validates the
* caps that is set on the srcpad
*/
static void
deinterlace_set_caps_and_check (GstCaps * input, gboolean must_deinterlace)
{
GstCaps *othercaps = NULL;
fail_unless (gst_pad_set_caps (sinkpad, input));
g_object_get (srcpad, "caps", &othercaps, NULL);
if (must_deinterlace) {
fail_if (gst_caps_is_interlaced (othercaps));
} else {
GstStructure *s;
fail_unless (gst_caps_is_interlaced (input) ==
gst_caps_is_interlaced (othercaps));
othercaps = gst_caps_make_writable (othercaps);
s = gst_caps_get_structure (othercaps, 0);
gst_structure_remove_field (s, "interlace-mode");
input = gst_caps_make_writable (input);
s = gst_caps_get_structure (input, 0);
gst_structure_remove_field (s, "interlace-mode");
fail_unless (gst_caps_is_equal (input, othercaps));
}
gst_caps_unref (input);
gst_caps_unref (othercaps);
}
static void
deinterlace_set_string_caps_and_check (const gchar * input,
gboolean must_deinterlace)
{
deinterlace_set_caps_and_check (gst_caps_from_string (input),
must_deinterlace);
}
GST_START_TEST (test_mode_auto_accept_caps)
{
setup_deinterlace ();
/* auto mode */
g_object_set (deinterlace, "mode", 0, NULL);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_PLAYING) ==
GST_STATE_CHANGE_SUCCESS);
/* try to set non interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_IMAGE, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_IMAGE, FALSE);
/* now try to set interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED_IMAGE, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED_IMAGE, TRUE);
/* cleanup */
gst_object_unref (sinkpad);
gst_object_unref (srcpad);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_NULL) ==
GST_STATE_CHANGE_SUCCESS);
gst_object_unref (deinterlace);
}
GST_END_TEST;
GST_START_TEST (test_mode_forced_accept_caps)
{
setup_deinterlace ();
/* forced mode */
g_object_set (deinterlace, "mode", 1, NULL);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_PLAYING) ==
GST_STATE_CHANGE_SUCCESS);
/* try to set non interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_IMAGE, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_IMAGE, TRUE);
/* now try to set interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED_IMAGE, TRUE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED_IMAGE, TRUE);
/* cleanup */
gst_object_unref (sinkpad);
gst_object_unref (srcpad);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_NULL) ==
GST_STATE_CHANGE_SUCCESS);
gst_object_unref (deinterlace);
}
GST_END_TEST;
GST_START_TEST (test_mode_disabled_accept_caps)
{
setup_deinterlace ();
/* disabled mode */
g_object_set (deinterlace, "mode", 2, NULL);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_PLAYING) ==
GST_STATE_CHANGE_SUCCESS);
/* try to set non interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_IMAGE, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_IMAGE, FALSE);
/* now try to set interlaced caps */
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YVYU_INTERLACED_IMAGE, FALSE);
deinterlace_set_string_caps_and_check (CAPS_YUY2_INTERLACED_IMAGE, FALSE);
/* cleanup */
gst_object_unref (sinkpad);
gst_object_unref (srcpad);
fail_unless (gst_element_set_state (deinterlace, GST_STATE_NULL) ==
GST_STATE_CHANGE_SUCCESS);
gst_object_unref (deinterlace);
}
GST_END_TEST;
GST_START_TEST (test_mode_disabled_passthrough)
{
/* 2 is disabled mode */
deinterlace_check_passthrough (2, CAPS_YUY2_INTERLACED);
deinterlace_check_passthrough (2, CAPS_YVYU_INTERLACED);
deinterlace_check_passthrough (2, CAPS_YUY2);
deinterlace_check_passthrough (2, CAPS_YVYU);
deinterlace_check_passthrough (2, CAPS_YUY2_INTERLACED_IMAGE);
deinterlace_check_passthrough (2, CAPS_YVYU_INTERLACED_IMAGE);
deinterlace_check_passthrough (2, CAPS_YUY2_IMAGE);
deinterlace_check_passthrough (2, CAPS_YVYU_IMAGE);
}
GST_END_TEST;
GST_START_TEST (test_mode_auto_deinterlaced_passthrough)
{
/* 0 is auto mode */
deinterlace_check_passthrough (0, CAPS_YUY2);
deinterlace_check_passthrough (0, CAPS_YVYU);
deinterlace_check_passthrough (0, CAPS_YUY2_IMAGE);
deinterlace_check_passthrough (0, CAPS_YVYU_IMAGE);
}
GST_END_TEST;
static Suite *
deinterlace_suite (void)
{
Suite *s = suite_create ("deinterlace");
TCase *tc_chain = tcase_create ("general");
suite_add_tcase (s, tc_chain);
tcase_set_timeout (tc_chain, 180);
if (!gst_registry_check_feature_version (gst_registry_get (), "deinterlace",
GST_VERSION_MAJOR, GST_VERSION_MINOR, GST_VERSION_MICRO)) {
GST_ERROR ("FIXME: port deinterlace element");
return s;
}
tcase_add_test (tc_chain, test_create_and_unref);
tcase_add_test (tc_chain, test_mode_auto_accept_caps);
tcase_add_test (tc_chain, test_mode_forced_accept_caps);
tcase_add_test (tc_chain, test_mode_disabled_accept_caps);
tcase_add_test (tc_chain, test_mode_disabled_passthrough);
tcase_add_test (tc_chain, test_mode_auto_deinterlaced_passthrough);
return s;
}
GST_CHECK_MAIN (deinterlace);