libavformat/rtpdec_mpeg4.c
32f3c541
 /*
9b3788ef
  * Common code for the RTP depacketization of MPEG-4 formats.
  * Copyright (c) 2010 Fabrice Bellard
  *                    Romain Degez
  *
  * This file is part of FFmpeg.
  *
  * FFmpeg 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.
  *
  * FFmpeg 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 FFmpeg; if not, write to the Free Software
  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  */
 
 /**
  * @file
  * @brief MPEG4 / RTP Code
  * @author Fabrice Bellard
  * @author Romain Degez
  */
 
965a3ddb
 #include "rtpdec_formats.h"
1537817e
 #include "internal.h"
9b3788ef
 #include "libavutil/avstring.h"
73e6c53e
 #include "libavcodec/get_bits.h"
7fc8ac7f
 
ca937a55
 /** Structure listing useful vars to parse RTP packet payload*/
 struct PayloadContext
 {
     int sizelength;
     int indexlength;
     int indexdeltalength;
     int profile_level_id;
     int streamtype;
     int objecttype;
     char *mode;
 
     /** mpeg 4 AU headers */
     struct AUHeaders {
         int size;
         int index;
         int cts_flag;
         int cts;
         int dts_flag;
         int dts;
         int rap_flag;
         int streamstate;
     } *au_headers;
     int au_headers_allocated;
     int nb_au_headers;
     int au_headers_length_bytes;
     int cur_au_index;
 };
9b3788ef
 
7fc8ac7f
 typedef struct {
     const char *str;
     uint16_t    type;
     uint32_t    offset;
 } AttrNameMap;
 
 /* All known fmtp parameters and the corresponding RTPAttrTypeEnum */
 #define ATTR_NAME_TYPE_INT 0
 #define ATTR_NAME_TYPE_STR 1
 static const AttrNameMap attr_names[]=
 {
     { "SizeLength",       ATTR_NAME_TYPE_INT,
ca937a55
       offsetof(PayloadContext, sizelength) },
7fc8ac7f
     { "IndexLength",      ATTR_NAME_TYPE_INT,
ca937a55
       offsetof(PayloadContext, indexlength) },
7fc8ac7f
     { "IndexDeltaLength", ATTR_NAME_TYPE_INT,
ca937a55
       offsetof(PayloadContext, indexdeltalength) },
7fc8ac7f
     { "profile-level-id", ATTR_NAME_TYPE_INT,
ca937a55
       offsetof(PayloadContext, profile_level_id) },
7fc8ac7f
     { "StreamType",       ATTR_NAME_TYPE_INT,
ca937a55
       offsetof(PayloadContext, streamtype) },
7fc8ac7f
     { "mode",             ATTR_NAME_TYPE_STR,
ca937a55
       offsetof(PayloadContext, mode) },
7fc8ac7f
     { NULL, -1, -1 },
 };
 
ca937a55
 static PayloadContext *new_context(void)
 {
     return av_mallocz(sizeof(PayloadContext));
 }
 
 static void free_context(PayloadContext * data)
 {
     int i;
     for (i = 0; i < data->nb_au_headers; i++) {
          /* according to rtp_parse_mp4_au, we treat multiple
           * au headers as one, so nb_au_headers is always 1.
           * loop anyway in case this changes.
           * (note: changes done carelessly might lead to a double free)
           */
        av_free(&data->au_headers[i]);
     }
     av_free(data->mode);
     av_free(data);
 }
 
9b3788ef
 static int parse_fmtp_config(AVCodecContext * codec, char *value)
 {
     /* decode the hexa encoded parameter */
311baee7
     int len = ff_hex_to_data(NULL, value);
e31a7441
     av_free(codec->extradata);
9b3788ef
     codec->extradata = av_mallocz(len + FF_INPUT_BUFFER_PADDING_SIZE);
     if (!codec->extradata)
         return AVERROR(ENOMEM);
     codec->extradata_size = len;
311baee7
     ff_hex_to_data(codec->extradata, value);
9b3788ef
     return 0;
 }
 
c47f567c
 static int rtp_parse_mp4_au(PayloadContext *data, const uint8_t *buf)
73e6c53e
 {
     int au_headers_length, au_header_size, i;
     GetBitContext getbitcontext;
 
     /* decode the first 2 bytes where the AUHeader sections are stored
        length in bits */
     au_headers_length = AV_RB16(buf);
 
     if (au_headers_length > RTP_MAX_PACKET_LENGTH)
       return -1;
 
c47f567c
     data->au_headers_length_bytes = (au_headers_length + 7) / 8;
73e6c53e
 
     /* skip AU headers length section (2 bytes) */
     buf += 2;
 
c47f567c
     init_get_bits(&getbitcontext, buf, data->au_headers_length_bytes * 8);
73e6c53e
 
     /* XXX: Wrong if optionnal additional sections are present (cts, dts etc...) */
c47f567c
     au_header_size = data->sizelength + data->indexlength;
73e6c53e
     if (au_header_size <= 0 || (au_headers_length % au_header_size != 0))
         return -1;
 
c47f567c
     data->nb_au_headers = au_headers_length / au_header_size;
     if (!data->au_headers || data->au_headers_allocated < data->nb_au_headers) {
         av_free(data->au_headers);
         data->au_headers = av_malloc(sizeof(struct AUHeaders) * data->nb_au_headers);
         data->au_headers_allocated = data->nb_au_headers;
73e6c53e
     }
 
     /* XXX: We handle multiple AU Section as only one (need to fix this for interleaving)
        In my test, the FAAD decoder does not behave correctly when sending each AU one by one
        but does when sending the whole as one big packet...  */
c47f567c
     data->au_headers[0].size = 0;
     data->au_headers[0].index = 0;
     for (i = 0; i < data->nb_au_headers; ++i) {
         data->au_headers[0].size += get_bits_long(&getbitcontext, data->sizelength);
         data->au_headers[0].index = get_bits_long(&getbitcontext, data->indexlength);
73e6c53e
     }
 
c47f567c
     data->nb_au_headers = 1;
73e6c53e
 
     return 0;
 }
 
 
 /* Follows RFC 3640 */
 static int aac_parse_packet(AVFormatContext *ctx,
c47f567c
                             PayloadContext *data,
73e6c53e
                             AVStream *st,
                             AVPacket *pkt,
                             uint32_t *timestamp,
                             const uint8_t *buf, int len, int flags)
 {
c47f567c
     if (rtp_parse_mp4_au(data, buf))
73e6c53e
         return -1;
 
c47f567c
     buf += data->au_headers_length_bytes + 2;
     len -= data->au_headers_length_bytes + 2;
73e6c53e
 
     /* XXX: Fixme we only handle the case where rtp_parse_mp4_au define
                     one au_header */
c47f567c
     av_new_packet(pkt, data->au_headers[0].size);
     memcpy(pkt->data, buf, data->au_headers[0].size);
73e6c53e
 
     pkt->stream_index = st->index;
     return 0;
 }
 
efc6d451
 static int parse_fmtp(AVStream *stream, PayloadContext *data,
                       char *attr, char *value)
9b3788ef
 {
efc6d451
     AVCodecContext *codec = stream->codec;
     int res, i;
 
4332bfbf
     if (!strcmp(attr, "config")) {
         res = parse_fmtp_config(codec, value);
9b3788ef
 
4332bfbf
         if (res < 0)
             return res;
     }
7fc8ac7f
 
4332bfbf
     if (codec->codec_id == CODEC_ID_AAC) {
         /* Looking for a known attribute */
         for (i = 0; attr_names[i].str; ++i) {
96949daf
             if (!av_strcasecmp(attr, attr_names[i].str)) {
4332bfbf
                 if (attr_names[i].type == ATTR_NAME_TYPE_INT) {
                     *(int *)((char *)data+
                         attr_names[i].offset) = atoi(value);
                 } else if (attr_names[i].type == ATTR_NAME_TYPE_STR)
                     *(char **)((char *)data+
                         attr_names[i].offset) = av_strdup(value);
7fc8ac7f
             }
9b3788ef
         }
4332bfbf
     }
     return 0;
 }
9b3788ef
 
efc6d451
 static int parse_sdp_line(AVFormatContext *s, int st_index,
                           PayloadContext *data, const char *line)
 {
     const char *p;
 
45600148
     if (st_index < 0)
         return 0;
 
efc6d451
     if (av_strstart(line, "fmtp:", &p))
         return ff_parse_fmtp(s->streams[st_index], data, p, parse_fmtp);
9b3788ef
 
efc6d451
     return 0;
9b3788ef
 }
 
 RTPDynamicProtocolHandler ff_mp4v_es_dynamic_handler = {
     .enc_name           = "MP4V-ES",
     .codec_type         = AVMEDIA_TYPE_VIDEO,
     .codec_id           = CODEC_ID_MPEG4,
     .parse_sdp_a_line   = parse_sdp_line,
 };
 
 RTPDynamicProtocolHandler ff_mpeg4_generic_dynamic_handler = {
     .enc_name           = "mpeg4-generic",
     .codec_type         = AVMEDIA_TYPE_AUDIO,
     .codec_id           = CODEC_ID_AAC,
     .parse_sdp_a_line   = parse_sdp_line,
9261e6cf
     .alloc              = new_context,
     .free               = free_context,
73e6c53e
     .parse_packet       = aac_parse_packet
9b3788ef
 };