mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-22 00:06:36 +00:00
2428a1ca55
Original commit message from CVS: * gst/rtp/gstrtpL16depay.c: (gst_rtp_L16_depay_setcaps), (gst_rtp_L16_depay_process): Check if clock-rate and channels are valid. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpL16pay.c: (gst_rtp_L16_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpac3depay.c: (gst_rtp_ac3_depay_setcaps), (gst_rtp_ac3_depay_process): Don't ignore the return value of set_caps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtpamrdepay.c: (gst_rtp_amr_depay_setcaps), (gst_rtp_amr_depay_process): * gst/rtp/gstrtpamrdepay.h: Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. No need to set output caps on the buffers, the base class does that for us. The subclass will make sure we are negotiated. * gst/rtp/gstrtpdvdepay.c: (gst_rtp_dv_depay_setcaps), (gst_rtp_dv_depay_process), (gst_rtp_dv_depay_reset): * gst/rtp/gstrtpdvdepay.h: Clean up caps negotiation. The subclass will make sure we are negotiated. * gst/rtp/gstrtpg726depay.c: (gst_rtp_g726_depay_setcaps), (gst_rtp_g726_depay_process): Clean up caps negotiation. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpg729depay.c: (gst_rtp_g729_depay_init), (gst_rtp_g729_depay_setcaps), (gst_rtp_g729_depay_process): * gst/rtp/gstrtpg729depay.h: The subclass will make sure we are negotiated. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpgsmdepay.c: (gst_rtp_gsm_depay_setcaps), (gst_rtp_gsm_depay_process): Clean up caps negotiation. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpgsmpay.c: (gst_rtp_gsm_pay_setcaps): Clean up caps negotiation. Don't ignore the return value of set_outcaps. * gst/rtp/gstrtph263depay.c: (gst_rtp_h263_depay_setcaps), (gst_rtp_h263_depay_process): Clean up caps negotiation. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtph263pay.c: (gst_rtp_h263_pay_setcaps), (gst_rtp_h263_pay_flush), (gst_rtp_h263_pay_handle_buffer): * gst/rtp/gstrtph263pay.h: Don't ignore the return value of set_outcaps. Do some more timestamps. * gst/rtp/gstrtph263pdepay.c: (gst_rtp_h263p_depay_setcaps), (gst_rtp_h263p_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtph263ppay.c: (gst_rtp_h263p_pay_class_init), (gst_rtp_h263p_pay_setcaps), (gst_rtp_h263p_pay_flush), (gst_rtp_h263p_pay_handle_buffer): * gst/rtp/gstrtph263ppay.h: Don't ignore the return value of set_outcaps. Do some more timestamps. * gst/rtp/gstrtph264depay.c: (gst_rtp_h264_depay_setcaps), (gst_rtp_h264_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. Fix possible caps leak. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtph264pay.c: (gst_rtp_h264_pay_setcaps): Add some more debug info. * gst/rtp/gstrtpilbcdepay.c: (gst_rtp_ilbc_depay_setcaps), (gst_rtp_ilbc_depay_process): Clean up caps negotiation. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpilbcpay.c: (gst_rtpilbcpay_sink_setcaps): Clean up caps negotiation. * gst/rtp/gstrtpmp1sdepay.c: (gst_rtp_mp1s_depay_setcaps), (gst_rtp_mp1s_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. No need to set caps on buffers, subclass does that for us. * gst/rtp/gstrtpmp2tdepay.c: (gst_rtp_mp2t_depay_setcaps), (gst_rtp_mp2t_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. No need to set caps on buffers, subclass does that for us. * gst/rtp/gstrtpmp4adepay.c: (gst_rtp_mp4a_depay_setcaps), (gst_rtp_mp4a_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtpmp4apay.c: (gst_rtp_mp4a_pay_new_caps), (gst_rtp_mp4a_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpmp4gdepay.c: (gst_rtp_mp4g_depay_setcaps), (gst_rtp_mp4g_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. No need to set caps on buffers, subclass does that for us. * gst/rtp/gstrtpmp4gpay.c: (gst_rtp_mp4g_pay_finalize), (gst_rtp_mp4g_pay_new_caps), (gst_rtp_mp4g_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpmp4vdepay.c: (gst_rtp_mp4v_depay_setcaps), (gst_rtp_mp4v_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. No need to set caps on buffers, subclass does that for us. * gst/rtp/gstrtpmp4vpay.c: (gst_rtp_mp4v_pay_new_caps), (gst_rtp_mp4v_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpmpadepay.c: (gst_rtp_mpa_depay_setcaps), (gst_rtp_mpa_depay_process): Clean up caps negotiation. Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtpmpapay.c: (gst_rtp_mpa_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpmpvdepay.c: (gst_rtp_mpv_depay_setcaps), (gst_rtp_mpv_depay_process): Clean up caps negotiation. Actually set output caps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtpmpvpay.c: (gst_rtp_mpv_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtppcmadepay.c: (gst_rtp_pcma_depay_setcaps), (gst_rtp_pcma_depay_process): Clean up caps negotiation. Set output buffer duration because we can. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtppcmapay.c: (gst_rtp_pcma_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtppcmudepay.c: (gst_rtp_pcmu_depay_setcaps), (gst_rtp_pcmu_depay_process): Clean up caps negotiation. Use the marker bit to set the DISCONT flag on outgoing buffers. * gst/rtp/gstrtppcmupay.c: (gst_rtp_pcmu_pay_setcaps): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpspeexdepay.c: (gst_rtp_speex_depay_init), (gst_rtp_speex_depay_setcaps), (gst_rtp_speex_depay_process): Clean up caps negotiation. Set output caps on the pad and header buffers. Set duration on output buffers because we can. * gst/rtp/gstrtpspeexpay.c: (gst_rtp_speex_pay_parse_ident): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpsv3vdepay.c: (gst_rtp_sv3v_depay_setcaps), (gst_rtp_sv3v_depay_process): Clean up caps negotiation. No need to validate the buffer, the base class does that for us. No need to set caps out output buffers, subclass does that. * gst/rtp/gstrtptheoradepay.c: (gst_rtp_theora_depay_setcaps), (gst_rtp_theora_depay_process): Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtptheorapay.c: (gst_rtp_theora_pay_class_init), (gst_rtp_theora_pay_flush_packet), (encode_base64), (gst_rtp_theora_pay_finish_headers), (gst_rtp_theora_pay_parse_id), (gst_rtp_theora_pay_handle_buffer): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpvorbisdepay.c: (gst_rtp_vorbis_depay_setcaps), (gst_rtp_vorbis_depay_process): Don't ignore the return value of setcaps. No need to validate the buffer, the base class does that for us. * gst/rtp/gstrtpvorbispay.c: (gst_rtp_vorbis_pay_finish_headers): Don't ignore the return value of set_outcaps. * gst/rtp/gstrtpvrawdepay.c: (gst_rtp_vraw_depay_setcaps): Clean up caps negotiation, don't ignore setcaps return. * gst/rtp/gstrtpvrawpay.c: (gst_rtp_vraw_pay_setcaps): Don't ignore the return value of set_outcaps.
471 lines
13 KiB
C
471 lines
13 KiB
C
/* GStreamer
|
|
* Copyright (C) <2005> Wim Taymans <wim@fluendo.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., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <gst/rtp/gstrtpbuffer.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "gstrtpamrdepay.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (rtpamrdepay_debug);
|
|
#define GST_CAT_DEFAULT (rtpamrdepay_debug)
|
|
|
|
/* references:
|
|
*
|
|
* RFC 3267 - Real-Time Transport Protocol (RTP) Payload Format and File
|
|
* Storage Format for the Adaptive Multi-Rate (AMR) and Adaptive Multi-Rate
|
|
* Wideband (AMR-WB) Audio Codecs.
|
|
*/
|
|
|
|
/* elementfactory information */
|
|
static const GstElementDetails gst_rtp_amrdepay_details =
|
|
GST_ELEMENT_DETAILS ("RTP packet depayloader",
|
|
"Codec/Depayloader/Network",
|
|
"Extracts AMR or AMR-WB audio from RTP packets (RFC 3267)",
|
|
"Wim Taymans <wim@fluendo.com>");
|
|
|
|
/* RtpAMRDepay signals and args */
|
|
enum
|
|
{
|
|
/* FILL ME */
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
ARG_0
|
|
};
|
|
|
|
/* input is an RTP packet
|
|
*
|
|
* params see RFC 3267, section 8.1
|
|
*/
|
|
static GstStaticPadTemplate gst_rtp_amr_depay_sink_template =
|
|
GST_STATIC_PAD_TEMPLATE ("sink",
|
|
GST_PAD_SINK,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("application/x-rtp, "
|
|
"media = (string) \"audio\", "
|
|
"payload = (int) " GST_RTP_PAYLOAD_DYNAMIC_STRING ", "
|
|
"clock-rate = (int) 8000, "
|
|
"encoding-name = (string) \"AMR\", "
|
|
"encoding-params = (string) \"1\", "
|
|
/* NOTE that all values must be strings in orde to be able to do SDP <->
|
|
* GstCaps mapping. */
|
|
"octet-align = (string) \"1\", "
|
|
"crc = (string) { \"0\", \"1\" }, "
|
|
"robust-sorting = (string) \"0\", " "interleaving = (string) \"0\";"
|
|
/* following options are not needed for a decoder
|
|
*
|
|
"mode-set = (int) [ 0, 7 ], "
|
|
"mode-change-period = (int) [ 1, MAX ], "
|
|
"mode-change-neighbor = (boolean) { TRUE, FALSE }, "
|
|
"maxptime = (int) [ 20, MAX ], "
|
|
"ptime = (int) [ 20, MAX ]"
|
|
*/
|
|
"application/x-rtp, "
|
|
"media = (string) \"audio\", "
|
|
"payload = (int) " GST_RTP_PAYLOAD_DYNAMIC_STRING ", "
|
|
"clock-rate = (int) 16000, "
|
|
"encoding-name = (string) \"AMR-WB\", "
|
|
"encoding-params = (string) \"1\", "
|
|
/* NOTE that all values must be strings in orde to be able to do SDP <->
|
|
* GstCaps mapping. */
|
|
"octet-align = (string) \"1\", "
|
|
"crc = (string) { \"0\", \"1\" }, "
|
|
"robust-sorting = (string) \"0\", " "interleaving = (string) \"0\""
|
|
/* following options are not needed for a decoder
|
|
*
|
|
"mode-set = (int) [ 0, 7 ], "
|
|
"mode-change-period = (int) [ 1, MAX ], "
|
|
"mode-change-neighbor = (boolean) { TRUE, FALSE }, "
|
|
"maxptime = (int) [ 20, MAX ], "
|
|
"ptime = (int) [ 20, MAX ]"
|
|
*/
|
|
)
|
|
);
|
|
|
|
static GstStaticPadTemplate gst_rtp_amr_depay_src_template =
|
|
GST_STATIC_PAD_TEMPLATE ("src",
|
|
GST_PAD_SRC,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("audio/AMR, " "channels = (int) 1," "rate = (int) 8000;"
|
|
"audio/AMR-WB, " "channels = (int) 1," "rate = (int) 16000")
|
|
);
|
|
|
|
static gboolean gst_rtp_amr_depay_setcaps (GstBaseRTPDepayload * depayload,
|
|
GstCaps * caps);
|
|
static GstBuffer *gst_rtp_amr_depay_process (GstBaseRTPDepayload * depayload,
|
|
GstBuffer * buf);
|
|
|
|
GST_BOILERPLATE (GstRtpAMRDepay, gst_rtp_amr_depay, GstBaseRTPDepayload,
|
|
GST_TYPE_BASE_RTP_DEPAYLOAD);
|
|
|
|
static void
|
|
gst_rtp_amr_depay_base_init (gpointer klass)
|
|
{
|
|
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
|
|
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_static_pad_template_get (&gst_rtp_amr_depay_src_template));
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_static_pad_template_get (&gst_rtp_amr_depay_sink_template));
|
|
|
|
gst_element_class_set_details (element_class, &gst_rtp_amrdepay_details);
|
|
}
|
|
|
|
static void
|
|
gst_rtp_amr_depay_class_init (GstRtpAMRDepayClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GstElementClass *gstelement_class;
|
|
GstBaseRTPDepayloadClass *gstbasertpdepayload_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
gstelement_class = (GstElementClass *) klass;
|
|
gstbasertpdepayload_class = (GstBaseRTPDepayloadClass *) klass;
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
gstbasertpdepayload_class->process = gst_rtp_amr_depay_process;
|
|
gstbasertpdepayload_class->set_caps = gst_rtp_amr_depay_setcaps;
|
|
|
|
GST_DEBUG_CATEGORY_INIT (rtpamrdepay_debug, "rtpamrdepay", 0,
|
|
"AMR/AMR-WB RTP Depayloader");
|
|
}
|
|
|
|
static void
|
|
gst_rtp_amr_depay_init (GstRtpAMRDepay * rtpamrdepay,
|
|
GstRtpAMRDepayClass * klass)
|
|
{
|
|
GstBaseRTPDepayload *depayload;
|
|
|
|
depayload = GST_BASE_RTP_DEPAYLOAD (rtpamrdepay);
|
|
|
|
gst_pad_use_fixed_caps (GST_BASE_RTP_DEPAYLOAD_SRCPAD (depayload));
|
|
}
|
|
|
|
static gboolean
|
|
gst_rtp_amr_depay_setcaps (GstBaseRTPDepayload * depayload, GstCaps * caps)
|
|
{
|
|
GstStructure *structure;
|
|
GstCaps *srccaps;
|
|
GstRtpAMRDepay *rtpamrdepay;
|
|
const gchar *params;
|
|
const gchar *str, *type;
|
|
gint clock_rate, need_clock_rate;
|
|
gboolean res;
|
|
|
|
rtpamrdepay = GST_RTP_AMR_DEPAY (depayload);
|
|
|
|
structure = gst_caps_get_structure (caps, 0);
|
|
|
|
/* figure out the mode first and set the clock rates */
|
|
if ((str = gst_structure_get_string (structure, "encoding-name"))) {
|
|
if (strcmp (str, "AMR") == 0) {
|
|
rtpamrdepay->mode = GST_RTP_AMR_DP_MODE_NB;
|
|
need_clock_rate = 8000;
|
|
type = "audio/AMR";
|
|
} else if (strcmp (str, "AMR-WB") == 0) {
|
|
rtpamrdepay->mode = GST_RTP_AMR_DP_MODE_WB;
|
|
need_clock_rate = 16000;
|
|
type = "audio/AMR-WB";
|
|
} else
|
|
goto invalid_mode;
|
|
} else
|
|
goto invalid_mode;
|
|
|
|
if (!(str = gst_structure_get_string (structure, "octet-align")))
|
|
rtpamrdepay->octet_align = FALSE;
|
|
else
|
|
rtpamrdepay->octet_align = (atoi (str) == 1);
|
|
|
|
if (!(str = gst_structure_get_string (structure, "crc")))
|
|
rtpamrdepay->crc = FALSE;
|
|
else
|
|
rtpamrdepay->crc = (atoi (str) == 1);
|
|
|
|
if (rtpamrdepay->crc) {
|
|
/* crc mode implies octet aligned mode */
|
|
rtpamrdepay->octet_align = TRUE;
|
|
}
|
|
|
|
if (!(str = gst_structure_get_string (structure, "robust-sorting")))
|
|
rtpamrdepay->robust_sorting = FALSE;
|
|
else
|
|
rtpamrdepay->robust_sorting = (atoi (str) == 1);
|
|
|
|
if (rtpamrdepay->robust_sorting) {
|
|
/* robust_sorting mode implies octet aligned mode */
|
|
rtpamrdepay->octet_align = TRUE;
|
|
}
|
|
|
|
if (!(str = gst_structure_get_string (structure, "interleaving")))
|
|
rtpamrdepay->interleaving = FALSE;
|
|
else
|
|
rtpamrdepay->interleaving = (atoi (str) == 1);
|
|
|
|
if (rtpamrdepay->interleaving) {
|
|
/* interleaving mode implies octet aligned mode */
|
|
rtpamrdepay->octet_align = TRUE;
|
|
}
|
|
|
|
if (!(params = gst_structure_get_string (structure, "encoding-params")))
|
|
rtpamrdepay->channels = 1;
|
|
else {
|
|
rtpamrdepay->channels = atoi (params);
|
|
}
|
|
|
|
if (!gst_structure_get_int (structure, "clock-rate", &clock_rate))
|
|
clock_rate = need_clock_rate;
|
|
depayload->clock_rate = clock_rate;
|
|
|
|
/* we require 1 channel, 8000 Hz, octet aligned, no CRC,
|
|
* no robust sorting, no interleaving for now */
|
|
if (rtpamrdepay->channels != 1)
|
|
return FALSE;
|
|
if (clock_rate != need_clock_rate)
|
|
return FALSE;
|
|
if (rtpamrdepay->octet_align != TRUE)
|
|
return FALSE;
|
|
if (rtpamrdepay->robust_sorting != FALSE)
|
|
return FALSE;
|
|
if (rtpamrdepay->interleaving != FALSE)
|
|
return FALSE;
|
|
|
|
srccaps = gst_caps_new_simple (type,
|
|
"channels", G_TYPE_INT, rtpamrdepay->channels,
|
|
"rate", G_TYPE_INT, clock_rate, NULL);
|
|
res = gst_pad_set_caps (GST_BASE_RTP_DEPAYLOAD_SRCPAD (depayload), srccaps);
|
|
gst_caps_unref (srccaps);
|
|
|
|
return res;
|
|
|
|
/* ERRORS */
|
|
invalid_mode:
|
|
{
|
|
GST_ERROR_OBJECT (rtpamrdepay, "invalid encoding-name");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* -1 is invalid */
|
|
static gint nb_frame_size[16] = {
|
|
12, 13, 15, 17, 19, 20, 26, 31,
|
|
5, -1, -1, -1, -1, -1, -1, 0
|
|
};
|
|
static gint wb_frame_size[16] = {
|
|
17, 23, 32, 36, 40, 46, 50, 58,
|
|
60, -1, -1, -1, -1, -1, -1, 0
|
|
};
|
|
|
|
static GstBuffer *
|
|
gst_rtp_amr_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
|
{
|
|
GstRtpAMRDepay *rtpamrdepay;
|
|
GstBuffer *outbuf = NULL;
|
|
gint payload_len;
|
|
gint *frame_size;
|
|
|
|
rtpamrdepay = GST_RTP_AMR_DEPAY (depayload);
|
|
|
|
/* setup frame size pointer */
|
|
if (rtpamrdepay->mode == GST_RTP_AMR_DP_MODE_NB)
|
|
frame_size = nb_frame_size;
|
|
else
|
|
frame_size = wb_frame_size;
|
|
|
|
/* when we get here, 1 channel, 8000/16000 Hz, octet aligned, no CRC,
|
|
* no robust sorting, no interleaving data is to be depayloaded */
|
|
{
|
|
guint8 *payload, *p, *dp;
|
|
guint8 CMR;
|
|
gint i, num_packets, num_nonempty_packets;
|
|
gint amr_len;
|
|
gint ILL, ILP;
|
|
gboolean marker;
|
|
|
|
marker = gst_rtp_buffer_get_marker (buf);
|
|
payload_len = gst_rtp_buffer_get_payload_len (buf);
|
|
|
|
/* need at least 2 bytes for the header */
|
|
if (payload_len < 2)
|
|
goto too_small;
|
|
|
|
payload = gst_rtp_buffer_get_payload (buf);
|
|
|
|
/* depay CMR. The CMR is used by the sender to request
|
|
* a new encoding mode.
|
|
*
|
|
* 0 1 2 3 4 5 6 7
|
|
* +-+-+-+-+-+-+-+-+
|
|
* | CMR |R|R|R|R|
|
|
* +-+-+-+-+-+-+-+-+
|
|
*/
|
|
CMR = (payload[0] & 0xf0) >> 4;
|
|
|
|
/* strip CMR header now, pack FT and the data for the decoder */
|
|
payload_len -= 1;
|
|
payload += 1;
|
|
|
|
GST_DEBUG_OBJECT (rtpamrdepay, "payload len %d", payload_len);
|
|
|
|
if (rtpamrdepay->interleaving) {
|
|
ILL = (payload[0] & 0xf0) >> 4;
|
|
ILP = (payload[0] & 0x0f);
|
|
|
|
payload_len -= 1;
|
|
payload += 1;
|
|
|
|
if (ILP > ILL)
|
|
goto wrong_interleaving;
|
|
}
|
|
|
|
/*
|
|
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6
|
|
* +-+-+-+-+-+-+-+-+..
|
|
* |F| FT |Q|P|P| more FT..
|
|
* +-+-+-+-+-+-+-+-+..
|
|
*/
|
|
/* count number of packets by counting the FTs. Also
|
|
* count number of amr data bytes and number of non-empty
|
|
* packets (this is also the number of CRCs if present). */
|
|
amr_len = 0;
|
|
num_nonempty_packets = 0;
|
|
num_packets = 0;
|
|
for (i = 0; i < payload_len; i++) {
|
|
gint fr_size;
|
|
guint8 FT;
|
|
|
|
FT = (payload[i] & 0x78) >> 3;
|
|
|
|
fr_size = frame_size[FT];
|
|
GST_DEBUG_OBJECT (rtpamrdepay, "frame size %d", fr_size);
|
|
if (fr_size == -1)
|
|
goto wrong_framesize;
|
|
|
|
if (fr_size > 0) {
|
|
amr_len += fr_size;
|
|
num_nonempty_packets++;
|
|
}
|
|
num_packets++;
|
|
|
|
if ((payload[i] & 0x80) == 0)
|
|
break;
|
|
}
|
|
|
|
if (rtpamrdepay->crc) {
|
|
/* data len + CRC len + header bytes should be smaller than payload_len */
|
|
if (num_packets + num_nonempty_packets + amr_len > payload_len)
|
|
goto wrong_length_1;
|
|
} else {
|
|
/* data len + header bytes should be smaller than payload_len */
|
|
if (num_packets + amr_len > payload_len)
|
|
goto wrong_length_2;
|
|
}
|
|
|
|
outbuf = gst_buffer_new_and_alloc (payload_len);
|
|
|
|
/* point to destination */
|
|
p = GST_BUFFER_DATA (outbuf);
|
|
/* point to first data packet */
|
|
dp = payload + num_packets;
|
|
if (rtpamrdepay->crc) {
|
|
/* skip CRC if present */
|
|
dp += num_nonempty_packets;
|
|
}
|
|
|
|
for (i = 0; i < num_packets; i++) {
|
|
gint fr_size;
|
|
|
|
/* copy FT, clear F bit */
|
|
*p++ = payload[i] & 0x7f;
|
|
|
|
fr_size = frame_size[(payload[i] & 0x78) >> 3];
|
|
if (fr_size > 0) {
|
|
/* copy data packet, FIXME, calc CRC here. */
|
|
memcpy (p, dp, fr_size);
|
|
|
|
p += fr_size;
|
|
dp += fr_size;
|
|
}
|
|
}
|
|
/* we can set the duration because each packet is 20 milliseconds */
|
|
GST_BUFFER_DURATION (outbuf) = num_packets * 20 * GST_MSECOND;
|
|
|
|
if (marker) {
|
|
/* marker bit marks a discont buffer after a talkspurt. */
|
|
GST_DEBUG_OBJECT (depayload, "marker bit was set");
|
|
GST_BUFFER_FLAG_SET (outbuf, GST_BUFFER_FLAG_DISCONT);
|
|
marker = FALSE;
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (depayload, "pushing buffer of size %d",
|
|
GST_BUFFER_SIZE (outbuf));
|
|
}
|
|
return outbuf;
|
|
|
|
/* ERRORS */
|
|
too_small:
|
|
{
|
|
GST_ELEMENT_WARNING (rtpamrdepay, STREAM, DECODE,
|
|
(NULL), ("AMR RTP payload too small (%d)", payload_len));
|
|
goto bad_packet;
|
|
}
|
|
wrong_interleaving:
|
|
{
|
|
GST_ELEMENT_WARNING (rtpamrdepay, STREAM, DECODE,
|
|
(NULL), ("AMR RTP wrong interleaving"));
|
|
goto bad_packet;
|
|
}
|
|
wrong_framesize:
|
|
{
|
|
GST_ELEMENT_WARNING (rtpamrdepay, STREAM, DECODE,
|
|
(NULL), ("AMR RTP frame size == -1"));
|
|
goto bad_packet;
|
|
}
|
|
wrong_length_1:
|
|
{
|
|
GST_ELEMENT_WARNING (rtpamrdepay, STREAM, DECODE,
|
|
(NULL), ("AMR RTP wrong length 1"));
|
|
goto bad_packet;
|
|
}
|
|
wrong_length_2:
|
|
{
|
|
GST_ELEMENT_WARNING (rtpamrdepay, STREAM, DECODE,
|
|
(NULL), ("AMR RTP wrong length 2"));
|
|
goto bad_packet;
|
|
}
|
|
bad_packet:
|
|
{
|
|
/* no fatal error */
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gst_rtp_amr_depay_plugin_init (GstPlugin * plugin)
|
|
{
|
|
return gst_element_register (plugin, "rtpamrdepay",
|
|
GST_RANK_MARGINAL, GST_TYPE_RTP_AMR_DEPAY);
|
|
}
|