annotate src/ffmpeg/libavformat/segafilm.c @ 841:9b50e71f7307 trunk

[svn] - implement shorten support more correctly.
author nenolod
date Mon, 12 Mar 2007 17:37:02 -0700
parents e8776388b02a
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
808
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
1 /*
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
2 * Sega FILM Format (CPK) Demuxer
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
3 * Copyright (c) 2003 The ffmpeg Project
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
4 *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
5 * This file is part of FFmpeg.
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
6 *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
7 * FFmpeg is free software; you can redistribute it and/or
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
8 * modify it under the terms of the GNU Lesser General Public
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
9 * License as published by the Free Software Foundation; either
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
10 * version 2.1 of the License, or (at your option) any later version.
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
11 *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
12 * FFmpeg is distributed in the hope that it will be useful,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
15 * Lesser General Public License for more details.
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
16 *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
17 * You should have received a copy of the GNU Lesser General Public
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
18 * License along with FFmpeg; if not, write to the Free Software
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
20 */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
21
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
22 /**
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
23 * @file segafilm.c
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
24 * Sega FILM (.cpk) file demuxer
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
25 * by Mike Melanson (melanson@pcisys.net)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
26 * For more information regarding the Sega FILM file format, visit:
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
27 * http://www.pcisys.net/~melanson/codecs/
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
28 */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
29
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
30 #include "avformat.h"
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
31
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
32 #define FILM_TAG MKBETAG('F', 'I', 'L', 'M')
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
33 #define FDSC_TAG MKBETAG('F', 'D', 'S', 'C')
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
34 #define STAB_TAG MKBETAG('S', 'T', 'A', 'B')
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
35 #define CVID_TAG MKBETAG('c', 'v', 'i', 'd')
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
36
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
37 typedef struct {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
38 int stream;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
39 offset_t sample_offset;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
40 unsigned int sample_size;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
41 int64_t pts;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
42 int keyframe;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
43 } film_sample_t;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
44
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
45 typedef struct FilmDemuxContext {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
46 int video_stream_index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
47 int audio_stream_index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
48
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
49 unsigned int audio_type;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
50 unsigned int audio_samplerate;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
51 unsigned int audio_bits;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
52 unsigned int audio_channels;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
53
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
54 unsigned int video_type;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
55 unsigned int sample_count;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
56 film_sample_t *sample_table;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
57 unsigned int current_sample;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
58
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
59 unsigned int base_clock;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
60 unsigned int version;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
61 int cvid_extra_bytes; /* the number of bytes thrown into the Cinepak
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
62 * chunk header to throw off decoders */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
63
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
64 /* buffer used for interleaving stereo PCM data */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
65 unsigned char *stereo_buffer;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
66 int stereo_buffer_size;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
67 } FilmDemuxContext;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
68
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
69 static int film_probe(AVProbeData *p)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
70 {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
71 if (p->buf_size < 4)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
72 return 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
73
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
74 if (BE_32(&p->buf[0]) != FILM_TAG)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
75 return 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
76
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
77 return AVPROBE_SCORE_MAX;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
78 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
79
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
80 static int film_read_header(AVFormatContext *s,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
81 AVFormatParameters *ap)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
82 {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
83 FilmDemuxContext *film = (FilmDemuxContext *)s->priv_data;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
84 ByteIOContext *pb = &s->pb;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
85 AVStream *st;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
86 unsigned char scratch[256];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
87 int i;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
88 unsigned int data_offset;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
89 unsigned int audio_frame_counter;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
90
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
91 film->sample_table = NULL;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
92 film->stereo_buffer = NULL;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
93 film->stereo_buffer_size = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
94
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
95 /* load the main FILM header */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
96 if (get_buffer(pb, scratch, 16) != 16)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
97 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
98 data_offset = BE_32(&scratch[4]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
99 film->version = BE_32(&scratch[8]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
100
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
101 /* load the FDSC chunk */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
102 if (film->version == 0) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
103 /* special case for Lemmings .film files; 20-byte header */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
104 if (get_buffer(pb, scratch, 20) != 20)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
105 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
106 /* make some assumptions about the audio parameters */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
107 film->audio_type = CODEC_ID_PCM_S8;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
108 film->audio_samplerate = 22050;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
109 film->audio_channels = 1;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
110 film->audio_bits = 8;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
111 } else {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
112 /* normal Saturn .cpk files; 32-byte header */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
113 if (get_buffer(pb, scratch, 32) != 32)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
114 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
115 film->audio_samplerate = BE_16(&scratch[24]);;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
116 film->audio_channels = scratch[21];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
117 film->audio_bits = scratch[22];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
118 if (film->audio_bits == 8)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
119 film->audio_type = CODEC_ID_PCM_S8;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
120 else if (film->audio_bits == 16)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
121 film->audio_type = CODEC_ID_PCM_S16BE;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
122 else
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
123 film->audio_type = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
124 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
125
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
126 if (BE_32(&scratch[0]) != FDSC_TAG)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
127 return AVERROR_INVALIDDATA;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
128
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
129 film->cvid_extra_bytes = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
130 if (BE_32(&scratch[8]) == CVID_TAG) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
131 film->video_type = CODEC_ID_CINEPAK;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
132 if (film->version)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
133 film->cvid_extra_bytes = 2;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
134 else
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
135 film->cvid_extra_bytes = 6; /* Lemmings 3DO case */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
136 } else
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
137 film->video_type = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
138
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
139 /* initialize the decoder streams */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
140 if (film->video_type) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
141 st = av_new_stream(s, 0);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
142 if (!st)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
143 return AVERROR_NOMEM;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
144 film->video_stream_index = st->index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
145 st->codec->codec_type = CODEC_TYPE_VIDEO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
146 st->codec->codec_id = film->video_type;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
147 st->codec->codec_tag = 0; /* no fourcc */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
148 st->codec->width = BE_32(&scratch[16]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
149 st->codec->height = BE_32(&scratch[12]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
150 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
151
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
152 if (film->audio_type) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
153 st = av_new_stream(s, 0);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
154 if (!st)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
155 return AVERROR_NOMEM;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
156 film->audio_stream_index = st->index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
157 st->codec->codec_type = CODEC_TYPE_AUDIO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
158 st->codec->codec_id = film->audio_type;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
159 st->codec->codec_tag = 1;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
160 st->codec->channels = film->audio_channels;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
161 st->codec->bits_per_sample = film->audio_bits;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
162 st->codec->sample_rate = film->audio_samplerate;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
163 st->codec->bit_rate = st->codec->channels * st->codec->sample_rate *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
164 st->codec->bits_per_sample;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
165 st->codec->block_align = st->codec->channels *
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
166 st->codec->bits_per_sample / 8;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
167 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
168
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
169 /* load the sample table */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
170 if (get_buffer(pb, scratch, 16) != 16)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
171 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
172 if (BE_32(&scratch[0]) != STAB_TAG)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
173 return AVERROR_INVALIDDATA;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
174 film->base_clock = BE_32(&scratch[8]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
175 film->sample_count = BE_32(&scratch[12]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
176 if(film->sample_count >= UINT_MAX / sizeof(film_sample_t))
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
177 return -1;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
178 film->sample_table = av_malloc(film->sample_count * sizeof(film_sample_t));
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
179
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
180 for(i=0; i<s->nb_streams; i++)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
181 av_set_pts_info(s->streams[i], 33, 1, film->base_clock);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
182
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
183 audio_frame_counter = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
184 for (i = 0; i < film->sample_count; i++) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
185 /* load the next sample record and transfer it to an internal struct */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
186 if (get_buffer(pb, scratch, 16) != 16) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
187 av_free(film->sample_table);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
188 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
189 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
190 film->sample_table[i].sample_offset =
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
191 data_offset + BE_32(&scratch[0]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
192 film->sample_table[i].sample_size = BE_32(&scratch[4]);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
193 if (BE_32(&scratch[8]) == 0xFFFFFFFF) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
194 film->sample_table[i].stream = film->audio_stream_index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
195 film->sample_table[i].pts = audio_frame_counter;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
196 film->sample_table[i].pts *= film->base_clock;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
197 film->sample_table[i].pts /= film->audio_samplerate;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
198
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
199 audio_frame_counter += (film->sample_table[i].sample_size /
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
200 (film->audio_channels * film->audio_bits / 8));
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
201 } else {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
202 film->sample_table[i].stream = film->video_stream_index;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
203 film->sample_table[i].pts = BE_32(&scratch[8]) & 0x7FFFFFFF;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
204 film->sample_table[i].keyframe = (scratch[8] & 0x80) ? 0 : 1;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
205 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
206 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
207
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
208 film->current_sample = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
209
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
210 return 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
211 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
212
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
213 static int film_read_packet(AVFormatContext *s,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
214 AVPacket *pkt)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
215 {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
216 FilmDemuxContext *film = (FilmDemuxContext *)s->priv_data;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
217 ByteIOContext *pb = &s->pb;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
218 film_sample_t *sample;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
219 int ret = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
220 int i;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
221 int left, right;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
222
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
223 if (film->current_sample >= film->sample_count)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
224 return AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
225
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
226 sample = &film->sample_table[film->current_sample];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
227
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
228 /* position the stream (will probably be there anyway) */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
229 url_fseek(pb, sample->sample_offset, SEEK_SET);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
230
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
231 /* do a special song and dance when loading FILM Cinepak chunks */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
232 if ((sample->stream == film->video_stream_index) &&
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
233 (film->video_type == CODEC_ID_CINEPAK)) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
234 if (av_new_packet(pkt, sample->sample_size - film->cvid_extra_bytes))
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
235 return AVERROR_NOMEM;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
236 if(pkt->size < 10)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
237 return -1;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
238 pkt->pos= url_ftell(pb);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
239 ret = get_buffer(pb, pkt->data, 10);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
240 /* skip the non-spec CVID bytes */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
241 url_fseek(pb, film->cvid_extra_bytes, SEEK_CUR);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
242 ret += get_buffer(pb, pkt->data + 10,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
243 sample->sample_size - 10 - film->cvid_extra_bytes);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
244 if (ret != sample->sample_size - film->cvid_extra_bytes)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
245 ret = AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
246 } else if ((sample->stream == film->audio_stream_index) &&
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
247 (film->audio_channels == 2)) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
248 /* stereo PCM needs to be interleaved */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
249
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
250 if (av_new_packet(pkt, sample->sample_size))
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
251 return AVERROR_NOMEM;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
252
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
253 /* make sure the interleave buffer is large enough */
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
254 if (sample->sample_size > film->stereo_buffer_size) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
255 av_free(film->stereo_buffer);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
256 film->stereo_buffer_size = sample->sample_size;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
257 film->stereo_buffer = av_malloc(film->stereo_buffer_size);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
258 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
259
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
260 pkt->pos= url_ftell(pb);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
261 ret = get_buffer(pb, film->stereo_buffer, sample->sample_size);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
262 if (ret != sample->sample_size)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
263 ret = AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
264
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
265 left = 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
266 right = sample->sample_size / 2;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
267 for (i = 0; i < sample->sample_size; ) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
268 if (film->audio_bits == 8) {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
269 pkt->data[i++] = film->stereo_buffer[left++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
270 pkt->data[i++] = film->stereo_buffer[right++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
271 } else {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
272 pkt->data[i++] = film->stereo_buffer[left++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
273 pkt->data[i++] = film->stereo_buffer[left++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
274 pkt->data[i++] = film->stereo_buffer[right++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
275 pkt->data[i++] = film->stereo_buffer[right++];
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
276 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
277 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
278 } else {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
279 ret= av_get_packet(pb, pkt, sample->sample_size);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
280 if (ret != sample->sample_size)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
281 ret = AVERROR_IO;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
282 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
283
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
284 pkt->stream_index = sample->stream;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
285 pkt->pts = sample->pts;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
286
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
287 film->current_sample++;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
288
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
289 return ret;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
290 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
291
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
292 static int film_read_close(AVFormatContext *s)
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
293 {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
294 FilmDemuxContext *film = (FilmDemuxContext *)s->priv_data;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
295
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
296 av_free(film->sample_table);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
297 av_free(film->stereo_buffer);
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
298
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
299 return 0;
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
300 }
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
301
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
302 AVInputFormat segafilm_demuxer = {
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
303 "film_cpk",
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
304 "Sega FILM/CPK format",
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
305 sizeof(FilmDemuxContext),
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
306 film_probe,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
307 film_read_header,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
308 film_read_packet,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
309 film_read_close,
e8776388b02a [svn] - add ffmpeg
nenolod
parents:
diff changeset
310 };