annotate flic.c @ 390:3a40642dc4df libavformat

adjust_write_index() fix by ("Curi Fabio Eduardo (SFL)" <curif at TELEFONICA dot COM dot AR>)
author michael
date Thu, 18 Mar 2004 20:52:40 +0000
parents c152849ee643
children b69898ffc92a
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
1 /*
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
2 * FLI/FLC Animation File Demuxer
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
3 * Copyright (c) 2003 The ffmpeg Project
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
4 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
5 * This library is free software; you can redistribute it and/or
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
6 * modify it under the terms of the GNU Lesser General Public
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
7 * License as published by the Free Software Foundation; either
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
8 * version 2 of the License, or (at your option) any later version.
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
9 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
10 * This library is distributed in the hope that it will be useful,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
13 * Lesser General Public License for more details.
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
14 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
15 * You should have received a copy of the GNU Lesser General Public
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
16 * License along with this library; if not, write to the Free Software
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
18 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
19
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
20 /**
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
21 * @file flic.c
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
22 * FLI/FLC file demuxer
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
23 * by Mike Melanson (melanson@pcisys.net)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
24 * for more information on the .fli/.flc file format and all of its many
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
25 * variations, visit:
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
26 * http://www.compuphase.com/flic.htm
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
27 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
28 * This demuxer handles standard 0xAF11- and 0xAF12-type FLIs. It also
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
29 * handles special FLIs from the PC game "Magic Carpet".
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
30 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
31
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
32 #include "avformat.h"
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
33
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
34 #define FLIC_FILE_MAGIC_1 0xAF11
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
35 #define FLIC_FILE_MAGIC_2 0xAF12
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
36 #define FLIC_CHUNK_MAGIC_1 0xF1FA
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
37 #define FLIC_CHUNK_MAGIC_2 0xF5FA
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
38 #define FLIC_MC_PTS_INC 6000 /* pts increment for Magic Carpet game FLIs */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
39 #define FLIC_DEFAULT_PTS_INC 6000 /* for FLIs that have 0 speed */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
40
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
41 #define FLIC_HEADER_SIZE 128
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
42 #define FLIC_PREAMBLE_SIZE 6
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
43
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
44 typedef struct FlicDemuxContext {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
45 int frame_pts_inc;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
46 int64_t pts;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
47 int video_stream_index;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
48 } FlicDemuxContext;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
49
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
50 static int flic_probe(AVProbeData *p)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
51 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
52 int magic_number;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
53
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
54 if (p->buf_size < 6)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
55 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
56
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
57 magic_number = LE_16(&p->buf[4]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
58 if ((magic_number != FLIC_FILE_MAGIC_1) &&
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
59 (magic_number != FLIC_FILE_MAGIC_2))
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
60 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
61
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
62 return AVPROBE_SCORE_MAX;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
63 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
64
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
65 static int flic_read_header(AVFormatContext *s,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
66 AVFormatParameters *ap)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
67 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
68 FlicDemuxContext *flic = (FlicDemuxContext *)s->priv_data;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
69 ByteIOContext *pb = &s->pb;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
70 unsigned char header[FLIC_HEADER_SIZE];
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
71 AVStream *st;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
72 int speed;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
73 int magic_number;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
74
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
75 flic->pts = 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
76
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
77 /* load the whole header and pull out the width and height */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
78 if (get_buffer(pb, header, FLIC_HEADER_SIZE) != FLIC_HEADER_SIZE)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
79 return -EIO;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
80
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
81 magic_number = LE_16(&header[4]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
82 speed = LE_32(&header[0x10]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
83
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
84 /* initialize the decoder streams */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
85 st = av_new_stream(s, 0);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
86 if (!st)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
87 return AVERROR_NOMEM;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
88 flic->video_stream_index = st->index;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
89 st->codec.codec_type = CODEC_TYPE_VIDEO;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
90 st->codec.codec_id = CODEC_ID_FLIC;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
91 st->codec.codec_tag = 0; /* no fourcc */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
92 st->codec.width = LE_16(&header[0x08]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
93 st->codec.height = LE_16(&header[0x0A]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
94
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
95 if (!st->codec.width || !st->codec.height)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
96 return AVERROR_INVALIDDATA;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
97
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
98 /* send over the whole 128-byte FLIC header */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
99 st->codec.extradata_size = FLIC_HEADER_SIZE;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
100 st->codec.extradata = av_malloc(FLIC_HEADER_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
101 memcpy(st->codec.extradata, header, FLIC_HEADER_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
102
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
103 /* set the pts reference (1 pts = 1/90000) */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
104 s->pts_num = 1;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
105 s->pts_den = 90000;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
106
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
107 /* Time to figure out the framerate: If there is a FLIC chunk magic
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
108 * number at offset 0x10, assume this is from the Bullfrog game,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
109 * Magic Carpet. */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
110 if (LE_16(&header[0x10]) == FLIC_CHUNK_MAGIC_1) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
111
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
112 flic->frame_pts_inc = FLIC_MC_PTS_INC;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
113
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
114 /* rewind the stream since the first chunk is at offset 12 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
115 url_fseek(pb, 12, SEEK_SET);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
116
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
117 /* send over abbreviated FLIC header chunk */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
118 av_free(st->codec.extradata);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
119 st->codec.extradata_size = 12;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
120 st->codec.extradata = av_malloc(12);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
121 memcpy(st->codec.extradata, header, 12);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
122
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
123 } else if (magic_number == FLIC_FILE_MAGIC_1) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
124 /*
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
125 * in this case, the speed (n) is number of 1/70s ticks between frames:
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
126 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
127 * pts n * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
128 * -------- = ----------- => pts = n * (90000/70) * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
129 * 90000 70
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
130 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
131 * therefore, the frame pts increment = n * 1285.7
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
132 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
133 flic->frame_pts_inc = speed * 1285.7;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
134 } else if (magic_number == FLIC_FILE_MAGIC_2) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
135 /*
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
136 * in this case, the speed (n) is number of milliseconds between frames:
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
137 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
138 * pts n * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
139 * -------- = ----------- => pts = n * 90 * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
140 * 90000 1000
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
141 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
142 * therefore, the frame pts increment = n * 90
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
143 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
144 flic->frame_pts_inc = speed * 90;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
145 } else
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
146 return AVERROR_INVALIDDATA;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
147
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
148 if (flic->frame_pts_inc == 0)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
149 flic->frame_pts_inc = FLIC_DEFAULT_PTS_INC;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
150
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
151 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
152 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
153
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
154 static int flic_read_packet(AVFormatContext *s,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
155 AVPacket *pkt)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
156 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
157 FlicDemuxContext *flic = (FlicDemuxContext *)s->priv_data;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
158 ByteIOContext *pb = &s->pb;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
159 int packet_read = 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
160 unsigned int size;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
161 int magic;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
162 int ret = 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
163 unsigned char preamble[FLIC_PREAMBLE_SIZE];
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
164
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
165 while (!packet_read) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
166
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
167 if ((ret = get_buffer(pb, preamble, FLIC_PREAMBLE_SIZE)) !=
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
168 FLIC_PREAMBLE_SIZE) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
169 ret = -EIO;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
170 break;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
171 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
172
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
173 size = LE_32(&preamble[0]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
174 magic = LE_16(&preamble[4]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
175
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
176 if ((magic == FLIC_CHUNK_MAGIC_1) || (magic == FLIC_CHUNK_MAGIC_2)) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
177 if (av_new_packet(pkt, size)) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
178 ret = -EIO;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
179 break;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
180 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
181 pkt->stream_index = flic->video_stream_index;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
182 pkt->pts = flic->pts;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
183 memcpy(pkt->data, preamble, FLIC_PREAMBLE_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
184 ret = get_buffer(pb, pkt->data + FLIC_PREAMBLE_SIZE,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
185 size - FLIC_PREAMBLE_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
186 if (ret != size - FLIC_PREAMBLE_SIZE) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
187 av_free_packet(pkt);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
188 ret = -EIO;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
189 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
190 flic->pts += flic->frame_pts_inc;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
191 packet_read = 1;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
192 } else {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
193 /* not interested in this chunk */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
194 url_fseek(pb, size - 6, SEEK_CUR);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
195 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
196 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
197
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
198 return ret;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
199 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
200
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
201 static int flic_read_close(AVFormatContext *s)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
202 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
203 // FlicDemuxContext *flic = (FlicDemuxContext *)s->priv_data;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
204
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
205 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
206 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
207
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
208 static AVInputFormat flic_iformat = {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
209 "flic",
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
210 "FLI/FLC animation format",
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
211 sizeof(FlicDemuxContext),
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
212 flic_probe,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
213 flic_read_header,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
214 flic_read_packet,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
215 flic_read_close,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
216 };
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
217
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
218 int flic_init(void)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
219 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
220 av_register_input_format(&flic_iformat);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
221 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
222 }