libavformat/swfdec.c
3b35f4ab
 /*
376aefdd
  * Flash Compatible Streaming Format demuxer
406792e7
  * Copyright (c) 2000 Fabrice Bellard
  * Copyright (c) 2003 Tinic Uro
3b35f4ab
  *
  * 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
  */
 
6a5d31ac
 #include "libavutil/intreadwrite.h"
3b35f4ab
 #include "swf.h"
 
471fe57e
 static int get_swf_tag(AVIOContext *pb, int *len_ptr)
3b35f4ab
 {
     int tag, len;
 
     if (url_feof(pb))
         return -1;
 
e63a3628
     tag = avio_rl16(pb);
3b35f4ab
     len = tag & 0x3f;
     tag = tag >> 6;
     if (len == 0x3f) {
e63a3628
         len = avio_rl32(pb);
3b35f4ab
     }
 //    av_log(NULL, AV_LOG_DEBUG, "Tag: %d - Len: %d\n", tag, len);
     *len_ptr = len;
     return tag;
 }
 
 
 static int swf_probe(AVProbeData *p)
 {
     /* check file header */
     if ((p->buf[0] == 'F' || p->buf[0] == 'C') && p->buf[1] == 'W' &&
         p->buf[2] == 'S')
         return AVPROBE_SCORE_MAX;
     else
         return 0;
 }
 
 static int swf_read_header(AVFormatContext *s, AVFormatParameters *ap)
 {
     SWFContext *swf = s->priv_data;
471fe57e
     AVIOContext *pb = s->pb;
3b35f4ab
     int nbits, len, tag;
 
e63a3628
     tag = avio_rb32(pb) & 0xffffff00;
3b35f4ab
 
     if (tag == MKBETAG('C', 'W', 'S', 0)) {
         av_log(s, AV_LOG_ERROR, "Compressed SWF format not supported\n");
         return AVERROR(EIO);
     }
     if (tag != MKBETAG('F', 'W', 'S', 0))
         return AVERROR(EIO);
e63a3628
     avio_rl32(pb);
3b35f4ab
     /* skip rectangle size */
e63a3628
     nbits = avio_r8(pb) >> 3;
3b35f4ab
     len = (4 * nbits - 3 + 7) / 8;
45a8a02a
     avio_skip(pb, len);
e63a3628
     swf->frame_rate = avio_rl16(pb); /* 8.8 fixed */
     avio_rl16(pb); /* frame count */
3b35f4ab
 
     swf->samples_per_frame = 0;
     s->ctx_flags |= AVFMTCTX_NOHEADER;
     return 0;
 }
 
 static int swf_read_packet(AVFormatContext *s, AVPacket *pkt)
 {
     SWFContext *swf = s->priv_data;
471fe57e
     AVIOContext *pb = s->pb;
3b35f4ab
     AVStream *vst = NULL, *ast = NULL, *st = 0;
     int tag, len, i, frame, v;
 
     for(;;) {
384c9c2f
         uint64_t pos = avio_tell(pb);
3b35f4ab
         tag = get_swf_tag(pb, &len);
         if (tag < 0)
             return AVERROR(EIO);
4e35ffa9
         if (tag == TAG_VIDEOSTREAM) {
e63a3628
             int ch_id = avio_rl16(pb);
4e35ffa9
             len -= 2;
 
             for (i=0; i<s->nb_streams; i++) {
                 st = s->streams[i];
72415b2a
                 if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO && st->id == ch_id)
4e35ffa9
                     goto skip;
             }
 
e63a3628
             avio_rl16(pb);
             avio_rl16(pb);
             avio_rl16(pb);
             avio_r8(pb);
3b35f4ab
             /* Check for FLV1 */
             vst = av_new_stream(s, ch_id);
             if (!vst)
                 return -1;
72415b2a
             vst->codec->codec_type = AVMEDIA_TYPE_VIDEO;
e63a3628
             vst->codec->codec_id = ff_codec_get_id(swf_codec_tags, avio_r8(pb));
36a12218
             av_set_pts_info(vst, 16, 256, swf->frame_rate);
3b35f4ab
             vst->codec->time_base = (AVRational){ 256, swf->frame_rate };
4e35ffa9
             len -= 8;
         } else if (tag == TAG_STREAMHEAD || tag == TAG_STREAMHEAD2) {
3b35f4ab
             /* streaming found */
             int sample_rate_code;
4e35ffa9
 
             for (i=0; i<s->nb_streams; i++) {
                 st = s->streams[i];
72415b2a
                 if (st->codec->codec_type == AVMEDIA_TYPE_AUDIO && st->id == -1)
4e35ffa9
                     goto skip;
             }
 
e63a3628
             avio_r8(pb);
             v = avio_r8(pb);
             swf->samples_per_frame = avio_rl16(pb);
3b35f4ab
             ast = av_new_stream(s, -1); /* -1 to avoid clash with video stream ch_id */
             if (!ast)
                 return -1;
             ast->codec->channels = 1 + (v&1);
72415b2a
             ast->codec->codec_type = AVMEDIA_TYPE_AUDIO;
1a40491e
             ast->codec->codec_id = ff_codec_get_id(swf_audio_codec_tags, (v>>4) & 15);
3b35f4ab
             ast->need_parsing = AVSTREAM_PARSE_FULL;
             sample_rate_code= (v>>2) & 3;
             if (!sample_rate_code)
                 return AVERROR(EIO);
             ast->codec->sample_rate = 11025 << (sample_rate_code-1);
             av_set_pts_info(ast, 64, 1, ast->codec->sample_rate);
             len -= 4;
         } else if (tag == TAG_VIDEOFRAME) {
e63a3628
             int ch_id = avio_rl16(pb);
3b35f4ab
             len -= 2;
             for(i=0; i<s->nb_streams; i++) {
                 st = s->streams[i];
72415b2a
                 if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO && st->id == ch_id) {
e63a3628
                     frame = avio_rl16(pb);
3b35f4ab
                     av_get_packet(pb, pkt, len-2);
7e04cfba
                     pkt->pos = pos;
3b35f4ab
                     pkt->pts = frame;
                     pkt->stream_index = st->index;
                     return pkt->size;
                 }
             }
         } else if (tag == TAG_STREAMBLOCK) {
4e35ffa9
             for (i = 0; i < s->nb_streams; i++) {
                 st = s->streams[i];
72415b2a
                 if (st->codec->codec_type == AVMEDIA_TYPE_AUDIO && st->id == -1) {
3b35f4ab
             if (st->codec->codec_id == CODEC_ID_MP3) {
45a8a02a
                 avio_skip(pb, 4);
3b35f4ab
                 av_get_packet(pb, pkt, len-4);
             } else { // ADPCM, PCM
                 av_get_packet(pb, pkt, len);
             }
7e04cfba
             pkt->pos = pos;
3b35f4ab
             pkt->stream_index = st->index;
             return pkt->size;
4e35ffa9
                 }
             }
3b35f4ab
         } else if (tag == TAG_JPEG2) {
             for (i=0; i<s->nb_streams; i++) {
                 st = s->streams[i];
4e35ffa9
                 if (st->codec->codec_id == CODEC_ID_MJPEG && st->id == -2)
3b35f4ab
                     break;
             }
             if (i == s->nb_streams) {
                 vst = av_new_stream(s, -2); /* -2 to avoid clash with video stream and audio stream */
                 if (!vst)
                     return -1;
72415b2a
                 vst->codec->codec_type = AVMEDIA_TYPE_VIDEO;
3b35f4ab
                 vst->codec->codec_id = CODEC_ID_MJPEG;
                 av_set_pts_info(vst, 64, 256, swf->frame_rate);
                 vst->codec->time_base = (AVRational){ 256, swf->frame_rate };
                 st = vst;
             }
e63a3628
             avio_rl16(pb); /* BITMAP_ID */
3b35f4ab
             av_new_packet(pkt, len-2);
e63a3628
             avio_read(pb, pkt->data, 4);
3b35f4ab
             if (AV_RB32(pkt->data) == 0xffd8ffd9 ||
                 AV_RB32(pkt->data) == 0xffd9ffd8) {
                 /* old SWF files containing SOI/EOI as data start */
                 /* files created by swink have reversed tag */
                 pkt->size -= 4;
e63a3628
                 avio_read(pb, pkt->data, pkt->size);
3b35f4ab
             } else {
e63a3628
                 avio_read(pb, pkt->data + 4, pkt->size - 4);
3b35f4ab
             }
7e04cfba
             pkt->pos = pos;
3b35f4ab
             pkt->stream_index = st->index;
             return pkt->size;
         }
4e35ffa9
     skip:
45a8a02a
         avio_skip(pb, len);
3b35f4ab
     }
     return 0;
 }
 
66355be3
 AVInputFormat ff_swf_demuxer = {
3b35f4ab
     "swf",
bde15e74
     NULL_IF_CONFIG_SMALL("Flash format"),
3b35f4ab
     sizeof(SWFContext),
     swf_probe,
     swf_read_header,
     swf_read_packet,
 };