1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Common code for the RTP depacketization of MPEG-4 formats. 3cabdff1aSopenharmony_ci * Copyright (c) 2010 Fabrice Bellard 4cabdff1aSopenharmony_ci * Romain Degez 5cabdff1aSopenharmony_ci * 6cabdff1aSopenharmony_ci * This file is part of FFmpeg. 7cabdff1aSopenharmony_ci * 8cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 9cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 10cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 11cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 12cabdff1aSopenharmony_ci * 13cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 14cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 15cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16cabdff1aSopenharmony_ci * Lesser General Public License for more details. 17cabdff1aSopenharmony_ci * 18cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 19cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 20cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 21cabdff1aSopenharmony_ci */ 22cabdff1aSopenharmony_ci 23cabdff1aSopenharmony_ci/** 24cabdff1aSopenharmony_ci * @file 25cabdff1aSopenharmony_ci * @brief MPEG-4 / RTP Code 26cabdff1aSopenharmony_ci * @author Fabrice Bellard 27cabdff1aSopenharmony_ci * @author Romain Degez 28cabdff1aSopenharmony_ci */ 29cabdff1aSopenharmony_ci 30cabdff1aSopenharmony_ci#include "rtpdec_formats.h" 31cabdff1aSopenharmony_ci#include "internal.h" 32cabdff1aSopenharmony_ci#include "libavutil/attributes.h" 33cabdff1aSopenharmony_ci#include "libavutil/avstring.h" 34cabdff1aSopenharmony_ci#include "libavcodec/get_bits.h" 35cabdff1aSopenharmony_ci 36cabdff1aSopenharmony_ci#define MAX_AAC_HBR_FRAME_SIZE 8191 37cabdff1aSopenharmony_ci 38cabdff1aSopenharmony_ci/** Structure listing useful vars to parse RTP packet payload */ 39cabdff1aSopenharmony_cistruct PayloadContext { 40cabdff1aSopenharmony_ci int sizelength; 41cabdff1aSopenharmony_ci int indexlength; 42cabdff1aSopenharmony_ci int indexdeltalength; 43cabdff1aSopenharmony_ci int profile_level_id; 44cabdff1aSopenharmony_ci int streamtype; 45cabdff1aSopenharmony_ci int objecttype; 46cabdff1aSopenharmony_ci char *mode; 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci /** mpeg 4 AU headers */ 49cabdff1aSopenharmony_ci struct AUHeaders { 50cabdff1aSopenharmony_ci int size; 51cabdff1aSopenharmony_ci int index; 52cabdff1aSopenharmony_ci int cts_flag; 53cabdff1aSopenharmony_ci int cts; 54cabdff1aSopenharmony_ci int dts_flag; 55cabdff1aSopenharmony_ci int dts; 56cabdff1aSopenharmony_ci int rap_flag; 57cabdff1aSopenharmony_ci int streamstate; 58cabdff1aSopenharmony_ci } *au_headers; 59cabdff1aSopenharmony_ci int au_headers_allocated; 60cabdff1aSopenharmony_ci int nb_au_headers; 61cabdff1aSopenharmony_ci int au_headers_length_bytes; 62cabdff1aSopenharmony_ci int cur_au_index; 63cabdff1aSopenharmony_ci 64cabdff1aSopenharmony_ci uint8_t buf[FFMAX(RTP_MAX_PACKET_LENGTH, MAX_AAC_HBR_FRAME_SIZE)]; 65cabdff1aSopenharmony_ci int buf_pos, buf_size; 66cabdff1aSopenharmony_ci uint32_t timestamp; 67cabdff1aSopenharmony_ci}; 68cabdff1aSopenharmony_ci 69cabdff1aSopenharmony_citypedef struct AttrNameMap { 70cabdff1aSopenharmony_ci const char *str; 71cabdff1aSopenharmony_ci uint16_t type; 72cabdff1aSopenharmony_ci uint32_t offset; 73cabdff1aSopenharmony_ci 74cabdff1aSopenharmony_ci /** Range for integer values */ 75cabdff1aSopenharmony_ci struct Range { 76cabdff1aSopenharmony_ci int min; 77cabdff1aSopenharmony_ci int max; 78cabdff1aSopenharmony_ci } range; 79cabdff1aSopenharmony_ci} AttrNameMap; 80cabdff1aSopenharmony_ci 81cabdff1aSopenharmony_ci/* All known fmtp parameters and the corresponding RTPAttrTypeEnum */ 82cabdff1aSopenharmony_ci#define ATTR_NAME_TYPE_INT 0 83cabdff1aSopenharmony_ci#define ATTR_NAME_TYPE_STR 1 84cabdff1aSopenharmony_cistatic const AttrNameMap attr_names[] = { 85cabdff1aSopenharmony_ci { "SizeLength", ATTR_NAME_TYPE_INT, 86cabdff1aSopenharmony_ci offsetof(PayloadContext, sizelength), 87cabdff1aSopenharmony_ci {0, 32} }, // SizeLength number of bits used to encode AU-size integer value 88cabdff1aSopenharmony_ci { "IndexLength", ATTR_NAME_TYPE_INT, 89cabdff1aSopenharmony_ci offsetof(PayloadContext, indexlength), 90cabdff1aSopenharmony_ci {0, 32} }, // IndexLength number of bits used to encode AU-Index integer value 91cabdff1aSopenharmony_ci { "IndexDeltaLength", ATTR_NAME_TYPE_INT, 92cabdff1aSopenharmony_ci offsetof(PayloadContext, indexdeltalength), 93cabdff1aSopenharmony_ci {0, 32} }, // IndexDeltaLength number of bits to encode AU-Index-delta integer value 94cabdff1aSopenharmony_ci { "profile-level-id", ATTR_NAME_TYPE_INT, 95cabdff1aSopenharmony_ci offsetof(PayloadContext, profile_level_id), 96cabdff1aSopenharmony_ci {INT32_MIN, INT32_MAX} }, // It differs depending on StreamType 97cabdff1aSopenharmony_ci { "StreamType", ATTR_NAME_TYPE_INT, 98cabdff1aSopenharmony_ci offsetof(PayloadContext, streamtype), 99cabdff1aSopenharmony_ci {0x00, 0x3F} }, // Values from ISO/IEC 14496-1, 'StreamType Values' table 100cabdff1aSopenharmony_ci { "mode", ATTR_NAME_TYPE_STR, 101cabdff1aSopenharmony_ci offsetof(PayloadContext, mode), 102cabdff1aSopenharmony_ci {0} }, 103cabdff1aSopenharmony_ci { NULL, -1, -1, {0} }, 104cabdff1aSopenharmony_ci}; 105cabdff1aSopenharmony_ci 106cabdff1aSopenharmony_cistatic void close_context(PayloadContext *data) 107cabdff1aSopenharmony_ci{ 108cabdff1aSopenharmony_ci av_freep(&data->au_headers); 109cabdff1aSopenharmony_ci av_freep(&data->mode); 110cabdff1aSopenharmony_ci} 111cabdff1aSopenharmony_ci 112cabdff1aSopenharmony_cistatic int parse_fmtp_config(AVCodecParameters *par, const char *value) 113cabdff1aSopenharmony_ci{ 114cabdff1aSopenharmony_ci /* decode the hexa encoded parameter */ 115cabdff1aSopenharmony_ci int len = ff_hex_to_data(NULL, value), ret; 116cabdff1aSopenharmony_ci 117cabdff1aSopenharmony_ci if ((ret = ff_alloc_extradata(par, len)) < 0) 118cabdff1aSopenharmony_ci return ret; 119cabdff1aSopenharmony_ci ff_hex_to_data(par->extradata, value); 120cabdff1aSopenharmony_ci return 0; 121cabdff1aSopenharmony_ci} 122cabdff1aSopenharmony_ci 123cabdff1aSopenharmony_cistatic int rtp_parse_mp4_au(PayloadContext *data, const uint8_t *buf, int len) 124cabdff1aSopenharmony_ci{ 125cabdff1aSopenharmony_ci int au_headers_length, au_header_size, i; 126cabdff1aSopenharmony_ci GetBitContext getbitcontext; 127cabdff1aSopenharmony_ci int ret; 128cabdff1aSopenharmony_ci 129cabdff1aSopenharmony_ci if (len < 2) 130cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 131cabdff1aSopenharmony_ci 132cabdff1aSopenharmony_ci /* decode the first 2 bytes where the AUHeader sections are stored 133cabdff1aSopenharmony_ci length in bits */ 134cabdff1aSopenharmony_ci au_headers_length = AV_RB16(buf); 135cabdff1aSopenharmony_ci 136cabdff1aSopenharmony_ci if (au_headers_length > RTP_MAX_PACKET_LENGTH) 137cabdff1aSopenharmony_ci return -1; 138cabdff1aSopenharmony_ci 139cabdff1aSopenharmony_ci data->au_headers_length_bytes = (au_headers_length + 7) / 8; 140cabdff1aSopenharmony_ci 141cabdff1aSopenharmony_ci /* skip AU headers length section (2 bytes) */ 142cabdff1aSopenharmony_ci buf += 2; 143cabdff1aSopenharmony_ci len -= 2; 144cabdff1aSopenharmony_ci 145cabdff1aSopenharmony_ci if (len < data->au_headers_length_bytes) 146cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 147cabdff1aSopenharmony_ci 148cabdff1aSopenharmony_ci ret = init_get_bits(&getbitcontext, buf, data->au_headers_length_bytes * 8); 149cabdff1aSopenharmony_ci if (ret < 0) 150cabdff1aSopenharmony_ci return ret; 151cabdff1aSopenharmony_ci 152cabdff1aSopenharmony_ci /* XXX: Wrong if optional additional sections are present (cts, dts etc...) */ 153cabdff1aSopenharmony_ci au_header_size = data->sizelength + data->indexlength; 154cabdff1aSopenharmony_ci if (au_header_size <= 0 || (au_headers_length % au_header_size != 0)) 155cabdff1aSopenharmony_ci return -1; 156cabdff1aSopenharmony_ci 157cabdff1aSopenharmony_ci data->nb_au_headers = au_headers_length / au_header_size; 158cabdff1aSopenharmony_ci if (!data->au_headers || data->au_headers_allocated < data->nb_au_headers) { 159cabdff1aSopenharmony_ci av_free(data->au_headers); 160cabdff1aSopenharmony_ci data->au_headers = av_malloc(sizeof(struct AUHeaders) * data->nb_au_headers); 161cabdff1aSopenharmony_ci if (!data->au_headers) 162cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 163cabdff1aSopenharmony_ci data->au_headers_allocated = data->nb_au_headers; 164cabdff1aSopenharmony_ci } 165cabdff1aSopenharmony_ci 166cabdff1aSopenharmony_ci for (i = 0; i < data->nb_au_headers; ++i) { 167cabdff1aSopenharmony_ci data->au_headers[i].size = get_bits_long(&getbitcontext, data->sizelength); 168cabdff1aSopenharmony_ci data->au_headers[i].index = get_bits_long(&getbitcontext, data->indexlength); 169cabdff1aSopenharmony_ci } 170cabdff1aSopenharmony_ci 171cabdff1aSopenharmony_ci return 0; 172cabdff1aSopenharmony_ci} 173cabdff1aSopenharmony_ci 174cabdff1aSopenharmony_ci 175cabdff1aSopenharmony_ci/* Follows RFC 3640 */ 176cabdff1aSopenharmony_cistatic int aac_parse_packet(AVFormatContext *ctx, PayloadContext *data, 177cabdff1aSopenharmony_ci AVStream *st, AVPacket *pkt, uint32_t *timestamp, 178cabdff1aSopenharmony_ci const uint8_t *buf, int len, uint16_t seq, 179cabdff1aSopenharmony_ci int flags) 180cabdff1aSopenharmony_ci{ 181cabdff1aSopenharmony_ci int ret; 182cabdff1aSopenharmony_ci 183cabdff1aSopenharmony_ci 184cabdff1aSopenharmony_ci if (!buf) { 185cabdff1aSopenharmony_ci if (data->cur_au_index > data->nb_au_headers) { 186cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Invalid parser state\n"); 187cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 188cabdff1aSopenharmony_ci } 189cabdff1aSopenharmony_ci if (data->buf_size - data->buf_pos < data->au_headers[data->cur_au_index].size) { 190cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Invalid AU size\n"); 191cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 192cabdff1aSopenharmony_ci } 193cabdff1aSopenharmony_ci if ((ret = av_new_packet(pkt, data->au_headers[data->cur_au_index].size)) < 0) { 194cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Out of memory\n"); 195cabdff1aSopenharmony_ci return ret; 196cabdff1aSopenharmony_ci } 197cabdff1aSopenharmony_ci memcpy(pkt->data, &data->buf[data->buf_pos], data->au_headers[data->cur_au_index].size); 198cabdff1aSopenharmony_ci data->buf_pos += data->au_headers[data->cur_au_index].size; 199cabdff1aSopenharmony_ci pkt->stream_index = st->index; 200cabdff1aSopenharmony_ci data->cur_au_index++; 201cabdff1aSopenharmony_ci 202cabdff1aSopenharmony_ci if (data->cur_au_index == data->nb_au_headers) { 203cabdff1aSopenharmony_ci data->buf_pos = 0; 204cabdff1aSopenharmony_ci return 0; 205cabdff1aSopenharmony_ci } 206cabdff1aSopenharmony_ci 207cabdff1aSopenharmony_ci return 1; 208cabdff1aSopenharmony_ci } 209cabdff1aSopenharmony_ci 210cabdff1aSopenharmony_ci if (rtp_parse_mp4_au(data, buf, len)) { 211cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Error parsing AU headers\n"); 212cabdff1aSopenharmony_ci return -1; 213cabdff1aSopenharmony_ci } 214cabdff1aSopenharmony_ci 215cabdff1aSopenharmony_ci buf += data->au_headers_length_bytes + 2; 216cabdff1aSopenharmony_ci len -= data->au_headers_length_bytes + 2; 217cabdff1aSopenharmony_ci if (data->nb_au_headers == 1 && len < data->au_headers[0].size) { 218cabdff1aSopenharmony_ci /* Packet is fragmented */ 219cabdff1aSopenharmony_ci 220cabdff1aSopenharmony_ci if (!data->buf_pos) { 221cabdff1aSopenharmony_ci if (data->au_headers[0].size > MAX_AAC_HBR_FRAME_SIZE) { 222cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Invalid AU size\n"); 223cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 224cabdff1aSopenharmony_ci } 225cabdff1aSopenharmony_ci 226cabdff1aSopenharmony_ci data->buf_size = data->au_headers[0].size; 227cabdff1aSopenharmony_ci data->timestamp = *timestamp; 228cabdff1aSopenharmony_ci } 229cabdff1aSopenharmony_ci 230cabdff1aSopenharmony_ci if (data->timestamp != *timestamp || 231cabdff1aSopenharmony_ci data->au_headers[0].size != data->buf_size || 232cabdff1aSopenharmony_ci data->buf_pos + len > MAX_AAC_HBR_FRAME_SIZE) { 233cabdff1aSopenharmony_ci data->buf_pos = 0; 234cabdff1aSopenharmony_ci data->buf_size = 0; 235cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Invalid packet received\n"); 236cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 237cabdff1aSopenharmony_ci } 238cabdff1aSopenharmony_ci 239cabdff1aSopenharmony_ci memcpy(&data->buf[data->buf_pos], buf, len); 240cabdff1aSopenharmony_ci data->buf_pos += len; 241cabdff1aSopenharmony_ci 242cabdff1aSopenharmony_ci if (!(flags & RTP_FLAG_MARKER)) 243cabdff1aSopenharmony_ci return AVERROR(EAGAIN); 244cabdff1aSopenharmony_ci 245cabdff1aSopenharmony_ci if (data->buf_pos != data->buf_size) { 246cabdff1aSopenharmony_ci data->buf_pos = 0; 247cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Missed some packets, discarding frame\n"); 248cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 249cabdff1aSopenharmony_ci } 250cabdff1aSopenharmony_ci 251cabdff1aSopenharmony_ci data->buf_pos = 0; 252cabdff1aSopenharmony_ci ret = av_new_packet(pkt, data->buf_size); 253cabdff1aSopenharmony_ci if (ret < 0) { 254cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Out of memory\n"); 255cabdff1aSopenharmony_ci return ret; 256cabdff1aSopenharmony_ci } 257cabdff1aSopenharmony_ci pkt->stream_index = st->index; 258cabdff1aSopenharmony_ci 259cabdff1aSopenharmony_ci memcpy(pkt->data, data->buf, data->buf_size); 260cabdff1aSopenharmony_ci 261cabdff1aSopenharmony_ci return 0; 262cabdff1aSopenharmony_ci } 263cabdff1aSopenharmony_ci 264cabdff1aSopenharmony_ci if (len < data->au_headers[0].size) { 265cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "First AU larger than packet size\n"); 266cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 267cabdff1aSopenharmony_ci } 268cabdff1aSopenharmony_ci if ((ret = av_new_packet(pkt, data->au_headers[0].size)) < 0) { 269cabdff1aSopenharmony_ci av_log(ctx, AV_LOG_ERROR, "Out of memory\n"); 270cabdff1aSopenharmony_ci return ret; 271cabdff1aSopenharmony_ci } 272cabdff1aSopenharmony_ci memcpy(pkt->data, buf, data->au_headers[0].size); 273cabdff1aSopenharmony_ci len -= data->au_headers[0].size; 274cabdff1aSopenharmony_ci buf += data->au_headers[0].size; 275cabdff1aSopenharmony_ci pkt->stream_index = st->index; 276cabdff1aSopenharmony_ci 277cabdff1aSopenharmony_ci if (len > 0 && data->nb_au_headers > 1) { 278cabdff1aSopenharmony_ci data->buf_size = FFMIN(len, sizeof(data->buf)); 279cabdff1aSopenharmony_ci memcpy(data->buf, buf, data->buf_size); 280cabdff1aSopenharmony_ci data->cur_au_index = 1; 281cabdff1aSopenharmony_ci data->buf_pos = 0; 282cabdff1aSopenharmony_ci return 1; 283cabdff1aSopenharmony_ci } 284cabdff1aSopenharmony_ci 285cabdff1aSopenharmony_ci return 0; 286cabdff1aSopenharmony_ci} 287cabdff1aSopenharmony_ci 288cabdff1aSopenharmony_cistatic int parse_fmtp(AVFormatContext *s, 289cabdff1aSopenharmony_ci AVStream *stream, PayloadContext *data, 290cabdff1aSopenharmony_ci const char *attr, const char *value) 291cabdff1aSopenharmony_ci{ 292cabdff1aSopenharmony_ci AVCodecParameters *par = stream->codecpar; 293cabdff1aSopenharmony_ci int res, i; 294cabdff1aSopenharmony_ci 295cabdff1aSopenharmony_ci if (!strcmp(attr, "config")) { 296cabdff1aSopenharmony_ci res = parse_fmtp_config(par, value); 297cabdff1aSopenharmony_ci 298cabdff1aSopenharmony_ci if (res < 0) 299cabdff1aSopenharmony_ci return res; 300cabdff1aSopenharmony_ci } 301cabdff1aSopenharmony_ci 302cabdff1aSopenharmony_ci if (par->codec_id == AV_CODEC_ID_AAC) { 303cabdff1aSopenharmony_ci /* Looking for a known attribute */ 304cabdff1aSopenharmony_ci for (i = 0; attr_names[i].str; ++i) { 305cabdff1aSopenharmony_ci if (!av_strcasecmp(attr, attr_names[i].str)) { 306cabdff1aSopenharmony_ci if (attr_names[i].type == ATTR_NAME_TYPE_INT) { 307cabdff1aSopenharmony_ci char *end_ptr = NULL; 308cabdff1aSopenharmony_ci long long int val = strtoll(value, &end_ptr, 10); 309cabdff1aSopenharmony_ci if (end_ptr == value || end_ptr[0] != '\0') { 310cabdff1aSopenharmony_ci av_log(s, AV_LOG_ERROR, 311cabdff1aSopenharmony_ci "The %s field value is not a valid number: %s\n", 312cabdff1aSopenharmony_ci attr, value); 313cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 314cabdff1aSopenharmony_ci } 315cabdff1aSopenharmony_ci if (val < attr_names[i].range.min || 316cabdff1aSopenharmony_ci val > attr_names[i].range.max) { 317cabdff1aSopenharmony_ci av_log(s, AV_LOG_ERROR, 318cabdff1aSopenharmony_ci "fmtp field %s should be in range [%d,%d] (provided value: %lld)", 319cabdff1aSopenharmony_ci attr, attr_names[i].range.min, attr_names[i].range.max, val); 320cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 321cabdff1aSopenharmony_ci } 322cabdff1aSopenharmony_ci 323cabdff1aSopenharmony_ci *(int *)((char *)data+ 324cabdff1aSopenharmony_ci attr_names[i].offset) = (int) val; 325cabdff1aSopenharmony_ci } else if (attr_names[i].type == ATTR_NAME_TYPE_STR) { 326cabdff1aSopenharmony_ci char *val = av_strdup(value); 327cabdff1aSopenharmony_ci if (!val) 328cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 329cabdff1aSopenharmony_ci *(char **)((char *)data+ 330cabdff1aSopenharmony_ci attr_names[i].offset) = val; 331cabdff1aSopenharmony_ci } 332cabdff1aSopenharmony_ci } 333cabdff1aSopenharmony_ci } 334cabdff1aSopenharmony_ci } 335cabdff1aSopenharmony_ci return 0; 336cabdff1aSopenharmony_ci} 337cabdff1aSopenharmony_ci 338cabdff1aSopenharmony_cistatic int parse_sdp_line(AVFormatContext *s, int st_index, 339cabdff1aSopenharmony_ci PayloadContext *data, const char *line) 340cabdff1aSopenharmony_ci{ 341cabdff1aSopenharmony_ci const char *p; 342cabdff1aSopenharmony_ci 343cabdff1aSopenharmony_ci if (st_index < 0) 344cabdff1aSopenharmony_ci return 0; 345cabdff1aSopenharmony_ci 346cabdff1aSopenharmony_ci if (av_strstart(line, "fmtp:", &p)) 347cabdff1aSopenharmony_ci return ff_parse_fmtp(s, s->streams[st_index], data, p, parse_fmtp); 348cabdff1aSopenharmony_ci 349cabdff1aSopenharmony_ci return 0; 350cabdff1aSopenharmony_ci} 351cabdff1aSopenharmony_ci 352cabdff1aSopenharmony_ciconst RTPDynamicProtocolHandler ff_mp4v_es_dynamic_handler = { 353cabdff1aSopenharmony_ci .enc_name = "MP4V-ES", 354cabdff1aSopenharmony_ci .codec_type = AVMEDIA_TYPE_VIDEO, 355cabdff1aSopenharmony_ci .codec_id = AV_CODEC_ID_MPEG4, 356cabdff1aSopenharmony_ci .need_parsing = AVSTREAM_PARSE_FULL, 357cabdff1aSopenharmony_ci .priv_data_size = sizeof(PayloadContext), 358cabdff1aSopenharmony_ci .parse_sdp_a_line = parse_sdp_line, 359cabdff1aSopenharmony_ci}; 360cabdff1aSopenharmony_ci 361cabdff1aSopenharmony_ciconst RTPDynamicProtocolHandler ff_mpeg4_generic_dynamic_handler = { 362cabdff1aSopenharmony_ci .enc_name = "mpeg4-generic", 363cabdff1aSopenharmony_ci .codec_type = AVMEDIA_TYPE_AUDIO, 364cabdff1aSopenharmony_ci .codec_id = AV_CODEC_ID_AAC, 365cabdff1aSopenharmony_ci .priv_data_size = sizeof(PayloadContext), 366cabdff1aSopenharmony_ci .parse_sdp_a_line = parse_sdp_line, 367cabdff1aSopenharmony_ci .close = close_context, 368cabdff1aSopenharmony_ci .parse_packet = aac_parse_packet, 369cabdff1aSopenharmony_ci}; 370