blob: a4d8b4d567e1859ec27ce96a4e00e395b284bf34 [file] [log] [blame]
/* GStreamer unit tests for the srtp elements
* Copyright (C) 2007 Tim-Philipp Müller <tim centricular net>
* Copyright (C) 2008 Sebastian Dröge <slomo@circular-chaos.org>
* Copyright (C) 2016 Collabora Ltd <vincent.penquerch@collabora.co.uk>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_VALGRIND
# include <valgrind/valgrind.h>
#endif
#include <gst/check/gstcheck.h>
#include <gst/check/gstharness.h>
GST_START_TEST (test_create_and_unref)
{
GstElement *e;
e = gst_element_factory_make ("srtpenc", NULL);
fail_unless (e != NULL);
gst_element_set_state (e, GST_STATE_NULL);
gst_object_unref (e);
e = gst_element_factory_make ("srtpdec", NULL);
fail_unless (e != NULL);
gst_element_set_state (e, GST_STATE_NULL);
gst_object_unref (e);
}
GST_END_TEST;
GST_START_TEST (test_play)
{
GstElement *source_pipeline, *sink_pipeline;
GstBus *source_bus;
GstMessage *msg;
source_pipeline =
gst_parse_launch
("audiotestsrc num-buffers=50 ! alawenc ! rtppcmapay ! application/x-rtp, payload=(int)8, ssrc=(uint)1356955624 ! srtpenc name=enc key=012345678901234567890123456789012345678901234567890123456789 ! udpsink port=5004 sync=false",
NULL);
sink_pipeline =
gst_parse_launch
("udpsrc port=5004 caps=\"application/x-srtp, payload=(int)8, ssrc=(uint)1356955624, srtp-key=(buffer)012345678901234567890123456789012345678901234567890123456789, srtp-cipher=(string)aes-128-icm, srtp-auth=(string)hmac-sha1-80, srtcp-cipher=(string)aes-128-icm, srtcp-auth=(string)hmac-sha1-80\" ! srtpdec name=dec ! rtppcmadepay ! alawdec ! fakesink",
NULL);
fail_unless (gst_element_set_state (source_pipeline,
GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
fail_unless (gst_element_set_state (sink_pipeline,
GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
source_bus = gst_pipeline_get_bus (GST_PIPELINE (source_pipeline));
msg =
gst_bus_timed_pop_filtered (source_bus, GST_CLOCK_TIME_NONE,
GST_MESSAGE_ERROR | GST_MESSAGE_EOS);
fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
gst_message_unref (msg);
gst_object_unref (source_bus);
gst_element_set_state (source_pipeline, GST_STATE_NULL);
gst_element_set_state (sink_pipeline, GST_STATE_NULL);
gst_object_unref (source_pipeline);
gst_object_unref (sink_pipeline);
}
GST_END_TEST;
typedef struct
{
guint counter;
guint start_roc;
} roc_check_data;
static guint
get_roc (GstElement * e)
{
const GstStructure *s, *ss;
const GValue *v;
guint roc = 0;
g_object_get (e, "stats", &s, NULL);
v = gst_structure_get_value (s, "streams");
fail_unless (v);
v = gst_value_array_get_value (v, 0);
ss = gst_value_get_structure (v);
gst_structure_get_uint (ss, "roc", &roc);
return roc;
}
static GstPadProbeReturn
roc_check_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
{
roc_check_data *data = user_data;
GstElement *e = GST_PAD_PARENT (pad);
if (G_UNLIKELY (data->counter % 8192 == 0))
GST_DEBUG_OBJECT (pad, "counter at %d", data->counter);
/* record first roc, then wait for 2^16 packets to pass */
if (data->counter == 0) {
data->start_roc = get_roc (e);
} else if (data->counter == 65536) {
/* get roc and check it's one more than what we started with */
fail_unless ((get_roc (e) & 0xffff) == ((data->start_roc + 1) & 0xffff));
}
data->counter++;
return GST_PAD_PROBE_OK;
}
static GstCaps *
request_key (void)
{
GstCaps *caps;
caps =
gst_caps_from_string
("application/x-srtp, payload=(int)8, ssrc=(uint)1356955624, srtp-key=(buffer)012345678901234567890123456789012345678901234567890123456789, srtp-cipher=(string)aes-128-icm, srtp-auth=(string)hmac-sha1-80, srtcp-cipher=(string)aes-128-icm, srtcp-auth=(string)hmac-sha1-80");
return caps;
}
GST_START_TEST (test_roc)
{
GstElement *source_pipeline, *sink_pipeline;
GstElement *srtpenc, *srtpdec;
GstBus *source_bus, *sink_bus;
GstMessage *msg;
GstPad *pad;
roc_check_data source_roc_check_data, sink_roc_check_data;
source_pipeline =
gst_parse_launch
("audiotestsrc num-buffers=65555 ! alawenc ! rtppcmapay ! application/x-rtp, payload=(int)8, ssrc=(uint)1356955624 ! srtpenc name=enc key=012345678901234567890123456789012345678901234567890123456789 ! udpsink port=5004 sync=false",
NULL);
sink_pipeline =
gst_parse_launch
("udpsrc port=5004 caps=\"application/x-srtp, payload=(int)8, ssrc=(uint)1356955624, srtp-key=(buffer)012345678901234567890123456789012345678901234567890123456789, srtp-cipher=(string)aes-128-icm, srtp-auth=(string)hmac-sha1-80, srtcp-cipher=(string)aes-128-icm, srtcp-auth=(string)hmac-sha1-80\" ! srtpdec name=dec ! rtppcmadepay ! alawdec ! fakesink",
NULL);
fail_unless (gst_element_set_state (source_pipeline,
GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
fail_unless (gst_element_set_state (sink_pipeline,
GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
source_bus = gst_pipeline_get_bus (GST_PIPELINE (source_pipeline));
sink_bus = gst_pipeline_get_bus (GST_PIPELINE (sink_pipeline));
/* install a pad probe on the srtp elements' source pads */
srtpenc = gst_bin_get_by_name (GST_BIN (source_pipeline), "enc");
fail_unless (srtpenc != NULL);
pad = gst_element_get_static_pad (srtpenc, "rtp_src_0");
fail_unless (pad != NULL);
source_roc_check_data.counter = 0;
gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_BUFFER, roc_check_probe,
&source_roc_check_data, NULL);
gst_object_unref (pad);
gst_object_unref (srtpenc);
srtpdec = gst_bin_get_by_name (GST_BIN (sink_pipeline), "dec");
fail_unless (srtpdec != NULL);
g_signal_connect (srtpdec, "request_key", G_CALLBACK (request_key),
GINT_TO_POINTER (0));
pad = gst_element_get_static_pad (srtpdec, "rtp_src");
sink_roc_check_data.counter = 0;
gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_BUFFER, roc_check_probe,
&sink_roc_check_data, NULL);
fail_unless (pad != NULL);
gst_object_unref (pad);
gst_object_unref (srtpdec);
msg =
gst_bus_timed_pop_filtered (source_bus, GST_CLOCK_TIME_NONE,
GST_MESSAGE_ERROR | GST_MESSAGE_EOS);
fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
gst_message_unref (msg);
gst_object_unref (source_bus);
gst_object_unref (sink_bus);
gst_element_set_state (source_pipeline, GST_STATE_NULL);
gst_element_set_state (sink_pipeline, GST_STATE_NULL);
gst_object_unref (source_pipeline);
gst_object_unref (sink_pipeline);
}
GST_END_TEST;
static Suite *
srtp_suite (void)
{
Suite *s = suite_create ("srtp");
TCase *tc_chain = tcase_create ("general");
suite_add_tcase (s, tc_chain);
tcase_set_timeout (tc_chain, 180);
tcase_add_test (tc_chain, test_create_and_unref);
tcase_add_test (tc_chain, test_play);
tcase_add_test (tc_chain, test_roc);
return s;
}
GST_CHECK_MAIN (srtp);