annotate flic.c @ 643:253b5292946a libavformat

various security fixes and precautionary checks
author michael
date Wed, 12 Jan 2005 00:16:25 +0000
parents 0fdc96c2f2fe
children c5077fdab490
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)
482
0fdc96c2f2fe sweeping change from -EIO -> AVERROR_IO
melanson
parents: 462
diff changeset
79 return AVERROR_IO;
315
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
462
b69898ffc92a move time_base (pts_num/pts_den) from AVFormatContext -> AVStream
michael
parents: 386
diff changeset
103 av_set_pts_info(st, 33, 1, 90000);
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
104
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
105 /* Time to figure out the framerate: If there is a FLIC chunk magic
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
106 * number at offset 0x10, assume this is from the Bullfrog game,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
107 * Magic Carpet. */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
108 if (LE_16(&header[0x10]) == FLIC_CHUNK_MAGIC_1) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
109
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
110 flic->frame_pts_inc = FLIC_MC_PTS_INC;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
111
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
112 /* rewind the stream since the first chunk is at offset 12 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
113 url_fseek(pb, 12, SEEK_SET);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
114
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
115 /* send over abbreviated FLIC header chunk */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
116 av_free(st->codec.extradata);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
117 st->codec.extradata_size = 12;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
118 st->codec.extradata = av_malloc(12);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
119 memcpy(st->codec.extradata, header, 12);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
120
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
121 } else if (magic_number == FLIC_FILE_MAGIC_1) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
122 /*
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
123 * in this case, the speed (n) is number of 1/70s ticks between frames:
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
124 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
125 * pts n * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
126 * -------- = ----------- => pts = n * (90000/70) * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
127 * 90000 70
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
128 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
129 * therefore, the frame pts increment = n * 1285.7
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
130 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
131 flic->frame_pts_inc = speed * 1285.7;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
132 } else if (magic_number == FLIC_FILE_MAGIC_2) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
133 /*
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
134 * in this case, the speed (n) is number of milliseconds between frames:
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
135 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
136 * pts n * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
137 * -------- = ----------- => pts = n * 90 * frame #
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
138 * 90000 1000
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
139 *
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
140 * therefore, the frame pts increment = n * 90
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
141 */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
142 flic->frame_pts_inc = speed * 90;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
143 } else
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
144 return AVERROR_INVALIDDATA;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
145
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
146 if (flic->frame_pts_inc == 0)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
147 flic->frame_pts_inc = FLIC_DEFAULT_PTS_INC;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
148
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
149 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
150 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
151
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
152 static int flic_read_packet(AVFormatContext *s,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
153 AVPacket *pkt)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
154 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
155 FlicDemuxContext *flic = (FlicDemuxContext *)s->priv_data;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
156 ByteIOContext *pb = &s->pb;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
157 int packet_read = 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
158 unsigned int size;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
159 int magic;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
160 int ret = 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
161 unsigned char preamble[FLIC_PREAMBLE_SIZE];
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
162
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
163 while (!packet_read) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
164
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
165 if ((ret = get_buffer(pb, preamble, FLIC_PREAMBLE_SIZE)) !=
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
166 FLIC_PREAMBLE_SIZE) {
482
0fdc96c2f2fe sweeping change from -EIO -> AVERROR_IO
melanson
parents: 462
diff changeset
167 ret = AVERROR_IO;
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
168 break;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
169 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
170
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
171 size = LE_32(&preamble[0]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
172 magic = LE_16(&preamble[4]);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
173
643
253b5292946a various security fixes and precautionary checks
michael
parents: 482
diff changeset
174 if (((magic == FLIC_CHUNK_MAGIC_1) || (magic == FLIC_CHUNK_MAGIC_2)) && size > FLIC_PREAMBLE_SIZE) {
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
175 if (av_new_packet(pkt, size)) {
482
0fdc96c2f2fe sweeping change from -EIO -> AVERROR_IO
melanson
parents: 462
diff changeset
176 ret = AVERROR_IO;
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
177 break;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
178 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
179 pkt->stream_index = flic->video_stream_index;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
180 pkt->pts = flic->pts;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
181 memcpy(pkt->data, preamble, FLIC_PREAMBLE_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
182 ret = get_buffer(pb, pkt->data + FLIC_PREAMBLE_SIZE,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
183 size - FLIC_PREAMBLE_SIZE);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
184 if (ret != size - FLIC_PREAMBLE_SIZE) {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
185 av_free_packet(pkt);
482
0fdc96c2f2fe sweeping change from -EIO -> AVERROR_IO
melanson
parents: 462
diff changeset
186 ret = AVERROR_IO;
315
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
187 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
188 flic->pts += flic->frame_pts_inc;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
189 packet_read = 1;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
190 } else {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
191 /* not interested in this chunk */
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
192 url_fseek(pb, size - 6, SEEK_CUR);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
193 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
194 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
195
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
196 return ret;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
197 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
198
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
199 static int flic_read_close(AVFormatContext *s)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
200 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
201 // FlicDemuxContext *flic = (FlicDemuxContext *)s->priv_data;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
202
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
203 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
204 }
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
205
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
206 static AVInputFormat flic_iformat = {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
207 "flic",
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
208 "FLI/FLC animation format",
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
209 sizeof(FlicDemuxContext),
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
210 flic_probe,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
211 flic_read_header,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
212 flic_read_packet,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
213 flic_read_close,
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
214 };
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
215
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
216 int flic_init(void)
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
217 {
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
218 av_register_input_format(&flic_iformat);
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
219 return 0;
ff595fc022ff yep, FLI support
melanson
parents:
diff changeset
220 }