mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 13:08:49 +00:00
b0afaffc5d
This allows downstream of a payloader to know the RTP header's marker flag without first having to map the buffer and parse the RTP header. Especially inside RTP header extension implementations this can be useful to decide which packet corresponds to e.g. the last packet of a video frame. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1776>
449 lines
13 KiB
C
449 lines
13 KiB
C
/* GStreamer plugin for forward error correction
|
|
* Copyright (C) 2017 Pexip
|
|
*
|
|
* 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
|
|
*
|
|
* Author: Mikhail Fludkov <misha@pexip.com>
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include "rtpulpfeccommon.h"
|
|
|
|
#define MIN_RTP_HEADER_LEN 12
|
|
|
|
typedef struct
|
|
{
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
unsigned int csrc_count:4; /* CSRC count */
|
|
unsigned int extension:1; /* header extension flag */
|
|
unsigned int padding:1; /* padding flag */
|
|
unsigned int version:2; /* protocol version */
|
|
unsigned int payload_type:7; /* payload type */
|
|
unsigned int marker:1; /* marker bit */
|
|
#elif G_BYTE_ORDER == G_BIG_ENDIAN
|
|
unsigned int version:2; /* protocol version */
|
|
unsigned int padding:1; /* padding flag */
|
|
unsigned int extension:1; /* header extension flag */
|
|
unsigned int csrc_count:4; /* CSRC count */
|
|
unsigned int marker:1; /* marker bit */
|
|
unsigned int payload_type:7; /* payload type */
|
|
#else
|
|
#error "G_BYTE_ORDER should be big or little endian."
|
|
#endif
|
|
unsigned int seq:16; /* sequence number */
|
|
unsigned int timestamp:32; /* timestamp */
|
|
unsigned int ssrc:32; /* synchronization source */
|
|
guint8 csrclist[4]; /* optional CSRC list, 32 bits each */
|
|
} RtpHeader;
|
|
|
|
static gsize
|
|
fec_level_hdr_get_size (gboolean l_bit)
|
|
{
|
|
return sizeof (RtpUlpFecLevelHeader) - (l_bit ? 0 : 4);
|
|
}
|
|
|
|
static guint64
|
|
fec_level_hdr_get_mask (RtpUlpFecLevelHeader const *fec_lvl_hdr, gboolean l_bit)
|
|
{
|
|
return ((guint64) g_ntohs (fec_lvl_hdr->mask) << 32) |
|
|
(l_bit ? g_ntohl (fec_lvl_hdr->mask_continued) : 0);
|
|
}
|
|
|
|
static void
|
|
fec_level_hdr_set_mask (RtpUlpFecLevelHeader * fec_lvl_hdr, gboolean l_bit,
|
|
guint64 mask)
|
|
{
|
|
fec_lvl_hdr->mask = g_htons (mask >> 32);
|
|
if (l_bit)
|
|
fec_lvl_hdr->mask_continued = g_htonl (mask);
|
|
}
|
|
|
|
static guint16
|
|
fec_level_hdr_get_protection_len (RtpUlpFecLevelHeader * fec_lvl_hdr)
|
|
{
|
|
return g_ntohs (fec_lvl_hdr->protection_len);
|
|
}
|
|
|
|
static void
|
|
fec_level_hdr_set_protection_len (RtpUlpFecLevelHeader * fec_lvl_hdr,
|
|
guint16 len)
|
|
{
|
|
fec_lvl_hdr->protection_len = g_htons (len);
|
|
}
|
|
|
|
static RtpUlpFecLevelHeader *
|
|
fec_hdr_get_level_hdr (RtpUlpFecHeader const *fec_hdr)
|
|
{
|
|
return (RtpUlpFecLevelHeader *) (fec_hdr + 1);
|
|
}
|
|
|
|
static guint64
|
|
fec_hdr_get_mask (RtpUlpFecHeader const *fec_hdr)
|
|
{
|
|
return fec_level_hdr_get_mask (fec_hdr_get_level_hdr (fec_hdr), fec_hdr->L);
|
|
}
|
|
|
|
static guint16
|
|
fec_hdr_get_seq_base (RtpUlpFecHeader const *fec_hdr, gboolean is_ulpfec,
|
|
guint16 fec_seq)
|
|
{
|
|
guint16 seq = g_ntohs (fec_hdr->seq);
|
|
if (is_ulpfec)
|
|
return seq;
|
|
return fec_seq - seq;
|
|
}
|
|
|
|
static guint16
|
|
fec_hdr_get_packets_len_recovery (RtpUlpFecHeader const *fec_hdr)
|
|
{
|
|
return g_htons (fec_hdr->len);
|
|
}
|
|
|
|
static guint32
|
|
fec_hdr_get_timestamp_recovery (RtpUlpFecHeader const *fec_hdr)
|
|
{
|
|
return g_ntohl (fec_hdr->timestamp);
|
|
}
|
|
|
|
static void
|
|
_xor_mem (guint8 * restrict dst, const guint8 * restrict src, gsize length)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 0; i < (length / sizeof (guint64)); ++i) {
|
|
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
|
|
GST_WRITE_UINT64_LE (dst,
|
|
GST_READ_UINT64_LE (dst) ^ GST_READ_UINT64_LE (src));
|
|
#else
|
|
GST_WRITE_UINT64_BE (dst,
|
|
GST_READ_UINT64_BE (dst) ^ GST_READ_UINT64_BE (src));
|
|
#endif
|
|
dst += sizeof (guint64);
|
|
src += sizeof (guint64);
|
|
}
|
|
for (i = 0; i < (length % sizeof (guint64)); ++i)
|
|
dst[i] ^= src[i];
|
|
}
|
|
|
|
guint16
|
|
rtp_ulpfec_hdr_get_protection_len (RtpUlpFecHeader const *fec_hdr)
|
|
{
|
|
return fec_level_hdr_get_protection_len (fec_hdr_get_level_hdr (fec_hdr));
|
|
}
|
|
|
|
RtpUlpFecHeader *
|
|
rtp_ulpfec_buffer_get_fechdr (GstRTPBuffer * rtp)
|
|
{
|
|
return (RtpUlpFecHeader *) gst_rtp_buffer_get_payload (rtp);
|
|
}
|
|
|
|
guint64
|
|
rtp_ulpfec_buffer_get_mask (GstRTPBuffer * rtp)
|
|
{
|
|
return fec_hdr_get_mask (rtp_ulpfec_buffer_get_fechdr (rtp));
|
|
}
|
|
|
|
guint16
|
|
rtp_ulpfec_buffer_get_seq_base (GstRTPBuffer * rtp)
|
|
{
|
|
return g_ntohs (rtp_ulpfec_buffer_get_fechdr (rtp)->seq);
|
|
}
|
|
|
|
guint
|
|
rtp_ulpfec_get_headers_len (gboolean fec_mask_long)
|
|
{
|
|
return sizeof (RtpUlpFecHeader) + fec_level_hdr_get_size (fec_mask_long);
|
|
}
|
|
|
|
#define ONE_64BIT G_GUINT64_CONSTANT(1)
|
|
|
|
guint64
|
|
rtp_ulpfec_packet_mask_from_seqnum (guint16 seq,
|
|
guint16 fec_seq_base, gboolean fec_mask_long)
|
|
{
|
|
gint seq_delta = gst_rtp_buffer_compare_seqnum (fec_seq_base, seq);
|
|
if (seq_delta >= 0
|
|
&& seq_delta <= RTP_ULPFEC_SEQ_BASE_OFFSET_MAX (fec_mask_long)) {
|
|
return ONE_64BIT << (RTP_ULPFEC_SEQ_BASE_OFFSET_MAX (TRUE) - seq_delta);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
gboolean
|
|
rtp_ulpfec_mask_is_long (guint64 mask)
|
|
{
|
|
return (mask & 0xffffffff) ? TRUE : FALSE;
|
|
}
|
|
|
|
gboolean
|
|
rtp_ulpfec_buffer_is_valid (GstRTPBuffer * rtp)
|
|
{
|
|
guint payload_len = gst_rtp_buffer_get_payload_len (rtp);
|
|
RtpUlpFecHeader *fec_hdr;
|
|
guint fec_hdrs_len;
|
|
guint fec_packet_len;
|
|
|
|
if (payload_len < sizeof (RtpUlpFecHeader))
|
|
goto toosmall;
|
|
|
|
fec_hdr = rtp_ulpfec_buffer_get_fechdr (rtp);
|
|
if (fec_hdr->E)
|
|
goto invalidcontent;
|
|
|
|
fec_hdrs_len = rtp_ulpfec_get_headers_len (fec_hdr->L);
|
|
if (payload_len < fec_hdrs_len)
|
|
goto toosmall;
|
|
|
|
fec_packet_len = fec_hdrs_len + rtp_ulpfec_hdr_get_protection_len (fec_hdr);
|
|
if (fec_packet_len != payload_len)
|
|
goto lengthmismatch;
|
|
|
|
return TRUE;
|
|
toosmall:
|
|
GST_WARNING ("FEC packet too small");
|
|
return FALSE;
|
|
|
|
lengthmismatch:
|
|
GST_WARNING ("invalid FEC packet (declared length %u, real length %u)",
|
|
fec_packet_len, payload_len);
|
|
return FALSE;
|
|
|
|
invalidcontent:
|
|
GST_WARNING ("FEC Header contains invalid fields: %u", fec_hdr->E);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
void
|
|
rtp_buffer_to_ulpfec_bitstring (GstRTPBuffer * rtp, GArray * dst_arr,
|
|
gboolean fec_buffer, gboolean fec_mask_long)
|
|
{
|
|
if (G_UNLIKELY (fec_buffer)) {
|
|
guint payload_len = gst_rtp_buffer_get_payload_len (rtp);
|
|
g_array_set_size (dst_arr, MAX (payload_len, dst_arr->len));
|
|
memcpy (dst_arr->data, gst_rtp_buffer_get_payload (rtp), payload_len);
|
|
} else {
|
|
const guint8 *src = rtp->data[0];
|
|
guint len = gst_rtp_buffer_get_packet_len (rtp) - MIN_RTP_HEADER_LEN;
|
|
guint dst_offset = rtp_ulpfec_get_headers_len (fec_mask_long);
|
|
guint src_offset = MIN_RTP_HEADER_LEN;
|
|
guint8 *dst;
|
|
|
|
g_array_set_size (dst_arr, MAX (dst_offset + len, dst_arr->len));
|
|
dst = (guint8 *) dst_arr->data;
|
|
|
|
*((guint64 *) dst) ^= *((const guint64 *) src);
|
|
((RtpUlpFecHeader *) dst)->len ^= g_htons (len);
|
|
_xor_mem (dst + dst_offset, src + src_offset, len);
|
|
}
|
|
}
|
|
|
|
GstBuffer *
|
|
rtp_ulpfec_bitstring_to_media_rtp_buffer (GArray * arr,
|
|
gboolean fec_mask_long, guint32 ssrc, guint16 seq)
|
|
{
|
|
guint fec_hdrs_len = rtp_ulpfec_get_headers_len (fec_mask_long);
|
|
guint payload_len =
|
|
fec_hdr_get_packets_len_recovery ((RtpUlpFecHeader *) arr->data);
|
|
GstMapInfo ret_info = GST_MAP_INFO_INIT;
|
|
GstMemory *ret_mem;
|
|
GstBuffer *ret;
|
|
|
|
if (payload_len > arr->len - fec_hdrs_len)
|
|
return NULL; // Not enough data
|
|
|
|
ret_mem = gst_allocator_alloc (NULL, MIN_RTP_HEADER_LEN + payload_len, NULL);
|
|
gst_memory_map (ret_mem, &ret_info, GST_MAP_READWRITE);
|
|
|
|
/* Filling 12 bytes of RTP header */
|
|
*((guint64 *) ret_info.data) = *((guint64 *) arr->data);
|
|
((RtpHeader *) ret_info.data)->version = 2;
|
|
((RtpHeader *) ret_info.data)->seq = g_htons (seq);
|
|
((RtpHeader *) ret_info.data)->ssrc = g_htonl (ssrc);
|
|
/* Filling payload */
|
|
memcpy (ret_info.data + MIN_RTP_HEADER_LEN,
|
|
arr->data + fec_hdrs_len, payload_len);
|
|
|
|
gst_memory_unmap (ret_mem, &ret_info);
|
|
ret = gst_buffer_new ();
|
|
gst_buffer_append_memory (ret, ret_mem);
|
|
return ret;
|
|
}
|
|
|
|
GstBuffer *
|
|
rtp_ulpfec_bitstring_to_fec_rtp_buffer (GArray * arr,
|
|
guint16 seq_base, gboolean fec_mask_long, guint64 fec_mask,
|
|
gboolean marker, guint8 pt, guint16 seq, guint32 timestamp, guint32 ssrc)
|
|
{
|
|
GstRTPBuffer rtp = GST_RTP_BUFFER_INIT;
|
|
GstBuffer *ret;
|
|
|
|
/* Filling FEC headers */
|
|
{
|
|
RtpUlpFecHeader *hdr = (RtpUlpFecHeader *) arr->data;
|
|
RtpUlpFecLevelHeader *lvlhdr;
|
|
hdr->E = 0;
|
|
hdr->L = fec_mask_long;
|
|
hdr->seq = g_htons (seq_base);
|
|
|
|
lvlhdr = fec_hdr_get_level_hdr (hdr);
|
|
fec_level_hdr_set_protection_len (lvlhdr,
|
|
arr->len - rtp_ulpfec_get_headers_len (fec_mask_long));
|
|
fec_level_hdr_set_mask (lvlhdr, fec_mask_long, fec_mask);
|
|
}
|
|
|
|
/* Filling RTP header, copying payload */
|
|
ret = gst_rtp_buffer_new_allocate (arr->len, 0, 0);
|
|
if (!gst_rtp_buffer_map (ret, GST_MAP_READWRITE, &rtp))
|
|
g_assert_not_reached ();
|
|
|
|
gst_rtp_buffer_set_marker (&rtp, marker);
|
|
if (marker)
|
|
GST_BUFFER_FLAG_SET (ret, GST_BUFFER_FLAG_MARKER);
|
|
gst_rtp_buffer_set_payload_type (&rtp, pt);
|
|
gst_rtp_buffer_set_seq (&rtp, seq);
|
|
gst_rtp_buffer_set_timestamp (&rtp, timestamp);
|
|
gst_rtp_buffer_set_ssrc (&rtp, ssrc);
|
|
|
|
memcpy (gst_rtp_buffer_get_payload (&rtp), arr->data, arr->len);
|
|
|
|
gst_rtp_buffer_unmap (&rtp);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* rtp_ulpfec_map_info_map:
|
|
* @buffer: (transfer: full) #GstBuffer
|
|
* @info: #RtpUlpFecMapInfo
|
|
*
|
|
* Maps the contents of @buffer into @info. If @buffer made of many #GstMemory
|
|
* objects, merges them together to create a new buffer made of single
|
|
* continious #GstMemory.
|
|
*
|
|
* Returns: %TRUE if @buffer could be mapped
|
|
**/
|
|
gboolean
|
|
rtp_ulpfec_map_info_map (GstBuffer * buffer, RtpUlpFecMapInfo * info)
|
|
{
|
|
/* We need to make sure we are working with continious memory chunk.
|
|
* If not merge all memories together */
|
|
if (gst_buffer_n_memory (buffer) > 1) {
|
|
GstBuffer *new_buffer = gst_buffer_new ();
|
|
GstMemory *mem = gst_buffer_get_all_memory (buffer);
|
|
gst_buffer_append_memory (new_buffer, mem);
|
|
|
|
/* We supposed to own the old buffer, but we don't use it here, so unref */
|
|
gst_buffer_unref (buffer);
|
|
buffer = new_buffer;
|
|
}
|
|
|
|
if (!gst_rtp_buffer_map (buffer,
|
|
GST_MAP_READ | GST_RTP_BUFFER_MAP_FLAG_SKIP_PADDING, &info->rtp)) {
|
|
/* info->rtp.buffer = NULL is an indication for rtp_ulpfec_map_info_unmap()
|
|
* that mapping has failed */
|
|
g_assert (NULL == info->rtp.buffer);
|
|
gst_buffer_unref (buffer);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* rtp_ulpfec_map_info_unmap:
|
|
* @info: #RtpUlpFecMapInfo
|
|
*
|
|
* Unmap @info previously mapped with rtp_ulpfec_map_info_map() and unrefs the
|
|
* buffer. For convenience can even be called even if rtp_ulpfec_map_info_map
|
|
* returned FALSE
|
|
**/
|
|
void
|
|
rtp_ulpfec_map_info_unmap (RtpUlpFecMapInfo * info)
|
|
{
|
|
GstBuffer *buffer = info->rtp.buffer;
|
|
|
|
if (buffer) {
|
|
gst_rtp_buffer_unmap (&info->rtp);
|
|
gst_buffer_unref (buffer);
|
|
}
|
|
}
|
|
|
|
#ifndef GST_DISABLE_GST_DEBUG
|
|
void
|
|
rtp_ulpfec_log_rtppacket (GstDebugCategory * cat, GstDebugLevel level,
|
|
gpointer object, const gchar * name, GstRTPBuffer * rtp)
|
|
{
|
|
guint seq;
|
|
guint ssrc;
|
|
guint timestamp;
|
|
guint pt;
|
|
|
|
if (level > gst_debug_category_get_threshold (cat))
|
|
return;
|
|
|
|
seq = gst_rtp_buffer_get_seq (rtp);
|
|
ssrc = gst_rtp_buffer_get_ssrc (rtp);
|
|
timestamp = gst_rtp_buffer_get_timestamp (rtp);
|
|
pt = gst_rtp_buffer_get_payload_type (rtp);
|
|
|
|
GST_CAT_LEVEL_LOG (cat, level, object,
|
|
"%-22s: [%c%c%c%c] ssrc=0x%08x pt=%u tstamp=%u seq=%u size=%u(%u,%u)",
|
|
name,
|
|
gst_rtp_buffer_get_marker (rtp) ? 'M' : ' ',
|
|
gst_rtp_buffer_get_extension (rtp) ? 'X' : ' ',
|
|
gst_rtp_buffer_get_padding (rtp) ? 'P' : ' ',
|
|
gst_rtp_buffer_get_csrc_count (rtp) > 0 ? 'C' : ' ',
|
|
ssrc, pt, timestamp, seq,
|
|
gst_rtp_buffer_get_packet_len (rtp),
|
|
gst_rtp_buffer_get_packet_len (rtp) - MIN_RTP_HEADER_LEN,
|
|
gst_rtp_buffer_get_payload_len (rtp));
|
|
}
|
|
#endif /* GST_DISABLE_GST_DEBUG */
|
|
|
|
#ifndef GST_DISABLE_GST_DEBUG
|
|
void
|
|
rtp_ulpfec_log_fec_packet (GstDebugCategory * cat, GstDebugLevel level,
|
|
gpointer object, GstRTPBuffer * fecrtp)
|
|
{
|
|
RtpUlpFecHeader *fec_hdr;
|
|
RtpUlpFecLevelHeader *fec_level_hdr;
|
|
|
|
if (level > gst_debug_category_get_threshold (cat))
|
|
return;
|
|
|
|
fec_hdr = gst_rtp_buffer_get_payload (fecrtp);
|
|
GST_CAT_LEVEL_LOG (cat, level, object,
|
|
"%-22s: [%c%c%c%c%c%c] pt=%u tstamp=%u seq=%u recovery_len=%u",
|
|
"fec header",
|
|
fec_hdr->E ? 'E' : ' ',
|
|
fec_hdr->L ? 'L' : ' ',
|
|
fec_hdr->P ? 'P' : ' ',
|
|
fec_hdr->X ? 'X' : ' ',
|
|
fec_hdr->CC ? 'C' : ' ',
|
|
fec_hdr->M ? 'M' : ' ',
|
|
fec_hdr->pt,
|
|
fec_hdr_get_timestamp_recovery (fec_hdr),
|
|
fec_hdr_get_seq_base (fec_hdr, TRUE,
|
|
gst_rtp_buffer_get_seq (fecrtp)),
|
|
fec_hdr_get_packets_len_recovery (fec_hdr));
|
|
|
|
fec_level_hdr = fec_hdr_get_level_hdr (fec_hdr);
|
|
GST_CAT_LEVEL_LOG (cat, level, object,
|
|
"%-22s: protection_len=%u mask=0x%012" G_GINT64_MODIFIER "x",
|
|
"fec level header",
|
|
g_ntohs (fec_level_hdr->protection_len),
|
|
fec_level_hdr_get_mask (fec_level_hdr, fec_hdr->L));
|
|
}
|
|
#endif /* GST_DISABLE_GST_DEBUG */
|