mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 09:55:36 +00:00
rtpopuspay: add DTX support
If enabled, the payloader won't transmit empty frames. Can be tested using: opusenc dtx=true bitrate-type=vbr ! rtpopuspay dtx=true Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-good/-/merge_requests/967>
This commit is contained in:
parent
a0067316e7
commit
41ba8c1b00
4 changed files with 176 additions and 1 deletions
|
@ -14854,7 +14854,20 @@
|
||||||
"presence": "always"
|
"presence": "always"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"properties": {},
|
"properties": {
|
||||||
|
"dtx": {
|
||||||
|
"blurb": "If enabled, the payloader will not transmit empty packets",
|
||||||
|
"conditionally-available": false,
|
||||||
|
"construct": false,
|
||||||
|
"construct-only": false,
|
||||||
|
"controllable": false,
|
||||||
|
"default": "false",
|
||||||
|
"mutable": "playing",
|
||||||
|
"readable": true,
|
||||||
|
"type": "gboolean",
|
||||||
|
"writable": true
|
||||||
|
}
|
||||||
|
},
|
||||||
"rank": "primary"
|
"rank": "primary"
|
||||||
},
|
},
|
||||||
"rtppcmadepay": {
|
"rtppcmadepay": {
|
||||||
|
|
|
@ -58,6 +58,13 @@
|
||||||
GST_DEBUG_CATEGORY_STATIC (rtpopuspay_debug);
|
GST_DEBUG_CATEGORY_STATIC (rtpopuspay_debug);
|
||||||
#define GST_CAT_DEFAULT (rtpopuspay_debug)
|
#define GST_CAT_DEFAULT (rtpopuspay_debug)
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PROP_0,
|
||||||
|
PROP_DTX,
|
||||||
|
};
|
||||||
|
|
||||||
|
#define DEFAULT_DTX FALSE
|
||||||
|
|
||||||
static GstStaticPadTemplate gst_rtp_opus_pay_sink_template =
|
static GstStaticPadTemplate gst_rtp_opus_pay_sink_template =
|
||||||
GST_STATIC_PAD_TEMPLATE ("sink",
|
GST_STATIC_PAD_TEMPLATE ("sink",
|
||||||
|
@ -90,24 +97,77 @@ G_DEFINE_TYPE (GstRtpOPUSPay, gst_rtp_opus_pay, GST_TYPE_RTP_BASE_PAYLOAD);
|
||||||
GST_ELEMENT_REGISTER_DEFINE_WITH_CODE (rtpopuspay, "rtpopuspay",
|
GST_ELEMENT_REGISTER_DEFINE_WITH_CODE (rtpopuspay, "rtpopuspay",
|
||||||
GST_RANK_PRIMARY, GST_TYPE_RTP_OPUS_PAY, rtp_element_init (plugin));
|
GST_RANK_PRIMARY, GST_TYPE_RTP_OPUS_PAY, rtp_element_init (plugin));
|
||||||
|
|
||||||
|
#define GST_RTP_OPUS_PAY_CAST(obj) ((GstRtpOPUSPay *)(obj))
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_rtp_opus_pay_set_property (GObject * object,
|
||||||
|
guint prop_id, const GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
GstRtpOPUSPay *self = GST_RTP_OPUS_PAY (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case PROP_DTX:
|
||||||
|
self->dtx = g_value_get_boolean (value);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_rtp_opus_pay_get_property (GObject * object,
|
||||||
|
guint prop_id, GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
GstRtpOPUSPay *self = GST_RTP_OPUS_PAY (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case PROP_DTX:
|
||||||
|
g_value_set_boolean (value, self->dtx);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_rtp_opus_pay_class_init (GstRtpOPUSPayClass * klass)
|
gst_rtp_opus_pay_class_init (GstRtpOPUSPayClass * klass)
|
||||||
{
|
{
|
||||||
GstRTPBasePayloadClass *gstbasertppayload_class;
|
GstRTPBasePayloadClass *gstbasertppayload_class;
|
||||||
GstElementClass *element_class;
|
GstElementClass *element_class;
|
||||||
|
GObjectClass *gobject_class;
|
||||||
|
|
||||||
gstbasertppayload_class = (GstRTPBasePayloadClass *) klass;
|
gstbasertppayload_class = (GstRTPBasePayloadClass *) klass;
|
||||||
element_class = GST_ELEMENT_CLASS (klass);
|
element_class = GST_ELEMENT_CLASS (klass);
|
||||||
|
gobject_class = (GObjectClass *) klass;
|
||||||
|
|
||||||
gstbasertppayload_class->set_caps = gst_rtp_opus_pay_setcaps;
|
gstbasertppayload_class->set_caps = gst_rtp_opus_pay_setcaps;
|
||||||
gstbasertppayload_class->get_caps = gst_rtp_opus_pay_getcaps;
|
gstbasertppayload_class->get_caps = gst_rtp_opus_pay_getcaps;
|
||||||
gstbasertppayload_class->handle_buffer = gst_rtp_opus_pay_handle_buffer;
|
gstbasertppayload_class->handle_buffer = gst_rtp_opus_pay_handle_buffer;
|
||||||
|
|
||||||
|
gobject_class->set_property = gst_rtp_opus_pay_set_property;
|
||||||
|
gobject_class->get_property = gst_rtp_opus_pay_get_property;
|
||||||
|
|
||||||
gst_element_class_add_static_pad_template (element_class,
|
gst_element_class_add_static_pad_template (element_class,
|
||||||
&gst_rtp_opus_pay_src_template);
|
&gst_rtp_opus_pay_src_template);
|
||||||
gst_element_class_add_static_pad_template (element_class,
|
gst_element_class_add_static_pad_template (element_class,
|
||||||
&gst_rtp_opus_pay_sink_template);
|
&gst_rtp_opus_pay_sink_template);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GstRtpOPUSPay:dtx:
|
||||||
|
*
|
||||||
|
* If enabled, the payloader will not transmit empty packets.
|
||||||
|
*
|
||||||
|
* Since: 1.20
|
||||||
|
*/
|
||||||
|
g_object_class_install_property (gobject_class, PROP_DTX,
|
||||||
|
g_param_spec_boolean ("dtx", "Discontinuous Transmission",
|
||||||
|
"If enabled, the payloader will not transmit empty packets",
|
||||||
|
DEFAULT_DTX,
|
||||||
|
G_PARAM_READWRITE | GST_PARAM_MUTABLE_PLAYING |
|
||||||
|
G_PARAM_STATIC_STRINGS));
|
||||||
|
|
||||||
gst_element_class_set_static_metadata (element_class,
|
gst_element_class_set_static_metadata (element_class,
|
||||||
"RTP Opus payloader",
|
"RTP Opus payloader",
|
||||||
"Codec/Payloader/Network/RTP",
|
"Codec/Payloader/Network/RTP",
|
||||||
|
@ -121,6 +181,7 @@ gst_rtp_opus_pay_class_init (GstRtpOPUSPayClass * klass)
|
||||||
static void
|
static void
|
||||||
gst_rtp_opus_pay_init (GstRtpOPUSPay * rtpopuspay)
|
gst_rtp_opus_pay_init (GstRtpOPUSPay * rtpopuspay)
|
||||||
{
|
{
|
||||||
|
rtpopuspay->dtx = DEFAULT_DTX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
|
@ -236,9 +297,18 @@ static GstFlowReturn
|
||||||
gst_rtp_opus_pay_handle_buffer (GstRTPBasePayload * basepayload,
|
gst_rtp_opus_pay_handle_buffer (GstRTPBasePayload * basepayload,
|
||||||
GstBuffer * buffer)
|
GstBuffer * buffer)
|
||||||
{
|
{
|
||||||
|
GstRtpOPUSPay *self = GST_RTP_OPUS_PAY_CAST (basepayload);
|
||||||
GstBuffer *outbuf;
|
GstBuffer *outbuf;
|
||||||
GstClockTime pts, dts, duration;
|
GstClockTime pts, dts, duration;
|
||||||
|
|
||||||
|
/* DTX packets are zero-length frames, with a 1 or 2-bytes header */
|
||||||
|
if (self->dtx && gst_buffer_get_size (buffer) <= 2) {
|
||||||
|
GST_LOG_OBJECT (self,
|
||||||
|
"discard empty buffer as DTX is enabled: %" GST_PTR_FORMAT, buffer);
|
||||||
|
gst_buffer_unref (buffer);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
||||||
pts = GST_BUFFER_PTS (buffer);
|
pts = GST_BUFFER_PTS (buffer);
|
||||||
dts = GST_BUFFER_DTS (buffer);
|
dts = GST_BUFFER_DTS (buffer);
|
||||||
duration = GST_BUFFER_DURATION (buffer);
|
duration = GST_BUFFER_DURATION (buffer);
|
||||||
|
|
|
@ -44,6 +44,8 @@ typedef struct _GstRtpOPUSPayClass GstRtpOPUSPayClass;
|
||||||
struct _GstRtpOPUSPay
|
struct _GstRtpOPUSPay
|
||||||
{
|
{
|
||||||
GstRTPBasePayload payload;
|
GstRTPBasePayload payload;
|
||||||
|
|
||||||
|
gboolean dtx;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _GstRtpOPUSPayClass
|
struct _GstRtpOPUSPayClass
|
||||||
|
|
|
@ -21,6 +21,7 @@
|
||||||
#include <gst/check/gstharness.h>
|
#include <gst/check/gstharness.h>
|
||||||
#include <gst/audio/audio.h>
|
#include <gst/audio/audio.h>
|
||||||
#include <gst/base/base.h>
|
#include <gst/base/base.h>
|
||||||
|
#include <gst/rtp/gstrtpbuffer.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#define RELEASE_ELEMENT(x) if(x) {gst_object_unref(x); x = NULL;}
|
#define RELEASE_ELEMENT(x) if(x) {gst_object_unref(x); x = NULL;}
|
||||||
|
@ -1670,6 +1671,93 @@ GST_START_TEST (rtp_vorbis_renegotiate)
|
||||||
|
|
||||||
GST_END_TEST;
|
GST_END_TEST;
|
||||||
|
|
||||||
|
static guint16
|
||||||
|
pull_rtp_buffer (GstHarness * h)
|
||||||
|
{
|
||||||
|
gint16 seq;
|
||||||
|
GstBuffer *buf;
|
||||||
|
GstRTPBuffer rtp = GST_RTP_BUFFER_INIT;
|
||||||
|
|
||||||
|
buf = gst_harness_try_pull (h);
|
||||||
|
fail_unless (buf);
|
||||||
|
|
||||||
|
fail_unless (gst_rtp_buffer_map (buf, GST_MAP_READ, &rtp));
|
||||||
|
seq = gst_rtp_buffer_get_seq (&rtp);
|
||||||
|
gst_rtp_buffer_unmap (&rtp);
|
||||||
|
|
||||||
|
gst_buffer_unref (buf);
|
||||||
|
return seq;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_rtp_opus_dtx (gboolean dtx)
|
||||||
|
{
|
||||||
|
GstHarness *h;
|
||||||
|
GstBuffer *buf;
|
||||||
|
/* generated with a muted mic using:
|
||||||
|
* gst-launch-1.0 pulsesrc ! opusenc dtx=true bitrate-type=vbr ! fakesink silent=false dump=true -v
|
||||||
|
*/
|
||||||
|
static const guint8 opus_empty[] = { 0xf8 };
|
||||||
|
static const guint8 opus_frame[] = { 0xf8, 0xff, 0xfe };
|
||||||
|
guint16 seq, expected_seq;
|
||||||
|
|
||||||
|
h = gst_harness_new_parse ("rtpopuspay");
|
||||||
|
fail_unless (h);
|
||||||
|
|
||||||
|
gst_harness_set (h, "rtpopuspay", "dtx", dtx, NULL);
|
||||||
|
|
||||||
|
gst_harness_set_caps_str (h,
|
||||||
|
"audio/x-opus, rate=48000, channels=1, channel-mapping-family=0",
|
||||||
|
"application/x-rtp, media=audio, clock-rate=48000, encoding-name=OPUS, sprop-stereo=(string)0, encoding-params=(string)2, sprop-maxcapturerate=(string)48000, payload=96");
|
||||||
|
|
||||||
|
/* push first opus frame */
|
||||||
|
buf =
|
||||||
|
gst_buffer_new_wrapped (g_memdup (opus_frame, sizeof (opus_frame)),
|
||||||
|
sizeof (opus_frame));
|
||||||
|
fail_unless_equals_int (gst_harness_push (h, buf), GST_FLOW_OK);
|
||||||
|
seq = pull_rtp_buffer (h);
|
||||||
|
expected_seq = seq + 1;
|
||||||
|
|
||||||
|
/* push empty frame */
|
||||||
|
buf =
|
||||||
|
gst_buffer_new_wrapped (g_memdup (opus_empty, sizeof (opus_empty)),
|
||||||
|
sizeof (opus_empty));
|
||||||
|
fail_unless_equals_int (gst_harness_push (h, buf), GST_FLOW_OK);
|
||||||
|
if (dtx) {
|
||||||
|
/* buffer is not transmitted if dtx is enabled */
|
||||||
|
buf = gst_harness_try_pull (h);
|
||||||
|
fail_if (buf);
|
||||||
|
} else {
|
||||||
|
seq = pull_rtp_buffer (h);
|
||||||
|
fail_unless_equals_int (seq, expected_seq);
|
||||||
|
expected_seq++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* push second opus frame */
|
||||||
|
buf =
|
||||||
|
gst_buffer_new_wrapped (g_memdup (opus_frame, sizeof (opus_frame)),
|
||||||
|
sizeof (opus_frame));
|
||||||
|
fail_unless_equals_int (gst_harness_push (h, buf), GST_FLOW_OK);
|
||||||
|
seq = pull_rtp_buffer (h);
|
||||||
|
fail_unless_equals_int (seq, expected_seq);
|
||||||
|
|
||||||
|
gst_harness_teardown (h);
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_START_TEST (rtp_opus_dtx_disabled)
|
||||||
|
{
|
||||||
|
test_rtp_opus_dtx (FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_END_TEST;
|
||||||
|
|
||||||
|
GST_START_TEST (rtp_opus_dtx_enabled)
|
||||||
|
{
|
||||||
|
test_rtp_opus_dtx (TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_END_TEST;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Creates the test suite.
|
* Creates the test suite.
|
||||||
*
|
*
|
||||||
|
@ -1734,6 +1822,8 @@ rtp_payloading_suite (void)
|
||||||
tcase_add_test (tc_chain, rtp_g729);
|
tcase_add_test (tc_chain, rtp_g729);
|
||||||
tcase_add_test (tc_chain, rtp_gst_custom_event);
|
tcase_add_test (tc_chain, rtp_gst_custom_event);
|
||||||
tcase_add_test (tc_chain, rtp_vorbis_renegotiate);
|
tcase_add_test (tc_chain, rtp_vorbis_renegotiate);
|
||||||
|
tcase_add_test (tc_chain, rtp_opus_dtx_disabled);
|
||||||
|
tcase_add_test (tc_chain, rtp_opus_dtx_enabled);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue