Mercurial > libavformat.hg
annotate avidec.c @ 79:8bdaec4e3e4b libavformat
fixing frame_rate accuracy
author | michaelni |
---|---|
date | Sat, 08 Mar 2003 14:29:12 +0000 |
parents | f1ea9a26edca |
children | 63807035cdb5 |
rev | line source |
---|---|
0 | 1 /* |
2 * AVI decoder. | |
3 * Copyright (c) 2001 Fabrice Bellard. | |
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 #include "avi.h" | |
21 | |
22 //#define DEBUG | |
23 | |
24 typedef struct AVIIndex { | |
25 unsigned char tag[4]; | |
26 unsigned int flags, pos, len; | |
27 struct AVIIndex *next; | |
28 } AVIIndex; | |
29 | |
30 typedef struct { | |
65 | 31 int64_t movi_end; |
0 | 32 offset_t movi_list; |
33 AVIIndex *first, *last; | |
34 } AVIContext; | |
35 | |
36 #ifdef DEBUG | |
37 static void print_tag(const char *str, unsigned int tag, int size) | |
38 { | |
39 printf("%s: tag=%c%c%c%c size=0x%x\n", | |
40 str, tag & 0xff, | |
41 (tag >> 8) & 0xff, | |
42 (tag >> 16) & 0xff, | |
43 (tag >> 24) & 0xff, | |
44 size); | |
45 } | |
46 #endif | |
47 | |
48 static int avi_read_header(AVFormatContext *s, AVFormatParameters *ap) | |
49 { | |
50 AVIContext *avi = s->priv_data; | |
51 ByteIOContext *pb = &s->pb; | |
65 | 52 uint32_t tag, tag1; |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
53 int codec_type, stream_index, frame_period, bit_rate, scale, rate; |
45 | 54 unsigned int size; |
0 | 55 int i; |
56 AVStream *st; | |
57 | |
58 /* check RIFF header */ | |
59 tag = get_le32(pb); | |
60 | |
61 if (tag != MKTAG('R', 'I', 'F', 'F')) | |
62 return -1; | |
63 get_le32(pb); /* file size */ | |
64 tag = get_le32(pb); | |
65 if (tag != MKTAG('A', 'V', 'I', ' ')) | |
66 return -1; | |
67 | |
68 /* first list tag */ | |
69 stream_index = -1; | |
70 codec_type = -1; | |
71 frame_period = 0; | |
72 for(;;) { | |
73 if (url_feof(pb)) | |
74 goto fail; | |
75 tag = get_le32(pb); | |
76 size = get_le32(pb); | |
77 #ifdef DEBUG | |
78 print_tag("tag", tag, size); | |
79 #endif | |
80 | |
81 switch(tag) { | |
82 case MKTAG('L', 'I', 'S', 'T'): | |
83 /* ignored, except when start of video packets */ | |
84 tag1 = get_le32(pb); | |
85 #ifdef DEBUG | |
86 print_tag("list", tag1, 0); | |
87 #endif | |
88 if (tag1 == MKTAG('m', 'o', 'v', 'i')) { | |
89 avi->movi_end = url_ftell(pb) + size - 4; | |
90 #ifdef DEBUG | |
91 printf("movi end=%Lx\n", avi->movi_end); | |
92 #endif | |
93 goto end_of_header; | |
94 } | |
95 break; | |
96 case MKTAG('a', 'v', 'i', 'h'): | |
97 /* avi header */ | |
98 /* using frame_period is bad idea */ | |
99 frame_period = get_le32(pb); | |
100 bit_rate = get_le32(pb) * 8; | |
101 url_fskip(pb, 4 * 4); | |
102 s->nb_streams = get_le32(pb); | |
103 for(i=0;i<s->nb_streams;i++) { | |
104 AVStream *st = av_mallocz(sizeof(AVStream)); | |
105 if (!st) | |
106 goto fail; | |
5 | 107 avcodec_get_context_defaults(&st->codec); |
108 | |
0 | 109 s->streams[i] = st; |
110 } | |
111 url_fskip(pb, size - 7 * 4); | |
112 break; | |
113 case MKTAG('s', 't', 'r', 'h'): | |
114 /* stream header */ | |
115 stream_index++; | |
116 tag1 = get_le32(pb); | |
117 switch(tag1) { | |
118 case MKTAG('v', 'i', 'd', 's'): | |
119 codec_type = CODEC_TYPE_VIDEO; | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
120 |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
121 if (stream_index >= s->nb_streams) { |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
122 url_fskip(pb, size - 4); |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
123 break; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
124 } |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
125 |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
126 st = s->streams[stream_index]; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
127 |
0 | 128 get_le32(pb); /* codec tag */ |
129 get_le32(pb); /* flags */ | |
130 get_le16(pb); /* priority */ | |
131 get_le16(pb); /* language */ | |
132 get_le32(pb); /* XXX: initial frame ? */ | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
133 scale= get_le32(pb); /* scale */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
134 rate= get_le32(pb); /* rate */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
135 |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
136 if(scale && rate) |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
137 st->codec.frame_rate= (rate * (uint64_t)FRAME_RATE_BASE + scale/2) / scale; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
138 else if(frame_period) |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
139 st->codec.frame_rate = (1000000LL * FRAME_RATE_BASE + frame_period/2) / frame_period; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
140 else |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
141 st->codec.frame_rate = 25 * FRAME_RATE_BASE; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
142 |
73
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
143 url_fskip(pb, size - 7 * 4); |
0 | 144 break; |
145 case MKTAG('a', 'u', 'd', 's'): | |
146 codec_type = CODEC_TYPE_AUDIO; | |
147 /* nothing really useful */ | |
73
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
148 url_fskip(pb, size - 4); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
149 break; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
150 default: |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
151 goto fail; |
0 | 152 } |
153 break; | |
154 case MKTAG('s', 't', 'r', 'f'): | |
155 /* stream header */ | |
156 if (stream_index >= s->nb_streams) { | |
157 url_fskip(pb, size); | |
158 } else { | |
159 st = s->streams[stream_index]; | |
160 switch(codec_type) { | |
161 case CODEC_TYPE_VIDEO: | |
162 get_le32(pb); /* size */ | |
163 st->codec.width = get_le32(pb); | |
164 st->codec.height = get_le32(pb); | |
165 get_le16(pb); /* panes */ | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
166 st->codec.bits_per_sample= get_le16(pb); /* depth */ |
0 | 167 tag1 = get_le32(pb); |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
168 get_le32(pb); /* ImageSize */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
169 get_le32(pb); /* XPelsPerMeter */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
170 get_le32(pb); /* YPelsPerMeter */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
171 get_le32(pb); /* ClrUsed */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
172 get_le32(pb); /* ClrImportant */ |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
173 |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
174 st->codec.extradata_size= size - 10*4; |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
175 st->codec.extradata= av_malloc(st->codec.extradata_size); //FIXME where should we free this? |
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
176 get_buffer(pb, st->codec.extradata, st->codec.extradata_size); |
76 | 177 |
178 if(st->codec.extradata_size & 1) //FIXME check if the encoder really did this correctly | |
179 get_byte(pb); | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
180 |
0 | 181 #ifdef DEBUG |
182 print_tag("video", tag1, 0); | |
183 #endif | |
184 st->codec.codec_type = CODEC_TYPE_VIDEO; | |
185 st->codec.codec_tag = tag1; | |
186 st->codec.codec_id = codec_get_id(codec_bmp_tags, tag1); | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
187 // url_fskip(pb, size - 5 * 4); |
0 | 188 break; |
73
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
189 case CODEC_TYPE_AUDIO: |
0 | 190 get_wav_header(pb, &st->codec, (size >= 18)); |
13
8a5285a0ca2f
Fix for odd strf tag in Stargate SG-1 - 3x18 - Shades of Grey.avi
mmu_man
parents:
5
diff
changeset
|
191 if (size%2) /* 2-aligned (fix for Stargate SG-1 - 3x18 - Shades of Grey.avi) */ |
8a5285a0ca2f
Fix for odd strf tag in Stargate SG-1 - 3x18 - Shades of Grey.avi
mmu_man
parents:
5
diff
changeset
|
192 url_fskip(pb, 1); |
0 | 193 break; |
194 default: | |
195 url_fskip(pb, size); | |
196 break; | |
197 } | |
198 } | |
199 break; | |
200 default: | |
201 /* skip tag */ | |
202 size += (size & 1); | |
203 url_fskip(pb, size); | |
204 break; | |
205 } | |
206 } | |
207 end_of_header: | |
208 /* check stream number */ | |
209 if (stream_index != s->nb_streams - 1) { | |
210 fail: | |
211 for(i=0;i<s->nb_streams;i++) { | |
212 av_freep(&s->streams[i]); | |
78 | 213 av_freep(&s->streams[i]->codec.extradata); |
0 | 214 } |
215 return -1; | |
216 } | |
217 | |
218 return 0; | |
219 } | |
220 | |
221 static int avi_read_packet(AVFormatContext *s, AVPacket *pkt) | |
222 { | |
223 AVIContext *avi = s->priv_data; | |
224 ByteIOContext *pb = &s->pb; | |
225 int n, d1, d2, size; | |
75
78bec272ce3a
read BITMAPINFOHEADER extra stuff (huffyuv decoding fixed)
michaelni
parents:
73
diff
changeset
|
226 |
73
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
227 find_next: |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
228 if (url_feof(pb) || url_ftell(pb) >= avi->movi_end) |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
229 return -1; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
230 d1 = get_byte(pb); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
231 if (d1 < '0' || d1 > '9') |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
232 goto find_next; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
233 d2 = get_byte(pb); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
234 if (d2 < '0' || d2 > '9') |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
235 goto find_next; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
236 n = (d1 - '0') * 10 + (d2 - '0'); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
237 |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
238 if (n < 0 || n >= s->nb_streams) |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
239 goto find_next; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
240 |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
241 d1 = get_byte(pb); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
242 d2 = get_byte(pb); |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
243 if ((d1 != 'd' && d2 != 'c') && |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
244 (d1 != 'w' && d2 != 'b')) |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
245 goto find_next; |
d40ddc73858a
reversing not yet reversed changes from r1.7 -> r1.8 except the static/const stuff
michaelni
parents:
65
diff
changeset
|
246 |
0 | 247 size = get_le32(pb); |
248 av_new_packet(pkt, size); | |
249 pkt->stream_index = n; | |
250 | |
251 get_buffer(pb, pkt->data, pkt->size); | |
252 | |
253 if (size & 1) | |
254 get_byte(pb); | |
255 | |
256 return 0; | |
257 } | |
258 | |
259 static int avi_read_close(AVFormatContext *s) | |
260 { | |
261 return 0; | |
262 } | |
263 | |
264 static int avi_probe(AVProbeData *p) | |
265 { | |
266 /* check file header */ | |
267 if (p->buf_size <= 32) | |
268 return 0; | |
269 if (p->buf[0] == 'R' && p->buf[1] == 'I' && | |
270 p->buf[2] == 'F' && p->buf[3] == 'F' && | |
271 p->buf[8] == 'A' && p->buf[9] == 'V' && | |
272 p->buf[10] == 'I' && p->buf[11] == ' ') | |
273 return AVPROBE_SCORE_MAX; | |
274 else | |
275 return 0; | |
276 } | |
277 | |
278 static AVInputFormat avi_iformat = { | |
279 "avi", | |
280 "avi format", | |
281 sizeof(AVIContext), | |
282 avi_probe, | |
283 avi_read_header, | |
284 avi_read_packet, | |
285 avi_read_close, | |
286 }; | |
287 | |
288 int avidec_init(void) | |
289 { | |
290 av_register_input_format(&avi_iformat); | |
291 return 0; | |
292 } |