mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-25 00:28:21 +00:00
214691b972
on average period calculation. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1073>
849 lines
27 KiB
C
849 lines
27 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 "../../../ext/avtp/gstavtpcrfbase.c"
|
|
#undef GST_CAT_DEFAULT
|
|
|
|
#include <glib.h>
|
|
#include <avtp_crf.h>
|
|
#include <gst/check/gstcheck.h>
|
|
#include <gst/check/gstharness.h>
|
|
|
|
static struct avtp_crf_pdu *
|
|
generate_crf_pdu (int data_len, guint64 first_tstamp)
|
|
{
|
|
const guint64 base_freq = 48000;
|
|
const guint64 interval = 160;
|
|
const gdouble interval_time = 1.0e9 / base_freq * interval;
|
|
|
|
struct avtp_crf_pdu *crf_pdu =
|
|
g_malloc0 (sizeof (struct avtp_crf_pdu) + data_len);
|
|
|
|
avtp_crf_pdu_init (crf_pdu);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_SV, 1);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_STREAM_ID, 0xABCD1234ABCD1234);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_TYPE, AVTP_CRF_TYPE_AUDIO_SAMPLE);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_BASE_FREQ, base_freq);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_PULL, 1);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_CRF_DATA_LEN, data_len);
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_TIMESTAMP_INTERVAL, interval);
|
|
for (int i = 0; i < data_len / 8; i++) {
|
|
const guint64 offset = i * interval_time;
|
|
crf_pdu->crf_data[i] = htobe64 (first_tstamp + offset);
|
|
}
|
|
|
|
return crf_pdu;
|
|
}
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_success)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == TRUE);
|
|
|
|
/* Validate thread_data fields. */
|
|
fail_unless (thread_data->base_freq == 48000);
|
|
fail_unless (thread_data->pull == 1);
|
|
fail_unless (thread_data->type == AVTP_CRF_TYPE_AUDIO_SAMPLE);
|
|
fail_unless (thread_data->mr == 0);
|
|
fail_unless (thread_data->timestamp_interval == 160);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_multiple_packets_success)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu1 = generate_crf_pdu (64, 1000);
|
|
struct avtp_crf_pdu *crf_pdu2 = generate_crf_pdu (64, 1800);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu1,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == TRUE);
|
|
|
|
/* Validate thread_data fields. */
|
|
fail_unless (thread_data->base_freq == 48000);
|
|
fail_unless (thread_data->pull == 1);
|
|
fail_unless (thread_data->type == AVTP_CRF_TYPE_AUDIO_SAMPLE);
|
|
fail_unless (thread_data->mr == 0);
|
|
fail_unless (thread_data->timestamp_interval == 160);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu2,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == TRUE);
|
|
|
|
g_free (crf_pdu1);
|
|
g_free (crf_pdu2);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_wrong_subtype)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_pdu_set ((struct avtp_common_pdu *) crf_pdu, AVTP_FIELD_SUBTYPE,
|
|
AVTP_SUBTYPE_CVF);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_streamid_invalid)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_SV, 0);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_streamid_different)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCDABCD;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_data_len_too_long)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 40);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_timestamp_interval_zero)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_TIMESTAMP_INTERVAL, 0);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_base_freq_zero)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_BASE_FREQ, 0);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_pull_invalid)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_PULL, 7);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_type_invalid)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_TYPE, 8);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_data_len_invalid)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_CRF_DATA_LEN, 20);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_timestamp_interval_mismatch)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 120;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->num_pkt_tstamps = 8;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_base_freq_mismatch)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->base_freq = 44100;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->num_pkt_tstamps = 8;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_pull_mismatch)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 2;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->num_pkt_tstamps = 8;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_type_mismatch)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (64, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_VIDEO_FRAME;
|
|
thread_data->num_pkt_tstamps = 8;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + 64);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_data_len_mismatch)
|
|
{
|
|
int data_len = 48;
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->num_pkt_tstamps = 6;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_CRF_DATA_LEN, 20);
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + data_len);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_validate_crf_pdu_tstamps_not_monotonic)
|
|
{
|
|
int data_len = 48;
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 1000);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
gboolean ret;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->num_pkt_tstamps = 6;
|
|
|
|
crf_pdu->crf_data[3] = 1145;
|
|
|
|
ret =
|
|
validate_crf_pdu (avtpcrfbase, crf_pdu,
|
|
sizeof (struct avtp_crf_pdu) + data_len);
|
|
fail_unless (ret == FALSE);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_gst_base_freq_multiplier)
|
|
{
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
gdouble ret;
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 0);
|
|
fail_unless_equals_float (ret, 1.0);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 1);
|
|
fail_unless_equals_float (ret, 1 / 1.001);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 2);
|
|
fail_unless_equals_float (ret, 1.001);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 3);
|
|
fail_unless_equals_float (ret, 24.0 / 25);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 4);
|
|
fail_unless_equals_float (ret, 25.0 / 24);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 5);
|
|
fail_unless_equals_float (ret, 1.0 / 8);
|
|
|
|
ret = get_base_freq_multiplier (avtpcrfbase, 6);
|
|
fail_unless_equals_float (ret, -1);
|
|
gst_object_unref (avtpcrfbase);
|
|
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static void
|
|
setup_thread_defaults (GstAvtpCrfBase * avtpcrfbase, gdouble * past_periods)
|
|
{
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
avtpcrfbase->streamid = 0xABCD1234ABCD1234;
|
|
thread_data->base_freq = 48000;
|
|
thread_data->pull = 1;
|
|
thread_data->type = AVTP_CRF_TYPE_AUDIO_SAMPLE;
|
|
thread_data->past_periods = past_periods;
|
|
}
|
|
|
|
/*
|
|
* Test for more than 1 timestamp per CRF AVTPDU. This is just a simple success
|
|
* case.
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_multiple_crf_tstamps)
|
|
{
|
|
int data_len = 64;
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 1000);
|
|
gdouble past_periods[10] = { 21000, 20500, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->num_pkt_tstamps = 6;
|
|
thread_data->past_periods_iter = 2;
|
|
thread_data->periods_stored = 2;
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
fail_unless_equals_float (thread_data->average_period, 20777.7775);
|
|
fail_unless_equals_float (thread_data->past_periods[2], 20833.3325);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 1000);
|
|
|
|
gst_object_unref (avtpcrfbase);
|
|
g_free (crf_pdu);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test for rounding error
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_rounding_error)
|
|
{
|
|
/* the presentation time in ns */
|
|
const GstClockTimeDiff ptime = 50000000;
|
|
/* the time in ns of one sync event e.g. one audio sample @48kHz */
|
|
const gdouble event_interval = 1.0e9 / 48000;
|
|
/* the presentation time measured in sync events (e.g. sample rate)
|
|
* for class B traffic with a presentation time of 50ms.
|
|
*/
|
|
const GstClockTime ptime_in_events = ptime / event_interval;
|
|
|
|
/* With 4 timestamps generate_crf_pdu() multiples the interval time
|
|
* with 3. This results into an integer time stamp in nsi without decimal
|
|
* digits. Therefore the rounding issue when generating the timestamps for
|
|
* the CRF PDU is avoided here.
|
|
*/
|
|
int data_len = 32;
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 1000);
|
|
gdouble past_periods[10] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->num_pkt_tstamps = data_len / sizeof (uint64_t);
|
|
thread_data->past_periods_iter = 0;
|
|
thread_data->periods_stored = 0;
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
|
|
/* When internally using integer for average_period calculation the following
|
|
* multiplication will result to (20833 * 2400=) 49999200ns. This value
|
|
* differs by 800ns from the original presentation time of 50ms. When using
|
|
* double this rounding error is avoided.
|
|
*/
|
|
fail_unless_equals_float ((thread_data->average_period * ptime_in_events),
|
|
ptime);
|
|
|
|
gst_object_unref (avtpcrfbase);
|
|
g_free (crf_pdu);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test for an overflow in the 64-bit CRF timestamp in the CRF AVTPDU when
|
|
* there are multiple CRF timestamps in a packet.
|
|
*/
|
|
GST_START_TEST
|
|
(test_calculate_average_period_multiple_crf_tstamps_64_bit_overflow) {
|
|
int data_len = 64;
|
|
struct avtp_crf_pdu *crf_pdu =
|
|
generate_crf_pdu (data_len, 18446744073709501615ULL);
|
|
gdouble past_periods[10] =
|
|
{ 21000, 20500, 21220, 21345, 20990, 21996, 20220, 20915, 21324, 23123 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->num_pkt_tstamps = 6;
|
|
thread_data->past_periods_iter = 5;
|
|
thread_data->periods_stored = 10;
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
fail_unless_equals_float (thread_data->average_period, 21147.03325);
|
|
fail_unless_equals_float (thread_data->past_periods[5], 20833.3325);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 18446744073709501615ULL);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test case for single timestamp per CRF AVTPDU. This is just a simple success
|
|
* case.
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_single_crf_tstamp)
|
|
{
|
|
int data_len = 8;
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 21833);
|
|
gdouble past_periods[10] = { 21000, 20500, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 1;
|
|
thread_data->num_pkt_tstamps = 1;
|
|
thread_data->past_periods_iter = 2;
|
|
thread_data->periods_stored = 2;
|
|
thread_data->last_received_tstamp = 1000;
|
|
thread_data->last_seqnum = 9;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_SEQ_NUM, 10);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
fail_unless_equals_float (thread_data->average_period, 20777.6666666);
|
|
fail_unless_equals_float (thread_data->past_periods[2], 20833);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 10);
|
|
fail_unless_equals_uint64 (thread_data->last_received_tstamp, 21833);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 21833);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test to ensure all members of thread_data struct are initialized as expected
|
|
* when receiving multiple CRF AVTPDUs with single CRF timestamp.
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_single_crf_tstamp_init)
|
|
{
|
|
int data_len = 8;
|
|
struct avtp_crf_pdu *crf_pdu1 = generate_crf_pdu (data_len, 1000);
|
|
struct avtp_crf_pdu *crf_pdu2 = generate_crf_pdu (data_len, 21833);
|
|
gdouble past_periods[10] = { 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 1;
|
|
thread_data->num_pkt_tstamps = 1;
|
|
|
|
avtp_crf_pdu_set (crf_pdu1, AVTP_CRF_FIELD_SEQ_NUM, 10);
|
|
avtp_crf_pdu_set (crf_pdu2, AVTP_CRF_FIELD_SEQ_NUM, 11);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu1);
|
|
fail_unless_equals_float (thread_data->past_periods[0], 0);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 10);
|
|
fail_unless_equals_float (thread_data->average_period, 20854);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 1000);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu2);
|
|
fail_unless_equals_float (thread_data->past_periods[0], 20833);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 11);
|
|
fail_unless_equals_float (thread_data->average_period, 20833);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 21833);
|
|
|
|
g_free (crf_pdu1);
|
|
g_free (crf_pdu2);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test to ensure average_period is calculated correctly
|
|
* when receiving multiple CRF AVTPDUs with single CRF timestamp
|
|
* with timestamp_interval > 1
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_single_crf_tstamp_interval)
|
|
{
|
|
int data_len = 8;
|
|
struct avtp_crf_pdu *crf_pdu1 = generate_crf_pdu (data_len, 1000);
|
|
/* Used timestamp
|
|
* = sample_time * timestamp_interval + first_tstamp
|
|
* = 1/48kHz * 160 + 1000
|
|
*/
|
|
struct avtp_crf_pdu *crf_pdu2 = generate_crf_pdu (data_len, 3334280);
|
|
gdouble past_periods[10] = { 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 160;
|
|
thread_data->num_pkt_tstamps = 1;
|
|
|
|
avtp_crf_pdu_set (crf_pdu1, AVTP_CRF_FIELD_SEQ_NUM, 10);
|
|
avtp_crf_pdu_set (crf_pdu2, AVTP_CRF_FIELD_SEQ_NUM, 11);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu1);
|
|
fail_unless_equals_float (thread_data->past_periods[0], 0);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 10);
|
|
fail_unless_equals_float (thread_data->average_period, 20854);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 1000);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu2);
|
|
fail_unless_equals_float (thread_data->past_periods[0], 20833);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 11);
|
|
fail_unless_equals_float (thread_data->average_period, 20833);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 3334280);
|
|
|
|
g_free (crf_pdu1);
|
|
g_free (crf_pdu2);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test for an overflow in the 64-bit CRF timestamp in the CRF AVTPDU when
|
|
* there is a single CRF timestamp in a packet.
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_single_crf_tstamp_64_bit_overflow)
|
|
{
|
|
int data_len = 8;
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 20833);
|
|
gdouble past_periods[10] = { 21000, 20500, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 1;
|
|
thread_data->num_pkt_tstamps = 1;
|
|
thread_data->past_periods_iter = 2;
|
|
thread_data->periods_stored = 2;
|
|
thread_data->last_received_tstamp = 18446744073709551615ULL;
|
|
thread_data->last_seqnum = 9;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_SEQ_NUM, 10);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
fail_unless_equals_float (thread_data->average_period, 20778);
|
|
fail_unless_equals_float (thread_data->past_periods[2], 20834);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 10);
|
|
fail_unless_equals_uint64 (thread_data->last_received_tstamp, 20833);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 20833);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
/*
|
|
* Test to ensure expected behevior when a sequence number is skipped (probably
|
|
* due to packet loss or delay) in CRF AVTPDUs with single timestamp per
|
|
* AVTPDU.
|
|
*/
|
|
GST_START_TEST (test_calculate_average_period_single_crf_tstamp_seq_num_skip)
|
|
{
|
|
int data_len = 8;
|
|
struct avtp_crf_pdu *crf_pdu = generate_crf_pdu (data_len, 21833);
|
|
gdouble past_periods[10] = { 21000, 20500, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
GstAvtpCrfBase *avtpcrfbase = g_object_new (GST_TYPE_AVTP_CRF_BASE, NULL);
|
|
GstAvtpCrfThreadData *thread_data = &avtpcrfbase->thread_data;
|
|
|
|
setup_thread_defaults (avtpcrfbase, past_periods);
|
|
|
|
thread_data->timestamp_interval = 1;
|
|
thread_data->num_pkt_tstamps = 1;
|
|
thread_data->past_periods_iter = 2;
|
|
thread_data->last_received_tstamp = 1000;
|
|
thread_data->last_seqnum = 9;
|
|
thread_data->average_period = 20750;
|
|
|
|
avtp_crf_pdu_set (crf_pdu, AVTP_CRF_FIELD_SEQ_NUM, 12);
|
|
|
|
calculate_average_period (avtpcrfbase, crf_pdu);
|
|
fail_unless_equals_float (thread_data->average_period, 20750);
|
|
fail_unless_equals_float (thread_data->past_periods[2], 0);
|
|
fail_unless_equals_uint64 (thread_data->last_seqnum, 12);
|
|
fail_unless_equals_uint64 (thread_data->last_received_tstamp, 21833);
|
|
fail_unless_equals_uint64 (thread_data->current_ts, 21833);
|
|
|
|
g_free (crf_pdu);
|
|
gst_object_unref (avtpcrfbase);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
avtpcrfbase_suite (void)
|
|
{
|
|
Suite *s = suite_create ("avtpcrfbase");
|
|
TCase *tc_chain = tcase_create ("general");
|
|
|
|
GST_DEBUG_CATEGORY_INIT (avtpcrfbase_debug, "avtpcrfbase", 0, "CRF Base");
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_success);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_multiple_packets_success);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_wrong_subtype);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_streamid_invalid);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_streamid_different);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_data_len_too_long);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_timestamp_interval_zero);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_base_freq_zero);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_pull_invalid);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_type_invalid);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_data_len_invalid);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_timestamp_interval_mismatch);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_base_freq_mismatch);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_pull_mismatch);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_type_mismatch);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_data_len_mismatch);
|
|
tcase_add_test (tc_chain, test_validate_crf_pdu_tstamps_not_monotonic);
|
|
tcase_add_test (tc_chain, test_gst_base_freq_multiplier);
|
|
tcase_add_test (tc_chain, test_calculate_average_period_multiple_crf_tstamps);
|
|
tcase_add_test (tc_chain, test_calculate_average_period_rounding_error);
|
|
tcase_add_test (tc_chain,
|
|
test_calculate_average_period_multiple_crf_tstamps_64_bit_overflow);
|
|
tcase_add_test (tc_chain, test_calculate_average_period_single_crf_tstamp);
|
|
tcase_add_test (tc_chain,
|
|
test_calculate_average_period_single_crf_tstamp_init);
|
|
tcase_add_test (tc_chain,
|
|
test_calculate_average_period_single_crf_tstamp_interval);
|
|
tcase_add_test (tc_chain,
|
|
test_calculate_average_period_single_crf_tstamp_64_bit_overflow);
|
|
tcase_add_test (tc_chain,
|
|
test_calculate_average_period_single_crf_tstamp_seq_num_skip);
|
|
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (avtpcrfbase);
|