Statistics
| Branch: | Revision:

ffmpeg / libavformat / flvdec.c @ 34f633df

History | View | Annotate | Download (5.34 KB)

1 d4f5d74a Garrick Meeker
/*
2
 * FLV encoder.
3
 * Copyright (c) 2003 The FFmpeg Project.
4
 *
5
 * This library is free software; you can redistribute it and/or
6
 * modify it under the terms of the GNU Lesser General Public
7
 * License as published by the Free Software Foundation; either
8
 * version 2 of the License, or (at your option) any later version.
9
 *
10
 * This library is distributed in the hope that it will be useful,
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13
 * Lesser General Public License for more details.
14
 *
15
 * You should have received a copy of the GNU Lesser General Public
16
 * License along with this library; if not, write to the Free Software
17
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18
 */
19
#include "avformat.h"
20
21
unsigned int get_be24(ByteIOContext *s)
22
{
23
    unsigned int val;
24
    val = get_byte(s) << 16;
25
    val |= get_byte(s) << 8;
26
    val |= get_byte(s);
27
    return val;
28
}
29
30
static int flv_probe(AVProbeData *p)
31
{
32
    const uint8_t *d;
33
34
    if (p->buf_size < 6)
35
        return 0;
36
    d = p->buf;
37
    if (d[0] == 'F' && d[1] == 'L' && d[2] == 'V') {
38
        return 50;
39
    }
40
    return 0;
41
}
42
43
static int flv_read_header(AVFormatContext *s,
44
                           AVFormatParameters *ap)
45
{
46
    int offset, flags;
47 bb01a3f0 Michael Niedermayer
    
48
    s->ctx_flags |= AVFMTCTX_NOHEADER; //ok we have a header but theres no fps, codec type, sample_rate, ...
49 d4f5d74a Garrick Meeker
50
    url_fskip(&s->pb, 4);
51
    flags = get_byte(&s->pb);
52
53
    offset = get_be32(&s->pb);
54
    url_fseek(&s->pb, offset, SEEK_SET);
55
56
    return 0;
57
}
58
59
static int flv_read_packet(AVFormatContext *s, AVPacket *pkt)
60
{
61 068f2a22 Michael Niedermayer
    int ret, i, type, size, pts, flags, is_audio;
62 923bd441 Alex Beregszaszi
    AVStream *st = NULL;
63 d4f5d74a Garrick Meeker
    
64 068f2a22 Michael Niedermayer
 for(;;){
65 d4f5d74a Garrick Meeker
    url_fskip(&s->pb, 4); /* size of previous packet */
66
    type = get_byte(&s->pb);
67
    size = get_be24(&s->pb);
68
    pts = get_be24(&s->pb);
69 92a26775 Michael Niedermayer
//    av_log(s, AV_LOG_DEBUG, "type:%d, size:%d, pts:%d\n", type, size, pts);
70 d4f5d74a Garrick Meeker
    if (url_feof(&s->pb))
71 0bd586c5 Mike Melanson
        return AVERROR_IO;
72 d4f5d74a Garrick Meeker
    url_fskip(&s->pb, 4); /* reserved */
73
    flags = 0;
74 068f2a22 Michael Niedermayer
    
75
    if(size == 0)
76
        continue;
77
    
78 d4f5d74a Garrick Meeker
    if (type == 8) {
79 068f2a22 Michael Niedermayer
        is_audio=1;
80 d4f5d74a Garrick Meeker
        flags = get_byte(&s->pb);
81
        size--;
82
    } else if (type == 9) {
83 068f2a22 Michael Niedermayer
        is_audio=0;
84 d4f5d74a Garrick Meeker
        flags = get_byte(&s->pb);
85
        size--;
86
    } else {
87
        /* skip packet */
88 bc874dae Michel Bardiaux
        av_log(s, AV_LOG_ERROR, "skipping flv packet: type %d, size %d, flags %d\n", type, size, flags);
89 d4f5d74a Garrick Meeker
        url_fskip(&s->pb, size);
90 068f2a22 Michael Niedermayer
        continue;
91 d4f5d74a Garrick Meeker
    }
92
93
    /* now find stream */
94
    for(i=0;i<s->nb_streams;i++) {
95
        st = s->streams[i];
96 068f2a22 Michael Niedermayer
        if (st->id == is_audio)
97
            break;
98 d4f5d74a Garrick Meeker
    }
99 068f2a22 Michael Niedermayer
    if(i == s->nb_streams){
100
        st = av_new_stream(s, is_audio);
101
        if (!st)
102
            return AVERROR_NOMEM;
103 9ee91c2f Michael Niedermayer
104
        av_set_pts_info(st, 24, 1, 1000); /* 24 bit pts in ms */
105 c0df9d75 Michael Niedermayer
        st->codec.time_base= (AVRational){1,1000};
106 6ed08157 Michael Niedermayer
    }
107 f3356e9c Michael Niedermayer
//    av_log(NULL, AV_LOG_DEBUG, "%d %X %d \n", is_audio, flags, st->discard);
108
    if(  (st->discard >= AVDISCARD_NONKEY && !((flags >> 4)==1 ||  is_audio))
109
       ||(st->discard >= AVDISCARD_BIDIR  &&  ((flags >> 4)==3 && !is_audio))
110
       || st->discard >= AVDISCARD_ALL
111
       ){
112 b9866ebc Michael Niedermayer
        url_fskip(&s->pb, size);
113
        continue;
114
    }
115 068f2a22 Michael Niedermayer
    break;
116
 }
117
118
    if(is_audio){
119
        if(st->codec.sample_rate == 0){
120
            st->codec.codec_type = CODEC_TYPE_AUDIO;
121
            st->codec.channels = (flags&1)+1;
122
            if((flags >> 4) == 5)
123
                st->codec.sample_rate= 8000;
124
            else
125
                st->codec.sample_rate = (44100<<((flags>>2)&3))>>3;
126
            switch(flags >> 4){/* 0: uncompressed 1: ADPCM 2: mp3 5: Nellymoser 8kHz mono 6: Nellymoser*/
127 923bd441 Alex Beregszaszi
            case 0: if (flags&2) st->codec.codec_id = CODEC_ID_PCM_S16BE;
128
                    else st->codec.codec_id = CODEC_ID_PCM_S8; break;
129 ae26a016 Alex Beregszaszi
            case 1: st->codec.codec_id = CODEC_ID_ADPCM_SWF; break;
130 068f2a22 Michael Niedermayer
            case 2: st->codec.codec_id = CODEC_ID_MP3; break;
131 923bd441 Alex Beregszaszi
            // this is not listed at FLV but at SWF, strange...
132
            case 3: if (flags&2) st->codec.codec_id = CODEC_ID_PCM_S16LE;
133
                    else st->codec.codec_id = CODEC_ID_PCM_S8; break;
134 068f2a22 Michael Niedermayer
            default:
135 ae26a016 Alex Beregszaszi
                    av_log(s, AV_LOG_INFO, "Unsupported audio codec (%x)\n", flags >> 4);
136 068f2a22 Michael Niedermayer
                st->codec.codec_tag= (flags >> 4);
137
            }
138 2f5132e7 Alex Beregszaszi
            st->codec.bits_per_sample = (flags & 2) ? 16 : 8;
139 068f2a22 Michael Niedermayer
        }
140
    }else{
141
            st->codec.codec_type = CODEC_TYPE_VIDEO;
142
            switch(flags & 0xF){
143
            case 2: st->codec.codec_id = CODEC_ID_FLV1; break;
144
            default:
145 ae26a016 Alex Beregszaszi
                    av_log(s, AV_LOG_INFO, "Unsupported video codec (%x)\n", flags & 0xf);
146 068f2a22 Michael Niedermayer
                st->codec.codec_tag= flags & 0xF;
147
            }
148 bb01a3f0 Michael Niedermayer
    }
149
150 d4f5d74a Garrick Meeker
    if (av_new_packet(pkt, size) < 0)
151 0bd586c5 Mike Melanson
        return AVERROR_IO;
152 d4f5d74a Garrick Meeker
153
    ret = get_buffer(&s->pb, pkt->data, size);
154
    if (ret <= 0) {
155
        av_free_packet(pkt);
156 0bd586c5 Mike Melanson
        return AVERROR_IO;
157 d4f5d74a Garrick Meeker
    }
158
    /* note: we need to modify the packet size here to handle the last
159
       packet */
160
    pkt->size = ret;
161
    pkt->pts = pts;
162
    pkt->stream_index = st->index;
163 d08f4bcb Alex Beregszaszi
    
164 f3356e9c Michael Niedermayer
    if (is_audio || ((flags >> 4)==1))
165 d08f4bcb Alex Beregszaszi
        pkt->flags |= PKT_FLAG_KEY;
166
    
167 d4f5d74a Garrick Meeker
    return ret;
168
}
169
170
static int flv_read_close(AVFormatContext *s)
171
{
172
    return 0;
173
}
174
175
AVInputFormat flv_iformat = {
176
    "flv",
177
    "flv format",
178
    0,
179
    flv_probe,
180
    flv_read_header,
181
    flv_read_packet,
182
    flv_read_close,
183
    .extensions = "flv",
184
    .value = CODEC_ID_FLV1,
185
};
186
187
int flvdec_init(void)
188
{
189
    av_register_input_format(&flv_iformat);
190
    return 0;
191
}