mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-03 14:08:56 +00:00
aaee569087
- Replace all id type from gint to guint - Fix annotation for GstAnalyticsODMtdData Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/5758>
978 lines
31 KiB
C
978 lines
31 KiB
C
/* GStreamer
|
|
* Copyright (C) 2022 Collabora Ltd
|
|
*
|
|
* analyticmeta.c
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <gst/check/gstcheck.h>
|
|
#include <gst/analytics/analytics.h>
|
|
|
|
GST_START_TEST (test_add_classification_meta)
|
|
{
|
|
/* Verify we can create a relation metadata
|
|
* and attach it classification mtd
|
|
*/
|
|
GstBuffer *buf;
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
GstAnalyticsClsMtd cls_mtd;
|
|
gfloat conf_lvl[] = { 0.5f, 0.5f };
|
|
GQuark class_quarks[2];
|
|
gboolean ret;
|
|
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
buf = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta (buf);
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, conf_lvl,
|
|
class_quarks, &cls_mtd);
|
|
fail_unless (ret == TRUE);
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_meta_pooled)
|
|
{
|
|
GstBufferPool *pool;
|
|
GstStructure *config;
|
|
GstBuffer *buf;
|
|
GstAnalyticsRelationMeta *rmeta1, *rmeta2;
|
|
|
|
pool = gst_buffer_pool_new ();
|
|
config = gst_buffer_pool_get_config (pool);
|
|
gst_buffer_pool_config_set_params (config, NULL, 1, 1, 1);
|
|
gst_buffer_pool_set_config (pool, config);
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
rmeta1 = gst_buffer_add_analytics_relation_meta (buf);
|
|
gst_buffer_unref (buf);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
|
|
rmeta2 = gst_buffer_add_analytics_relation_meta (buf);
|
|
|
|
fail_unless (rmeta1 == rmeta2);
|
|
gst_buffer_unref (buf);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_classification_meta_classes)
|
|
{
|
|
/* Verify we can retrieve classification data
|
|
* from the relation metadata
|
|
*/
|
|
GstBuffer *buf;
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
gboolean ret;
|
|
GQuark class_quarks[2];
|
|
GstAnalyticsClsMtd cls_mtd, cls_mtd2;
|
|
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
buf = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta (buf);
|
|
gfloat conf_lvl[] = { 0.6f, 0.4f };
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, conf_lvl,
|
|
class_quarks, &cls_mtd);
|
|
fail_unless (ret == TRUE);
|
|
fail_unless (gst_analytics_relation_get_length (rmeta) == 1);
|
|
|
|
gint dogIndex = gst_analytics_cls_mtd_get_index_by_quark (&cls_mtd,
|
|
class_quarks[0]);
|
|
fail_unless (dogIndex == 0);
|
|
gfloat confLvl = gst_analytics_cls_mtd_get_level (&cls_mtd, dogIndex);
|
|
GST_LOG ("dog:%f", confLvl);
|
|
assert_equals_float (confLvl, 0.6f);
|
|
|
|
gint catIndex = gst_analytics_cls_mtd_get_index_by_quark (&cls_mtd,
|
|
g_quark_from_string ("cat"));
|
|
confLvl = gst_analytics_cls_mtd_get_level (&cls_mtd, catIndex);
|
|
GST_LOG ("Cat:%f", confLvl);
|
|
assert_equals_float (confLvl, 0.4f);
|
|
assert_equals_int (gst_analytics_mtd_get_id ((GstAnalyticsMtd *) & cls_mtd),
|
|
0);
|
|
|
|
conf_lvl[0] = 0.1f;
|
|
conf_lvl[1] = 0.9f;
|
|
ret =
|
|
gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, conf_lvl,
|
|
class_quarks, &cls_mtd2);
|
|
fail_unless (ret == TRUE);
|
|
fail_unless (gst_analytics_relation_get_length (rmeta) == 2);
|
|
|
|
dogIndex = gst_analytics_cls_mtd_get_index_by_quark (&cls_mtd2,
|
|
class_quarks[0]);
|
|
confLvl = gst_analytics_cls_mtd_get_level (&cls_mtd2, dogIndex);
|
|
assert_equals_float (confLvl, 0.1f);
|
|
|
|
catIndex = gst_analytics_cls_mtd_get_index_by_quark (&cls_mtd2,
|
|
class_quarks[0]);
|
|
confLvl = gst_analytics_cls_mtd_get_level (&cls_mtd2, catIndex);
|
|
assert_equals_float (confLvl, 0.1f);
|
|
|
|
/* Verify the relation meta contain the correct number of relatable metadata */
|
|
fail_unless (gst_analytics_relation_get_length (rmeta) == 2);
|
|
|
|
/* Verify first relatable metadata has the correct id. */
|
|
assert_equals_int (gst_analytics_mtd_get_id ((GstAnalyticsMtd *) & cls_mtd),
|
|
0);
|
|
|
|
/* Verify second relatable metadata has the correct id. */
|
|
assert_equals_int (gst_analytics_mtd_get_id (
|
|
(GstAnalyticsMtd *) & cls_mtd2), 1);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_add_relation_meta)
|
|
{
|
|
/* Verify we set a relation between relatable metadata. */
|
|
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *relations;
|
|
GQuark class_quarks[2];
|
|
gboolean ret;
|
|
|
|
buf = gst_buffer_new ();
|
|
relations = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
gfloat conf_lvl[] = { 0.6f, 0.4f };
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
fail_unless (ret == TRUE);
|
|
gst_analytics_mtd_get_id ((GstAnalyticsMtd *)
|
|
& cls_mtd[0]);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
|
|
class_quarks[0] = g_quark_from_string ("plant");
|
|
class_quarks[1] = g_quark_from_string ("animal");
|
|
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
gst_analytics_mtd_get_id ((GstAnalyticsMtd *)
|
|
& cls_mtd[1]);
|
|
|
|
fail_unless (gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id,
|
|
cls_mtd[1].id) == TRUE);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_add_relation_inefficiency_reporting_cases)
|
|
{
|
|
/*
|
|
* Verify inefficiency of relation order is reported.
|
|
*/
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 2, 10 };
|
|
GstAnalyticsRelationMeta *relations;
|
|
gboolean ret;
|
|
GQuark class_quarks[2];
|
|
|
|
|
|
buf = gst_buffer_new ();
|
|
relations = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
gfloat conf_lvl[] = { 0.6f, 0.4f };
|
|
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
fail_unless (gst_analytics_relation_get_length (relations) == 1);
|
|
fail_unless (ret == TRUE);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("plant");
|
|
class_quarks[1] = g_quark_from_string ("animal");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
fail_unless (gst_analytics_relation_get_length (relations) == 2);
|
|
fail_unless (ret == TRUE);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("male");
|
|
class_quarks[1] = g_quark_from_string ("female");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[2]);
|
|
fail_unless (gst_analytics_relation_get_length (relations) == 3);
|
|
fail_unless (ret == TRUE);
|
|
|
|
fail_unless (gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id, cls_mtd[1].id)
|
|
== TRUE);
|
|
fail_unless (gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id, cls_mtd[2].id)
|
|
== TRUE);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_query_relation_meta_cases)
|
|
{
|
|
/* Verify we can query existence of direct and indirect relation */
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 2, 150 };
|
|
GstAnalyticsRelationMeta *relations;
|
|
gboolean ret;
|
|
GQuark class_quarks[2];
|
|
|
|
buf = gst_buffer_new ();
|
|
relations = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
gfloat conf_lvl[] = { 0.6f, 0.4f };
|
|
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
fail_unless (ret == TRUE);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("plant");
|
|
class_quarks[1] = g_quark_from_string ("animal");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
fail_unless (ret == TRUE);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("male");
|
|
class_quarks[1] = g_quark_from_string ("female");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[2]);
|
|
fail_unless (ret == TRUE);
|
|
|
|
// Pet is part of kingdom
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id, cls_mtd[1].id);
|
|
|
|
// Kingdom contain pet
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, cls_mtd[1].id, cls_mtd[0].id);
|
|
|
|
// Pet contain gender
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, cls_mtd[0].id, cls_mtd[2].id);
|
|
|
|
/* Query if pet relate kingdom through a IS_PART relation with a maximum
|
|
* relation span of 1. Max relation span of 1 mean they directly related.*/
|
|
gboolean exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[1].id, 1, GST_ANALYTICS_REL_TYPE_IS_PART_OF, NULL);
|
|
fail_unless (exist == TRUE);
|
|
|
|
/* Query if pet relate to gender through a IS_PART relation. */
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[2].id, 1, GST_ANALYTICS_REL_TYPE_IS_PART_OF, NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
/* Query if pet relate to kingdom through a CONTAIN relation. */
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[1].id, 1, GST_ANALYTICS_REL_TYPE_CONTAIN, NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
GstAnalyticsRelTypes cond =
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF | GST_ANALYTICS_REL_TYPE_CONTAIN |
|
|
GST_ANALYTICS_REL_TYPE_RELATE_TO;
|
|
|
|
/* Query if pet relate to gender through IS_PART or CONTAIN or
|
|
* RELATE_TO relation. */
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[2].id, 1, cond, NULL);
|
|
fail_unless (exist == TRUE);
|
|
|
|
/* Query if pet relate to kindom through CONTAIN or RELATE_TO relation */
|
|
cond = GST_ANALYTICS_REL_TYPE_CONTAIN | GST_ANALYTICS_REL_TYPE_RELATE_TO;
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[1].id, 1, cond, NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
/* Query if kingdom relate to gender through a CONTAIN relation with a maximum
|
|
* relation span of 1. */
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[1].id,
|
|
cls_mtd[2].id, 1, GST_ANALYTICS_REL_TYPE_CONTAIN, NULL);
|
|
|
|
/* We expect this to fail because kingdom relate to gender CONTAIN relations
|
|
* but indirectly (via pet) and we set the max relation span to 1*/
|
|
fail_unless (exist == FALSE);
|
|
|
|
/* Same has previous check but using INFINIT relation span */
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[1].id,
|
|
cls_mtd[2].id, GST_INF_RELATION_SPAN, GST_ANALYTICS_REL_TYPE_CONTAIN,
|
|
NULL);
|
|
fail_unless (exist == TRUE);
|
|
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[2].id,
|
|
cls_mtd[1].id, GST_INF_RELATION_SPAN, GST_ANALYTICS_REL_TYPE_CONTAIN,
|
|
NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_path_relation_meta)
|
|
{
|
|
/* Verify we can retrieve relation path */
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 2, 150 };
|
|
GstAnalyticsRelationMeta *relations;
|
|
gboolean ret;
|
|
GQuark class_quarks[2];
|
|
|
|
buf = gst_buffer_new ();
|
|
relations = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
gfloat conf_lvl[] = { 0.6f, 0.4f };
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
fail_unless (ret == TRUE);
|
|
fail_unless (cls_mtd[0].id == 0);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("plant");
|
|
class_quarks[1] = g_quark_from_string ("animal");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
fail_unless (ret == TRUE);
|
|
fail_unless (cls_mtd[1].id == 1);
|
|
|
|
conf_lvl[0] = 0.6f;
|
|
conf_lvl[1] = 0.4f;
|
|
class_quarks[0] = g_quark_from_string ("male");
|
|
class_quarks[1] = g_quark_from_string ("female");
|
|
|
|
ret = gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[2]);
|
|
fail_unless (ret == TRUE);
|
|
fail_unless (cls_mtd[2].id == 2);
|
|
|
|
// Pet is part of kingdom
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id, cls_mtd[1].id);
|
|
|
|
// Kingdom contain pet
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, cls_mtd[1].id, cls_mtd[0].id);
|
|
|
|
// Pet contain gender
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, cls_mtd[0].id, cls_mtd[2].id);
|
|
|
|
GArray *path = NULL;
|
|
GstAnalyticsRelTypes cond = GST_ANALYTICS_REL_TYPE_CONTAIN;
|
|
gboolean exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[2].id, GST_INF_RELATION_SPAN, cond, &path);
|
|
if (exist) {
|
|
fail_unless (path != NULL);
|
|
gint i;
|
|
guint path_check_ids[] = { 0, 2 };
|
|
fail_unless (path->len == 2);
|
|
for (i = 0; i < path->len; i++) {
|
|
fail_unless (path_check_ids[i] == g_array_index (path, guint, i));
|
|
}
|
|
g_array_free (g_steal_pointer (&path), TRUE);
|
|
fail_unless (i == 2);
|
|
}
|
|
|
|
cond = GST_ANALYTICS_REL_TYPE_CONTAIN;
|
|
exist = gst_analytics_relation_meta_exist (relations, cls_mtd[1].id,
|
|
cls_mtd[2].id, GST_INF_RELATION_SPAN, cond, &path);
|
|
if (exist) {
|
|
gint i;
|
|
guint path_check_ids[] = { 1, 0, 2 };
|
|
fail_unless (path != NULL);
|
|
fail_unless (path->len == 3);
|
|
for (i = 0; i < path->len; i++) {
|
|
fail_unless (path_check_ids[i] == g_array_index (path, guint, i));
|
|
}
|
|
g_array_free (g_steal_pointer (&path), TRUE);
|
|
fail_unless (i == 3);
|
|
}
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_cyclic_relation_meta)
|
|
{
|
|
/* Verify we can discover cycle in relation and not report multiple time
|
|
* the same node and get into an infinit exploration */
|
|
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 2, 150 };
|
|
GstAnalyticsRelationMeta *relations;
|
|
gfloat conf_lvl[2];
|
|
GQuark class_quarks[2];
|
|
|
|
class_quarks[0] = g_quark_from_string ("attr1");
|
|
class_quarks[1] = g_quark_from_string ("attr2");
|
|
|
|
buf = gst_buffer_new ();
|
|
relations = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
conf_lvl[0] = 0.5f;
|
|
conf_lvl[1] = 0.5f;
|
|
gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
|
|
gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
|
|
gst_analytics_relation_meta_add_cls_mtd (relations, 2, conf_lvl,
|
|
class_quarks, &cls_mtd[2]);
|
|
|
|
// (0) -> (1)
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[0].id, cls_mtd[1].id);
|
|
|
|
// (1)->(2)
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[1].id, cls_mtd[2].id);
|
|
|
|
// (2) -> (0)
|
|
gst_analytics_relation_meta_set_relation (relations,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd[2].id, cls_mtd[0].id);
|
|
|
|
GArray *path = NULL;
|
|
GstAnalyticsRelTypes cond = GST_ANALYTICS_REL_TYPE_CONTAIN;
|
|
gboolean exist = gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[2].id, GST_INF_RELATION_SPAN, cond, &path);
|
|
fail_unless (exist == FALSE);
|
|
|
|
cond = GST_ANALYTICS_REL_TYPE_IS_PART_OF;
|
|
exist =
|
|
gst_analytics_relation_meta_exist (relations, cls_mtd[0].id,
|
|
cls_mtd[2].id, GST_INF_RELATION_SPAN, cond, &path);
|
|
fail_unless (exist == TRUE);
|
|
if (exist) {
|
|
gint i;
|
|
guint path_ids[] = { 0, 1, 2 };
|
|
fail_unless (path->len == 3);
|
|
for (i = 0; i < path->len; i++) {
|
|
fail_unless (path_ids[i] == g_array_index (path, guint, i));
|
|
}
|
|
g_array_free (g_steal_pointer (&path), TRUE);
|
|
fail_unless (i == 3);
|
|
}
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_add_od_meta)
|
|
{
|
|
/* Verity we can add Object Detection relatable metadata to a relation
|
|
* metadata */
|
|
GstBuffer *buf;
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
GstAnalyticsODMtd od_mtd;
|
|
gboolean ret;
|
|
buf = gst_buffer_new ();
|
|
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
GQuark type = g_quark_from_string ("dog");
|
|
gint x = 20;
|
|
gint y = 20;
|
|
gint w = 10;
|
|
gint h = 15;
|
|
gfloat loc_conf_lvl = 0.6f;
|
|
ret = gst_analytics_relation_meta_add_od_mtd (rmeta, type, x, y,
|
|
w, h, loc_conf_lvl, &od_mtd);
|
|
fail_unless (ret == TRUE);
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_od_meta_fields)
|
|
{
|
|
/* Verify we can readback fields of object detection metadata */
|
|
GstBuffer *buf;
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
GstAnalyticsODMtd od_mtd;
|
|
gboolean ret;
|
|
buf = gst_buffer_new ();
|
|
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
GQuark type = g_quark_from_string ("dog");
|
|
gint x = 21;
|
|
gint y = 20;
|
|
gint w = 10;
|
|
gint h = 15;
|
|
gfloat loc_conf_lvl = 0.6f;
|
|
ret = gst_analytics_relation_meta_add_od_mtd (rmeta, type, x, y,
|
|
w, h, loc_conf_lvl, &od_mtd);
|
|
|
|
fail_unless (ret == TRUE);
|
|
|
|
gint _x, _y, _w, _h;
|
|
gfloat _loc_conf_lvl;
|
|
gst_analytics_od_mtd_get_location (&od_mtd, &_x, &_y, &_w, &_h,
|
|
&_loc_conf_lvl);
|
|
|
|
fail_unless (_x == x);
|
|
fail_unless (_y == y);
|
|
fail_unless (_w == w);
|
|
fail_unless (_h == h);
|
|
fail_unless (_loc_conf_lvl == loc_conf_lvl);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_od_cls_relation)
|
|
{
|
|
/* Verify we can add a object detection and classification metadata to
|
|
* a relation metadata */
|
|
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd;
|
|
GstAnalyticsODMtd od_mtd;
|
|
/* We intentionally set buffer small than required to verify sanity
|
|
* with re-allocation */
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
gboolean ret;
|
|
GArray *path = NULL;
|
|
gboolean exist;
|
|
gint _x, _y, _w, _h;
|
|
gfloat _loc_conf_lvl;
|
|
GQuark class_quarks[2];
|
|
|
|
buf = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
gfloat conf_lvl[] = { 0.7f, 0.3f };
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, conf_lvl,
|
|
class_quarks, &cls_mtd);
|
|
|
|
GQuark type = g_quark_from_string ("dog");
|
|
gint x = 21;
|
|
gint y = 20;
|
|
gint w = 10;
|
|
gint h = 15;
|
|
gfloat loc_conf_lvl = 0.6f;
|
|
gst_analytics_relation_meta_add_od_mtd (rmeta, type, x, y, w, h,
|
|
loc_conf_lvl, &od_mtd);
|
|
|
|
ret = gst_analytics_relation_meta_set_relation (rmeta,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, od_mtd.id, cls_mtd.id);
|
|
fail_unless (ret == TRUE);
|
|
|
|
ret = gst_analytics_relation_meta_set_relation (rmeta,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, cls_mtd.id, od_mtd.id);
|
|
fail_unless (ret == TRUE);
|
|
|
|
/* Verify OD relate to CLS only through a CONTAIN relation */
|
|
exist = gst_analytics_relation_meta_exist (rmeta,
|
|
od_mtd.id, cls_mtd.id, GST_INF_RELATION_SPAN,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
exist = gst_analytics_relation_meta_exist (rmeta,
|
|
od_mtd.id, cls_mtd.id, GST_INF_RELATION_SPAN,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, &path);
|
|
fail_unless (exist == TRUE);
|
|
|
|
/* Query the relation path and verify it is correct */
|
|
guint ids[2];
|
|
gint i;
|
|
fail_unless (path->len == 2);
|
|
for (i = 0; i < path->len; i++) {
|
|
ids[i] = g_array_index (path, guint, i);
|
|
GST_LOG ("id=%u", ids[i]);
|
|
}
|
|
g_array_free (g_steal_pointer (&path), TRUE);
|
|
fail_unless (ids[0] == 1);
|
|
fail_unless (ids[1] == 0);
|
|
|
|
GstAnalyticsMtd rlt_mtd;
|
|
exist = gst_analytics_relation_meta_get_mtd (rmeta, ids[0], 0, &rlt_mtd);
|
|
fail_unless (exist == TRUE);
|
|
|
|
GQuark mtd_type = gst_analytics_mtd_get_type_quark (&rlt_mtd);
|
|
|
|
/* Verify relatable meta with id == 1 is of type Object Detection */
|
|
fail_unless (mtd_type == gst_analytics_od_mtd_get_type_quark ());
|
|
|
|
gst_analytics_od_mtd_get_location ((GstAnalyticsODMtd *) & rlt_mtd, &_x, &_y,
|
|
&_w, &_h, &_loc_conf_lvl);
|
|
fail_unless (_x == x);
|
|
fail_unless (_y == y);
|
|
fail_unless (_w == w);
|
|
fail_unless (_h == h);
|
|
fail_unless (_loc_conf_lvl == loc_conf_lvl);
|
|
|
|
GST_LOG ("mtd_type:%s", g_quark_to_string (mtd_type));
|
|
|
|
exist = gst_analytics_relation_meta_get_mtd (rmeta, ids[1], 0, &rlt_mtd);
|
|
fail_unless (exist == TRUE);
|
|
mtd_type = gst_analytics_mtd_get_type_quark (&rlt_mtd);
|
|
|
|
/* Verify relatable meta with id == 0 is of type classification */
|
|
fail_unless (mtd_type == gst_analytics_cls_mtd_get_type_quark ());
|
|
gint index =
|
|
gst_analytics_cls_mtd_get_index_by_quark ((GstAnalyticsClsMtd *) &
|
|
rlt_mtd,
|
|
g_quark_from_string ("dog"));
|
|
gfloat lvl =
|
|
gst_analytics_cls_mtd_get_level ((GstAnalyticsClsMtd *) & rlt_mtd, index);
|
|
GST_LOG ("dog %f [%d, %d %d, %d", lvl, _x, _y, _w, _h);
|
|
|
|
fail_unless (lvl == 0.7f);
|
|
index =
|
|
gst_analytics_cls_mtd_get_index_by_quark ((GstAnalyticsClsMtd *) &
|
|
rlt_mtd, g_quark_from_string ("cat"));
|
|
lvl =
|
|
gst_analytics_cls_mtd_get_level ((GstAnalyticsClsMtd *) & rlt_mtd, index);
|
|
fail_unless (lvl == 0.3f);
|
|
|
|
GST_LOG ("mtd_type:%s", g_quark_to_string (mtd_type));
|
|
GST_LOG ("cat %f [%d, %d %d, %d", lvl, _x, _y, _w, _h);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_multi_od_cls_relation)
|
|
{
|
|
GstBuffer *buf;
|
|
GstAnalyticsClsMtd cls_mtd[3];
|
|
GstAnalyticsODMtd od_mtd[2];
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
guint cls_id, ids[2], i;
|
|
gboolean ret;
|
|
const gint dog_cls_index = 0;
|
|
const gint cat_cls_index = 1;
|
|
gfloat cls_conf_lvl[2], lvl;
|
|
GArray *path = NULL;
|
|
gfloat _loc_conf_lvl;
|
|
gint x, _x, y, _y, w, _w, h, _h;
|
|
GQuark mtd_type, cls_type;
|
|
GstAnalyticsMtd mtd;
|
|
gpointer state = NULL;
|
|
GQuark class_quarks[2];
|
|
class_quarks[0] = g_quark_from_string ("dog");
|
|
class_quarks[1] = g_quark_from_string ("cat");
|
|
|
|
|
|
buf = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf, &init_params);
|
|
|
|
/* Define first relation ObjectDetection -contain-> Classification */
|
|
cls_conf_lvl[dog_cls_index] = 0.7f;
|
|
cls_conf_lvl[cat_cls_index] = 0.3f;
|
|
|
|
gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, cls_conf_lvl,
|
|
class_quarks, &cls_mtd[0]);
|
|
|
|
cls_type = g_quark_from_string ("dog");
|
|
x = 21;
|
|
y = 20;
|
|
w = 10;
|
|
h = 15;
|
|
gfloat loc_conf_lvl = 0.6f;
|
|
gst_analytics_relation_meta_add_od_mtd (rmeta, cls_type, x, y, w,
|
|
h, loc_conf_lvl, &od_mtd[0]);
|
|
|
|
ret = gst_analytics_relation_meta_set_relation (rmeta,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, od_mtd[0].id, cls_mtd[0].id);
|
|
fail_unless (ret == TRUE);
|
|
GST_LOG ("Set rel Obj:%d -c-> Cls:%d", od_mtd[0].id, cls_mtd[0].id);
|
|
|
|
/* Define second relation ObjectDetection -contain-> Classification */
|
|
cls_conf_lvl[dog_cls_index] = 0.1f;
|
|
cls_conf_lvl[cat_cls_index] = 0.9f;
|
|
gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, cls_conf_lvl,
|
|
class_quarks, &cls_mtd[1]);
|
|
|
|
cls_type = g_quark_from_string ("cat");
|
|
x = 50;
|
|
y = 21;
|
|
w = 11;
|
|
h = 16;
|
|
loc_conf_lvl = 0.7f;
|
|
gst_analytics_relation_meta_add_od_mtd (rmeta, cls_type, x, y, w,
|
|
h, loc_conf_lvl, &od_mtd[1]);
|
|
|
|
ret = gst_analytics_relation_meta_set_relation (rmeta,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, od_mtd[1].id, cls_mtd[1].id);
|
|
|
|
GST_LOG ("Set rel Obj:%d -c-> Cls:%d", od_mtd[1].id, cls_mtd[1].id);
|
|
fail_unless (ret == TRUE);
|
|
|
|
/* Query relations */
|
|
|
|
/* Query relation between first object detection and first classification
|
|
* and verify they are only related by CONTAIN relation OD relate to
|
|
* CLASSIFICATION through a CONTAIN relation. */
|
|
gboolean exist =
|
|
gst_analytics_relation_meta_exist (rmeta, od_mtd[0].id, cls_mtd[0].id,
|
|
GST_INF_RELATION_SPAN,
|
|
GST_ANALYTICS_REL_TYPE_IS_PART_OF, NULL);
|
|
fail_unless (exist == FALSE);
|
|
|
|
exist =
|
|
gst_analytics_relation_meta_exist (rmeta, od_mtd[0].id, cls_mtd[0].id,
|
|
GST_INF_RELATION_SPAN, GST_ANALYTICS_REL_TYPE_CONTAIN, NULL);
|
|
fail_unless (exist == TRUE);
|
|
|
|
|
|
/* Query relation between second object detection and second classification
|
|
* and verify they are only related by CONTAIN relation OD relate to
|
|
* CLASSIFICATION through a CONTAIN relation.
|
|
*/
|
|
exist =
|
|
gst_analytics_relation_meta_exist (rmeta, od_mtd[1].id, cls_mtd[1].id,
|
|
GST_INF_RELATION_SPAN, GST_ANALYTICS_REL_TYPE_CONTAIN, &path);
|
|
fail_unless (exist == TRUE);
|
|
|
|
/* Verify relation path between OD second (id=3) and Cls second (id=2)
|
|
* is correct
|
|
*/
|
|
fail_unless (path->len == 2);
|
|
for (i = 0; i < path->len; i++) {
|
|
ids[i] = g_array_index (path, guint, i);
|
|
GST_LOG ("id=%u", ids[i]);
|
|
}
|
|
g_array_free (g_steal_pointer (&path), TRUE);
|
|
fail_unless (ids[0] == 3);
|
|
fail_unless (ids[1] == 2);
|
|
|
|
/* Verify the relatable metadata 3 is of correct type
|
|
* (ObjectDetection). Verify it describe the correct
|
|
* the correct data.
|
|
*/
|
|
gst_analytics_relation_meta_get_mtd (rmeta, ids[0], 0, &mtd);
|
|
mtd_type = gst_analytics_mtd_get_type_quark (&mtd);
|
|
fail_unless (mtd_type == gst_analytics_od_mtd_get_type_quark ());
|
|
|
|
gst_analytics_od_mtd_get_location ((GstAnalyticsODMtd *) & mtd, &_x, &_y, &_w,
|
|
&_h, &_loc_conf_lvl);
|
|
fail_unless (_x == 50);
|
|
fail_unless (_y == 21);
|
|
fail_unless (_w == 11);
|
|
fail_unless (_h == 16);
|
|
fail_unless (_loc_conf_lvl == 0.7f);
|
|
|
|
GST_LOG ("mtd_type:%s", g_quark_to_string (mtd_type));
|
|
|
|
/* Verify the relatable metadata 2 is of correct type
|
|
* (ObjectDetection).
|
|
*/
|
|
gst_analytics_relation_meta_get_mtd (rmeta, ids[1], 0, &mtd);
|
|
mtd_type = gst_analytics_mtd_get_type_quark (&mtd);
|
|
fail_unless (mtd_type == gst_analytics_cls_mtd_get_type_quark ());
|
|
|
|
/* Verify data of the CLASSIFICATION retrieved */
|
|
gint index =
|
|
gst_analytics_cls_mtd_get_index_by_quark ((GstAnalyticsClsMtd *) & mtd,
|
|
g_quark_from_string ("dog"));
|
|
lvl = gst_analytics_cls_mtd_get_level ((GstAnalyticsClsMtd *) & mtd, index);
|
|
GST_LOG ("dog %f [%d, %d %d, %d", lvl, _x, _y, _w, _h);
|
|
fail_unless (lvl == 0.1f);
|
|
|
|
/* Verify data of the CLASSIFICATION retrieved */
|
|
index =
|
|
gst_analytics_cls_mtd_get_index_by_quark ((GstAnalyticsClsMtd *) & mtd,
|
|
g_quark_from_string ("cat"));
|
|
lvl = gst_analytics_cls_mtd_get_level ((GstAnalyticsClsMtd *) & mtd, index);
|
|
GST_LOG ("mtd_type:%s", g_quark_to_string (mtd_type));
|
|
GST_LOG ("cat %f [%d, %d %d, %d", lvl, _x, _y, _w, _h);
|
|
fail_unless (lvl == 0.9f);
|
|
|
|
/* Retrieve relatable metadata related to the first object detection
|
|
* through a CONTAIN relation of type CLASSIFICATION
|
|
* Verify it's the first classification metadata
|
|
*/
|
|
gst_analytics_relation_meta_get_direct_related (rmeta, od_mtd[0].id,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, gst_analytics_cls_mtd_get_type_quark (),
|
|
&state, &mtd);
|
|
|
|
cls_id = gst_analytics_mtd_get_id (&mtd);
|
|
GST_LOG ("Obj:%d -> Cls:%d", od_mtd[0].id, cls_id);
|
|
fail_unless (cls_id == cls_mtd[0].id);
|
|
|
|
state = NULL;
|
|
/* Retrieve relatable metadata related to the second object detection
|
|
* through a CONTAIN relation of type CLASSIFICATION
|
|
* Verify it's the first classification metadata
|
|
*/
|
|
gst_analytics_relation_meta_get_direct_related (rmeta, od_mtd[1].id,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, gst_analytics_cls_mtd_get_type_quark (),
|
|
&state, &mtd);
|
|
cls_id = gst_analytics_mtd_get_id (&mtd);
|
|
|
|
GST_LOG ("Obj:%d -> Cls:%d", od_mtd[1].id, cls_id);
|
|
fail_unless (cls_id == cls_mtd[1].id);
|
|
|
|
cls_conf_lvl[dog_cls_index] = 0.2f;
|
|
cls_conf_lvl[cat_cls_index] = 0.8f;
|
|
class_quarks[0] = g_quark_from_string ("canine");
|
|
class_quarks[1] = g_quark_from_string ("feline");
|
|
gst_analytics_relation_meta_add_cls_mtd (rmeta, 2, cls_conf_lvl,
|
|
class_quarks, &cls_mtd[2]);
|
|
|
|
ret = gst_analytics_relation_meta_set_relation (rmeta,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, od_mtd[1].id, cls_mtd[2].id);
|
|
|
|
state = NULL;
|
|
ret = gst_analytics_relation_meta_get_direct_related (rmeta, od_mtd[1].id,
|
|
GST_ANALYTICS_REL_TYPE_CONTAIN, gst_analytics_od_mtd_get_type_quark (),
|
|
&state, &mtd);
|
|
|
|
fail_unless (ret == FALSE);
|
|
|
|
gst_buffer_unref (buf);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_add_tracking_meta)
|
|
{
|
|
/* Verify we can add tracking relatable meta to relation meta */
|
|
GstBuffer *buf1, *buf2;
|
|
GstAnalyticsRelationMetaInitParams init_params = { 5, 150 };
|
|
GstAnalyticsRelationMeta *rmeta;
|
|
GstAnalyticsTrackingMtd tracking_mtd;
|
|
guint tracking_id;
|
|
GstClockTime tracking_observation_time_1;
|
|
gboolean ret;
|
|
|
|
/* Verify we can add multiple trackings to relation metadata
|
|
*/
|
|
|
|
buf1 = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf1, &init_params);
|
|
tracking_id = 1;
|
|
tracking_observation_time_1 = GST_BUFFER_TIMESTAMP (buf1);
|
|
ret = gst_analytics_relation_meta_add_tracking_mtd (rmeta, tracking_id,
|
|
tracking_observation_time_1, &tracking_mtd);
|
|
fail_unless (ret == TRUE);
|
|
|
|
gst_buffer_unref (buf1);
|
|
|
|
buf2 = gst_buffer_new ();
|
|
rmeta = gst_buffer_add_analytics_relation_meta_full (buf2, &init_params);
|
|
tracking_id = 1;
|
|
ret = gst_analytics_relation_meta_add_tracking_mtd (rmeta, tracking_id,
|
|
tracking_observation_time_1, &tracking_mtd);
|
|
fail_unless (ret == TRUE);
|
|
|
|
gst_buffer_unref (buf2);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
analyticmeta_suite (void)
|
|
{
|
|
|
|
Suite *s;
|
|
TCase *tc_chain_cls;
|
|
TCase *tc_chain_relation;
|
|
TCase *tc_chain_od;
|
|
TCase *tc_chain_od_cls;
|
|
TCase *tc_chain_tracking;
|
|
|
|
s = suite_create ("Analytic Meta Library");
|
|
|
|
tc_chain_cls = tcase_create ("Classification Mtd");
|
|
suite_add_tcase (s, tc_chain_cls);
|
|
tcase_add_test (tc_chain_cls, test_add_classification_meta);
|
|
tcase_add_test (tc_chain_cls, test_classification_meta_classes);
|
|
|
|
tcase_add_test (tc_chain_cls, test_meta_pooled);
|
|
|
|
tc_chain_relation = tcase_create ("Relation Meta");
|
|
suite_add_tcase (s, tc_chain_relation);
|
|
tcase_add_test (tc_chain_relation, test_add_relation_meta);
|
|
tcase_add_test (tc_chain_relation,
|
|
test_add_relation_inefficiency_reporting_cases);
|
|
tcase_add_test (tc_chain_relation, test_query_relation_meta_cases);
|
|
tcase_add_test (tc_chain_relation, test_path_relation_meta);
|
|
tcase_add_test (tc_chain_relation, test_cyclic_relation_meta);
|
|
|
|
tc_chain_od = tcase_create ("Object Detection Mtd");
|
|
suite_add_tcase (s, tc_chain_od);
|
|
tcase_add_test (tc_chain_od, test_add_od_meta);
|
|
tcase_add_test (tc_chain_od, test_od_meta_fields);
|
|
|
|
tc_chain_od_cls = tcase_create ("Object Detection <-> Classification Mtd");
|
|
suite_add_tcase (s, tc_chain_od_cls);
|
|
tcase_add_test (tc_chain_od_cls, test_od_cls_relation);
|
|
tcase_add_test (tc_chain_od_cls, test_multi_od_cls_relation);
|
|
|
|
tc_chain_tracking = tcase_create ("Tracking Mtd");
|
|
suite_add_tcase (s, tc_chain_tracking);
|
|
tcase_add_test (tc_chain_tracking, test_add_tracking_meta);
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (analyticmeta);
|