mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-15 03:45:38 +00:00
1502 lines
56 KiB
C
1502 lines
56 KiB
C
/*
|
|
* GStreamer AVTP Plugin
|
|
* Copyright (C) 2019 Intel Corporation
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <gst/check/gstcheck.h>
|
|
#include <gst/check/gstharness.h>
|
|
|
|
#include <avtp.h>
|
|
#include <avtp_cvf.h>
|
|
|
|
#define AVTP_CVF_H264_HEADER_SIZE (sizeof(struct avtp_stream_pdu) + sizeof(guint32))
|
|
#define STREAM_ID 0xAABBCCDDEEFF0000
|
|
|
|
static gboolean
|
|
check_nal_filling (GstBuffer * buffer, guint8 first)
|
|
{
|
|
GstMapInfo map;
|
|
gboolean result = TRUE;
|
|
gsize offset = 5; /* 4 bytes for the nal size and one with nal type */
|
|
int i;
|
|
|
|
gst_buffer_map (buffer, &map, GST_MAP_READ);
|
|
|
|
for (i = offset; i < map.size; i++) {
|
|
if (map.data[i] != first++) {
|
|
result = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
gst_buffer_unmap (buffer, &map);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
fill_nal (guint8 * buf, gsize size, guint8 first)
|
|
{
|
|
while (size--) {
|
|
*buf++ = first++;
|
|
}
|
|
}
|
|
|
|
static gsize
|
|
nal_size (GstBuffer * buffer)
|
|
{
|
|
guint8 nal_size[4];
|
|
|
|
gst_buffer_extract (buffer, 0, nal_size, 4);
|
|
return GST_READ_UINT32_BE (nal_size);
|
|
}
|
|
|
|
static gsize
|
|
nal_type (GstBuffer * buffer)
|
|
{
|
|
guint8 nal_type;
|
|
|
|
gst_buffer_extract (buffer, 4, &nal_type, 1);
|
|
return nal_type & 0x1f;
|
|
}
|
|
|
|
static GstBuffer *
|
|
fetch_nal (GstBuffer * buffer, gsize * offset)
|
|
{
|
|
gsize nal_size;
|
|
GstBuffer *ret;
|
|
guint8 buf[4];
|
|
|
|
if (*offset >= (gst_buffer_get_size (buffer) - 4))
|
|
return NULL;
|
|
|
|
gst_buffer_extract (buffer, *offset, buf, 4);
|
|
nal_size = GST_READ_UINT32_BE (buf);
|
|
|
|
ret =
|
|
gst_buffer_copy_region (buffer, GST_BUFFER_COPY_MEMORY, *offset,
|
|
nal_size + 4);
|
|
*offset += nal_size + 4;
|
|
|
|
return ret;
|
|
}
|
|
|
|
GST_START_TEST (test_depayloader_fragment_and_single)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in;
|
|
const gint DATA_LEN = sizeof (guint32) + 10;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay ! fakesink num-buffers=1");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 10);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
/* Start with a single NAL */
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_OK);
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Then a fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_OK);
|
|
|
|
/* Third and last AVTPDU, again a single NAL */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_EOS);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_fragmented_two_start_eos)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in;
|
|
const gint DATA_LEN = sizeof (guint32) + 10;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay ! fakesink num-buffers=1");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 10);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
/* Start with a single NAL */
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_OK);
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Then a fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_OK);
|
|
|
|
/* Third and last AVTPDU, another fragment with start bit set */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 16);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_EOS);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_multiple_lost_eos)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay ! fakesink num-buffers=1");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x7; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second AVTPDU, but skipping one seqnum */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_EOS);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_fragmented_eos)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in;
|
|
const gint DATA_LEN = sizeof (guint32) + 10;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay ! fakesink num-buffers=1");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 10);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_OK);
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second and last AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 16);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, gst_buffer_copy (in)),
|
|
GST_FLOW_EOS);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Tests a big fragmented NAL scenario */
|
|
GST_START_TEST (test_depayloader_single_eos)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay ! fakesink num-buffers=1");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
fail_unless_equals_int (gst_harness_push (h, in), GST_FLOW_EOS);
|
|
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_invalid_avtpdu)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *small;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_MJPEG);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* Invalid CVF subtype */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid subtype */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_FORMAT_SUBTYPE,
|
|
AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_pdu_set ((struct avtp_common_pdu *) pdu, AVTP_FIELD_SUBTYPE,
|
|
AVTP_SUBTYPE_CRF);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid CVF type */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_pdu_set ((struct avtp_common_pdu *) pdu, AVTP_FIELD_SUBTYPE,
|
|
AVTP_SUBTYPE_CVF);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_FORMAT, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid AVTP version */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_FORMAT, AVTP_CVF_FORMAT_RFC);
|
|
avtp_pdu_set ((struct avtp_common_pdu *) pdu, AVTP_FIELD_VERSION, 3);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid SV */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_pdu_set ((struct avtp_common_pdu *) pdu, AVTP_FIELD_VERSION, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SV, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid stream id */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, 0xAABBCCDDEEFF0001);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid stream data len */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, 100);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (STAP-A) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 24;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (STAP-B) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 25;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (MTAP16) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 26;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (MTAP24) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 27;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (FU-B) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 29;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid NAL type (STAP-A) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 24;
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid buffer size (too small to fit an AVTP header) */
|
|
small = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE / 2);
|
|
gst_harness_push (h, small);
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
/* Invalid buffer size (too small to fit a fragment header) */
|
|
small = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 1);
|
|
gst_buffer_map (small, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, sizeof (guint32) + 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 28;
|
|
gst_buffer_unmap (small, &map);
|
|
|
|
gst_harness_push (h, small);
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 0);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* This test will send some invalid fragments, but with valid seqnum
|
|
* (misbehaving payloader).*/
|
|
GST_START_TEST (test_depayloader_lost_fragments)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 10;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 10);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
/* First fragment doesn't have start bit set, so it should be ignored */
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = 4; /* S = 0, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second AVTPDU - but this should be also ignored as it doesn't have the
|
|
* start bit set */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = 4; /* type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 8);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send third AVTPDU, with end bit set, but it should be discarded as there
|
|
* was no start fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 16);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* Ensure no buffer came out */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Now, let's send an invalid one, with both start and end bits set */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 3);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (3 << 6) | 4; /* S = E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 24);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send a fragment with proper start */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 4);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 32);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* But send start again. Previous one should be dropped */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 5);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 40);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Finally, send ending fragment. It should come out a buffer
|
|
* whose content starts on 40 (starting of start fragment) */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 6);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 48);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
/* NAL is composed of 8 bytes fragment + reconstructed NAL header, so 17 bytes */
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 17);
|
|
fail_unless (check_nal_filling (nal, 40) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 4);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* This test jumps one seq_num, thus simulating a lost packet */
|
|
GST_START_TEST (test_depayloader_lost_packet)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x7; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send third and last AVTPDU, but jumping one SEQ_NUM.
|
|
* This should make the first two NAL units to be flushed,
|
|
* despite M not being set on this third packet.
|
|
* Also, this NAL is not filled from 0, so if it somehow
|
|
* leaks - it's not supposed to go outside of the avtpcvdepay
|
|
* as it doesn't have M bit set - we can catch on checks below */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 3);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 5);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_int (gst_harness_buffers_received (h), 1);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
/* Validate each NAL unit size and content */
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 7);
|
|
gst_buffer_unref (nal);
|
|
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 7);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* This test simulates a scenario in which one single NAL unit is sent,
|
|
* followed by a fragment without start bit set, so fragment is discarded
|
|
* and previous single NAL is sent to the pipeline, as avtpcvfdepay is not
|
|
* sure about the sanity of the data anymore - but hopes h264decoder knows
|
|
* what to do. This scenario emerges from misbehaving payloaders. */
|
|
GST_START_TEST (test_depayloader_single_and_messed_fragments)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* First, we send a single NAL with M = 0, so nothing should come out */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Then, we send invalid fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = 4; /* S = 0, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* When we push it, it should be discarded, but previous single NAL
|
|
* should come out */
|
|
out = gst_harness_push_and_pull (h, gst_buffer_copy (in));
|
|
|
|
/* Check that we got the right one */
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless_equals_uint64 (nal_type (nal), 1);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* This test explores the case in which a fragment is followed by
|
|
* a single NAL - and not by an ending fragment. Fragments stored
|
|
* so far are dropped, and things shall flow normally for the single NAL.
|
|
* This can be created by a misbehaving payloader */
|
|
GST_START_TEST (test_depayloader_single_and_messed_fragments_2)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* Send a perfectly valid start fragment */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Then, we send a single NAL. Previous fragment should be dropped */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x2; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 5);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* When we push it, it should come out as it has M = 1 */
|
|
out = gst_harness_push_and_pull (h, gst_buffer_copy (in));
|
|
|
|
/* Check that we got the right one - its NAL filling should start with 5 */
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless_equals_uint64 (nal_type (nal), 2);
|
|
fail_unless (check_nal_filling (nal, 5) == TRUE);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
/* To be really sure, send an ending fragment. It should be dropped,
|
|
* as there should not be any previous fragment on the wait */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 2);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* This test ensures that, if a fragment is dropped due arrival of a single
|
|
* NAL (and fragment was never completed), any previous single NAL waiting
|
|
* for M set NAL are flushed to the pipeline. avtpcvfdepay never sents known
|
|
* incomplete NAL units to the pipeline, but should not hold forever NALs
|
|
* waiting for an M set NAL - specially after something wrong already happened */
|
|
GST_START_TEST (test_depayloader_single_and_messed_fragments_3)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x2; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* Send a single NAL with M = 0, so nothing will come out */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send a valid start fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send a single NAL without ending fragment. So, both first NAL and second should
|
|
* come out, on two different buffers. Fragment should be gone. */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x3; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 7);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_uint64 (gst_harness_buffers_received (h), 2);
|
|
|
|
/* Check that we got the right ones. First has nal_type 2, and second 3.
|
|
* Second also has its nal filling starting from 7 */
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless_equals_uint64 (nal_type (nal), 2);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
gst_buffer_unref (out);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless_equals_uint64 (nal_type (nal), 3);
|
|
fail_unless (check_nal_filling (nal, 7) == TRUE);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
gst_buffer_unref (out);
|
|
|
|
/* To be really sure, send an ending fragment. It should be dropped,
|
|
* as there should not be any previous fragment on the wait */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 3);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 2);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_property)
|
|
{
|
|
GstHarness *h;
|
|
GstElement *element;
|
|
guint64 streamid;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new_parse ("avtpcvfdepay streamid=0xAABBCCDDEEFF0001");
|
|
|
|
/* Check if property was properly set up */
|
|
element = gst_harness_find_element (h, "avtpcvfdepay");
|
|
g_object_get (G_OBJECT (element), "streamid", &streamid, NULL);
|
|
fail_unless_equals_uint64 (streamid, 0xAABBCCDDEEFF0001);
|
|
|
|
gst_object_unref (element);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Tests if everything goes right when a single NAL unit without M bit is
|
|
* followed by fragments that, when merged, have the M bit set */
|
|
GST_START_TEST (test_depayloader_single_and_fragmented)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* First, we send a single NAL with M = 0, so nothing should come out */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Then, we send first fragment */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* And last */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 2, 2);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_int (gst_harness_buffers_received (h), 1);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
/* Validate each NAL unit size and content */
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 1);
|
|
gst_buffer_unref (nal);
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 5);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 4);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Tests a simple fragmented NAL scenario */
|
|
GST_START_TEST (test_depayloader_fragmented)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 10;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 10);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = 4; /* type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 8);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send third and last AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], 8, 16);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_int (gst_harness_buffers_received (h), 1);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 25);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 4);
|
|
gst_buffer_unref (nal);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Tests a big fragmented NAL scenario */
|
|
GST_START_TEST (test_depayloader_fragmented_big)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = 1470;
|
|
struct avtp_stream_pdu *pdu;
|
|
/* 12000 * 1468 > 2^24 - so we can check if nal size is retrieved correctly */
|
|
const gint nal_count = 12000;
|
|
guint8 seq_num = 0;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
gint i;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + DATA_LEN);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN,
|
|
DATA_LEN + sizeof (guint32));
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 3 << 5 | 28; /* NAL type FU-A, NRI 3 */
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 7) | 4; /* S = 1, type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], DATA_LEN - 2, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Loop sending fragments. The idea is to create a NAL unit big enough
|
|
* to use the 4 bytes of nal_length_size */
|
|
for (i = 0; i < nal_count - 1; i++) {
|
|
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, ++seq_num);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = 4; /* type 4 */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 2], DATA_LEN - 2,
|
|
(guint8) ((DATA_LEN - 2) * seq_num));
|
|
|
|
/* Last one is special - need to set M and TV, etc */
|
|
if (i == nal_count - 2) {
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE + 1] = (1 << 6) | 4; /* E = 1, type 4 */
|
|
}
|
|
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
if (i < nal_count - 2)
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
}
|
|
|
|
/* After last one was sent, we check everything */
|
|
fail_unless_equals_int (gst_harness_buffers_received (h), 1);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), (DATA_LEN - 2) * nal_count + 1);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 4);
|
|
gst_buffer_unref (nal);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/* Tests several single NAL units. They should be grouped and delivered
|
|
* to the pipeline only when one NAL unit with M bit set arrives */
|
|
GST_START_TEST (test_depayloader_multiple_single)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out, *nal;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
gsize offset;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 0);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x7; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
/* We push a copy so that we can change only what is necessary on our buffer */
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send second AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 1);
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless (gst_harness_try_pull (h) == NULL);
|
|
|
|
/* Send third and last AVTPDU */
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_SEQ_NUM, 2);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
gst_harness_push (h, gst_buffer_copy (in));
|
|
fail_unless_equals_int (gst_harness_buffers_received (h), 1);
|
|
|
|
out = gst_harness_pull (h);
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
|
|
/* Validate each NAL unit size and content */
|
|
offset = 0;
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 7);
|
|
gst_buffer_unref (nal);
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 7);
|
|
gst_buffer_unref (nal);
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless_equals_uint64 (nal_size (nal), 4);
|
|
fail_unless (check_nal_filling (nal, 0) == TRUE);
|
|
fail_unless_equals_uint64 (nal_type (nal), 1);
|
|
gst_buffer_unref (nal);
|
|
|
|
/* Ensure no other NAL units are present */
|
|
nal = fetch_nal (out, &offset);
|
|
fail_unless (nal == NULL);
|
|
|
|
gst_buffer_unref (out);
|
|
gst_buffer_unref (in);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_depayloader_single)
|
|
{
|
|
GstHarness *h;
|
|
GstBuffer *in, *out;
|
|
const gint DATA_LEN = sizeof (guint32) + 4;
|
|
struct avtp_stream_pdu *pdu;
|
|
GstMapInfo map;
|
|
|
|
/* Create the harness for the avtpcvfpay */
|
|
h = gst_harness_new ("avtpcvfdepay");
|
|
gst_harness_set_src_caps_str (h, "application/x-avtp");
|
|
|
|
/* Create the input AVTPDU header */
|
|
in = gst_harness_create_buffer (h, AVTP_CVF_H264_HEADER_SIZE + 4);
|
|
gst_buffer_map (in, &map, GST_MAP_READWRITE);
|
|
pdu = (struct avtp_stream_pdu *) map.data;
|
|
|
|
avtp_cvf_pdu_init (pdu, AVTP_CVF_FORMAT_SUBTYPE_H264);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_ID, STREAM_ID);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_M, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_TIMESTAMP, 1000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_PTV, 1);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_H264_TIMESTAMP, 2000000);
|
|
avtp_cvf_pdu_set (pdu, AVTP_CVF_FIELD_STREAM_DATA_LEN, DATA_LEN);
|
|
map.data[AVTP_CVF_H264_HEADER_SIZE] = 0x1; /* Add NAL type */
|
|
fill_nal (&map.data[AVTP_CVF_H264_HEADER_SIZE + 1], 3, 0);
|
|
gst_buffer_unmap (in, &map);
|
|
|
|
out = gst_harness_push_and_pull (h, in);
|
|
|
|
fail_unless_equals_uint64 (GST_BUFFER_DTS (out), 1000000);
|
|
fail_unless_equals_uint64 (GST_BUFFER_PTS (out), 2000000);
|
|
fail_unless_equals_uint64 (nal_size (out), 4);
|
|
fail_unless_equals_uint64 (nal_type (out), 1);
|
|
fail_unless (check_nal_filling (out, 0) == TRUE);
|
|
gst_buffer_unref (out);
|
|
gst_harness_teardown (h);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
avtpcvfdepay_suite (void)
|
|
{
|
|
Suite *s = suite_create ("avtpcvfdepay");
|
|
TCase *tc_chain = tcase_create ("general");
|
|
TCase *tc_slow = tcase_create ("slow");
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
tcase_add_test (tc_chain, test_depayloader_single);
|
|
tcase_add_test (tc_chain, test_depayloader_multiple_single);
|
|
tcase_add_test (tc_chain, test_depayloader_fragmented);
|
|
tcase_add_test (tc_chain, test_depayloader_single_and_fragmented);
|
|
tcase_add_test (tc_chain, test_depayloader_property);
|
|
tcase_add_test (tc_chain, test_depayloader_lost_packet);
|
|
tcase_add_test (tc_chain, test_depayloader_lost_fragments);
|
|
tcase_add_test (tc_chain, test_depayloader_single_and_messed_fragments);
|
|
tcase_add_test (tc_chain, test_depayloader_single_and_messed_fragments_2);
|
|
tcase_add_test (tc_chain, test_depayloader_single_and_messed_fragments_3);
|
|
tcase_add_test (tc_chain, test_depayloader_invalid_avtpdu);
|
|
tcase_add_test (tc_chain, test_depayloader_single_eos);
|
|
tcase_add_test (tc_chain, test_depayloader_fragmented_eos);
|
|
tcase_add_test (tc_chain, test_depayloader_fragmented_two_start_eos);
|
|
tcase_add_test (tc_chain, test_depayloader_multiple_lost_eos);
|
|
tcase_add_test (tc_chain, test_depayloader_fragment_and_single);
|
|
|
|
suite_add_tcase (s, tc_slow);
|
|
/* 'fragmented_big' may take some time to run, so give it a bit more time */
|
|
tcase_set_timeout (tc_slow, 20);
|
|
tcase_add_test (tc_slow, test_depayloader_fragmented_big);
|
|
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (avtpcvfdepay);
|