aboutsummaryrefslogtreecommitdiffstats
path: root/libavformat/ipmovie.c
diff options
context:
space:
mode:
authorMike Melanson <mike@multimedia.cx>2003-10-02 04:16:59 +0000
committerMike Melanson <mike@multimedia.cx>2003-10-02 04:16:59 +0000
commitdda61423618e67c5e96ccf3376b1fe85f4ad0a3a (patch)
tree376bf704d3a2064fd69ccceb33394cd7876896b0 /libavformat/ipmovie.c
parent68a43d044a0b8505f8fdef44880faa5c98388a72 (diff)
downloadffmpeg-dda61423618e67c5e96ccf3376b1fe85f4ad0a3a.tar.gz
properly demux silent files; implemented precise framerate calculation
Originally committed as revision 2330 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavformat/ipmovie.c')
-rw-r--r--libavformat/ipmovie.c57
1 files changed, 36 insertions, 21 deletions
diff --git a/libavformat/ipmovie.c b/libavformat/ipmovie.c
index 45b6883b1e..e8ba0b7043 100644
--- a/libavformat/ipmovie.c
+++ b/libavformat/ipmovie.c
@@ -96,7 +96,7 @@ typedef struct IPMVEContext {
unsigned char *buf;
int buf_size;
- int fps;
+ float fps;
int frame_pts_inc;
unsigned int video_width;
@@ -199,6 +199,9 @@ static int load_ipmovie_packet(IPMVEContext *s, ByteIOContext *pb,
pkt->stream_index = s->video_stream_index;
pkt->pts = s->video_pts;
+ debug_ipmovie("sending video frame with pts %lld\n",
+ pkt->pts);
+
s->video_pts += s->frame_pts_inc;
chunk_type = CHUNK_VIDEO;
@@ -330,10 +333,9 @@ static int process_ipmovie_chunk(IPMVEContext *s, ByteIOContext *pb,
chunk_type = CHUNK_BAD;
break;
}
- s->fps = 1000000 / (LE_32(&scratch[0]) * LE_16(&scratch[4]));
- s->fps++; /* above calculation usually yields 14.9; we need 15 */
+ s->fps = 1000000.0 / (LE_32(&scratch[0]) * LE_16(&scratch[4]));
s->frame_pts_inc = 90000 / s->fps;
- debug_ipmovie("%d frames/second (timer div = %d, subdiv = %d)\n",
+ debug_ipmovie(" %.2f frames/second (timer div = %d, subdiv = %d)\n",
s->fps, LE_32(&scratch[0]), LE_16(&scratch[4]));
break;
@@ -527,6 +529,8 @@ static int ipmovie_read_header(AVFormatContext *s,
ByteIOContext *pb = &s->pb;
AVPacket pkt;
AVStream *st;
+ unsigned char chunk_preamble[CHUNK_PREAMBLE_SIZE];
+ int chunk_type;
/* initialize private context members */
ipmovie->video_pts = ipmovie->audio_frame_count = 0;
@@ -540,8 +544,17 @@ static int ipmovie_read_header(AVFormatContext *s,
if (process_ipmovie_chunk(ipmovie, pb, &pkt) != CHUNK_INIT_VIDEO)
return AVERROR_INVALIDDATA;
- /* process the next chunk which should be CHUNK_INIT_AUDIO */
- if (process_ipmovie_chunk(ipmovie, pb, &pkt) != CHUNK_INIT_AUDIO)
+ /* peek ahead to the next chunk-- if it is an init audio chunk, process
+ * it; if it is the first video chunk, this is a silent file */
+ if (get_buffer(pb, chunk_preamble, CHUNK_PREAMBLE_SIZE) !=
+ CHUNK_PREAMBLE_SIZE)
+ return -EIO;
+ chunk_type = LE_16(&chunk_preamble[2]);
+ url_fseek(pb, -CHUNK_PREAMBLE_SIZE, SEEK_CUR);
+
+ if (chunk_type == CHUNK_VIDEO)
+ ipmovie->audio_type = 0; /* no audio */
+ else if (process_ipmovie_chunk(ipmovie, pb, &pkt) != CHUNK_INIT_AUDIO)
return AVERROR_INVALIDDATA;
/* set the pts reference (1 pts = 1/90000) */
@@ -563,21 +576,23 @@ static int ipmovie_read_header(AVFormatContext *s,
st->codec.extradata_size = sizeof(AVPaletteControl);
st->codec.extradata = &ipmovie->palette_control;
- st = av_new_stream(s, 0);
- if (!st)
- return AVERROR_NOMEM;
- ipmovie->audio_stream_index = st->index;
- st->codec.codec_type = CODEC_TYPE_AUDIO;
- st->codec.codec_id = ipmovie->audio_type;
- st->codec.codec_tag = 0; /* no tag */
- st->codec.channels = ipmovie->audio_channels;
- st->codec.sample_rate = ipmovie->audio_sample_rate;
- st->codec.bits_per_sample = ipmovie->audio_bits;
- st->codec.bit_rate = st->codec.channels * st->codec.sample_rate *
- st->codec.bits_per_sample;
- if (st->codec.codec_id == CODEC_ID_INTERPLAY_DPCM)
- st->codec.bit_rate /= 2;
- st->codec.block_align = st->codec.channels * st->codec.bits_per_sample;
+ if (ipmovie->audio_type) {
+ st = av_new_stream(s, 0);
+ if (!st)
+ return AVERROR_NOMEM;
+ ipmovie->audio_stream_index = st->index;
+ st->codec.codec_type = CODEC_TYPE_AUDIO;
+ st->codec.codec_id = ipmovie->audio_type;
+ st->codec.codec_tag = 0; /* no tag */
+ st->codec.channels = ipmovie->audio_channels;
+ st->codec.sample_rate = ipmovie->audio_sample_rate;
+ st->codec.bits_per_sample = ipmovie->audio_bits;
+ st->codec.bit_rate = st->codec.channels * st->codec.sample_rate *
+ st->codec.bits_per_sample;
+ if (st->codec.codec_id == CODEC_ID_INTERPLAY_DPCM)
+ st->codec.bit_rate /= 2;
+ st->codec.block_align = st->codec.channels * st->codec.bits_per_sample;
+ }
return 0;
}