diff options
author | Luca Abeni <lucabe72@email.it> | 2008-01-21 14:28:58 +0000 |
---|---|---|
committer | Luca Abeni <lucabe72@email.it> | 2008-01-21 14:28:58 +0000 |
commit | 29e35d67766edf67ba470add620e3c3e879e8195 (patch) | |
tree | 3b2d5877da52f35d40b07589986fdee90ccbbf94 /libavformat/sdp.c | |
parent | f79bfe481d48cd9a24f2241bd25e59897f878ab8 (diff) | |
download | ffmpeg-29e35d67766edf67ba470add620e3c3e879e8195.tar.gz |
Support out-of-band parameter sets in SDP for H.264 video
Originally committed as revision 11590 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavformat/sdp.c')
-rw-r--r-- | libavformat/sdp.c | 48 |
1 files changed, 48 insertions, 0 deletions
diff --git a/libavformat/sdp.c b/libavformat/sdp.c index a9ba2e68de..0f01a7de43 100644 --- a/libavformat/sdp.c +++ b/libavformat/sdp.c @@ -20,6 +20,8 @@ #include "avstring.h" #include "avformat.h" +#include "avc.h" +#include "base64.h" #include "rtp.h" #ifdef CONFIG_RTP_MUXER @@ -90,6 +92,49 @@ static int get_address(char *dest_addr, int size, int *ttl, const char *url) return port; } +#define MAX_PSET_SIZE 1024 +static char *extradata2psets(AVCodecContext *c) +{ + char *psets, *p; + uint8_t *r; + const char *pset_string = "; sprop-parameter-sets="; + + if (c->extradata_size > MAX_EXTRADATA_SIZE) { + av_log(c, AV_LOG_ERROR, "Too many extra data!\n"); + + return NULL; + } + + psets = av_mallocz(MAX_PSET_SIZE); + if (psets == NULL) { + av_log(c, AV_LOG_ERROR, "Cannot allocate memory for the parameter sets\n"); + return NULL; + } + memcpy(psets, pset_string, strlen(pset_string)); + p = psets + strlen(pset_string); + r = ff_avc_find_startcode(c->extradata, c->extradata + c->extradata_size); + while (r < c->extradata + c->extradata_size) { + uint8_t *r1; + + while (!*(r++)); + r1 = ff_avc_find_startcode(r, c->extradata + c->extradata_size); + if (p != (psets + strlen(pset_string))) { + *p = ','; + p++; + } + if (av_base64_encode(p, MAX_PSET_SIZE - (p - psets), r, r1 - r) == NULL) { + av_log(c, AV_LOG_ERROR, "Cannot BASE64 encode %d %d!\n", MAX_PSET_SIZE - (p - psets), r1 - r); + av_free(psets); + + return NULL; + } + p += strlen(p); + r = r1; + } + + return psets; +} + static void digit_to_char(char *dst, uint8_t src) { if (src < 10) { @@ -138,6 +183,9 @@ static char *sdp_media_attributes(char *buff, int size, AVCodecContext *c, int p switch (c->codec_id) { case CODEC_ID_H264: + if (c->extradata_size) { + config = extradata2psets(c); + } av_strlcatf(buff, size, "a=rtpmap:%d H264/90000\r\n" "a=fmtp:%d packetization-mode=1%s\r\n", payload_type, |