mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-13 19:05:37 +00:00
pad-monitor: add check for setcaps passing audio/video fields
Checks that the common audio/video fields are correctly passed downstream after a setcaps
This commit is contained in:
parent
0aaf23cb0f
commit
c650034fb4
2 changed files with 133 additions and 0 deletions
|
@ -40,6 +40,8 @@ GST_DEBUG_CATEGORY_STATIC (gst_qa_pad_monitor_debug);
|
||||||
G_DEFINE_TYPE_WITH_CODE (GstQaPadMonitor, gst_qa_pad_monitor,
|
G_DEFINE_TYPE_WITH_CODE (GstQaPadMonitor, gst_qa_pad_monitor,
|
||||||
GST_TYPE_QA_MONITOR, _do_init);
|
GST_TYPE_QA_MONITOR, _do_init);
|
||||||
|
|
||||||
|
#define PENDING_FIELDS "pending-fields"
|
||||||
|
|
||||||
static gboolean gst_qa_pad_monitor_do_setup (GstQaMonitor * monitor);
|
static gboolean gst_qa_pad_monitor_do_setup (GstQaMonitor * monitor);
|
||||||
|
|
||||||
/* This was copied from gstpad.c and might need
|
/* This was copied from gstpad.c and might need
|
||||||
|
@ -332,6 +334,8 @@ gst_qa_pad_monitor_dispose (GObject * object)
|
||||||
if (monitor->expected_segment)
|
if (monitor->expected_segment)
|
||||||
gst_event_unref (monitor->expected_segment);
|
gst_event_unref (monitor->expected_segment);
|
||||||
|
|
||||||
|
gst_structure_free (monitor->pending_setcaps_fields);
|
||||||
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->dispose (object);
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
||||||
}
|
}
|
||||||
|
@ -353,6 +357,8 @@ gst_qa_pad_monitor_class_init (GstQaPadMonitorClass * klass)
|
||||||
static void
|
static void
|
||||||
gst_qa_pad_monitor_init (GstQaPadMonitor * pad_monitor)
|
gst_qa_pad_monitor_init (GstQaPadMonitor * pad_monitor)
|
||||||
{
|
{
|
||||||
|
pad_monitor->pending_setcaps_fields =
|
||||||
|
gst_structure_empty_new (PENDING_FIELDS);
|
||||||
gst_segment_init (&pad_monitor->segment, GST_FORMAT_BYTES);
|
gst_segment_init (&pad_monitor->segment, GST_FORMAT_BYTES);
|
||||||
pad_monitor->first_buffer = TRUE;
|
pad_monitor->first_buffer = TRUE;
|
||||||
|
|
||||||
|
@ -590,6 +596,86 @@ gst_qa_pad_monitor_check_aggregated_return (GstQaPadMonitor * monitor,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (GstQaPadMonitor * monitor,
|
||||||
|
GstStructure * structure, const gchar * field)
|
||||||
|
{
|
||||||
|
GstIterator *iter;
|
||||||
|
gboolean done;
|
||||||
|
GstPad *otherpad;
|
||||||
|
GstQaPadMonitor *othermonitor;
|
||||||
|
const GValue *v;
|
||||||
|
|
||||||
|
v = gst_structure_get_value (structure, field);
|
||||||
|
if (v == NULL) {
|
||||||
|
GST_DEBUG_OBJECT (monitor, "Not adding pending field %s as it isn't "
|
||||||
|
"present on structure %" GST_PTR_FORMAT, field, structure);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
iter = gst_pad_iterate_internal_links (GST_QA_PAD_MONITOR_GET_PAD (monitor));
|
||||||
|
done = FALSE;
|
||||||
|
while (!done) {
|
||||||
|
switch (gst_iterator_next (iter, (gpointer *) & otherpad)) {
|
||||||
|
case GST_ITERATOR_OK:
|
||||||
|
othermonitor = g_object_get_data ((GObject *) otherpad, "qa-monitor");
|
||||||
|
if (othermonitor) {
|
||||||
|
g_assert (othermonitor->pending_setcaps_fields != NULL);
|
||||||
|
gst_structure_set_value (othermonitor->pending_setcaps_fields,
|
||||||
|
field, v);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_RESYNC:
|
||||||
|
gst_iterator_resync (iter);
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_ERROR:
|
||||||
|
GST_WARNING_OBJECT (monitor, "Internal links pad iteration error");
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_DONE:
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
gst_iterator_free (iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_qa_pad_monitor_otherpad_clear_pending_fields (GstQaPadMonitor * monitor)
|
||||||
|
{
|
||||||
|
GstIterator *iter;
|
||||||
|
gboolean done;
|
||||||
|
GstPad *otherpad;
|
||||||
|
GstQaPadMonitor *othermonitor;
|
||||||
|
|
||||||
|
iter = gst_pad_iterate_internal_links (GST_QA_PAD_MONITOR_GET_PAD (monitor));
|
||||||
|
done = FALSE;
|
||||||
|
while (!done) {
|
||||||
|
switch (gst_iterator_next (iter, (gpointer *) & otherpad)) {
|
||||||
|
case GST_ITERATOR_OK:
|
||||||
|
othermonitor = g_object_get_data ((GObject *) otherpad, "qa-monitor");
|
||||||
|
if (othermonitor) {
|
||||||
|
g_assert (othermonitor->pending_setcaps_fields != NULL);
|
||||||
|
gst_structure_free (othermonitor->pending_setcaps_fields);
|
||||||
|
othermonitor->pending_setcaps_fields =
|
||||||
|
gst_structure_empty_new (PENDING_FIELDS);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_RESYNC:
|
||||||
|
gst_iterator_resync (iter);
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_ERROR:
|
||||||
|
GST_WARNING_OBJECT (monitor, "Internal links pad iteration error");
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_DONE:
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
gst_iterator_free (iter);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_qa_pad_monitor_add_expected_newsegment (GstQaPadMonitor * monitor,
|
gst_qa_pad_monitor_add_expected_newsegment (GstQaPadMonitor * monitor,
|
||||||
GstEvent * event)
|
GstEvent * event)
|
||||||
|
@ -1024,11 +1110,56 @@ gst_qa_pad_monitor_setcaps_func (GstPad * pad, GstCaps * caps)
|
||||||
GstQaPadMonitor *pad_monitor =
|
GstQaPadMonitor *pad_monitor =
|
||||||
g_object_get_data ((GObject *) pad, "qa-monitor");
|
g_object_get_data ((GObject *) pad, "qa-monitor");
|
||||||
gboolean ret = TRUE;
|
gboolean ret = TRUE;
|
||||||
|
GstStructure *structure;
|
||||||
|
|
||||||
|
if (caps) {
|
||||||
|
structure = gst_caps_get_structure (caps, 0);
|
||||||
|
if (gst_structure_n_fields (pad_monitor->pending_setcaps_fields)) {
|
||||||
|
gint i;
|
||||||
|
for (i = 0; i < gst_structure_n_fields (structure); i++) {
|
||||||
|
const gchar *name = gst_structure_nth_field_name (structure, i);
|
||||||
|
const GValue *v = gst_structure_get_value (structure, name);
|
||||||
|
const GValue *otherv = gst_structure_get_value (structure, name);
|
||||||
|
|
||||||
|
if (!gst_value_compare (v, otherv) != GST_VALUE_EQUAL) {
|
||||||
|
GST_QA_MONITOR_REPORT_WARNING (pad_monitor, FALSE, CAPS_NEGOTIATION,
|
||||||
|
MISSING_FIELD,
|
||||||
|
"Field %s is missing from setcaps %" GST_PTR_FORMAT, name, caps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gst_qa_pad_monitor_pad_should_proxy_othercaps (pad_monitor)) {
|
||||||
|
/* TODO iterate structure and add pending expected fields */
|
||||||
|
if (_structure_is_video (structure)) {
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"width");
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"height");
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"framerate");
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"pixel-aspect-ratio");
|
||||||
|
} else if (_structure_is_audio (structure)) {
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"rate");
|
||||||
|
gst_qa_pad_monitor_otherpad_add_pending_field (pad_monitor, structure,
|
||||||
|
"channels");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_structure_free (pad_monitor->pending_setcaps_fields);
|
||||||
|
pad_monitor->pending_setcaps_fields =
|
||||||
|
gst_structure_empty_new (PENDING_FIELDS);
|
||||||
|
|
||||||
if (pad_monitor->setcaps_func) {
|
if (pad_monitor->setcaps_func) {
|
||||||
ret = pad_monitor->setcaps_func (pad, caps);
|
ret = pad_monitor->setcaps_func (pad, caps);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!ret)
|
||||||
|
gst_qa_pad_monitor_otherpad_clear_pending_fields (pad_monitor);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -81,6 +81,8 @@ struct _GstQaPadMonitor {
|
||||||
|
|
||||||
GstEvent *expected_segment;
|
GstEvent *expected_segment;
|
||||||
|
|
||||||
|
GstStructure *pending_setcaps_fields;
|
||||||
|
|
||||||
/* tracked data */
|
/* tracked data */
|
||||||
GstSegment segment;
|
GstSegment segment;
|
||||||
GstClockTime current_timestamp;
|
GstClockTime current_timestamp;
|
||||||
|
|
Loading…
Reference in a new issue