mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-04 23:46:43 +00:00
1041 lines
32 KiB
C
1041 lines
32 KiB
C
/* GStreamer
|
|
* Copyright (C) <2005> Thomas Vander Stichele <thomas at apestaart dot org>
|
|
*
|
|
* gststructure.c: Unit tests for GstStructure
|
|
*
|
|
* 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 <gst/gststructure.h>
|
|
#include <gst/check/gstcheck.h>
|
|
|
|
|
|
GST_START_TEST (test_from_string_int)
|
|
{
|
|
const char *strings[] = {
|
|
"video/x-raw, width = (int) 123456",
|
|
"video/x-raw, stride = (int) -123456",
|
|
"video/x-raw, red_mask = (int) 0xFFFF",
|
|
"video/x-raw, red_mask = (int) 0x0000FFFF",
|
|
"video/x-raw, red_mask = (int) 0x7FFFFFFF",
|
|
"video/x-raw, red_mask = (int) 0x80000000",
|
|
"video/x-raw, red_mask = (int) 0xFF000000",
|
|
/* result from
|
|
* gst-launch ... ! "video/x-raw, red_mask=(int)0xFF000000" ! ... */
|
|
"video/x-raw,\\ red_mask=(int)0xFF000000",
|
|
};
|
|
gint results[] = {
|
|
123456,
|
|
-123456,
|
|
0xFFFF,
|
|
0xFFFF,
|
|
0x7FFFFFFF,
|
|
(gint) 0x80000000,
|
|
(gint) 0xFF000000,
|
|
(gint) 0xFF000000,
|
|
};
|
|
GstStructure *structure;
|
|
int i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (strings); ++i) {
|
|
const char *s;
|
|
const gchar *name;
|
|
gint value;
|
|
|
|
s = strings[i];
|
|
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
name = gst_structure_nth_field_name (structure, 0);
|
|
fail_unless (gst_structure_get_int (structure, name, &value));
|
|
fail_unless (value == results[i],
|
|
"Value %d is not the expected result %d for string %s",
|
|
value, results[i], s);
|
|
|
|
/* cleanup */
|
|
gst_structure_free (structure);
|
|
}
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_from_string_uint)
|
|
{
|
|
const char *strings[] = {
|
|
"taglist, bar = (uint) 123456",
|
|
"taglist, bar = (uint) 0xFFFF",
|
|
"taglist, bar = (uint) 0x0000FFFF",
|
|
"taglist, bar = (uint) 0x7FFFFFFF",
|
|
"taglist, bar = (uint) 0x80000000",
|
|
"taglist, bar = (uint) 0xFF000000"
|
|
};
|
|
guint results[] = {
|
|
123456,
|
|
0xFFFF,
|
|
0xFFFF,
|
|
0x7FFFFFFF,
|
|
0x80000000,
|
|
0xFF000000,
|
|
};
|
|
GstStructure *structure;
|
|
int i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (strings); ++i) {
|
|
const char *s;
|
|
const gchar *name;
|
|
guint value;
|
|
|
|
s = strings[i];
|
|
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
name = gst_structure_nth_field_name (structure, 0);
|
|
fail_unless (gst_structure_get_uint (structure, name, &value));
|
|
fail_unless (value == results[i],
|
|
"Value %u is not the expected result %u for string %s",
|
|
value, results[i], s);
|
|
|
|
/* cleanup */
|
|
gst_structure_free (structure);
|
|
}
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Test type conversions from string */
|
|
GST_START_TEST (test_from_string)
|
|
{
|
|
GstStructure *structure;
|
|
const gchar *s;
|
|
const GValue *val;
|
|
|
|
s = "test-string,value=1";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_INT (val));
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=1.0";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_DOUBLE (val));
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=1/1";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (GST_VALUE_HOLDS_FRACTION (val));
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=bar";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_STRING (val));
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=true";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_BOOLEAN (val));
|
|
fail_unless_equals_int (g_value_get_boolean (val), TRUE);
|
|
gst_structure_free (structure);
|
|
|
|
/* Test trailing comas */
|
|
s = "test-string,";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,;";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=true,";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_BOOLEAN (val));
|
|
fail_unless_equals_int (g_value_get_boolean (val), TRUE);
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=true,;";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_BOOLEAN (val));
|
|
fail_unless_equals_int (g_value_get_boolean (val), TRUE);
|
|
gst_structure_free (structure);
|
|
|
|
s = "test-string,value=true,,";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL, "Created structure from string %s", s);
|
|
|
|
/* Tests for flagset deserialisation */
|
|
s = "foobar,value=0010:ffff";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (GST_VALUE_HOLDS_FLAG_SET (val));
|
|
gst_structure_free (structure);
|
|
|
|
/* In the presence of the hex values, the strings don't matter as long as they
|
|
* have the right form */
|
|
s = "foobar,value=0010:ffff:+random+other/not-the-other";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "value")) != NULL);
|
|
fail_unless (GST_VALUE_HOLDS_FLAG_SET (val));
|
|
gst_structure_free (structure);
|
|
|
|
/* Test that a timecode string is deserialised as a string, not a flagset:
|
|
* https://bugzilla.gnome.org/show_bug.cgi?id=779755 */
|
|
s = "foobar,timecode=00:01:00:00";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
fail_unless ((val = gst_structure_get_value (structure, "timecode")) != NULL);
|
|
fail_unless (G_VALUE_HOLDS_STRING (val));
|
|
gst_structure_free (structure);
|
|
|
|
s = "0.10:decoder-video/mpeg, abc=(boolean)false";
|
|
structure = NULL;
|
|
ASSERT_CRITICAL (structure = gst_structure_from_string (s, NULL));
|
|
fail_unless (structure == NULL, "Could not get structure from string %s", s);
|
|
|
|
/* make sure we bail out correctly in case of an error or if parsing fails */
|
|
s = "***foo***, abc=(boolean)false";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL);
|
|
|
|
/* assert that we get a warning if the structure wasn't entirely consumed, but
|
|
* we didn't provide an end pointer */
|
|
s = "foo/bar; other random data";
|
|
ASSERT_WARNING (structure = gst_structure_from_string (s, NULL));
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
gst_structure_free (structure);
|
|
|
|
/* make sure we handle \ as last character in various things, run with valgrind */
|
|
s = "foo,test=\"foobar\\";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL);
|
|
s = "\\";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL);
|
|
s = "foobar,test\\";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL);
|
|
s = "foobar,test=(string)foo\\";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_unless (structure == NULL);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_to_string)
|
|
{
|
|
GstStructure *st1 = NULL;
|
|
|
|
ASSERT_CRITICAL (st1 = gst_structure_new_empty ("Foo\nwith-newline"));
|
|
fail_unless (st1 == NULL);
|
|
|
|
ASSERT_CRITICAL (st1 = gst_structure_new_empty ("Foo with whitespace"));
|
|
fail_unless (st1 == NULL);
|
|
ASSERT_CRITICAL (st1 = gst_structure_new_empty ("1st"));
|
|
fail_unless (st1 == NULL);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_to_from_string)
|
|
{
|
|
GstStructure *st1, *st2;
|
|
gchar *str;
|
|
|
|
/* test escaping/unescaping */
|
|
st1 = gst_structure_new ("FooBar-123/0_1", "num", G_TYPE_INT, 9173,
|
|
"string", G_TYPE_STRING, "Something Like Face/Off", NULL);
|
|
str = gst_structure_to_string (st1);
|
|
st2 = gst_structure_from_string (str, NULL);
|
|
g_free (str);
|
|
|
|
fail_unless (st2 != NULL);
|
|
fail_unless (gst_structure_is_equal (st1, st2),
|
|
"Structures did not match:\n\tStructure 1: %" GST_PTR_FORMAT
|
|
"\n\tStructure 2: %" GST_PTR_FORMAT "\n", st1, st2);
|
|
|
|
gst_structure_free (st1);
|
|
gst_structure_free (st2);
|
|
|
|
/* Test NULL strings */
|
|
st1 = gst_structure_new ("test", "mynullstr", G_TYPE_STRING, NULL, NULL);
|
|
fail_unless (st1 != NULL);
|
|
str = gst_structure_to_string (st1);
|
|
fail_unless (strcmp (str, "test, mynullstr=(string)NULL;") == 0,
|
|
"Failed to serialize to right string: %s", str);
|
|
|
|
st2 = gst_structure_from_string (str, NULL);
|
|
fail_unless (st2 != NULL);
|
|
g_free (str);
|
|
|
|
fail_unless (gst_structure_is_equal (st1, st2),
|
|
"Structures did not match:\n\tStructure 1: %" GST_PTR_FORMAT
|
|
"\n\tStructure 2: %" GST_PTR_FORMAT "\n", st1, st2);
|
|
|
|
gst_structure_free (st1);
|
|
gst_structure_free (st2);
|
|
|
|
/* pointers are serialized but not deserialized */
|
|
st1 = gst_structure_new ("test", "ptr", G_TYPE_POINTER, 0xdeadbeef, NULL);
|
|
str = gst_structure_to_string (st1);
|
|
/* The way pointers are serialized may be plateform specific so just check
|
|
* if it contains the address */
|
|
fail_unless (g_strrstr (str, "deadbeef") || g_strrstr (str, "DEADBEEF"),
|
|
"Failed to serialize to right string: %s", str);
|
|
|
|
st2 = gst_structure_from_string (str, NULL);
|
|
fail_unless (!st2);
|
|
|
|
gst_structure_free (st1);
|
|
g_free (str);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Added to make sure taglists are properly serialized/deserialized after bug
|
|
* https://bugzilla.gnome.org/show_bug.cgi?id=733131 */
|
|
GST_START_TEST (test_to_from_string_tag_event)
|
|
{
|
|
GstEvent *tagevent;
|
|
GstTagList *taglist;
|
|
GstStructure *st1, *st2;
|
|
gchar *str;
|
|
|
|
/* empty taglist */
|
|
taglist = gst_tag_list_new_empty ();
|
|
tagevent = gst_event_new_tag (taglist);
|
|
|
|
st1 = (GstStructure *) gst_event_get_structure (tagevent);
|
|
str = gst_structure_to_string (st1);
|
|
fail_unless (str != NULL);
|
|
|
|
st2 = gst_structure_new_from_string (str);
|
|
fail_unless (st2 != NULL);
|
|
fail_unless (gst_structure_is_equal (st1, st2));
|
|
gst_event_unref (tagevent);
|
|
gst_structure_free (st2);
|
|
g_free (str);
|
|
|
|
/* taglist with data */
|
|
taglist = gst_tag_list_new ("title", "TEST TITLE", NULL);
|
|
tagevent = gst_event_new_tag (taglist);
|
|
|
|
st1 = (GstStructure *) gst_event_get_structure (tagevent);
|
|
str = gst_structure_to_string (st1);
|
|
fail_unless (str != NULL);
|
|
|
|
st2 = gst_structure_new_from_string (str);
|
|
fail_unless (st2 != NULL);
|
|
fail_unless (gst_structure_is_equal (st1, st2));
|
|
gst_event_unref (tagevent);
|
|
gst_structure_free (st2);
|
|
g_free (str);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_complete_structure)
|
|
{
|
|
GstStructure *structure;
|
|
const gchar *s;
|
|
|
|
s = "GstEventSeek, rate=(double)1, format=(GstFormat)GST_FORMAT_TIME, flags=(GstSeekFlags)GST_SEEK_FLAG_NONE, start_type=(GstSeekType)GST_SEEK_TYPE_SET, start=(gint64)1000000000, stop_type=(GstSeekType)GST_SEEK_TYPE_NONE, stop=(gint64)0";
|
|
structure = gst_structure_from_string (s, NULL);
|
|
fail_if (structure == NULL, "Could not get structure from string %s", s);
|
|
/* FIXME: TODO: add checks for correct serialization of members ? */
|
|
gst_structure_free (structure);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_string_properties)
|
|
{
|
|
GstStructure *st1, *st2;
|
|
gchar *str;
|
|
|
|
/* test escaping/unescaping */
|
|
st1 = gst_structure_new ("RandomStructure", "prop1", G_TYPE_STRING, "foo",
|
|
"prop2", G_TYPE_STRING, "", "prop3", G_TYPE_STRING, NULL,
|
|
"prop4", G_TYPE_STRING, "NULL", NULL);
|
|
str = gst_structure_to_string (st1);
|
|
st2 = gst_structure_from_string (str, NULL);
|
|
g_free (str);
|
|
|
|
fail_unless (st2 != NULL);
|
|
fail_unless (gst_structure_is_equal (st1, st2),
|
|
"Structures did not match:\n\tStructure 1: %" GST_PTR_FORMAT
|
|
"\n\tStructure 2: %" GST_PTR_FORMAT "\n", st1, st2);
|
|
|
|
gst_structure_free (st1);
|
|
gst_structure_free (st2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_structure_new)
|
|
{
|
|
GstStructure *s;
|
|
GError *e;
|
|
GQuark domain;
|
|
gboolean bool;
|
|
gint num, den;
|
|
GstClockTime clocktime;
|
|
guint64 uint64;
|
|
|
|
s = gst_structure_new ("name",
|
|
"key", G_TYPE_STRING, "value",
|
|
"bool", G_TYPE_BOOLEAN, TRUE,
|
|
"fraction", GST_TYPE_FRACTION, 1, 5,
|
|
"clocktime", GST_TYPE_CLOCK_TIME, GST_CLOCK_TIME_NONE,
|
|
"uint64", G_TYPE_UINT64, (guint64) 1234, NULL);
|
|
|
|
fail_unless (gst_structure_get_field_type (s, "unknown") == G_TYPE_INVALID);
|
|
/* test setting a different name */
|
|
gst_structure_set_name (s, "newname");
|
|
fail_unless (strcmp (gst_structure_get_string (s, "key"), "value") == 0);
|
|
fail_unless (gst_structure_has_field (s, "key"));
|
|
fail_unless_equals_int (gst_structure_n_fields (s), 5);
|
|
/* test removing a field */
|
|
gst_structure_remove_field (s, "key");
|
|
fail_if (gst_structure_get_string (s, "key"));
|
|
fail_if (gst_structure_has_field (s, "key"));
|
|
fail_unless_equals_int (gst_structure_n_fields (s), 4);
|
|
|
|
fail_unless (gst_structure_get_boolean (s, "bool", &bool));
|
|
fail_unless (bool);
|
|
|
|
fail_unless (gst_structure_get_fraction (s, "fraction", &num, &den));
|
|
fail_unless_equals_int (num, 1);
|
|
fail_unless_equals_int (den, 5);
|
|
|
|
fail_unless (gst_structure_get_clock_time (s, "clocktime", &clocktime));
|
|
fail_unless_equals_uint64 (clocktime, GST_CLOCK_TIME_NONE);
|
|
|
|
fail_unless (gst_structure_get_uint64 (s, "uint64", &uint64));
|
|
fail_unless_equals_uint64 (uint64, 1234);
|
|
|
|
gst_structure_free (s);
|
|
|
|
domain = g_quark_from_static_string ("test");
|
|
e = g_error_new (domain, 0, "a test error");
|
|
s = gst_structure_new ("name", "key", G_TYPE_ERROR, e, NULL);
|
|
g_error_free (e);
|
|
gst_structure_free (s);
|
|
|
|
ASSERT_CRITICAL (gst_structure_free (gst_structure_new_empty
|
|
("0.10:decoder-video/mpeg")));
|
|
|
|
/* make sure we bail out correctly in case of an error or if parsing fails */
|
|
s = NULL;
|
|
ASSERT_CRITICAL (s = gst_structure_new ("^joo\nba\ndoo^",
|
|
"abc", G_TYPE_BOOLEAN, FALSE, NULL));
|
|
fail_unless (s == NULL);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_fixate)
|
|
{
|
|
GstStructure *s;
|
|
gint i;
|
|
|
|
s = gst_structure_new ("name",
|
|
"int", G_TYPE_INT, 5,
|
|
"intrange", GST_TYPE_INT_RANGE, 10, 20,
|
|
"intrange2", GST_TYPE_INT_RANGE, 10, 20,
|
|
"intrange3", GST_TYPE_INT_RANGE, 10, 20, NULL);
|
|
|
|
/* basic test */
|
|
fail_if (gst_structure_fixate_field_nearest_int (s, "int", 5));
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange", 15));
|
|
fail_if (gst_structure_fixate_field_nearest_int (s, "intrange", 15));
|
|
fail_unless (gst_structure_get_int (s, "intrange", &i));
|
|
fail_unless_equals_int (i, 15);
|
|
|
|
/* test range min */
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange2", 5));
|
|
fail_unless (gst_structure_get_int (s, "intrange2", &i));
|
|
fail_unless_equals_int (i, 10);
|
|
|
|
/* test range max */
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange3", 25));
|
|
fail_unless (gst_structure_get_int (s, "intrange3", &i));
|
|
fail_unless_equals_int (i, 20);
|
|
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_fixate_stepped)
|
|
{
|
|
GstStructure *s;
|
|
GValue v = G_VALUE_INIT;
|
|
gint i;
|
|
|
|
s = gst_structure_new_empty ("name");
|
|
|
|
g_value_init (&v, GST_TYPE_INT_RANGE);
|
|
gst_value_set_int_range_step (&v, 10, 20, 5);
|
|
gst_structure_set_value (s, "intrange1", &v);
|
|
gst_structure_set_value (s, "intrange2", &v);
|
|
gst_structure_set_value (s, "intrange3", &v);
|
|
g_value_unset (&v);
|
|
|
|
/* exact */
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange1", 15));
|
|
fail_unless (gst_structure_get_int (s, "intrange1", &i));
|
|
fail_unless_equals_int (i, 15);
|
|
|
|
/* round down */
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange2", 12));
|
|
fail_unless (gst_structure_get_int (s, "intrange2", &i));
|
|
fail_unless_equals_int (i, 10);
|
|
|
|
/* round up */
|
|
fail_unless (gst_structure_fixate_field_nearest_int (s, "intrange3", 13));
|
|
fail_unless (gst_structure_get_int (s, "intrange3", &i));
|
|
fail_unless_equals_int (i, 15);
|
|
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_fixate_frac_list)
|
|
{
|
|
GstStructure *s, *s2;
|
|
GValue list = { 0 };
|
|
GValue frac = { 0 };
|
|
gchar *str;
|
|
gint num, denom;
|
|
|
|
g_value_init (&list, GST_TYPE_LIST);
|
|
g_value_init (&frac, GST_TYPE_FRACTION);
|
|
|
|
gst_value_set_fraction (&frac, 30, 1);
|
|
gst_value_list_append_value (&list, &frac);
|
|
gst_value_set_fraction (&frac, 15, 1);
|
|
gst_value_list_append_value (&list, &frac);
|
|
gst_value_set_fraction (&frac, 10, 1);
|
|
gst_value_list_append_value (&list, &frac);
|
|
|
|
s = gst_structure_new_empty ("name");
|
|
gst_structure_set_value (s, "frac", &list);
|
|
g_value_unset (&frac);
|
|
g_value_unset (&list);
|
|
|
|
str = gst_structure_to_string (s);
|
|
GST_DEBUG ("list %s", str);
|
|
g_free (str);
|
|
|
|
/* take copy */
|
|
s2 = gst_structure_copy (s);
|
|
|
|
/* fixate to the nearest fraction, this should give 15/1 */
|
|
fail_unless (gst_structure_fixate_field_nearest_fraction (s, "frac", 14, 1));
|
|
|
|
fail_unless (gst_structure_get_fraction (s, "frac", &num, &denom));
|
|
fail_unless (num == 15);
|
|
fail_unless (denom == 1);
|
|
|
|
gst_structure_free (s);
|
|
s = s2;
|
|
|
|
/* fixate to the nearest fraction, this should give 30/1 */
|
|
fail_unless (gst_structure_fixate_field_nearest_fraction (s, "frac", G_MAXINT,
|
|
1));
|
|
|
|
fail_unless (gst_structure_get_fraction (s, "frac", &num, &denom));
|
|
fail_unless (num == 30);
|
|
fail_unless (denom == 1);
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_is_subset_equal_array_list)
|
|
{
|
|
GstStructure *s1, *s2;
|
|
|
|
s1 = gst_structure_from_string ("test/test, channels=(int){ 1, 2 }", NULL);
|
|
fail_if (s1 == NULL);
|
|
s2 = gst_structure_from_string ("test/test, channels=(int)[ 1, 2 ]", NULL);
|
|
fail_if (s2 == NULL);
|
|
|
|
fail_unless (gst_structure_is_subset (s1, s2));
|
|
|
|
gst_structure_free (s1);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_is_subset_different_name)
|
|
{
|
|
GstStructure *s1, *s2;
|
|
|
|
s1 = gst_structure_from_string ("test/test, channels=(int)1", NULL);
|
|
fail_if (s1 == NULL);
|
|
s2 = gst_structure_from_string ("test/baz, channels=(int)1", NULL);
|
|
fail_if (s2 == NULL);
|
|
|
|
fail_unless (!gst_structure_is_subset (s1, s2));
|
|
|
|
gst_structure_free (s1);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_is_subset_superset_missing_fields)
|
|
{
|
|
GstStructure *s1, *s2;
|
|
|
|
/* a missing field is equivalent to any value */
|
|
s1 = gst_structure_from_string ("test/test, channels=(int)1, rate=(int)1",
|
|
NULL);
|
|
fail_if (s1 == NULL);
|
|
s2 = gst_structure_from_string ("test/test, channels=(int)1", NULL);
|
|
fail_if (s2 == NULL);
|
|
|
|
fail_unless (gst_structure_is_subset (s1, s2));
|
|
|
|
gst_structure_free (s1);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_is_subset_superset_extra_fields)
|
|
{
|
|
GstStructure *s1, *s2;
|
|
|
|
/* a missing field is equivalent to any value */
|
|
s1 = gst_structure_from_string ("test/test, channels=(int)1", NULL);
|
|
fail_if (s1 == NULL);
|
|
s2 = gst_structure_from_string ("test/test, channels=(int)1, rate=(int)1",
|
|
NULL);
|
|
fail_if (s2 == NULL);
|
|
|
|
fail_unless (!gst_structure_is_subset (s1, s2));
|
|
|
|
gst_structure_free (s1);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_is_subset_superset_extra_values)
|
|
{
|
|
GstStructure *s1, *s2;
|
|
|
|
s1 = gst_structure_from_string ("test/test, channels=(int)1", NULL);
|
|
fail_if (s1 == NULL);
|
|
s2 = gst_structure_from_string ("test/test, channels=(int)[ 1, 2 ]", NULL);
|
|
fail_if (s2 == NULL);
|
|
|
|
fail_unless (gst_structure_is_subset (s1, s2));
|
|
|
|
gst_structure_free (s1);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_structure_nested)
|
|
{
|
|
GstStructure *sp, *sc1, *sc2;
|
|
gchar *str;
|
|
|
|
sc1 = gst_structure_new ("Camera",
|
|
"XResolution", G_TYPE_INT, 72, "YResolution", G_TYPE_INT, 73, NULL);
|
|
fail_unless (sc1 != NULL);
|
|
|
|
sc2 = gst_structure_new ("Image-Data",
|
|
"Orientation", G_TYPE_STRING, "top-left",
|
|
"Comment", G_TYPE_STRING, "super photo", NULL);
|
|
fail_unless (sc2 != NULL);
|
|
|
|
sp = gst_structure_new ("Exif", "Camera", GST_TYPE_STRUCTURE, sc1,
|
|
"Image Data", GST_TYPE_STRUCTURE, sc2, NULL);
|
|
fail_unless (sp != NULL);
|
|
|
|
fail_unless (gst_structure_n_fields (sp) == 2);
|
|
|
|
fail_unless (gst_structure_has_field_typed (sp, "Camera",
|
|
GST_TYPE_STRUCTURE));
|
|
|
|
str = gst_structure_to_string (sp);
|
|
fail_unless (str != NULL);
|
|
|
|
GST_DEBUG ("serialized to '%s'", str);
|
|
|
|
fail_unless (g_str_equal (str,
|
|
"Exif"
|
|
", Camera=(structure)\"Camera\\,\\ XResolution\\=\\(int\\)72\\,\\ YResolution\\=\\(int\\)73\\;\""
|
|
", Image Data=(structure)\"Image-Data\\,\\ Orientation\\=\\(string\\)top-left\\,\\ Comment\\=\\(string\\)\\\"super\\\\\\ photo\\\"\\;\";"));
|
|
|
|
g_free (str);
|
|
str = NULL;
|
|
|
|
gst_structure_free (sc1);
|
|
gst_structure_free (sc2);
|
|
gst_structure_free (sp);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_structure_nested_from_and_to_string)
|
|
{
|
|
GstStructure *s;
|
|
const gchar *str1;
|
|
gchar *str2, *end = NULL;
|
|
|
|
str1 = "main"
|
|
", main-sub1=(structure)\"type-b\\,\\ machine-type\\=\\(int\\)0\\;\""
|
|
", main-sub2=(structure)\"type-a\\,\\ plugin-filename\\=\\(string\\)\\\"/home/user/lib/lib\\\\\\ with\\\\\\ spaces.dll\\\"\\,\\ machine-type\\=\\(int\\)1\\;\""
|
|
", main-sub3=(structure)\"type-b\\,\\ plugin-filename\\=\\(string\\)/home/user/lib/lib_no_spaces.so\\,\\ machine-type\\=\\(int\\)1\\;\""
|
|
";";
|
|
|
|
s = gst_structure_from_string (str1, &end);
|
|
fail_unless (s != NULL);
|
|
|
|
GST_DEBUG ("not parsed part : %s", end);
|
|
fail_unless (*end == '\0');
|
|
|
|
fail_unless (gst_structure_n_fields (s) == 3);
|
|
|
|
fail_unless (gst_structure_has_field_typed (s, "main-sub1",
|
|
GST_TYPE_STRUCTURE));
|
|
|
|
str2 = gst_structure_to_string (s);
|
|
fail_unless (str2 != NULL);
|
|
|
|
fail_unless (g_str_equal (str1, str2));
|
|
|
|
g_free (str2);
|
|
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_serialize_nested_structures)
|
|
{
|
|
GstStructure *s;
|
|
const gchar *str1;
|
|
gchar *str2, *end = NULL;
|
|
|
|
str1 = "main"
|
|
", main-sub1=(structure)[type-b, machine-type=(int)0;]"
|
|
", main-sub2=(structure)[type-a, plugin-filename=(string)\"/home/user/lib/lib\\ with\\ spaces.dll\", machine-type=(int)1;]"
|
|
", main-sub3=(structure)[type-b, plugin-filename=(string)/home/user/lib/lib_no_spaces.so, machine-type=(int)1;]"
|
|
";";
|
|
|
|
s = gst_structure_from_string (str1, &end);
|
|
fail_unless (s != NULL);
|
|
|
|
GST_DEBUG ("not parsed part : %s", end);
|
|
fail_unless (*end == '\0');
|
|
|
|
fail_unless (gst_structure_n_fields (s) == 3);
|
|
|
|
fail_unless (gst_structure_has_field_typed (s, "main-sub1",
|
|
GST_TYPE_STRUCTURE));
|
|
|
|
str2 = gst_structure_serialize (s, GST_SERIALIZE_FLAG_NONE);
|
|
fail_unless (str2 != NULL);
|
|
|
|
fail_unless (g_str_equal (str1, str2));
|
|
|
|
g_free (str2);
|
|
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_vararg_getters)
|
|
{
|
|
GstStructure *s;
|
|
GstBuffer *buf, *buf2;
|
|
gboolean ret;
|
|
GstCaps *caps, *caps2;
|
|
GstMapInfo info;
|
|
gdouble d;
|
|
gint64 i64;
|
|
gchar *c;
|
|
gint i, num, denom;
|
|
guint8 *data;
|
|
|
|
buf = gst_buffer_new_and_alloc (3);
|
|
|
|
fail_unless (gst_buffer_map (buf, &info, GST_MAP_WRITE));
|
|
data = info.data;
|
|
data[0] = 0xf0;
|
|
data[1] = 0x66;
|
|
data[2] = 0x0d;
|
|
gst_buffer_unmap (buf, &info);
|
|
|
|
caps = gst_caps_new_empty_simple ("video/x-foo");
|
|
|
|
s = gst_structure_new ("test", "int", G_TYPE_INT, 12345678, "string",
|
|
G_TYPE_STRING, "Hello World!", "buf", GST_TYPE_BUFFER, buf, "caps",
|
|
GST_TYPE_CAPS, caps, "int64", G_TYPE_INT64, G_GINT64_CONSTANT (-99),
|
|
"double", G_TYPE_DOUBLE, G_MAXDOUBLE, "frag", GST_TYPE_FRACTION, 39, 14,
|
|
NULL);
|
|
|
|
/* first the plain one */
|
|
ret = gst_structure_get (s, "double", G_TYPE_DOUBLE, &d, "string",
|
|
G_TYPE_STRING, &c, "caps", GST_TYPE_CAPS, &caps2, "buf",
|
|
GST_TYPE_BUFFER, &buf2, "frag", GST_TYPE_FRACTION, &num, &denom, "int",
|
|
G_TYPE_INT, &i, "int64", G_TYPE_INT64, &i64, NULL);
|
|
|
|
fail_unless (ret);
|
|
fail_unless_equals_string (c, "Hello World!");
|
|
fail_unless_equals_int (i, 12345678);
|
|
fail_unless_equals_float (d, G_MAXDOUBLE);
|
|
fail_unless_equals_int (num, 39);
|
|
fail_unless_equals_int (denom, 14);
|
|
fail_unless (i64 == -99);
|
|
fail_unless (caps == caps2);
|
|
fail_unless (buf == buf2);
|
|
|
|
/* expected failures */
|
|
ASSERT_CRITICAL (gst_structure_get (s, NULL, G_TYPE_INT, &i, NULL));
|
|
fail_if (gst_structure_get (s, "int", G_TYPE_INT, &i, "double",
|
|
G_TYPE_FLOAT, &d, NULL));
|
|
fail_if (gst_structure_get (s, "int", G_TYPE_INT, &i, "dooble",
|
|
G_TYPE_DOUBLE, &d, NULL));
|
|
|
|
g_free (c);
|
|
c = NULL;
|
|
gst_caps_unref (caps2);
|
|
caps2 = NULL;
|
|
gst_buffer_unref (buf2);
|
|
buf2 = NULL;
|
|
|
|
/* and now the _id variant */
|
|
ret = gst_structure_id_get (s, g_quark_from_static_string ("double"),
|
|
G_TYPE_DOUBLE, &d, g_quark_from_static_string ("string"), G_TYPE_STRING,
|
|
&c, g_quark_from_static_string ("caps"), GST_TYPE_CAPS, &caps2,
|
|
g_quark_from_static_string ("buf"), GST_TYPE_BUFFER, &buf2,
|
|
g_quark_from_static_string ("int"), G_TYPE_INT, &i,
|
|
g_quark_from_static_string ("int64"), G_TYPE_INT64, &i64, NULL);
|
|
|
|
fail_unless (ret);
|
|
fail_unless_equals_string (c, "Hello World!");
|
|
fail_unless_equals_int (i, 12345678);
|
|
fail_unless_equals_float (d, G_MAXDOUBLE);
|
|
fail_unless (i64 == -99);
|
|
fail_unless (caps == caps2);
|
|
fail_unless (buf == buf2);
|
|
|
|
/* expected failures */
|
|
ASSERT_CRITICAL (gst_structure_get (s, 0, G_TYPE_INT, &i, NULL));
|
|
fail_if (gst_structure_id_get (s, g_quark_from_static_string ("int"),
|
|
G_TYPE_INT, &i, g_quark_from_static_string ("double"), G_TYPE_FLOAT,
|
|
&d, NULL));
|
|
fail_if (gst_structure_id_get (s, g_quark_from_static_string ("int"),
|
|
G_TYPE_INT, &i, g_quark_from_static_string ("dooble"), G_TYPE_DOUBLE,
|
|
&d, NULL));
|
|
|
|
g_free (c);
|
|
gst_caps_unref (caps2);
|
|
gst_buffer_unref (buf2);
|
|
|
|
/* finally make sure NULL as return location is handled gracefully */
|
|
ret = gst_structure_get (s, "double", G_TYPE_DOUBLE, NULL, "string",
|
|
G_TYPE_STRING, NULL, "caps", GST_TYPE_CAPS, NULL, "buf",
|
|
GST_TYPE_BUFFER, NULL, "int", G_TYPE_INT, &i, "frag", GST_TYPE_FRACTION,
|
|
NULL, NULL, "int64", G_TYPE_INT64, &i64, NULL);
|
|
|
|
ASSERT_WARNING (gst_structure_get (s, "frag", GST_TYPE_FRACTION, NULL,
|
|
&denom, NULL));
|
|
ASSERT_WARNING (gst_structure_get (s, "frag", GST_TYPE_FRACTION, &num,
|
|
NULL, NULL));
|
|
|
|
/* clean up */
|
|
gst_caps_unref (caps);
|
|
gst_buffer_unref (buf);
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static gboolean
|
|
foreach_func (GQuark field_id, const GValue * value, gpointer user_data)
|
|
{
|
|
gint *sum = user_data;
|
|
gint v = 0;
|
|
|
|
if (G_VALUE_HOLDS_INT (value))
|
|
v = g_value_get_int (value);
|
|
*sum += v;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
GST_START_TEST (test_foreach)
|
|
{
|
|
GstStructure *s;
|
|
gint sum = 0;
|
|
|
|
s = gst_structure_new ("foo/bar", "baz", G_TYPE_INT, 1, "bla", G_TYPE_INT, 3,
|
|
NULL);
|
|
fail_unless (gst_structure_foreach (s, foreach_func, &sum));
|
|
fail_unless_equals_int (sum, 4);
|
|
gst_structure_free (s);
|
|
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static gboolean
|
|
map_func (GQuark field_id, GValue * value, gpointer user_data)
|
|
{
|
|
if (G_VALUE_HOLDS_INT (value))
|
|
g_value_set_int (value, 123);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
GST_START_TEST (test_map_in_place)
|
|
{
|
|
GstStructure *s, *s2;
|
|
|
|
s = gst_structure_new ("foo/bar", "baz", G_TYPE_INT, 1, "bla", G_TYPE_INT, 3,
|
|
NULL);
|
|
s2 = gst_structure_new ("foo/bar", "baz", G_TYPE_INT, 123, "bla", G_TYPE_INT,
|
|
123, NULL);
|
|
fail_unless (gst_structure_map_in_place (s, map_func, NULL));
|
|
fail_unless (gst_structure_is_equal (s, s2));
|
|
gst_structure_free (s);
|
|
gst_structure_free (s2);
|
|
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static gboolean
|
|
filter_map_func (GQuark field_id, GValue * value, gpointer user_data)
|
|
{
|
|
if (strcmp (g_quark_to_string (field_id), "bla") == 0)
|
|
return FALSE;
|
|
|
|
if (G_VALUE_HOLDS_INT (value))
|
|
g_value_set_int (value, 2);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
GST_START_TEST (test_filter_and_map_in_place)
|
|
{
|
|
GstStructure *s, *s2;
|
|
|
|
s = gst_structure_new ("foo/bar", "baz", G_TYPE_INT, 1, "bla", G_TYPE_INT, 3,
|
|
NULL);
|
|
s2 = gst_structure_new ("foo/bar", "baz", G_TYPE_INT, 2, NULL);
|
|
gst_structure_filter_and_map_in_place (s, filter_map_func, NULL);
|
|
fail_unless (gst_structure_is_equal (s, s2));
|
|
gst_structure_free (s);
|
|
gst_structure_free (s2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_flagset)
|
|
{
|
|
GstStructure *s;
|
|
GType test_flagset_type;
|
|
guint test_flags =
|
|
GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_SKIP | GST_SEEK_FLAG_SNAP_AFTER;
|
|
guint test_mask = GST_FLAG_SET_MASK_EXACT;
|
|
guint out_flags, out_mask;
|
|
|
|
test_flagset_type = gst_flagset_register (GST_TYPE_SEEK_FLAGS);
|
|
fail_unless (g_type_is_a (test_flagset_type, GST_TYPE_FLAG_SET));
|
|
|
|
/* Check that we can retrieve a non-standard flagset from the structure */
|
|
s = gst_structure_new ("test-struct", "test-flagset", test_flagset_type,
|
|
test_flags, test_mask, NULL);
|
|
fail_unless (gst_structure_get_flagset (s, "test-flagset", &out_flags,
|
|
&out_mask));
|
|
|
|
fail_unless (out_flags == test_flags);
|
|
fail_unless (out_mask == test_mask);
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
gst_structure_suite (void)
|
|
{
|
|
Suite *s = suite_create ("GstStructure");
|
|
TCase *tc_chain = tcase_create ("general");
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
tcase_add_test (tc_chain, test_from_string_int);
|
|
tcase_add_test (tc_chain, test_from_string_uint);
|
|
tcase_add_test (tc_chain, test_from_string);
|
|
tcase_add_test (tc_chain, test_to_string);
|
|
tcase_add_test (tc_chain, test_to_from_string);
|
|
tcase_add_test (tc_chain, test_to_from_string_tag_event);
|
|
tcase_add_test (tc_chain, test_string_properties);
|
|
tcase_add_test (tc_chain, test_complete_structure);
|
|
tcase_add_test (tc_chain, test_structure_new);
|
|
tcase_add_test (tc_chain, test_fixate);
|
|
tcase_add_test (tc_chain, test_fixate_stepped);
|
|
tcase_add_test (tc_chain, test_fixate_frac_list);
|
|
tcase_add_test (tc_chain, test_is_subset_equal_array_list);
|
|
tcase_add_test (tc_chain, test_is_subset_different_name);
|
|
tcase_add_test (tc_chain, test_is_subset_superset_missing_fields);
|
|
tcase_add_test (tc_chain, test_is_subset_superset_extra_fields);
|
|
tcase_add_test (tc_chain, test_is_subset_superset_extra_values);
|
|
tcase_add_test (tc_chain, test_structure_nested);
|
|
tcase_add_test (tc_chain, test_structure_nested_from_and_to_string);
|
|
tcase_add_test (tc_chain, test_serialize_nested_structures);
|
|
tcase_add_test (tc_chain, test_vararg_getters);
|
|
tcase_add_test (tc_chain, test_foreach);
|
|
tcase_add_test (tc_chain, test_map_in_place);
|
|
tcase_add_test (tc_chain, test_filter_and_map_in_place);
|
|
tcase_add_test (tc_chain, test_flagset);
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (gst_structure);
|