Commit aa90239f authored by Baptiste Coudurier's avatar Baptiste Coudurier

mov vfr muxing

Originally committed as revision 6854 to svn://svn.ffmpeg.org/ffmpeg/trunk
parent d55f46e5
...@@ -69,7 +69,6 @@ unassigned TODO: (unordered) ...@@ -69,7 +69,6 @@ unassigned TODO: (unordered)
- MPEG4 GMC encoding support - MPEG4 GMC encoding support
- macroblock based pixel format (better cache locality, somewhat complex, one paper claimed it faster for high res) - macroblock based pixel format (better cache locality, somewhat complex, one paper claimed it faster for high res)
- finish NUT implementation - finish NUT implementation
- movenc.c fix stts so that variable framerate files work
- seeking regression test - seeking regression test
- regression tests for codecs which dont have an encoder (I+P frame bitstream in svn) - regression tests for codecs which dont have an encoder (I+P frame bitstream in svn)
- add support for using mplayers video filters to ffmpeg - add support for using mplayers video filters to ffmpeg
......
...@@ -44,6 +44,7 @@ typedef struct MOVIentry { ...@@ -44,6 +44,7 @@ typedef struct MOVIentry {
char key_frame; char key_frame;
unsigned int entries; unsigned int entries;
int64_t cts; int64_t cts;
int64_t dts;
} MOVIentry; } MOVIentry;
typedef struct MOVIndex { typedef struct MOVIndex {
...@@ -65,6 +66,7 @@ typedef struct MOVIndex { ...@@ -65,6 +66,7 @@ typedef struct MOVIndex {
int vosLen; int vosLen;
uint8_t *vosData; uint8_t *vosData;
MOVIentry *cluster; MOVIentry *cluster;
int audio_vbr;
} MOVTrack; } MOVTrack;
typedef struct MOVContext { typedef struct MOVContext {
...@@ -704,18 +706,46 @@ static int mov_write_ctts_tag(ByteIOContext *pb, MOVTrack* track) ...@@ -704,18 +706,46 @@ static int mov_write_ctts_tag(ByteIOContext *pb, MOVTrack* track)
return atom_size; return atom_size;
} }
/* TODO: */
/* Time to sample atom */ /* Time to sample atom */
static int mov_write_stts_tag(ByteIOContext *pb, MOVTrack* track) static int mov_write_stts_tag(ByteIOContext *pb, MOVTrack* track)
{ {
put_be32(pb, 0x18); /* size */ Time2Sample *stts_entries;
uint32_t entries = -1;
uint32_t atom_size;
int i;
if (track->enc->codec_type == CODEC_TYPE_AUDIO && !track->audio_vbr) {
stts_entries = av_malloc(sizeof(*stts_entries)); /* one entry */
stts_entries[0].count = track->sampleCount;
stts_entries[0].duration = 1;
entries = 1;
} else {
stts_entries = av_malloc(track->entry * sizeof(*stts_entries)); /* worst case */
for (i=0; i<track->entry; i++) {
int64_t duration = i + 1 == track->entry ?
track->trackDuration - track->cluster[i].dts + track->cluster[0].dts : /* readjusting */
track->cluster[i+1].dts - track->cluster[i].dts;
if (i && duration == stts_entries[entries].duration) {
stts_entries[entries].count++; /* compress */
} else {
entries++;
stts_entries[entries].duration = duration;
stts_entries[entries].count = 1;
}
}
entries++; /* last one */
}
atom_size = 16 + (entries * 8);
put_be32(pb, atom_size); /* size */
put_tag(pb, "stts"); put_tag(pb, "stts");
put_be32(pb, 0); /* version & flags */ put_be32(pb, 0); /* version & flags */
put_be32(pb, 1); /* entry count */ put_be32(pb, entries); /* entry count */
for (i=0; i<entries; i++) {
put_be32(pb, track->sampleCount); /* sample count */ put_be32(pb, stts_entries[i].count);
put_be32(pb, track->sampleDuration); /* sample duration */ put_be32(pb, stts_entries[i].duration);
return 0x18; }
av_free(stts_entries);
return atom_size;
} }
static int mov_write_dref_tag(ByteIOContext *pb) static int mov_write_dref_tag(ByteIOContext *pb)
...@@ -1301,8 +1331,6 @@ static int mov_write_moov_tag(ByteIOContext *pb, MOVContext *mov, ...@@ -1301,8 +1331,6 @@ static int mov_write_moov_tag(ByteIOContext *pb, MOVContext *mov,
for (i=0; i<mov->nb_streams; i++) { for (i=0; i<mov->nb_streams; i++) {
if(mov->tracks[i].entry <= 0) continue; if(mov->tracks[i].entry <= 0) continue;
mov->tracks[i].trackDuration =
(int64_t)mov->tracks[i].sampleCount * mov->tracks[i].sampleDuration;
mov->tracks[i].time = mov->time; mov->tracks[i].time = mov->time;
mov->tracks[i].trackID = i+1; mov->tracks[i].trackID = i+1;
} }
...@@ -1470,7 +1498,16 @@ static int mov_write_header(AVFormatContext *s) ...@@ -1470,7 +1498,16 @@ static int mov_write_header(AVFormatContext *s)
track->timescale = st->codec->sample_rate; track->timescale = st->codec->sample_rate;
track->sampleDuration = st->codec->frame_size; track->sampleDuration = st->codec->frame_size;
av_set_pts_info(st, 64, 1, st->codec->sample_rate); av_set_pts_info(st, 64, 1, st->codec->sample_rate);
track->sampleSize = (av_get_bits_per_sample(st->codec->codec_id) >> 3) * st->codec->channels; switch(track->enc->codec_id){
case CODEC_ID_MP3:
case CODEC_ID_AAC:
case CODEC_ID_AMR_NB:
case CODEC_ID_AMR_WB:
track->audio_vbr = 1;
break;
default:
track->sampleSize = (av_get_bits_per_sample(st->codec->codec_id) >> 3) * st->codec->channels;
}
} }
if (!track->sampleDuration) { if (!track->sampleDuration) {
av_log(s, AV_LOG_ERROR, "track %d: sample duration is not set\n", i); av_log(s, AV_LOG_ERROR, "track %d: sample duration is not set\n", i);
...@@ -1543,6 +1580,9 @@ static int mov_write_packet(AVFormatContext *s, AVPacket *pkt) ...@@ -1543,6 +1580,9 @@ static int mov_write_packet(AVFormatContext *s, AVPacket *pkt)
trk->cluster[trk->entry].samplesInChunk = samplesInChunk; trk->cluster[trk->entry].samplesInChunk = samplesInChunk;
trk->cluster[trk->entry].size = size; trk->cluster[trk->entry].size = size;
trk->cluster[trk->entry].entries = samplesInChunk; trk->cluster[trk->entry].entries = samplesInChunk;
trk->cluster[trk->entry].dts = pkt->dts;
trk->trackDuration = pkt->dts - trk->cluster[0].dts + pkt->duration;
if(enc->codec_type == CODEC_TYPE_VIDEO) { if(enc->codec_type == CODEC_TYPE_VIDEO) {
if (pkt->dts != pkt->pts) if (pkt->dts != pkt->pts)
trk->hasBframes = 1; trk->hasBframes = 1;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment