mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-15 11:55:32 +00:00
03b6226772
If a stream has an 'irregular' frame rate (e.g. metadata) RTCP SR may be generated way too early, before the RTPSource has received the first packet after Latency was configured in the pipeline. We skip such RTPSources in the RTCP generation. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/7740>
2229 lines
68 KiB
C
2229 lines
68 KiB
C
/* GStreamer
|
|
* Copyright (C) <2007> Wim Taymans <wim.taymans@gmail.com>
|
|
* Copyright (C) 2015 Kurento (http://kurento.org/)
|
|
* @author: Miguel París <mparisdiaz@gmail.com>
|
|
*
|
|
* 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.
|
|
*/
|
|
#include <string.h>
|
|
|
|
#include <gst/rtp/gstrtpbuffer.h>
|
|
#include <gst/rtp/gstrtcpbuffer.h>
|
|
|
|
#include "rtpsource.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (rtp_source_debug);
|
|
#define GST_CAT_DEFAULT rtp_source_debug
|
|
|
|
#define RTP_MAX_PROBATION_LEN 32
|
|
|
|
/* signals and args */
|
|
enum
|
|
{
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
#define DEFAULT_SSRC 0
|
|
#define DEFAULT_IS_CSRC FALSE
|
|
#define DEFAULT_IS_VALIDATED FALSE
|
|
#define DEFAULT_IS_SENDER FALSE
|
|
#define DEFAULT_SDES NULL
|
|
#define DEFAULT_PROBATION RTP_DEFAULT_PROBATION
|
|
#define DEFAULT_MAX_DROPOUT_TIME 60000
|
|
#define DEFAULT_MAX_MISORDER_TIME 2000
|
|
#define DEFAULT_DISABLE_RTCP FALSE
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_SSRC,
|
|
PROP_IS_CSRC,
|
|
PROP_IS_VALIDATED,
|
|
PROP_IS_SENDER,
|
|
PROP_SDES,
|
|
PROP_STATS,
|
|
PROP_PROBATION,
|
|
PROP_MAX_DROPOUT_TIME,
|
|
PROP_MAX_MISORDER_TIME,
|
|
PROP_DISABLE_RTCP
|
|
};
|
|
|
|
/* GObject vmethods */
|
|
static void rtp_source_finalize (GObject * object);
|
|
static void rtp_source_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec);
|
|
static void rtp_source_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
|
|
/* static guint rtp_source_signals[LAST_SIGNAL] = { 0 }; */
|
|
|
|
G_DEFINE_TYPE (RTPSource, rtp_source, G_TYPE_OBJECT);
|
|
|
|
static void
|
|
rtp_source_class_init (RTPSourceClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
|
|
gobject_class->finalize = rtp_source_finalize;
|
|
|
|
gobject_class->set_property = rtp_source_set_property;
|
|
gobject_class->get_property = rtp_source_get_property;
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SSRC,
|
|
g_param_spec_uint ("ssrc", "SSRC",
|
|
"The SSRC of this source", 0, G_MAXUINT, DEFAULT_SSRC,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_IS_CSRC,
|
|
g_param_spec_boolean ("is-csrc", "Is CSRC",
|
|
"If this SSRC is acting as a contributing source",
|
|
DEFAULT_IS_CSRC, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_IS_VALIDATED,
|
|
g_param_spec_boolean ("is-validated", "Is Validated",
|
|
"If this SSRC is validated", DEFAULT_IS_VALIDATED,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_IS_SENDER,
|
|
g_param_spec_boolean ("is-sender", "Is Sender",
|
|
"If this SSRC is a sender", DEFAULT_IS_SENDER,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* RTPSource:sdes
|
|
*
|
|
* The current SDES items of the source. Returns a structure with name
|
|
* application/x-rtp-source-sdes and may contain the following fields:
|
|
*
|
|
* 'cname' G_TYPE_STRING : The canonical name in the form user@host
|
|
* 'name' G_TYPE_STRING : The user name
|
|
* 'email' G_TYPE_STRING : The user's electronic mail address
|
|
* 'phone' G_TYPE_STRING : The user's phone number
|
|
* 'location' G_TYPE_STRING : The geographic user location
|
|
* 'tool' G_TYPE_STRING : The name of application or tool
|
|
* 'note' G_TYPE_STRING : A notice about the source
|
|
*
|
|
* Other fields may be present and these represent private items in
|
|
* the SDES where the field name is the prefix.
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_SDES,
|
|
g_param_spec_boxed ("sdes", "SDES",
|
|
"The SDES information for this source",
|
|
GST_TYPE_STRUCTURE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* RTPSource:stats
|
|
*
|
|
* This property returns a GstStructure named application/x-rtp-source-stats with
|
|
* fields useful for statistics and diagnostics.
|
|
*
|
|
* Take note of each respective field's units:
|
|
*
|
|
* - NTP times are in the appropriate 32-bit or 64-bit fixed-point format
|
|
* starting from January 1, 1970 (except for timespans).
|
|
* - RTP times are in clock rate units (i.e. clock rate = 1 second)
|
|
* starting at a random offset.
|
|
* - For fields indicating packet loss, note that late packets are not considered lost,
|
|
* and duplicates are not taken into account. Hence, the loss may be negative
|
|
* if there are duplicates.
|
|
*
|
|
* The following fields are always present.
|
|
*
|
|
* * "ssrc" G_TYPE_UINT the SSRC of this source
|
|
* * "internal" G_TYPE_BOOLEAN this source is a source of the session
|
|
* * "validated" G_TYPE_BOOLEAN the source is validated
|
|
* * "received-bye" G_TYPE_BOOLEAN we received a BYE from this source
|
|
* * "is-csrc" G_TYPE_BOOLEAN this source was found as CSRC
|
|
* * "is-sender" G_TYPE_BOOLEAN this source is a sender
|
|
* * "seqnum-base" G_TYPE_INT first seqnum if known
|
|
* * "clock-rate" G_TYPE_INT the clock rate of the media
|
|
*
|
|
* The following fields are only present when known.
|
|
*
|
|
* * "rtp-from" G_TYPE_STRING where we received the last RTP packet from
|
|
* * "rtcp-from" G_TYPE_STRING where we received the last RTCP packet from
|
|
*
|
|
* The following fields make sense for internal sources and will only increase
|
|
* when "is-sender" is TRUE.
|
|
*
|
|
* * "octets-sent" G_TYPE_UINT64 number of payload bytes we sent
|
|
* * "packets-sent" G_TYPE_UINT64 number of packets we sent
|
|
*
|
|
* The following fields make sense for non-internal sources and will only
|
|
* increase when "is-sender" is TRUE.
|
|
*
|
|
* * "octets-received" G_TYPE_UINT64 total number of payload bytes received
|
|
* * "packets-received" G_TYPE_UINT64 total number of packets received
|
|
* * "bytes-received" G_TYPE_UINT64 total number of bytes received including lower level headers overhead
|
|
*
|
|
* Following fields are updated when "is-sender" is TRUE.
|
|
*
|
|
* * "bitrate" G_TYPE_UINT64 bitrate in bits per second
|
|
* * "jitter" G_TYPE_UINT estimated jitter (in clock rate units)
|
|
* * "packets-lost" G_TYPE_INT estimated amount of packets lost
|
|
*
|
|
* The last SR report this source sent. This only updates when "is-sender" is
|
|
* TRUE.
|
|
*
|
|
* * "have-sr" G_TYPE_BOOLEAN the source has sent SR
|
|
* * "sr-ntptime" G_TYPE_UINT64 NTP time of SR (in NTP Timestamp Format, 32.32 fixed point)
|
|
* * "sr-rtptime" G_TYPE_UINT RTP time of SR (in clock rate units)
|
|
* * "sr-octet-count" G_TYPE_UINT the number of bytes in the SR
|
|
* * "sr-packet-count" G_TYPE_UINT the number of packets in the SR
|
|
*
|
|
* The following fields are only present for non-internal sources and
|
|
* represent the content of the last RB packet that was sent to this source.
|
|
* These values are only updated when the source is sending.
|
|
*
|
|
* * "sent-rb" G_TYPE_BOOLEAN we have sent an RB
|
|
* * "sent-rb-fractionlost" G_TYPE_UINT calculated lost 8-bit fraction
|
|
* * "sent-rb-packetslost" G_TYPE_INT lost packets
|
|
* * "sent-rb-exthighestseq" G_TYPE_UINT last seen seqnum
|
|
* * "sent-rb-jitter" G_TYPE_UINT jitter (in clock rate units)
|
|
* * "sent-rb-lsr" G_TYPE_UINT last SR time (seconds in NTP Short Format, 16.16 fixed point)
|
|
* * "sent-rb-dlsr" G_TYPE_UINT delay since last SR (seconds in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
* The following fields are present for non-internal and internal sources, but
|
|
* the meaning is different.
|
|
* non-internal: it represents the last RB (RR in this case) that this source sent. This is
|
|
* only updated when the source is receiving data and sending RB blocks. It is
|
|
* deprecated and is present for backwards compatibility. Statistics about internal local sources
|
|
* should be retrieved from the source.
|
|
* internal: it represents the last RB (RR in this case) received with remote statistics about this source.
|
|
*
|
|
* * "have-rb" G_TYPE_BOOLEAN the source has sent RB
|
|
* * "rb-ssrc" G_TYPE_UINT The SSRC of the source the RB is about
|
|
* * "rb-fractionlost" G_TYPE_UINT lost 8-bit fraction
|
|
* * "rb-packetslost" G_TYPE_INT lost packets
|
|
* * "rb-exthighestseq" G_TYPE_UINT highest received seqnum
|
|
* * "rb-jitter" G_TYPE_UINT reception jitter (in clock rate units)
|
|
* * "rb-lsr" G_TYPE_UINT last SR time (seconds in NTP Short Format, 16.16 fixed point)
|
|
* * "rb-dlsr" G_TYPE_UINT delay since last SR (seconds in NTP Short Format, 16.16 fixed point)
|
|
* * "rb-round-trip" G_TYPE_UINT the round-trip time (seconds in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
* The round trip of this source is calculated from the last RB
|
|
* values and the reception time of the last RB packet.
|
|
*
|
|
* The following field is present only for internal sources and
|
|
* contains an array of the the most recent receiver reports from each peer. In unicast
|
|
* scenarios this will be a single entry that is identical to the data provided by the have-rb and rb-*
|
|
* fields, but in multicast there will be one entry in the array for each peer that is sending
|
|
* receiver reports.
|
|
*
|
|
* * "received-rr" GST_TYPE_LIST Array of GstStructure entries, one for each peer that sent an RR.
|
|
*
|
|
* Each entry in the array contains the following fields:
|
|
*
|
|
* * "rb-ssrc" G_TYPE_UINT The SSRC of this source
|
|
* * "rb-sender-ssrc" G_TYPE_UINT The SSRC of the sender of this RR
|
|
* * "rb-fractionlost" G_TYPE_UINT lost 8-bit fraction
|
|
* * "rb-packetslost" G_TYPE_INT lost packets
|
|
* * "rb-exthighestseq" G_TYPE_UINT highest received seqnum
|
|
* * "rb-jitter" G_TYPE_UINT reception jitter (in clock rate units)
|
|
* * "rb-lsr" G_TYPE_UINT last SR time (seconds in NTP Short Format, 16.16 fixed point)
|
|
* * "rb-dlsr" G_TYPE_UINT delay since last SR (seconds in NTP Short Format, 16.16 fixed point)
|
|
* * "rb-round-trip" G_TYPE_UINT the round-trip time (seconds in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
*
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_STATS,
|
|
g_param_spec_boxed ("stats", "Stats",
|
|
"The stats of this source", GST_TYPE_STRUCTURE,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_PROBATION,
|
|
g_param_spec_uint ("probation", "Number of probations",
|
|
"Consecutive packet sequence numbers to accept the source",
|
|
0, G_MAXUINT, DEFAULT_PROBATION,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_MAX_DROPOUT_TIME,
|
|
g_param_spec_uint ("max-dropout-time", "Max dropout time",
|
|
"The maximum time (milliseconds) of missing packets tolerated.",
|
|
0, G_MAXUINT, DEFAULT_MAX_DROPOUT_TIME,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_MAX_MISORDER_TIME,
|
|
g_param_spec_uint ("max-misorder-time", "Max misorder time",
|
|
"The maximum time (milliseconds) of misordered packets tolerated.",
|
|
0, G_MAXUINT, DEFAULT_MAX_MISORDER_TIME,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* RTPSource:disable-rtcp:
|
|
*
|
|
* Allow disabling the sending of RTCP packets for this source.
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_DISABLE_RTCP,
|
|
g_param_spec_boolean ("disable-rtcp", "Disable RTCP",
|
|
"Disable sending RTCP packets for this source",
|
|
DEFAULT_DISABLE_RTCP, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
GST_DEBUG_CATEGORY_INIT (rtp_source_debug, "rtpsource", 0, "RTP Source");
|
|
}
|
|
|
|
/**
|
|
* rtp_source_reset:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Reset the stats of @src.
|
|
*/
|
|
void
|
|
rtp_source_reset (RTPSource * src)
|
|
{
|
|
src->marked_bye = FALSE;
|
|
if (src->bye_reason)
|
|
g_free (src->bye_reason);
|
|
src->bye_reason = NULL;
|
|
src->sent_bye = FALSE;
|
|
g_hash_table_remove_all (src->reported_in_sr_of);
|
|
g_hash_table_remove_all (src->received_rr);
|
|
g_queue_foreach (src->retained_feedback, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_clear (src->retained_feedback);
|
|
src->last_rtptime = -1;
|
|
|
|
src->stats.cycles = -1;
|
|
src->stats.jitter = 0;
|
|
src->stats.transit = -1;
|
|
src->stats.curr_sr = 0;
|
|
src->stats.sr[0].is_valid = FALSE;
|
|
src->stats.curr_rr = 0;
|
|
src->stats.rr[0].is_valid = FALSE;
|
|
src->stats.prev_rtptime = GST_CLOCK_TIME_NONE;
|
|
src->stats.prev_rtcptime = GST_CLOCK_TIME_NONE;
|
|
src->stats.last_rtptime = GST_CLOCK_TIME_NONE;
|
|
src->stats.last_rtcptime = GST_CLOCK_TIME_NONE;
|
|
g_array_set_size (src->nacks, 0);
|
|
|
|
src->stats.sent_pli_count = 0;
|
|
src->stats.sent_fir_count = 0;
|
|
src->stats.sent_nack_count = 0;
|
|
src->stats.recv_nack_count = 0;
|
|
}
|
|
|
|
static void
|
|
rtp_source_init (RTPSource * src)
|
|
{
|
|
/* sources are initially on probation until we receive enough valid RTP
|
|
* packets or a valid RTCP packet */
|
|
src->validated = FALSE;
|
|
src->internal = FALSE;
|
|
src->probation = DEFAULT_PROBATION;
|
|
src->curr_probation = src->probation;
|
|
src->closing = FALSE;
|
|
src->max_dropout_time = DEFAULT_MAX_DROPOUT_TIME;
|
|
src->max_misorder_time = DEFAULT_MAX_MISORDER_TIME;
|
|
|
|
src->sdes = gst_structure_new_empty ("application/x-rtp-source-sdes");
|
|
|
|
src->payload = -1;
|
|
src->clock_rate = -1;
|
|
src->packets = g_queue_new ();
|
|
src->seqnum_offset = -1;
|
|
|
|
src->retained_feedback = g_queue_new ();
|
|
src->nacks = g_array_new (FALSE, FALSE, sizeof (guint16));
|
|
src->nack_deadlines = g_array_new (FALSE, FALSE, sizeof (GstClockTime));
|
|
|
|
src->reported_in_sr_of = g_hash_table_new (g_direct_hash, g_direct_equal);
|
|
src->received_rr =
|
|
g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL, g_free);
|
|
|
|
src->last_keyframe_request = GST_CLOCK_TIME_NONE;
|
|
|
|
rtp_source_reset (src);
|
|
|
|
src->pt_set = FALSE;
|
|
}
|
|
|
|
void
|
|
rtp_conflicting_address_free (RTPConflictingAddress * addr)
|
|
{
|
|
g_object_unref (addr->address);
|
|
g_free (addr);
|
|
}
|
|
|
|
static void
|
|
rtp_source_finalize (GObject * object)
|
|
{
|
|
RTPSource *src;
|
|
|
|
src = RTP_SOURCE_CAST (object);
|
|
|
|
g_queue_foreach (src->packets, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_free (src->packets);
|
|
|
|
gst_structure_free (src->sdes);
|
|
|
|
g_free (src->bye_reason);
|
|
|
|
gst_caps_replace (&src->caps, NULL);
|
|
|
|
g_list_free_full (src->conflicting_addresses,
|
|
(GDestroyNotify) rtp_conflicting_address_free);
|
|
g_queue_foreach (src->retained_feedback, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_free (src->retained_feedback);
|
|
|
|
g_array_free (src->nacks, TRUE);
|
|
g_array_free (src->nack_deadlines, TRUE);
|
|
|
|
if (src->rtp_from)
|
|
g_object_unref (src->rtp_from);
|
|
if (src->rtcp_from)
|
|
g_object_unref (src->rtcp_from);
|
|
|
|
g_hash_table_unref (src->reported_in_sr_of);
|
|
g_hash_table_unref (src->received_rr);
|
|
|
|
G_OBJECT_CLASS (rtp_source_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
rtp_source_get_rb_stats (RTPSource * src, GstStructure * s)
|
|
{
|
|
gboolean have_rb;
|
|
guint32 ssrc = 0;
|
|
guint8 fractionlost = 0;
|
|
gint32 packetslost = 0;
|
|
guint32 exthighestseq = 0;
|
|
guint32 jitter = 0;
|
|
guint32 lsr = 0;
|
|
guint32 dlsr = 0;
|
|
guint32 round_trip = 0;
|
|
|
|
have_rb = rtp_source_get_last_rb (src, &ssrc, &fractionlost,
|
|
&packetslost, &exthighestseq, &jitter, &lsr, &dlsr, &round_trip);
|
|
|
|
gst_structure_set (s,
|
|
"have-rb", G_TYPE_BOOLEAN, have_rb,
|
|
"rb-ssrc", G_TYPE_UINT, ssrc,
|
|
"rb-fractionlost", G_TYPE_UINT, (guint) fractionlost,
|
|
"rb-packetslost", G_TYPE_INT, (gint) packetslost,
|
|
"rb-exthighestseq", G_TYPE_UINT, (guint) exthighestseq,
|
|
"rb-jitter", G_TYPE_UINT, (guint) jitter,
|
|
"rb-lsr", G_TYPE_UINT, (guint) lsr,
|
|
"rb-dlsr", G_TYPE_UINT, (guint) dlsr,
|
|
"rb-round-trip", G_TYPE_UINT, (guint) round_trip, NULL);
|
|
}
|
|
|
|
static void
|
|
_create_rr_entry (gpointer * key, gpointer val, gpointer userdata)
|
|
{
|
|
GValue *res = userdata;
|
|
guint32 sender_ssrc = GPOINTER_TO_UINT (key);
|
|
RTPReceiverReport *rr = val;
|
|
GstStructure *s = gst_structure_new ("application/x-rtp-receiver-report",
|
|
"rb-ssrc", G_TYPE_UINT, (guint) rr->ssrc,
|
|
"rb-sender-ssrc", G_TYPE_UINT, (guint) sender_ssrc,
|
|
"rb-fractionlost", G_TYPE_UINT, (guint) rr->fractionlost,
|
|
"rb-packetslost", G_TYPE_INT, (gint) rr->packetslost,
|
|
"rb-exthighestseq", G_TYPE_UINT, (guint) rr->exthighestseq,
|
|
"rb-jitter", G_TYPE_UINT, (guint) rr->jitter,
|
|
"rb-lsr", G_TYPE_UINT, (guint) rr->lsr,
|
|
"rb-dlsr", G_TYPE_UINT, (guint) rr->dlsr,
|
|
"rb-round-trip", G_TYPE_UINT, (guint) rr->round_trip, NULL);
|
|
|
|
GValue v = G_VALUE_INIT;
|
|
g_value_init (&v, GST_TYPE_STRUCTURE);
|
|
gst_value_set_structure (&v, s);
|
|
gst_value_list_append_and_take_value (res, &v);
|
|
gst_structure_free (s);
|
|
}
|
|
|
|
static void
|
|
rtp_source_get_rr_stats (RTPSource * src, GstStructure * s)
|
|
{
|
|
GValue rr_list = G_VALUE_INIT;
|
|
g_value_init (&rr_list, GST_TYPE_LIST);
|
|
g_hash_table_foreach (src->received_rr, (GHFunc) _create_rr_entry, &rr_list);
|
|
|
|
gst_structure_take_value (s, "received-rr", &rr_list);
|
|
}
|
|
|
|
static GstStructure *
|
|
rtp_source_create_stats (RTPSource * src)
|
|
{
|
|
GstStructure *s;
|
|
gboolean is_sender = src->is_sender;
|
|
gboolean internal = src->internal;
|
|
gchar *address_str;
|
|
gboolean have_sr;
|
|
GstClockTime time = 0;
|
|
guint64 ntptime = 0;
|
|
guint32 rtptime = 0;
|
|
guint32 packet_count = 0;
|
|
guint32 octet_count = 0;
|
|
|
|
|
|
/* common data for all types of sources */
|
|
s = gst_structure_new ("application/x-rtp-source-stats",
|
|
"ssrc", G_TYPE_UINT, (guint) src->ssrc,
|
|
"internal", G_TYPE_BOOLEAN, internal,
|
|
"validated", G_TYPE_BOOLEAN, src->validated,
|
|
"received-bye", G_TYPE_BOOLEAN, src->marked_bye,
|
|
"is-csrc", G_TYPE_BOOLEAN, src->is_csrc,
|
|
"is-sender", G_TYPE_BOOLEAN, is_sender,
|
|
"seqnum-base", G_TYPE_INT, src->seqnum_offset,
|
|
"clock-rate", G_TYPE_INT, src->clock_rate, NULL);
|
|
|
|
/* add address and port */
|
|
if (src->rtp_from) {
|
|
address_str = __g_socket_address_to_string (src->rtp_from);
|
|
gst_structure_set (s, "rtp-from", G_TYPE_STRING, address_str, NULL);
|
|
g_free (address_str);
|
|
}
|
|
if (src->rtcp_from) {
|
|
address_str = __g_socket_address_to_string (src->rtcp_from);
|
|
gst_structure_set (s, "rtcp-from", G_TYPE_STRING, address_str, NULL);
|
|
g_free (address_str);
|
|
}
|
|
|
|
gst_structure_set (s,
|
|
"octets-sent", G_TYPE_UINT64, src->stats.octets_sent,
|
|
"packets-sent", G_TYPE_UINT64, src->stats.packets_sent,
|
|
"octets-received", G_TYPE_UINT64, src->stats.octets_received,
|
|
"packets-received", G_TYPE_UINT64, src->stats.packets_received,
|
|
"bytes-received", G_TYPE_UINT64, src->stats.bytes_received,
|
|
"bitrate", G_TYPE_UINT64, src->bitrate,
|
|
"packets-lost", G_TYPE_INT,
|
|
(gint) rtp_stats_get_packets_lost (&src->stats), "jitter", G_TYPE_UINT,
|
|
(guint) (src->stats.jitter >> 4),
|
|
"sent-pli-count", G_TYPE_UINT, src->stats.sent_pli_count,
|
|
"recv-pli-count", G_TYPE_UINT, src->stats.recv_pli_count,
|
|
"sent-fir-count", G_TYPE_UINT, src->stats.sent_fir_count,
|
|
"recv-fir-count", G_TYPE_UINT, src->stats.recv_fir_count,
|
|
"sent-nack-count", G_TYPE_UINT, src->stats.sent_nack_count,
|
|
"recv-nack-count", G_TYPE_UINT, src->stats.recv_nack_count,
|
|
"recv-packet-rate", G_TYPE_UINT,
|
|
gst_rtp_packet_rate_ctx_get (&src->packet_rate_ctx), NULL);
|
|
|
|
/* get the last SR. */
|
|
have_sr = rtp_source_get_last_sr (src, &time, &ntptime, &rtptime,
|
|
&packet_count, &octet_count);
|
|
gst_structure_set (s,
|
|
"have-sr", G_TYPE_BOOLEAN, have_sr,
|
|
"sr-ntptime", G_TYPE_UINT64, ntptime,
|
|
"sr-rtptime", G_TYPE_UINT, (guint) rtptime,
|
|
"sr-octet-count", G_TYPE_UINT, (guint) octet_count,
|
|
"sr-packet-count", G_TYPE_UINT, (guint) packet_count, NULL);
|
|
|
|
if (!internal) {
|
|
/* get the last RB we sent */
|
|
gst_structure_set (s,
|
|
"sent-rb", G_TYPE_BOOLEAN, src->last_rr.is_valid,
|
|
"sent-rb-fractionlost", G_TYPE_UINT, (guint) src->last_rr.fractionlost,
|
|
"sent-rb-packetslost", G_TYPE_INT, (gint) src->last_rr.packetslost,
|
|
"sent-rb-exthighestseq", G_TYPE_UINT,
|
|
(guint) src->last_rr.exthighestseq, "sent-rb-jitter", G_TYPE_UINT,
|
|
(guint) src->last_rr.jitter, "sent-rb-lsr", G_TYPE_UINT,
|
|
(guint) src->last_rr.lsr, "sent-rb-dlsr", G_TYPE_UINT,
|
|
(guint) src->last_rr.dlsr, NULL);
|
|
|
|
/* Deprecated, report block information is reported in each RTPSource */
|
|
/* get the last RB */
|
|
rtp_source_get_rb_stats (src, s);
|
|
} else {
|
|
rtp_source_get_rb_stats (src, s);
|
|
rtp_source_get_rr_stats (src, s);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_sdes_struct:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Get the SDES from @src. See the SDES property for more details.
|
|
*
|
|
* Returns: %GstStructure of type "application/x-rtp-source-sdes". The result is
|
|
* valid until the SDES items of @src are modified.
|
|
*/
|
|
const GstStructure *
|
|
rtp_source_get_sdes_struct (RTPSource * src)
|
|
{
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), NULL);
|
|
|
|
return src->sdes;
|
|
}
|
|
|
|
static gboolean
|
|
sdes_struct_compare_func (const GstIdStr * fieldname, const GValue * value,
|
|
gpointer user_data)
|
|
{
|
|
GstStructure *old;
|
|
|
|
old = GST_STRUCTURE (user_data);
|
|
|
|
if (!gst_structure_id_str_has_field (old, fieldname))
|
|
return FALSE;
|
|
|
|
g_assert (G_VALUE_HOLDS_STRING (value));
|
|
|
|
return strcmp (g_value_get_string (value), gst_structure_get_string (old,
|
|
gst_id_str_as_str (fieldname))) == 0;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_set_sdes_struct:
|
|
* @src: an #RTPSource
|
|
* @sdes: the SDES structure
|
|
*
|
|
* Store the @sdes in @src. @sdes must be a structure of type
|
|
* "application/x-rtp-source-sdes", see the SDES property for more details.
|
|
*
|
|
* This function takes ownership of @sdes.
|
|
*
|
|
* Returns: %FALSE if the SDES was unchanged.
|
|
*/
|
|
gboolean
|
|
rtp_source_set_sdes_struct (RTPSource * src, GstStructure * sdes)
|
|
{
|
|
gboolean changed;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
g_return_val_if_fail (strcmp (gst_structure_get_name (sdes),
|
|
"application/x-rtp-source-sdes") == 0, FALSE);
|
|
|
|
changed =
|
|
!gst_structure_foreach_id_str (sdes, sdes_struct_compare_func, src->sdes);
|
|
|
|
if (changed) {
|
|
gst_structure_free (src->sdes);
|
|
src->sdes = sdes;
|
|
} else {
|
|
gst_structure_free (sdes);
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
static void
|
|
rtp_source_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
RTPSource *src;
|
|
|
|
src = RTP_SOURCE (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_SSRC:
|
|
src->ssrc = g_value_get_uint (value);
|
|
break;
|
|
case PROP_PROBATION:
|
|
src->probation = g_value_get_uint (value);
|
|
break;
|
|
case PROP_MAX_DROPOUT_TIME:
|
|
src->max_dropout_time = g_value_get_uint (value);
|
|
break;
|
|
case PROP_MAX_MISORDER_TIME:
|
|
src->max_misorder_time = g_value_get_uint (value);
|
|
break;
|
|
case PROP_DISABLE_RTCP:
|
|
src->disable_rtcp = g_value_get_boolean (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
rtp_source_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
RTPSource *src;
|
|
|
|
src = RTP_SOURCE (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_SSRC:
|
|
g_value_set_uint (value, rtp_source_get_ssrc (src));
|
|
break;
|
|
case PROP_IS_CSRC:
|
|
g_value_set_boolean (value, rtp_source_is_as_csrc (src));
|
|
break;
|
|
case PROP_IS_VALIDATED:
|
|
g_value_set_boolean (value, rtp_source_is_validated (src));
|
|
break;
|
|
case PROP_IS_SENDER:
|
|
g_value_set_boolean (value, rtp_source_is_sender (src));
|
|
break;
|
|
case PROP_SDES:
|
|
g_value_set_boxed (value, rtp_source_get_sdes_struct (src));
|
|
break;
|
|
case PROP_STATS:
|
|
g_value_take_boxed (value, rtp_source_create_stats (src));
|
|
break;
|
|
case PROP_PROBATION:
|
|
g_value_set_uint (value, src->probation);
|
|
break;
|
|
case PROP_MAX_DROPOUT_TIME:
|
|
g_value_set_uint (value, src->max_dropout_time);
|
|
break;
|
|
case PROP_MAX_MISORDER_TIME:
|
|
g_value_set_uint (value, src->max_misorder_time);
|
|
break;
|
|
case PROP_DISABLE_RTCP:
|
|
g_value_set_boolean (value, src->disable_rtcp);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* rtp_source_new:
|
|
* @ssrc: an SSRC
|
|
*
|
|
* Create a #RTPSource with @ssrc.
|
|
*
|
|
* Returns: a new #RTPSource. Use g_object_unref() after usage.
|
|
*/
|
|
RTPSource *
|
|
rtp_source_new (guint32 ssrc)
|
|
{
|
|
RTPSource *src;
|
|
|
|
src = g_object_new (RTP_TYPE_SOURCE, NULL);
|
|
src->ssrc = ssrc;
|
|
|
|
return src;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_set_callbacks:
|
|
* @src: an #RTPSource
|
|
* @cb: callback functions
|
|
* @user_data: user data
|
|
*
|
|
* Set the callbacks for the source.
|
|
*/
|
|
void
|
|
rtp_source_set_callbacks (RTPSource * src, RTPSourceCallbacks * cb,
|
|
gpointer user_data)
|
|
{
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
src->callbacks.push_rtp = cb->push_rtp;
|
|
src->callbacks.caps = cb->caps;
|
|
src->user_data = user_data;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_ssrc:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Get the SSRC of @source.
|
|
*
|
|
* Returns: the SSRC of src.
|
|
*/
|
|
guint32
|
|
rtp_source_get_ssrc (RTPSource * src)
|
|
{
|
|
guint32 result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), 0);
|
|
|
|
result = src->ssrc;
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_set_as_csrc:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Configure @src as a CSRC, this will also validate @src.
|
|
*/
|
|
void
|
|
rtp_source_set_as_csrc (RTPSource * src)
|
|
{
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
src->validated = TRUE;
|
|
src->is_csrc = TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_is_as_csrc:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Check if @src is a contributing source.
|
|
*
|
|
* Returns: %TRUE if @src is acting as a contributing source.
|
|
*/
|
|
gboolean
|
|
rtp_source_is_as_csrc (RTPSource * src)
|
|
{
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
result = src->is_csrc;
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_is_active:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Check if @src is an active source. A source is active if it has been
|
|
* validated and has not yet received a BYE packet
|
|
*
|
|
* Returns: %TRUE if @src is an qactive source.
|
|
*/
|
|
gboolean
|
|
rtp_source_is_active (RTPSource * src)
|
|
{
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
result = RTP_SOURCE_IS_ACTIVE (src);
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_is_validated:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Check if @src is a validated source.
|
|
*
|
|
* Returns: %TRUE if @src is a validated source.
|
|
*/
|
|
gboolean
|
|
rtp_source_is_validated (RTPSource * src)
|
|
{
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
result = src->validated;
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_is_sender:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Check if @src is a sending source.
|
|
*
|
|
* Returns: %TRUE if @src is a sending source.
|
|
*/
|
|
gboolean
|
|
rtp_source_is_sender (RTPSource * src)
|
|
{
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
result = RTP_SOURCE_IS_SENDER (src);
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_is_marked_bye:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Check if @src is marked as leaving the session with a BYE packet.
|
|
*
|
|
* Returns: %TRUE if @src has been marked BYE.
|
|
*/
|
|
gboolean
|
|
rtp_source_is_marked_bye (RTPSource * src)
|
|
{
|
|
gboolean result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
result = RTP_SOURCE_IS_MARKED_BYE (src);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
/**
|
|
* rtp_source_get_bye_reason:
|
|
* @src: an #RTPSource
|
|
*
|
|
* Get the BYE reason for @src. Check if the source is marked as leaving the
|
|
* session with a BYE message first with rtp_source_is_marked_bye().
|
|
*
|
|
* Returns: The BYE reason or NULL when no reason was given or the source was
|
|
* not marked BYE yet. g_free() after usage.
|
|
*/
|
|
gchar *
|
|
rtp_source_get_bye_reason (RTPSource * src)
|
|
{
|
|
gchar *result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), NULL);
|
|
|
|
result = g_strdup (src->bye_reason);
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_update_caps:
|
|
* @src: an #RTPSource
|
|
* @caps: a #GstCaps
|
|
*
|
|
* Parse @caps and store all relevant information in @source.
|
|
*/
|
|
void
|
|
rtp_source_update_send_caps (RTPSource * src, GstCaps * caps)
|
|
{
|
|
GstStructure *s;
|
|
guint val;
|
|
gint ival;
|
|
guint ssrc, rtx_ssrc = -1;
|
|
gboolean rtx;
|
|
|
|
/* nothing changed, return */
|
|
if (caps == NULL || src->caps == caps)
|
|
return;
|
|
|
|
s = gst_caps_get_structure (caps, 0);
|
|
|
|
if (!gst_structure_get_uint (s, "ssrc", &ssrc))
|
|
return;
|
|
gst_structure_get_uint (s, "rtx-ssrc", &rtx_ssrc);
|
|
|
|
if (src->ssrc != ssrc && src->ssrc != rtx_ssrc) {
|
|
GST_WARNING ("got ssrc %u/%u that doesn't match with this source's ssrc %u",
|
|
ssrc, rtx_ssrc, src->ssrc);
|
|
return;
|
|
}
|
|
|
|
rtx = (rtx_ssrc == src->ssrc);
|
|
|
|
if (gst_structure_get_int (s, rtx ? "rtx-payload" : "payload", &ival))
|
|
src->payload = ival;
|
|
else
|
|
src->payload = -1;
|
|
|
|
GST_DEBUG ("got %spayload %d", rtx ? "rtx " : "", src->payload);
|
|
|
|
if (gst_structure_get_int (s, "clock-rate", &ival))
|
|
src->clock_rate = ival;
|
|
else
|
|
src->clock_rate = -1;
|
|
|
|
GST_DEBUG ("got clock-rate %d", src->clock_rate);
|
|
|
|
if (gst_structure_get_uint (s, rtx ? "rtx-seqnum-offset" : "seqnum-offset",
|
|
&val))
|
|
src->seqnum_offset = val;
|
|
else
|
|
src->seqnum_offset = -1;
|
|
|
|
GST_DEBUG ("got %sseqnum-offset %" G_GINT32_FORMAT, rtx ? "rtx " : "",
|
|
src->seqnum_offset);
|
|
|
|
gst_caps_replace (&src->caps, caps);
|
|
|
|
if (rtx) {
|
|
src->media_ssrc = ssrc;
|
|
} else {
|
|
src->media_ssrc = -1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* rtp_source_set_rtp_from:
|
|
* @src: an #RTPSource
|
|
* @address: the RTP address to set
|
|
*
|
|
* Set that @src is receiving RTP packets from @address. This is used for
|
|
* collistion checking.
|
|
*/
|
|
void
|
|
rtp_source_set_rtp_from (RTPSource * src, GSocketAddress * address)
|
|
{
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
if (src->rtp_from)
|
|
g_object_unref (src->rtp_from);
|
|
src->rtp_from = G_SOCKET_ADDRESS (g_object_ref (address));
|
|
}
|
|
|
|
/**
|
|
* rtp_source_set_rtcp_from:
|
|
* @src: an #RTPSource
|
|
* @address: the RTCP address to set
|
|
*
|
|
* Set that @src is receiving RTCP packets from @address. This is used for
|
|
* collistion checking.
|
|
*/
|
|
void
|
|
rtp_source_set_rtcp_from (RTPSource * src, GSocketAddress * address)
|
|
{
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
if (src->rtcp_from)
|
|
g_object_unref (src->rtcp_from);
|
|
src->rtcp_from = G_SOCKET_ADDRESS (g_object_ref (address));
|
|
}
|
|
|
|
static GstFlowReturn
|
|
push_packet (RTPSource * src, GstBuffer * buffer)
|
|
{
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
|
|
/* push queued packets first if any */
|
|
while (!g_queue_is_empty (src->packets)) {
|
|
GstBuffer *buffer = GST_BUFFER_CAST (g_queue_pop_head (src->packets));
|
|
|
|
GST_LOG ("pushing queued packet");
|
|
if (src->callbacks.push_rtp)
|
|
src->callbacks.push_rtp (src, buffer, src->user_data);
|
|
else
|
|
gst_buffer_unref (buffer);
|
|
}
|
|
GST_LOG ("pushing new packet");
|
|
/* push packet */
|
|
if (src->callbacks.push_rtp)
|
|
ret = src->callbacks.push_rtp (src, buffer, src->user_data);
|
|
else
|
|
gst_buffer_unref (buffer);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
fetch_caps_for_payload (RTPSource * src, guint8 payload)
|
|
{
|
|
if (src->payload == -1) {
|
|
/* first payload received, nothing was in the caps, lock on to this payload */
|
|
src->payload = payload;
|
|
GST_DEBUG ("first payload %d", payload);
|
|
} else if (payload != src->payload) {
|
|
/* we have a different payload than before, reset the clock-rate */
|
|
GST_DEBUG ("new payload %d", payload);
|
|
src->payload = payload;
|
|
src->clock_rate = -1;
|
|
src->stats.transit = -1;
|
|
}
|
|
|
|
if (src->clock_rate == -1 || !src->caps) {
|
|
GstCaps *caps = NULL;
|
|
|
|
if (src->callbacks.caps) {
|
|
caps = src->callbacks.caps (src, payload, src->user_data);
|
|
}
|
|
|
|
GST_DEBUG ("got caps %" GST_PTR_FORMAT, caps);
|
|
|
|
if (caps) {
|
|
const GstStructure *s;
|
|
gint clock_rate = -1;
|
|
const gchar *encoding_name;
|
|
|
|
s = gst_caps_get_structure (caps, 0);
|
|
|
|
if (gst_structure_get_int (s, "clock-rate", &clock_rate)) {
|
|
src->clock_rate = clock_rate;
|
|
gst_rtp_packet_rate_ctx_reset (&src->packet_rate_ctx, clock_rate);
|
|
} else {
|
|
GST_DEBUG ("No clock-rate in caps!");
|
|
}
|
|
|
|
encoding_name = gst_structure_get_string (s, "encoding-name");
|
|
/* Disable probation for RTX sources as packets will arrive very
|
|
* irregularly and waiting for a second packet usually exceeds the
|
|
* deadline of the retransmission */
|
|
if (g_strcmp0 (encoding_name, "rtx") == 0) {
|
|
src->probation = src->curr_probation = 0;
|
|
}
|
|
}
|
|
|
|
gst_caps_replace (&src->caps, caps);
|
|
gst_clear_caps (&caps);
|
|
}
|
|
}
|
|
|
|
/* Jitter is the variation in the delay of received packets in a flow. It is
|
|
* measured by comparing the interval when RTP packets were sent to the interval
|
|
* at which they were received. For instance, if packet #1 and packet #2 leave
|
|
* 50 milliseconds apart and arrive 60 milliseconds apart, then the jitter is 10
|
|
* milliseconds. */
|
|
static void
|
|
calculate_jitter (RTPSource * src, RTPPacketInfo * pinfo)
|
|
{
|
|
GstClockTime running_time;
|
|
guint32 rtparrival, transit, rtptime;
|
|
gint32 diff;
|
|
|
|
/* get arrival time */
|
|
if ((running_time = pinfo->running_time) == GST_CLOCK_TIME_NONE)
|
|
goto no_time;
|
|
|
|
GST_LOG ("SSRC %08x got payload %d", src->ssrc, pinfo->pt);
|
|
|
|
/* check if clock-rate is valid */
|
|
if (src->clock_rate == -1)
|
|
goto no_clock_rate;
|
|
|
|
rtptime = pinfo->rtptime;
|
|
|
|
/* convert arrival time to RTP timestamp units, truncate to 32 bits, we don't
|
|
* care about the absolute value, just the difference. */
|
|
rtparrival =
|
|
gst_util_uint64_scale_int (running_time, src->clock_rate, GST_SECOND);
|
|
|
|
/* transit time is difference with RTP timestamp */
|
|
transit = rtparrival - rtptime;
|
|
|
|
/* get ABS diff with previous transit time */
|
|
if (src->stats.transit != -1) {
|
|
if (transit > src->stats.transit)
|
|
diff = transit - src->stats.transit;
|
|
else
|
|
diff = src->stats.transit - transit;
|
|
} else
|
|
diff = 0;
|
|
|
|
src->stats.transit = transit;
|
|
|
|
/* update jitter, the value we store is scaled up so we can keep precision. */
|
|
src->stats.jitter += diff - ((src->stats.jitter + 8) >> 4);
|
|
|
|
src->stats.prev_rtptime = src->stats.last_rtptime;
|
|
src->stats.last_rtptime = rtparrival;
|
|
|
|
GST_LOG ("rtparrival %u, rtptime %u, clock-rate %d, diff %d, jitter: %f",
|
|
rtparrival, rtptime, src->clock_rate, diff, (src->stats.jitter) / 16.0);
|
|
|
|
return;
|
|
|
|
/* ERRORS */
|
|
no_time:
|
|
{
|
|
GST_WARNING ("cannot get current running_time");
|
|
return;
|
|
}
|
|
no_clock_rate:
|
|
{
|
|
GST_WARNING ("cannot get clock-rate for pt %d", pinfo->pt);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
update_queued_stats (GstBuffer * buffer, RTPSource * src)
|
|
{
|
|
GstRTPBuffer rtp = { NULL };
|
|
guint payload_len;
|
|
guint64 bytes;
|
|
|
|
/* no need to check the return value, a queued packet is a valid RTP one */
|
|
gst_rtp_buffer_map (buffer, GST_MAP_READ, &rtp);
|
|
payload_len = gst_rtp_buffer_get_payload_len (&rtp);
|
|
|
|
bytes = gst_buffer_get_size (buffer) + UDP_IP_HEADER_OVERHEAD;
|
|
|
|
src->stats.octets_received += payload_len;
|
|
src->stats.bytes_received += bytes;
|
|
src->stats.packets_received++;
|
|
/* for the bitrate estimation consider all lower level headers */
|
|
src->bytes_received += bytes;
|
|
|
|
gst_rtp_buffer_unmap (&rtp);
|
|
}
|
|
|
|
static void
|
|
init_seq (RTPSource * src, guint16 seq)
|
|
{
|
|
src->stats.base_seq = seq;
|
|
src->stats.max_seq = seq;
|
|
src->stats.bad_seq = RTP_SEQ_MOD + 1; /* so seq == bad_seq is false */
|
|
src->stats.cycles = 0;
|
|
src->stats.packets_received = 0;
|
|
src->stats.octets_received = 0;
|
|
src->stats.bytes_received = 0;
|
|
src->stats.prev_received = 0;
|
|
src->stats.prev_expected = 0;
|
|
src->stats.recv_pli_count = 0;
|
|
src->stats.recv_fir_count = 0;
|
|
|
|
/* if there are queued packets, consider them too in the stats */
|
|
g_queue_foreach (src->packets, (GFunc) update_queued_stats, src);
|
|
|
|
GST_DEBUG ("base_seq %d", seq);
|
|
}
|
|
|
|
#define BITRATE_INTERVAL (2 * GST_SECOND)
|
|
|
|
static void
|
|
do_bitrate_estimation (RTPSource * src, GstClockTime running_time,
|
|
guint64 * bytes_handled)
|
|
{
|
|
guint64 elapsed;
|
|
|
|
if (src->prev_rtime) {
|
|
elapsed = running_time - src->prev_rtime;
|
|
|
|
if (elapsed > BITRATE_INTERVAL) {
|
|
guint64 rate;
|
|
|
|
rate = gst_util_uint64_scale (*bytes_handled, 8 * GST_SECOND, elapsed);
|
|
|
|
GST_LOG ("Elapsed %" G_GUINT64_FORMAT ", bytes %" G_GUINT64_FORMAT
|
|
", rate %" G_GUINT64_FORMAT, elapsed, *bytes_handled, rate);
|
|
|
|
if (src->bitrate == 0)
|
|
src->bitrate = rate;
|
|
else
|
|
src->bitrate = ((src->bitrate * 3) + rate) / 4;
|
|
|
|
src->prev_rtime = running_time;
|
|
*bytes_handled = 0;
|
|
}
|
|
} else {
|
|
GST_LOG ("Reset bitrate measurement");
|
|
src->prev_rtime = running_time;
|
|
src->bitrate = 0;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
update_receiver_stats (RTPSource * src, RTPPacketInfo * pinfo,
|
|
gboolean is_receive)
|
|
{
|
|
guint16 seqnr, expected;
|
|
RTPSourceStats *stats;
|
|
gint16 delta;
|
|
gint32 packet_rate, max_dropout, max_misorder;
|
|
|
|
stats = &src->stats;
|
|
|
|
seqnr = pinfo->seqnum;
|
|
|
|
packet_rate =
|
|
gst_rtp_packet_rate_ctx_update (&src->packet_rate_ctx, pinfo->seqnum,
|
|
pinfo->rtptime);
|
|
max_dropout =
|
|
gst_rtp_packet_rate_ctx_get_max_dropout (&src->packet_rate_ctx,
|
|
src->max_dropout_time);
|
|
max_misorder =
|
|
gst_rtp_packet_rate_ctx_get_max_misorder (&src->packet_rate_ctx,
|
|
src->max_misorder_time);
|
|
GST_TRACE ("SSRC %08x, packet_rate: %d, max_dropout: %d, max_misorder: %d",
|
|
src->ssrc, packet_rate, max_dropout, max_misorder);
|
|
|
|
if (stats->cycles == -1) {
|
|
GST_DEBUG ("received first packet");
|
|
/* first time we heard of this source */
|
|
init_seq (src, seqnr);
|
|
src->stats.max_seq = seqnr - 1;
|
|
src->curr_probation = src->probation;
|
|
}
|
|
|
|
if (is_receive) {
|
|
expected = src->stats.max_seq + 1;
|
|
delta = gst_rtp_buffer_compare_seqnum (expected, seqnr);
|
|
|
|
/* if we are still on probation, check seqnum */
|
|
if (src->curr_probation) {
|
|
/* when in probation, we require consecutive seqnums */
|
|
if (delta == 0) {
|
|
/* expected packet */
|
|
GST_DEBUG ("probation: seqnr %d == expected %d", seqnr, expected);
|
|
src->curr_probation--;
|
|
if (seqnr < stats->max_seq) {
|
|
/* sequence number wrapped - count another 64K cycle. */
|
|
stats->cycles += RTP_SEQ_MOD;
|
|
}
|
|
src->stats.max_seq = seqnr;
|
|
|
|
if (src->curr_probation == 0) {
|
|
GST_DEBUG ("probation done!");
|
|
init_seq (src, seqnr);
|
|
} else {
|
|
GstBuffer *q;
|
|
|
|
GST_DEBUG ("probation %d: queue packet", src->curr_probation);
|
|
/* when still in probation, keep packets in a list. */
|
|
g_queue_push_tail (src->packets, pinfo->data);
|
|
pinfo->data = NULL;
|
|
/* remove packets from queue if there are too many */
|
|
while (g_queue_get_length (src->packets) > RTP_MAX_PROBATION_LEN) {
|
|
q = g_queue_pop_head (src->packets);
|
|
gst_buffer_unref (q);
|
|
}
|
|
goto done;
|
|
}
|
|
} else {
|
|
/* unexpected seqnum in probation
|
|
*
|
|
* There is no need to clean the queue at this point because the
|
|
* invalid packets in the queue are not going to be pushed as we are
|
|
* still in probation, and some cleanup will be performed at future
|
|
* probation attempts anyway if there are too many old packets in the
|
|
* queue.
|
|
*/
|
|
goto probation_seqnum;
|
|
}
|
|
} else if (delta >= 0 && delta < max_dropout) {
|
|
/* Clear bad packets */
|
|
stats->bad_seq = RTP_SEQ_MOD + 1; /* so seq == bad_seq is false */
|
|
g_queue_foreach (src->packets, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_clear (src->packets);
|
|
|
|
/* in order, with permissible gap */
|
|
if (seqnr < stats->max_seq) {
|
|
/* sequence number wrapped - count another 64K cycle. */
|
|
stats->cycles += RTP_SEQ_MOD;
|
|
}
|
|
stats->max_seq = seqnr;
|
|
} else if (delta < -max_misorder || delta >= max_dropout) {
|
|
/* the sequence number made a very large jump */
|
|
if (seqnr == stats->bad_seq && src->packets->head) {
|
|
/* two sequential packets -- assume that the other side
|
|
* restarted without telling us so just re-sync
|
|
* (i.e., pretend this was the first packet). */
|
|
init_seq (src, seqnr);
|
|
} else {
|
|
/* unacceptable jump */
|
|
stats->bad_seq = (seqnr + 1) & (RTP_SEQ_MOD - 1);
|
|
g_queue_foreach (src->packets, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_clear (src->packets);
|
|
g_queue_push_tail (src->packets, pinfo->data);
|
|
pinfo->data = NULL;
|
|
goto bad_sequence;
|
|
}
|
|
} else { /* delta < 0 && delta >= -max_misorder */
|
|
/* Clear bad packets */
|
|
stats->bad_seq = RTP_SEQ_MOD + 1; /* so seq == bad_seq is false */
|
|
g_queue_foreach (src->packets, (GFunc) gst_buffer_unref, NULL);
|
|
g_queue_clear (src->packets);
|
|
|
|
/* duplicate or reordered packet, will be filtered by jitterbuffer. */
|
|
GST_INFO ("duplicate or reordered packet (seqnr %u, expected %u)",
|
|
seqnr, expected);
|
|
}
|
|
} else {
|
|
/* Sender stats - update the outbound sequence number */
|
|
expected = src->stats.max_seq + 1;
|
|
delta = gst_rtp_buffer_compare_seqnum (expected, seqnr);
|
|
/* No probation for local senders, just check for lost / dropouts */
|
|
if (delta >= 0 && delta < max_dropout) {
|
|
stats->bad_seq = RTP_SEQ_MOD + 1; /* so seq == bad_seq is false */
|
|
/* in order, with permissible gap */
|
|
if (seqnr < stats->max_seq) {
|
|
/* sequence number wrapped - count another 64K cycle. */
|
|
stats->cycles += RTP_SEQ_MOD;
|
|
}
|
|
stats->max_seq = seqnr;
|
|
} else if (delta < -max_misorder || delta >= max_dropout) {
|
|
/* the sequence number made a very large jump */
|
|
if (seqnr != stats->bad_seq) {
|
|
/* unacceptable jump */
|
|
stats->bad_seq = (seqnr + 1) & (RTP_SEQ_MOD - 1);
|
|
}
|
|
} else { /* delta < 0 && delta >= -max_misorder */
|
|
stats->bad_seq = RTP_SEQ_MOD + 1; /* so seq == bad_seq is false */
|
|
}
|
|
}
|
|
|
|
src->stats.octets_received += pinfo->payload_len;
|
|
src->stats.bytes_received += pinfo->bytes;
|
|
src->stats.packets_received += pinfo->packets;
|
|
/* for the bitrate estimation consider all lower level headers */
|
|
src->bytes_received += pinfo->bytes;
|
|
|
|
GST_LOG ("seq %u, PC: %" G_GUINT64_FORMAT ", OC: %" G_GUINT64_FORMAT,
|
|
seqnr, src->stats.packets_received, src->stats.octets_received);
|
|
|
|
return TRUE;
|
|
|
|
/* ERRORS */
|
|
done:
|
|
{
|
|
return FALSE;
|
|
}
|
|
bad_sequence:
|
|
{
|
|
GST_WARNING
|
|
("unacceptable seqnum received (seqnr %u, delta %d, packet_rate: %d, max_dropout: %d, max_misorder: %d)",
|
|
seqnr, delta, packet_rate, max_dropout, max_misorder);
|
|
return FALSE;
|
|
}
|
|
probation_seqnum:
|
|
{
|
|
GST_WARNING ("probation: seqnr %d != expected %d "
|
|
"(SSRC %u curr_probation %i probation %i)", seqnr, expected, src->ssrc,
|
|
src->curr_probation, src->probation);
|
|
src->curr_probation = src->probation;
|
|
src->stats.max_seq = seqnr;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* rtp_source_process_rtp:
|
|
* @src: an #RTPSource
|
|
* @pinfo: an #RTPPacketInfo
|
|
*
|
|
* Let @src handle the incoming RTP packet described in @pinfo.
|
|
*
|
|
* Returns: a #GstFlowReturn.
|
|
*/
|
|
GstFlowReturn
|
|
rtp_source_process_rtp (RTPSource * src, RTPPacketInfo * pinfo)
|
|
{
|
|
GstFlowReturn result;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), GST_FLOW_ERROR);
|
|
g_return_val_if_fail (pinfo != NULL, GST_FLOW_ERROR);
|
|
|
|
fetch_caps_for_payload (src, pinfo->pt);
|
|
|
|
if (!update_receiver_stats (src, pinfo, TRUE))
|
|
return GST_FLOW_OK;
|
|
|
|
/* the source that sent the packet must be a sender */
|
|
src->is_sender = TRUE;
|
|
src->validated = TRUE;
|
|
|
|
do_bitrate_estimation (src, pinfo->running_time, &src->bytes_received);
|
|
|
|
/* calculate jitter for the stats */
|
|
calculate_jitter (src, pinfo);
|
|
|
|
/* we're ready to push the RTP packet now */
|
|
result = push_packet (src, pinfo->data);
|
|
pinfo->data = NULL;
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_mark_bye:
|
|
* @src: an #RTPSource
|
|
* @reason: the reason for leaving
|
|
*
|
|
* Mark @src in the BYE state. This can happen when the source wants to
|
|
* leave the session or when a BYE packets has been received.
|
|
*
|
|
* This will make the source inactive.
|
|
*/
|
|
void
|
|
rtp_source_mark_bye (RTPSource * src, const gchar * reason)
|
|
{
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
GST_DEBUG ("marking SSRC %08x as BYE, reason: %s", src->ssrc,
|
|
GST_STR_NULL (reason));
|
|
|
|
/* copy the reason and mark as bye */
|
|
g_free (src->bye_reason);
|
|
src->bye_reason = g_strdup (reason);
|
|
src->marked_bye = TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_send_rtp:
|
|
* @src: an #RTPSource
|
|
* @pinfo: an #RTPPacketInfo
|
|
*
|
|
* Send data (an RTP buffer or buffer list from @pinfo) originating from @src.
|
|
* This will make @src a sender. This function takes ownership of the data and
|
|
* modifies the SSRC in the RTP packet to that of @src when needed.
|
|
*
|
|
* Returns: a #GstFlowReturn.
|
|
*/
|
|
GstFlowReturn
|
|
rtp_source_send_rtp (RTPSource * src, RTPPacketInfo * pinfo)
|
|
{
|
|
GstFlowReturn result;
|
|
GstClockTime running_time;
|
|
guint32 rtptime;
|
|
guint64 ext_rtptime;
|
|
guint64 rt_diff, rtp_diff;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), GST_FLOW_ERROR);
|
|
|
|
/* we are a sender now */
|
|
src->is_sender = TRUE;
|
|
|
|
/* we are also a receiver of our packets */
|
|
if (!update_receiver_stats (src, pinfo, FALSE))
|
|
return GST_FLOW_OK;
|
|
|
|
if (src->pt_set && src->pt != pinfo->pt) {
|
|
GST_WARNING ("Changing pt from %u to %u for SSRC %u", src->pt, pinfo->pt,
|
|
src->ssrc);
|
|
}
|
|
|
|
src->pt = pinfo->pt;
|
|
src->pt_set = TRUE;
|
|
|
|
/* update stats for the SR */
|
|
src->stats.packets_sent += pinfo->packets;
|
|
src->stats.octets_sent += pinfo->payload_len;
|
|
src->bytes_sent += pinfo->bytes;
|
|
|
|
running_time = pinfo->running_time;
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (running_time))
|
|
do_bitrate_estimation (src, running_time, &src->bytes_sent);
|
|
|
|
rtptime = pinfo->rtptime;
|
|
|
|
ext_rtptime = src->last_rtptime;
|
|
ext_rtptime = gst_rtp_buffer_ext_timestamp (&ext_rtptime, rtptime);
|
|
|
|
GST_LOG ("SSRC %08x, RTP %" G_GUINT64_FORMAT ", running_time %"
|
|
GST_TIME_FORMAT, src->ssrc, ext_rtptime, GST_TIME_ARGS (running_time));
|
|
|
|
if (ext_rtptime > src->last_rtptime) {
|
|
rtp_diff = ext_rtptime - src->last_rtptime;
|
|
rt_diff =
|
|
GST_CLOCK_TIME_IS_VALID (running_time) ? running_time -
|
|
src->last_rtime : GST_CLOCK_TIME_NONE;
|
|
|
|
/* calc the diff so we can detect drift at the sender. This can also be used
|
|
* to guestimate the clock rate if the NTP time is locked to the RTP
|
|
* timestamps (as is the case when the capture device is providing the clock). */
|
|
GST_LOG ("SSRC %08x, diff RTP %" G_GUINT64_FORMAT ", diff running_time %"
|
|
GST_TIME_FORMAT, src->ssrc, rtp_diff, GST_TIME_ARGS (rt_diff));
|
|
}
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (running_time)) {
|
|
/* we keep track of the last received RTP timestamp and the corresponding
|
|
* buffer running_time so that we can use this info when constructing SR reports */
|
|
src->last_rtime = running_time;
|
|
src->last_rtptime = ext_rtptime;
|
|
}
|
|
|
|
/* push packet */
|
|
if (!src->callbacks.push_rtp)
|
|
goto no_callback;
|
|
|
|
GST_LOG ("pushing RTP %s %" G_GUINT64_FORMAT,
|
|
pinfo->is_list ? "list" : "packet", src->stats.packets_sent);
|
|
|
|
result = src->callbacks.push_rtp (src, pinfo->data, src->user_data);
|
|
pinfo->data = NULL;
|
|
|
|
return result;
|
|
|
|
/* ERRORS */
|
|
no_callback:
|
|
{
|
|
GST_WARNING ("no callback installed, dropping packet");
|
|
return GST_FLOW_OK;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* rtp_source_process_sr:
|
|
* @src: an #RTPSource
|
|
* @time: time of packet arrival
|
|
* @ntptime: the NTP time (in NTP Timestamp Format, 32.32 fixed point)
|
|
* @rtptime: the RTP time (in clock rate units)
|
|
* @packet_count: the packet count
|
|
* @octet_count: the octet count
|
|
*
|
|
* Update the sender report in @src.
|
|
*/
|
|
void
|
|
rtp_source_process_sr (RTPSource * src, GstClockTime time, guint64 ntptime,
|
|
guint32 rtptime, guint32 packet_count, guint32 octet_count)
|
|
{
|
|
RTPSenderReport *curr;
|
|
gint curridx;
|
|
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
GST_DEBUG ("got SR packet: SSRC %08x, NTP %08x:%08x, RTP %" G_GUINT32_FORMAT
|
|
", PC %" G_GUINT32_FORMAT ", OC %" G_GUINT32_FORMAT, src->ssrc,
|
|
(guint32) (ntptime >> 32), (guint32) (ntptime & 0xffffffff), rtptime,
|
|
packet_count, octet_count);
|
|
|
|
curridx = src->stats.curr_sr ^ 1;
|
|
curr = &src->stats.sr[curridx];
|
|
|
|
/* this is a sender now */
|
|
src->is_sender = TRUE;
|
|
|
|
/* update current */
|
|
curr->is_valid = TRUE;
|
|
curr->ntptime = ntptime;
|
|
curr->rtptime = rtptime;
|
|
curr->packet_count = packet_count;
|
|
curr->octet_count = octet_count;
|
|
curr->time = time;
|
|
|
|
/* make current */
|
|
src->stats.curr_sr = curridx;
|
|
|
|
src->stats.prev_rtcptime = src->stats.last_rtcptime;
|
|
src->stats.last_rtcptime = time;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_process_rb:
|
|
* @src: an #RTPSource
|
|
* @ssrc: SSRC of the local source for which this RB was sent
|
|
* @sender_ssrc: SSRC from which this RB was received
|
|
* @ntpnstime: the current time in nanoseconds since 1970
|
|
* @fractionlost: fraction lost since last SR/RR
|
|
* @packetslost: the cumulative number of packets lost
|
|
* @exthighestseq: the extended last sequence number received
|
|
* @jitter: the interarrival jitter (in clock rate units)
|
|
* @lsr: the time of the last SR packet on this source
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
* @dlsr: the delay since the last SR packet
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
* Update the report block in @src.
|
|
*/
|
|
void
|
|
rtp_source_process_rb (RTPSource * src, guint32 ssrc, guint32 sender_ssrc,
|
|
guint64 ntpnstime, guint8 fractionlost, gint32 packetslost,
|
|
guint32 exthighestseq, guint32 jitter, guint32 lsr, guint32 dlsr)
|
|
{
|
|
RTPReceiverReport *curr;
|
|
gint curridx;
|
|
guint32 ntp, A;
|
|
guint64 f_ntp;
|
|
|
|
g_return_if_fail (RTP_IS_SOURCE (src));
|
|
|
|
GST_DEBUG ("got RB packet: SSRC %08x, FL %2x, PL %d, HS %" G_GUINT32_FORMAT
|
|
", jitter %" G_GUINT32_FORMAT ", LSR %04x:%04x, DLSR %04x:%04x",
|
|
src->ssrc, fractionlost, packetslost, exthighestseq, jitter, lsr >> 16,
|
|
lsr & 0xffff, dlsr >> 16, dlsr & 0xffff);
|
|
|
|
curridx = src->stats.curr_rr ^ 1;
|
|
curr = &src->stats.rr[curridx];
|
|
|
|
/* update current */
|
|
curr->is_valid = TRUE;
|
|
curr->ssrc = ssrc;
|
|
curr->fractionlost = fractionlost;
|
|
curr->packetslost = packetslost;
|
|
curr->exthighestseq = exthighestseq;
|
|
curr->jitter = jitter;
|
|
curr->lsr = lsr;
|
|
curr->dlsr = dlsr;
|
|
|
|
/* convert the NTP time in nanoseconds to 32.32 fixed point */
|
|
f_ntp = gst_util_uint64_scale (ntpnstime, (1LL << 32), GST_SECOND);
|
|
/* calculate round trip, round the time up */
|
|
ntp = ((f_ntp + 0xffff) >> 16) & 0xffffffff;
|
|
|
|
A = dlsr + lsr;
|
|
if (A > 0 && ntp > A)
|
|
A = ntp - A;
|
|
else
|
|
A = 0;
|
|
curr->round_trip = A;
|
|
|
|
GST_DEBUG ("NTP %04x:%04x, round trip %04x:%04x", ntp >> 16, ntp & 0xffff,
|
|
A >> 16, A & 0xffff);
|
|
|
|
/* make current */
|
|
src->stats.curr_rr = curridx;
|
|
|
|
if (src->internal) {
|
|
/* Make a copy to store in the received rr's hash table, but only for
|
|
* internal sources to track which remote sources sent reports for this source */
|
|
RTPReceiverReport *copy = g_memdup2 (curr, sizeof (RTPReceiverReport));
|
|
g_hash_table_replace (src->received_rr, GUINT_TO_POINTER (sender_ssrc),
|
|
copy);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_new_sr:
|
|
* @src: an #RTPSource
|
|
* @ntpnstime: the current time in nanoseconds since 1970
|
|
* @running_time: the current running_time of the pipeline
|
|
* @ntptime: the NTP time (in NTP Timestamp Format, 32.32 fixed point)
|
|
* @rtptime: the RTP time corresponding to @ntptime (in clock rate units)
|
|
* @packet_count: the packet count
|
|
* @octet_count: the octet count
|
|
*
|
|
* Get new values to put into a new SR report from this source.
|
|
*
|
|
* @running_time and @ntpnstime are captured at the same time and represent the
|
|
* running time of the pipeline clock and the absolute current system time in
|
|
* nanoseconds respectively. Together with the last running_time and RTP timestamp
|
|
* we have observed in the source, we can generate @ntptime and @rtptime for an SR
|
|
* packet. @ntptime is basically the fixed point representation of @ntpnstime
|
|
* and @rtptime the associated RTP timestamp.
|
|
*
|
|
* Returns: %TRUE on success.
|
|
*/
|
|
gboolean
|
|
rtp_source_get_new_sr (RTPSource * src, guint64 ntpnstime,
|
|
GstClockTime running_time, guint64 * ntptime, guint32 * rtptime,
|
|
guint32 * packet_count, guint32 * octet_count)
|
|
{
|
|
guint64 t_rtp;
|
|
guint64 t_current_ntp;
|
|
GstClockTimeDiff diff;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
/* We last saw a buffer with last_rtptime at last_rtime. Given a running_time
|
|
* and an NTP time, we can scale the RTP timestamps so that they match the
|
|
* given NTP time. for scaling, we assume that the slope of the rtptime vs
|
|
* running_time vs ntptime curve is close to 1, which is certainly
|
|
* sufficient for the frequency at which we report SR and the rate we send
|
|
* out RTP packets. */
|
|
t_rtp = src->last_rtptime;
|
|
|
|
GST_DEBUG ("last_rtime %" GST_TIME_FORMAT ", last_rtptime %"
|
|
G_GUINT64_FORMAT, GST_TIME_ARGS (src->last_rtime), t_rtp);
|
|
|
|
if (src->clock_rate == -1 && src->pt_set) {
|
|
GST_INFO ("no clock-rate, getting for pt %u and SSRC %u", src->pt,
|
|
src->ssrc);
|
|
fetch_caps_for_payload (src, src->pt);
|
|
}
|
|
|
|
if (src->clock_rate != -1) {
|
|
/* if no running time has been set yet we wait until we get one */
|
|
if (src->last_rtime == -1) {
|
|
GST_WARNING ("running time not set, can not create SR for SSRC %u",
|
|
src->ssrc);
|
|
return FALSE;
|
|
}
|
|
|
|
/* get the diff between the clock running_time and the buffer running_time.
|
|
* This is the elapsed time, as measured against the pipeline clock, between
|
|
* when the rtp timestamp was observed and the current running_time.
|
|
*
|
|
* We need to apply this diff to the RTP timestamp to get the RTP timestamp
|
|
* for the given ntpnstime. */
|
|
diff = GST_CLOCK_DIFF (src->last_rtime, running_time);
|
|
GST_DEBUG ("running_time %" GST_TIME_FORMAT ", diff %" GST_STIME_FORMAT,
|
|
GST_TIME_ARGS (running_time), GST_STIME_ARGS (diff));
|
|
|
|
/* now translate the diff to RTP time, handle positive and negative cases.
|
|
* If there is no diff, we already set rtptime correctly above. */
|
|
if (diff > 0) {
|
|
t_rtp += gst_util_uint64_scale_int (diff, src->clock_rate, GST_SECOND);
|
|
} else {
|
|
diff = -diff;
|
|
t_rtp -= gst_util_uint64_scale_int (diff, src->clock_rate, GST_SECOND);
|
|
}
|
|
} else {
|
|
GST_WARNING ("no clock-rate, cannot interpolate rtp time for SSRC %u",
|
|
src->ssrc);
|
|
}
|
|
|
|
/* convert the NTP time in nanoseconds to 32.32 fixed point */
|
|
t_current_ntp = gst_util_uint64_scale (ntpnstime, (1LL << 32), GST_SECOND);
|
|
|
|
GST_DEBUG ("NTP %08x:%08x, RTP %" G_GUINT32_FORMAT,
|
|
(guint32) (t_current_ntp >> 32), (guint32) (t_current_ntp & 0xffffffff),
|
|
(guint32) t_rtp);
|
|
|
|
if (ntptime)
|
|
*ntptime = t_current_ntp;
|
|
if (rtptime)
|
|
*rtptime = t_rtp;
|
|
if (packet_count)
|
|
*packet_count = src->stats.packets_sent;
|
|
if (octet_count)
|
|
*octet_count = src->stats.octets_sent;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_new_rb:
|
|
* @src: an #RTPSource
|
|
* @time: the current time of the system clock
|
|
* @fractionlost: fraction lost since last SR/RR
|
|
* @packetslost: the cumulative number of packets lost
|
|
* @exthighestseq: the extended last sequence number received
|
|
* @jitter: the interarrival jitter (in clock rate units)
|
|
* @lsr: the time of the last SR packet on this source
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
* @dlsr: the delay since the last SR packet
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
* Get new values to put into a new report block from this source.
|
|
*
|
|
* Returns: %TRUE on success.
|
|
*/
|
|
gboolean
|
|
rtp_source_get_new_rb (RTPSource * src, GstClockTime time,
|
|
guint8 * fractionlost, gint32 * packetslost, guint32 * exthighestseq,
|
|
guint32 * jitter, guint32 * lsr, guint32 * dlsr)
|
|
{
|
|
RTPSourceStats *stats;
|
|
guint64 extended_max, expected;
|
|
guint64 expected_interval, received_interval, ntptime;
|
|
gint64 lost, lost_interval;
|
|
guint32 fraction, LSR, DLSR;
|
|
GstClockTime sr_time;
|
|
|
|
stats = &src->stats;
|
|
|
|
extended_max = stats->cycles + stats->max_seq;
|
|
expected = extended_max - stats->base_seq + 1;
|
|
|
|
GST_DEBUG ("ext_max %" G_GUINT64_FORMAT ", expected %" G_GUINT64_FORMAT
|
|
", received %" G_GUINT64_FORMAT ", base_seq %" G_GUINT32_FORMAT,
|
|
extended_max, expected, stats->packets_received, stats->base_seq);
|
|
|
|
lost = expected - stats->packets_received;
|
|
lost = CLAMP (lost, -0x800000, 0x7fffff);
|
|
|
|
expected_interval = expected - stats->prev_expected;
|
|
stats->prev_expected = expected;
|
|
received_interval = stats->packets_received - stats->prev_received;
|
|
stats->prev_received = stats->packets_received;
|
|
|
|
lost_interval = expected_interval - received_interval;
|
|
|
|
if (expected_interval == 0 || lost_interval <= 0)
|
|
fraction = 0;
|
|
else
|
|
fraction = (lost_interval << 8) / expected_interval;
|
|
|
|
GST_DEBUG ("add RR for SSRC %08x", src->ssrc);
|
|
/* we scaled the jitter up for additional precision */
|
|
GST_DEBUG ("fraction %" G_GUINT32_FORMAT ", lost %" G_GINT64_FORMAT
|
|
", extseq %" G_GUINT64_FORMAT ", jitter %d", fraction, lost,
|
|
extended_max, stats->jitter >> 4);
|
|
|
|
if (rtp_source_get_last_sr (src, &sr_time, &ntptime, NULL, NULL, NULL)) {
|
|
GstClockTime diff;
|
|
|
|
/* LSR is middle 32 bits of the last ntptime */
|
|
LSR = (ntptime >> 16) & 0xffffffff;
|
|
diff = time - sr_time;
|
|
GST_DEBUG ("last SR time diff %" GST_TIME_FORMAT, GST_TIME_ARGS (diff));
|
|
/* DLSR, delay since last SR is expressed in 1/65536 second units */
|
|
DLSR = gst_util_uint64_scale_int (diff, 65536, GST_SECOND);
|
|
} else {
|
|
/* No valid SR received, LSR/DLSR are set to 0 then */
|
|
GST_DEBUG ("no valid SR received");
|
|
LSR = 0;
|
|
DLSR = 0;
|
|
}
|
|
GST_DEBUG ("LSR %04x:%04x, DLSR %04x:%04x", LSR >> 16, LSR & 0xffff,
|
|
DLSR >> 16, DLSR & 0xffff);
|
|
|
|
if (fractionlost)
|
|
*fractionlost = fraction;
|
|
if (packetslost)
|
|
*packetslost = lost;
|
|
if (exthighestseq)
|
|
*exthighestseq = extended_max;
|
|
if (jitter)
|
|
*jitter = stats->jitter >> 4;
|
|
if (lsr)
|
|
*lsr = LSR;
|
|
if (dlsr)
|
|
*dlsr = DLSR;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_last_sr:
|
|
* @src: an #RTPSource
|
|
* @time: time of packet arrival
|
|
* @ntptime: the NTP time (in NTP Timestamp Format, 32.32 fixed point)
|
|
* @rtptime: the RTP time (in clock rate units)
|
|
* @packet_count: the packet count
|
|
* @octet_count: the octet count
|
|
*
|
|
* Get the values of the last sender report as set with rtp_source_process_sr().
|
|
*
|
|
* Returns: %TRUE if there was a valid SR report.
|
|
*/
|
|
gboolean
|
|
rtp_source_get_last_sr (RTPSource * src, GstClockTime * time, guint64 * ntptime,
|
|
guint32 * rtptime, guint32 * packet_count, guint32 * octet_count)
|
|
{
|
|
RTPSenderReport *curr;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
curr = &src->stats.sr[src->stats.curr_sr];
|
|
if (!curr->is_valid)
|
|
return FALSE;
|
|
|
|
if (ntptime)
|
|
*ntptime = curr->ntptime;
|
|
if (rtptime)
|
|
*rtptime = curr->rtptime;
|
|
if (packet_count)
|
|
*packet_count = curr->packet_count;
|
|
if (octet_count)
|
|
*octet_count = curr->octet_count;
|
|
if (time)
|
|
*time = curr->time;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_last_rb:
|
|
* @src: an #RTPSource
|
|
* @ssrc: SSRC of the local source for this this RB was sent
|
|
* @fractionlost: fraction lost since last SR/RR
|
|
* @packetslost: the cumulative number of packets lost
|
|
* @exthighestseq: the extended last sequence number received
|
|
* @jitter: the interarrival jitter (in clock rate units)
|
|
* @lsr: the time of the last SR packet on this source
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
* @dlsr: the delay since the last SR packet
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
* @round_trip: the round-trip time
|
|
* (in NTP Short Format, 16.16 fixed point)
|
|
*
|
|
* Get the values of the last RB report set with rtp_source_process_rb().
|
|
*
|
|
* Returns: %TRUE if there was a valid SB report.
|
|
*/
|
|
gboolean
|
|
rtp_source_get_last_rb (RTPSource * src, guint32 * ssrc,
|
|
guint8 * fractionlost, gint32 * packetslost, guint32 * exthighestseq,
|
|
guint32 * jitter, guint32 * lsr, guint32 * dlsr, guint32 * round_trip)
|
|
{
|
|
RTPReceiverReport *curr;
|
|
|
|
g_return_val_if_fail (RTP_IS_SOURCE (src), FALSE);
|
|
|
|
curr = &src->stats.rr[src->stats.curr_rr];
|
|
if (!curr->is_valid)
|
|
return FALSE;
|
|
|
|
if (ssrc)
|
|
*ssrc = curr->ssrc;
|
|
if (fractionlost)
|
|
*fractionlost = curr->fractionlost;
|
|
if (packetslost)
|
|
*packetslost = curr->packetslost;
|
|
if (exthighestseq)
|
|
*exthighestseq = curr->exthighestseq;
|
|
if (jitter)
|
|
*jitter = curr->jitter;
|
|
if (lsr)
|
|
*lsr = curr->lsr;
|
|
if (dlsr)
|
|
*dlsr = curr->dlsr;
|
|
if (round_trip)
|
|
*round_trip = curr->round_trip;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
find_conflicting_address (GList * conflicting_addresses,
|
|
GSocketAddress * address, GstClockTime time)
|
|
{
|
|
GList *item;
|
|
|
|
for (item = conflicting_addresses; item; item = g_list_next (item)) {
|
|
RTPConflictingAddress *known_conflict = item->data;
|
|
|
|
if (__g_socket_address_equal (address, known_conflict->address)) {
|
|
known_conflict->time = time;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
GList *
|
|
add_conflicting_address (GList * conflicting_addresses,
|
|
GSocketAddress * address, GstClockTime time)
|
|
{
|
|
RTPConflictingAddress *new_conflict;
|
|
|
|
new_conflict = g_new (RTPConflictingAddress, 1);
|
|
|
|
new_conflict->address = G_SOCKET_ADDRESS (g_object_ref (address));
|
|
new_conflict->time = time;
|
|
|
|
return g_list_prepend (conflicting_addresses, new_conflict);
|
|
}
|
|
|
|
GList *
|
|
timeout_conflicting_addresses (GList * conflicting_addresses,
|
|
GstClockTime current_time)
|
|
{
|
|
GList *item;
|
|
/* "a relatively long time" -- RFC 3550 section 8.2 */
|
|
const GstClockTime collision_timeout =
|
|
RTP_STATS_MIN_INTERVAL * GST_SECOND * 10;
|
|
|
|
item = g_list_first (conflicting_addresses);
|
|
while (item) {
|
|
RTPConflictingAddress *known_conflict = item->data;
|
|
GList *next_item = g_list_next (item);
|
|
|
|
if (known_conflict->time + collision_timeout < current_time) {
|
|
gchar *buf;
|
|
|
|
conflicting_addresses = g_list_delete_link (conflicting_addresses, item);
|
|
buf = __g_socket_address_to_string (known_conflict->address);
|
|
GST_DEBUG ("collision %p timed out: %s", known_conflict, buf);
|
|
g_free (buf);
|
|
rtp_conflicting_address_free (known_conflict);
|
|
}
|
|
item = next_item;
|
|
}
|
|
|
|
return conflicting_addresses;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_find_conflicting_address:
|
|
* @src: The source the packet came in
|
|
* @address: address to check for
|
|
* @time: The time when the packet that is possibly in conflict arrived
|
|
*
|
|
* Checks if an address which has a conflict is already known. If it is
|
|
* a known conflict, remember the time
|
|
*
|
|
* Returns: TRUE if it was a known conflict, FALSE otherwise
|
|
*/
|
|
gboolean
|
|
rtp_source_find_conflicting_address (RTPSource * src, GSocketAddress * address,
|
|
GstClockTime time)
|
|
{
|
|
return find_conflicting_address (src->conflicting_addresses, address, time);
|
|
}
|
|
|
|
/**
|
|
* rtp_source_add_conflicting_address:
|
|
* @src: The source the packet came in
|
|
* @address: address to remember
|
|
* @time: The time when the packet that is in conflict arrived
|
|
*
|
|
* Adds a new conflict address
|
|
*/
|
|
void
|
|
rtp_source_add_conflicting_address (RTPSource * src,
|
|
GSocketAddress * address, GstClockTime time)
|
|
{
|
|
src->conflicting_addresses =
|
|
add_conflicting_address (src->conflicting_addresses, address, time);
|
|
}
|
|
|
|
/**
|
|
* rtp_source_timeout:
|
|
* @src: The #RTPSource
|
|
* @current_time: The current time
|
|
* @feedback_retention_window: The running time before which retained feedback
|
|
* packets have to be discarded
|
|
*
|
|
* This is processed on each RTCP interval. It times out old collisions.
|
|
* It also times out old retained feedback packets
|
|
*/
|
|
void
|
|
rtp_source_timeout (RTPSource * src, GstClockTime current_time,
|
|
GstClockTime running_time, GstClockTime feedback_retention_window)
|
|
{
|
|
GstRTCPPacket *pkt;
|
|
GstClockTime max_pts_window;
|
|
guint pruned = 0;
|
|
|
|
src->conflicting_addresses =
|
|
timeout_conflicting_addresses (src->conflicting_addresses, current_time);
|
|
|
|
if (feedback_retention_window == GST_CLOCK_TIME_NONE ||
|
|
running_time < feedback_retention_window) {
|
|
return;
|
|
}
|
|
|
|
max_pts_window = running_time - feedback_retention_window;
|
|
|
|
/* Time out AVPF packets that are older than the desired length */
|
|
while ((pkt = g_queue_peek_head (src->retained_feedback)) &&
|
|
GST_BUFFER_PTS (pkt) < max_pts_window) {
|
|
gst_buffer_unref (g_queue_pop_head (src->retained_feedback));
|
|
pruned++;
|
|
}
|
|
|
|
GST_LOG_OBJECT (src,
|
|
"%u RTCP packets pruned with PTS less than %" GST_TIME_FORMAT
|
|
", queue len: %u", pruned, GST_TIME_ARGS (max_pts_window),
|
|
g_queue_get_length (src->retained_feedback));
|
|
}
|
|
|
|
static gint
|
|
compare_buffers (gconstpointer a, gconstpointer b, gpointer user_data)
|
|
{
|
|
const GstBuffer *bufa = a;
|
|
const GstBuffer *bufb = b;
|
|
|
|
g_return_val_if_fail (GST_BUFFER_PTS (bufa) != GST_CLOCK_TIME_NONE, -1);
|
|
g_return_val_if_fail (GST_BUFFER_PTS (bufb) != GST_CLOCK_TIME_NONE, 1);
|
|
|
|
if (GST_BUFFER_PTS (bufa) < GST_BUFFER_PTS (bufb)) {
|
|
return -1;
|
|
} else if (GST_BUFFER_PTS (bufa) > GST_BUFFER_PTS (bufb)) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
rtp_source_retain_rtcp_packet (RTPSource * src, GstRTCPPacket * packet,
|
|
GstClockTime running_time)
|
|
{
|
|
GstBuffer *buffer;
|
|
|
|
g_return_if_fail (running_time != GST_CLOCK_TIME_NONE);
|
|
|
|
buffer = gst_buffer_copy_region (packet->rtcp->buffer, GST_BUFFER_COPY_MEMORY,
|
|
packet->offset, (gst_rtcp_packet_get_length (packet) + 1) * 4);
|
|
|
|
GST_BUFFER_PTS (buffer) = running_time;
|
|
|
|
g_queue_insert_sorted (src->retained_feedback, buffer, compare_buffers, NULL);
|
|
|
|
GST_LOG_OBJECT (src, "RTCP packet retained with PTS: %" GST_TIME_FORMAT,
|
|
GST_TIME_ARGS (running_time));
|
|
}
|
|
|
|
gboolean
|
|
rtp_source_has_retained (RTPSource * src, GCompareFunc func, gconstpointer data)
|
|
{
|
|
if (g_queue_find_custom (src->retained_feedback, data, func))
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_register_nack:
|
|
* @src: The #RTPSource
|
|
* @seqnum: a seqnum
|
|
* @deadline: the deadline before which RTX is still possible
|
|
*
|
|
* Register that @seqnum has not been received from @src.
|
|
*/
|
|
void
|
|
rtp_source_register_nack (RTPSource * src, guint16 seqnum,
|
|
GstClockTime deadline)
|
|
{
|
|
gint i;
|
|
guint len;
|
|
gint diff = -1;
|
|
guint16 tseq;
|
|
|
|
len = src->nacks->len;
|
|
for (i = len - 1; i >= 0; i--) {
|
|
tseq = g_array_index (src->nacks, guint16, i);
|
|
diff = gst_rtp_buffer_compare_seqnum (tseq, seqnum);
|
|
|
|
GST_TRACE ("[%u] %u %u diff %i len %u", i, tseq, seqnum, diff, len);
|
|
|
|
if (diff >= 0)
|
|
break;
|
|
}
|
|
|
|
if (diff == 0) {
|
|
GST_DEBUG ("update NACK #%u deadline to %" GST_TIME_FORMAT, seqnum,
|
|
GST_TIME_ARGS (deadline));
|
|
g_array_index (src->nack_deadlines, GstClockTime, i) = deadline;
|
|
} else if (i == len - 1) {
|
|
GST_DEBUG ("append NACK #%u with deadline %" GST_TIME_FORMAT, seqnum,
|
|
GST_TIME_ARGS (deadline));
|
|
g_array_append_val (src->nacks, seqnum);
|
|
g_array_append_val (src->nack_deadlines, deadline);
|
|
} else {
|
|
GST_DEBUG ("insert NACK #%u with deadline %" GST_TIME_FORMAT, seqnum,
|
|
GST_TIME_ARGS (deadline));
|
|
g_array_insert_val (src->nacks, i + 1, seqnum);
|
|
g_array_insert_val (src->nack_deadlines, i + 1, deadline);
|
|
}
|
|
|
|
src->send_nack = TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_nacks:
|
|
* @src: The #RTPSource
|
|
* @n_nacks: result number of nacks
|
|
*
|
|
* Get the registered NACKS since the last rtp_source_clear_nacks().
|
|
*
|
|
* Returns: an array of @n_nacks seqnum values.
|
|
*/
|
|
guint16 *
|
|
rtp_source_get_nacks (RTPSource * src, guint * n_nacks)
|
|
{
|
|
if (n_nacks)
|
|
*n_nacks = src->nacks->len;
|
|
|
|
return (guint16 *) src->nacks->data;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_get_nack_deadlines:
|
|
* @src: The #RTPSource
|
|
* @n_nacks: result number of nacks
|
|
*
|
|
* Get the registered NACKS deadlines.
|
|
*
|
|
* Returns: an array of @n_nacks deadline values.
|
|
*/
|
|
GstClockTime *
|
|
rtp_source_get_nack_deadlines (RTPSource * src, guint * n_nacks)
|
|
{
|
|
if (n_nacks)
|
|
*n_nacks = src->nack_deadlines->len;
|
|
|
|
return (GstClockTime *) src->nack_deadlines->data;
|
|
}
|
|
|
|
/**
|
|
* rtp_source_clear_nacks:
|
|
* @src: The #RTPSource
|
|
* @n_nacks: number of nacks
|
|
*
|
|
* Remove @n_nacks oldest NACKS form array.
|
|
*/
|
|
void
|
|
rtp_source_clear_nacks (RTPSource * src, guint n_nacks)
|
|
{
|
|
g_return_if_fail (n_nacks <= src->nacks->len);
|
|
|
|
if (src->nacks->len == n_nacks) {
|
|
g_array_set_size (src->nacks, 0);
|
|
g_array_set_size (src->nack_deadlines, 0);
|
|
src->send_nack = FALSE;
|
|
} else {
|
|
g_array_remove_range (src->nacks, 0, n_nacks);
|
|
g_array_remove_range (src->nack_deadlines, 0, n_nacks);
|
|
}
|
|
}
|