movenc: fragment support
Reviewed-by: Baptiste Coudurier Tested-with-QT: Baptiste Coudurier Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
parent
c89488588b
commit
cf8ec019ab
@ -45,6 +45,7 @@ static const AVOption options[] = {
|
|||||||
{ "movflags", "MOV muxer flags", offsetof(MOVMuxContext, flags), AV_OPT_TYPE_FLAGS, {.dbl = 0}, INT_MIN, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, "movflags" },
|
{ "movflags", "MOV muxer flags", offsetof(MOVMuxContext, flags), AV_OPT_TYPE_FLAGS, {.dbl = 0}, INT_MIN, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, "movflags" },
|
||||||
{ "rtphint", "Add RTP hint tracks", 0, AV_OPT_TYPE_CONST, {.dbl = FF_MOV_FLAG_RTP_HINT}, INT_MIN, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, "movflags" },
|
{ "rtphint", "Add RTP hint tracks", 0, AV_OPT_TYPE_CONST, {.dbl = FF_MOV_FLAG_RTP_HINT}, INT_MIN, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, "movflags" },
|
||||||
{ "moov_size", "maximum moov size so it can be placed at the begin", offsetof(MOVMuxContext, reserved_moov_size), AV_OPT_TYPE_INT, {.dbl = 0}, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, 0 },
|
{ "moov_size", "maximum moov size so it can be placed at the begin", offsetof(MOVMuxContext, reserved_moov_size), AV_OPT_TYPE_INT, {.dbl = 0}, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, 0 },
|
||||||
|
{ "frag_duration", "maximum fragment duration", offsetof(MOVMuxContext, max_fragment_duration), AV_OPT_TYPE_INT, {.dbl = 0}, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM, 0 },
|
||||||
FF_RTP_FLAG_OPTS(MOVMuxContext, rtp_flags),
|
FF_RTP_FLAG_OPTS(MOVMuxContext, rtp_flags),
|
||||||
{ NULL },
|
{ NULL },
|
||||||
};
|
};
|
||||||
@ -57,6 +58,8 @@ static const AVClass flavor ## _muxer_class = {\
|
|||||||
.version = LIBAVUTIL_VERSION_INT,\
|
.version = LIBAVUTIL_VERSION_INT,\
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static int update_first_fragment(AVFormatContext *s);
|
||||||
|
|
||||||
//FIXME support 64 bit variant with wide placeholders
|
//FIXME support 64 bit variant with wide placeholders
|
||||||
static int64_t updateSize(AVIOContext *pb, int64_t pos)
|
static int64_t updateSize(AVIOContext *pb, int64_t pos)
|
||||||
{
|
{
|
||||||
@ -1391,13 +1394,14 @@ static int mov_write_udta_sdp(AVIOContext *pb, AVFormatContext *ctx, int index)
|
|||||||
return len + 24;
|
return len + 24;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mov_write_trak_tag(AVIOContext *pb, MOVTrack *track, AVStream *st)
|
static int mov_write_trak_tag(AVIOContext *pb, MOVMuxContext *mov, MOVTrack *track, AVStream *st)
|
||||||
{
|
{
|
||||||
int64_t pos = avio_tell(pb);
|
int64_t pos = avio_tell(pb);
|
||||||
avio_wb32(pb, 0); /* size */
|
avio_wb32(pb, 0); /* size */
|
||||||
ffio_wfourcc(pb, "trak");
|
ffio_wfourcc(pb, "trak");
|
||||||
mov_write_tkhd_tag(pb, track, st);
|
mov_write_tkhd_tag(pb, track, st);
|
||||||
if (track->mode == MODE_PSP || track->flags & MOV_TRACK_CTTS || track->cluster[0].dts)
|
if (track->mode == MODE_PSP || track->flags & MOV_TRACK_CTTS || track->cluster[0].dts)
|
||||||
|
if(!mov->fragments) // EDTS with fragments is tricky as we dont know the duration when its written
|
||||||
mov_write_edts_tag(pb, track); // PSP Movies require edts box
|
mov_write_edts_tag(pb, track); // PSP Movies require edts box
|
||||||
if (track->tref_tag)
|
if (track->tref_tag)
|
||||||
mov_write_tref_tag(pb, track);
|
mov_write_tref_tag(pb, track);
|
||||||
@ -1414,6 +1418,63 @@ static int mov_write_trak_tag(AVIOContext *pb, MOVTrack *track, AVStream *st)
|
|||||||
return updateSize(pb, pos);
|
return updateSize(pb, pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mov_write_tfhd_tag(AVIOContext *pb, MOVTrack *track, AVStream *st)
|
||||||
|
{
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
int flags=1;
|
||||||
|
|
||||||
|
avio_wb32(pb, 0);
|
||||||
|
ffio_wfourcc(pb, "tfhd");
|
||||||
|
avio_w8(pb, 0);
|
||||||
|
avio_wb24(pb, flags);
|
||||||
|
avio_wb32(pb, track->trackID);
|
||||||
|
|
||||||
|
track->base_data_offset_pos= avio_tell(pb);
|
||||||
|
if (flags & 0x01) avio_wb64(pb, 0);
|
||||||
|
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mov_write_trun_tag(AVIOContext *pb, MOVTrack *track)
|
||||||
|
{
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
int sample_count= track->entry - track->cluster_write_index;
|
||||||
|
int tr_flags=0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
tr_flags |= 0xF00; //FIXME
|
||||||
|
|
||||||
|
avio_wb32(pb, 0);
|
||||||
|
ffio_wfourcc(pb, "trun");
|
||||||
|
avio_w8(pb, 0);
|
||||||
|
avio_wb24(pb, tr_flags);
|
||||||
|
avio_wb32(pb, sample_count);
|
||||||
|
if(tr_flags&1) avio_wb32(pb, 0);
|
||||||
|
|
||||||
|
for(i=track->cluster_write_index; 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(tr_flags&0x100) avio_wb32(pb, duration);
|
||||||
|
if(tr_flags&0x200) avio_wb32(pb, track->cluster[i].size);
|
||||||
|
if(tr_flags&0x400) avio_wb32(pb, (track->cluster[i].flags&MOV_SYNC_SAMPLE) ? 0x02000000 : 0x01010000);
|
||||||
|
if(tr_flags&0x800) avio_wb32(pb, track->cluster[i].cts);
|
||||||
|
}
|
||||||
|
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mov_write_traf_tag(AVIOContext *pb, MOVTrack *track, AVStream *st)
|
||||||
|
{
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
avio_wb32(pb, 0); /* size */
|
||||||
|
ffio_wfourcc(pb, "traf");
|
||||||
|
mov_write_tfhd_tag(pb, track, st);
|
||||||
|
mov_write_trun_tag(pb, track);
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
/* TODO: Not sorted out, but not necessary either */
|
/* TODO: Not sorted out, but not necessary either */
|
||||||
static int mov_write_iods_tag(AVIOContext *pb, MOVMuxContext *mov)
|
static int mov_write_iods_tag(AVIOContext *pb, MOVMuxContext *mov)
|
||||||
@ -1827,6 +1888,39 @@ static void build_chunks(MOVTrack *trk)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mov_write_trex_tag(AVIOContext *pb, MOVTrack *track, AVStream *st)
|
||||||
|
{
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
avio_wb32(pb, 0); /* size */
|
||||||
|
ffio_wfourcc(pb, "trex");
|
||||||
|
|
||||||
|
avio_w8(pb, 0);
|
||||||
|
avio_wb24(pb, 0);
|
||||||
|
avio_wb32(pb, track->trackID);
|
||||||
|
avio_wb32(pb, 1); // stsd_id
|
||||||
|
avio_wb32(pb, 1); // duration
|
||||||
|
avio_wb32(pb, 1/*Size*/);
|
||||||
|
avio_wb32(pb, 1<<16);
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mov_write_mvex_tag(AVIOContext *pb, MOVMuxContext *mov,
|
||||||
|
AVFormatContext *s)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
avio_wb32(pb, 0); /* size placeholder*/
|
||||||
|
ffio_wfourcc(pb, "mvex");
|
||||||
|
|
||||||
|
for (i=0; i<mov->nb_streams; i++) {
|
||||||
|
if(mov->tracks[i].entry > 0) {
|
||||||
|
mov_write_trex_tag(pb, &(mov->tracks[i]), i < s->nb_streams ? s->streams[i] : NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
static int mov_write_moov_tag(AVIOContext *pb, MOVMuxContext *mov,
|
static int mov_write_moov_tag(AVIOContext *pb, MOVMuxContext *mov,
|
||||||
AVFormatContext *s)
|
AVFormatContext *s)
|
||||||
{
|
{
|
||||||
@ -1861,10 +1955,13 @@ static int mov_write_moov_tag(AVIOContext *pb, MOVMuxContext *mov,
|
|||||||
//mov_write_iods_tag(pb, mov);
|
//mov_write_iods_tag(pb, mov);
|
||||||
for (i=0; i<mov->nb_streams; i++) {
|
for (i=0; i<mov->nb_streams; i++) {
|
||||||
if(mov->tracks[i].entry > 0) {
|
if(mov->tracks[i].entry > 0) {
|
||||||
mov_write_trak_tag(pb, &(mov->tracks[i]), i < s->nb_streams ? s->streams[i] : NULL);
|
mov_write_trak_tag(pb, mov, &(mov->tracks[i]), i < s->nb_streams ? s->streams[i] : NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(mov->fragments)
|
||||||
|
mov_write_mvex_tag(pb, mov, s);
|
||||||
|
|
||||||
if (mov->mode == MODE_PSP)
|
if (mov->mode == MODE_PSP)
|
||||||
mov_write_uuidusmt_tag(pb, s);
|
mov_write_uuidusmt_tag(pb, s);
|
||||||
else
|
else
|
||||||
@ -1873,6 +1970,36 @@ static int mov_write_moov_tag(AVIOContext *pb, MOVMuxContext *mov,
|
|||||||
return updateSize(pb, pos);
|
return updateSize(pb, pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mov_write_mfhd_tag(AVIOContext *pb, MOVMuxContext *mov)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
avio_wb32(pb, 0); /* size placeholder*/
|
||||||
|
ffio_wfourcc(pb, "mfhd");
|
||||||
|
avio_wb32(pb, 0);
|
||||||
|
avio_wb32(pb, mov->frag_seq_num++);
|
||||||
|
|
||||||
|
return updateSize(pb, pos); //FIXME replace by hardcoded num also above
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mov_write_moof_tag(AVIOContext *pb, MOVMuxContext *mov,
|
||||||
|
AVFormatContext *s)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int64_t pos = avio_tell(pb);
|
||||||
|
avio_wb32(pb, 0); /* size placeholder*/
|
||||||
|
ffio_wfourcc(pb, "moof");
|
||||||
|
|
||||||
|
mov_write_mfhd_tag(pb, mov);
|
||||||
|
for (i=0; i<mov->nb_streams; i++) {
|
||||||
|
if(mov->tracks[i].entry > 0) {
|
||||||
|
mov_write_traf_tag(pb, &(mov->tracks[i]), i < s->nb_streams ? s->streams[i] : NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return updateSize(pb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
static int mov_write_mdat_tag(AVIOContext *pb, MOVMuxContext *mov)
|
static int mov_write_mdat_tag(AVIOContext *pb, MOVMuxContext *mov)
|
||||||
{
|
{
|
||||||
avio_wb32(pb, 8); // placeholder for extended size field (64 bit)
|
avio_wb32(pb, 8); // placeholder for extended size field (64 bit)
|
||||||
@ -2023,11 +2150,53 @@ static int mov_parse_mpeg2_frame(AVPacket *pkt, uint32_t *flags)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int flush_cluster_buffer(AVFormatContext *s){
|
||||||
|
MOVMuxContext *mov = s->priv_data;
|
||||||
|
int i, j;
|
||||||
|
int has_data=0;
|
||||||
|
|
||||||
|
for (i=0; i<mov->nb_streams; i++){
|
||||||
|
MOVTrack *track= &mov->tracks[i];
|
||||||
|
if(track->entry != track->cluster_write_index)
|
||||||
|
has_data=1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(mov->frag_seq_num==0){
|
||||||
|
update_first_fragment(s);
|
||||||
|
mov->frag_seq_num++;
|
||||||
|
}else if(has_data) {
|
||||||
|
mov_write_moof_tag(s->pb, mov, s);
|
||||||
|
mov_write_mdat_tag(s->pb, mov);
|
||||||
|
for (i=0; i<mov->nb_streams; i++){
|
||||||
|
MOVTrack *track= &mov->tracks[i];
|
||||||
|
if(track->entry > 0) {
|
||||||
|
int64_t pos= avio_tell(s->pb);
|
||||||
|
avio_seek(s->pb, track->base_data_offset_pos, SEEK_SET);
|
||||||
|
avio_wb64(s->pb, pos);
|
||||||
|
avio_seek(s->pb, pos, SEEK_SET);
|
||||||
|
for(j=track->cluster_write_index; j<track->entry; j++){
|
||||||
|
avio_write(s->pb, track->cluster[j].data, track->cluster[j].size);
|
||||||
|
av_freep(&track->cluster[j].data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
updateSize(s->pb, mov->mdat_pos);
|
||||||
|
}
|
||||||
|
for (i=0; i<mov->nb_streams; i++) {
|
||||||
|
MOVTrack *track= &mov->tracks[i];
|
||||||
|
track->cluster_write_index= track->entry;
|
||||||
|
}
|
||||||
|
avio_flush(s->pb);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int ff_mov_write_packet(AVFormatContext *s, AVPacket *pkt)
|
int ff_mov_write_packet(AVFormatContext *s, AVPacket *pkt)
|
||||||
{
|
{
|
||||||
MOVMuxContext *mov = s->priv_data;
|
MOVMuxContext *mov = s->priv_data;
|
||||||
AVIOContext *pb = s->pb;
|
AVIOContext *pb = s->pb;
|
||||||
MOVTrack *trk = &mov->tracks[pkt->stream_index];
|
MOVTrack *trk = &mov->tracks[pkt->stream_index];
|
||||||
|
AVStream *st = s->streams[pkt->stream_index];
|
||||||
AVCodecContext *enc = trk->enc;
|
AVCodecContext *enc = trk->enc;
|
||||||
unsigned int samplesInChunk = 0;
|
unsigned int samplesInChunk = 0;
|
||||||
int size= pkt->size;
|
int size= pkt->size;
|
||||||
@ -2035,6 +2204,11 @@ int ff_mov_write_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
if (!s->pb->seekable) return 0; /* Can't handle that */
|
if (!s->pb->seekable) return 0; /* Can't handle that */
|
||||||
if (!size) return 0; /* Discard 0 sized packets */
|
if (!size) return 0; /* Discard 0 sized packets */
|
||||||
|
|
||||||
|
if (mov->fragments && trk->entry > trk->cluster_write_index &&
|
||||||
|
av_rescale_q(pkt->dts - trk->cluster[ trk->cluster_write_index ].dts, st->time_base, AV_TIME_BASE_Q) >= mov->max_fragment_duration){
|
||||||
|
flush_cluster_buffer(s);
|
||||||
|
}
|
||||||
|
|
||||||
if (enc->codec_id == CODEC_ID_AMR_NB) {
|
if (enc->codec_id == CODEC_ID_AMR_NB) {
|
||||||
/* We must find out how many AMR blocks there are in one packet */
|
/* We must find out how many AMR blocks there are in one packet */
|
||||||
static uint16_t packed_size[16] =
|
static uint16_t packed_size[16] =
|
||||||
@ -2064,14 +2238,34 @@ int ff_mov_write_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
memcpy(trk->vosData, enc->extradata, trk->vosLen);
|
memcpy(trk->vosData, enc->extradata, trk->vosLen);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!(trk->entry % MOV_INDEX_CLUSTER_SIZE)) {
|
||||||
|
trk->cluster = av_realloc_f(trk->cluster, sizeof(*trk->cluster), (trk->entry + MOV_INDEX_CLUSTER_SIZE));
|
||||||
|
if (!trk->cluster)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
if (enc->codec_id == CODEC_ID_H264 && trk->vosLen > 0 && *(uint8_t *)trk->vosData != 1) {
|
if (enc->codec_id == CODEC_ID_H264 && trk->vosLen > 0 && *(uint8_t *)trk->vosData != 1) {
|
||||||
/* from x264 or from bytestream h264 */
|
/* from x264 or from bytestream h264 */
|
||||||
/* nal reformating needed */
|
/* nal reformating needed */
|
||||||
|
if(mov->frag_seq_num>0){
|
||||||
|
uint8_t *buf=NULL;
|
||||||
|
size= pkt->size;
|
||||||
|
if(ff_avc_parse_nal_units_buf(pkt->data, &buf, &size) < 0){
|
||||||
|
av_log(s, AV_LOG_ERROR, "malformated H264 bitstream\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
trk->cluster[trk->entry].data= buf;
|
||||||
|
}else
|
||||||
size = ff_avc_parse_nal_units(pb, pkt->data, pkt->size);
|
size = ff_avc_parse_nal_units(pb, pkt->data, pkt->size);
|
||||||
} else if (enc->codec_id == CODEC_ID_AAC && pkt->size > 2 &&
|
} else if (enc->codec_id == CODEC_ID_AAC && pkt->size > 2 &&
|
||||||
(AV_RB16(pkt->data) & 0xfff0) == 0xfff0) {
|
(AV_RB16(pkt->data) & 0xfff0) == 0xfff0) {
|
||||||
av_log(s, AV_LOG_ERROR, "malformated aac bitstream, use -absf aac_adtstoasc\n");
|
av_log(s, AV_LOG_ERROR, "malformated aac bitstream, use -absf aac_adtstoasc\n");
|
||||||
return -1;
|
return -1;
|
||||||
|
} else if (mov->frag_seq_num>0){
|
||||||
|
trk->cluster[trk->entry].data = av_malloc(size);
|
||||||
|
if (!trk->cluster[trk->entry].data)
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
memcpy(trk->cluster[trk->entry].data, pkt->data, size);
|
||||||
} else {
|
} else {
|
||||||
avio_write(pb, pkt->data, size);
|
avio_write(pb, pkt->data, size);
|
||||||
}
|
}
|
||||||
@ -2086,13 +2280,7 @@ int ff_mov_write_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
memcpy(trk->vosData, pkt->data, size);
|
memcpy(trk->vosData, pkt->data, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(trk->entry % MOV_INDEX_CLUSTER_SIZE)) {
|
trk->cluster[trk->entry].pos = avio_tell(pb) - (mov->frag_seq_num==0 ? size : 0);
|
||||||
trk->cluster = av_realloc_f(trk->cluster, sizeof(*trk->cluster), (trk->entry + MOV_INDEX_CLUSTER_SIZE));
|
|
||||||
if (!trk->cluster)
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
trk->cluster[trk->entry].pos = avio_tell(pb) - size;
|
|
||||||
trk->cluster[trk->entry].samplesInChunk = samplesInChunk;
|
trk->cluster[trk->entry].samplesInChunk = samplesInChunk;
|
||||||
trk->cluster[trk->entry].chunkNum = 0;
|
trk->cluster[trk->entry].chunkNum = 0;
|
||||||
trk->cluster[trk->entry].size = size;
|
trk->cluster[trk->entry].size = size;
|
||||||
@ -2181,6 +2369,10 @@ static int mov_write_header(AVFormatContext *s)
|
|||||||
/* Default mode == MP4 */
|
/* Default mode == MP4 */
|
||||||
mov->mode = MODE_MP4;
|
mov->mode = MODE_MP4;
|
||||||
|
|
||||||
|
if(mov->max_fragment_duration){
|
||||||
|
mov->fragments= 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (s->oformat != NULL) {
|
if (s->oformat != NULL) {
|
||||||
if (!strcmp("3gp", s->oformat->name)) mov->mode = MODE_3GP;
|
if (!strcmp("3gp", s->oformat->name)) mov->mode = MODE_3GP;
|
||||||
else if (!strcmp("3g2", s->oformat->name)) mov->mode = MODE_3GP|MODE_3G2;
|
else if (!strcmp("3g2", s->oformat->name)) mov->mode = MODE_3GP|MODE_3G2;
|
||||||
@ -2339,7 +2531,7 @@ static int mov_write_header(AVFormatContext *s)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mov_write_trailer(AVFormatContext *s)
|
static int update_first_fragment(AVFormatContext *s)
|
||||||
{
|
{
|
||||||
MOVMuxContext *mov = s->priv_data;
|
MOVMuxContext *mov = s->priv_data;
|
||||||
AVIOContext *pb = s->pb;
|
AVIOContext *pb = s->pb;
|
||||||
@ -2375,6 +2567,18 @@ static int mov_write_trailer(AVFormatContext *s)
|
|||||||
avio_seek(pb, moov_pos, SEEK_SET);
|
avio_seek(pb, moov_pos, SEEK_SET);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mov_write_trailer(AVFormatContext *s)
|
||||||
|
{
|
||||||
|
MOVMuxContext *mov = s->priv_data;
|
||||||
|
AVIOContext *pb = s->pb;
|
||||||
|
int res = 0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
flush_cluster_buffer(s);
|
||||||
|
|
||||||
if (mov->chapter_track)
|
if (mov->chapter_track)
|
||||||
av_freep(&mov->tracks[mov->chapter_track].enc);
|
av_freep(&mov->tracks[mov->chapter_track].enc);
|
||||||
|
|
||||||
|
|||||||
@ -50,6 +50,7 @@ typedef struct MOVIentry {
|
|||||||
#define MOV_SYNC_SAMPLE 0x0001
|
#define MOV_SYNC_SAMPLE 0x0001
|
||||||
#define MOV_PARTIAL_SYNC_SAMPLE 0x0002
|
#define MOV_PARTIAL_SYNC_SAMPLE 0x0002
|
||||||
uint32_t flags;
|
uint32_t flags;
|
||||||
|
uint8_t *data;
|
||||||
} MOVIentry;
|
} MOVIentry;
|
||||||
|
|
||||||
typedef struct HintSample {
|
typedef struct HintSample {
|
||||||
@ -86,6 +87,7 @@ typedef struct MOVIndex {
|
|||||||
|
|
||||||
int vosLen;
|
int vosLen;
|
||||||
uint8_t *vosData;
|
uint8_t *vosData;
|
||||||
|
int cluster_write_index;
|
||||||
MOVIentry *cluster;
|
MOVIentry *cluster;
|
||||||
int audio_vbr;
|
int audio_vbr;
|
||||||
int height; ///< active picture (w/o VBI) height for D-10/IMX
|
int height; ///< active picture (w/o VBI) height for D-10/IMX
|
||||||
@ -98,6 +100,7 @@ typedef struct MOVIndex {
|
|||||||
uint32_t prev_rtp_ts;
|
uint32_t prev_rtp_ts;
|
||||||
int64_t cur_rtp_ts_unwrapped;
|
int64_t cur_rtp_ts_unwrapped;
|
||||||
uint32_t max_packet_size;
|
uint32_t max_packet_size;
|
||||||
|
int64_t base_data_offset_pos;
|
||||||
|
|
||||||
HintSampleQueue sample_queue;
|
HintSampleQueue sample_queue;
|
||||||
} MOVTrack;
|
} MOVTrack;
|
||||||
@ -111,11 +114,14 @@ typedef struct MOVMuxContext {
|
|||||||
int64_t mdat_pos;
|
int64_t mdat_pos;
|
||||||
uint64_t mdat_size;
|
uint64_t mdat_size;
|
||||||
MOVTrack *tracks;
|
MOVTrack *tracks;
|
||||||
|
int fragments;
|
||||||
|
int frag_seq_num;
|
||||||
|
|
||||||
int flags;
|
int flags;
|
||||||
int rtp_flags;
|
int rtp_flags;
|
||||||
int reserved_moov_size;
|
int reserved_moov_size;
|
||||||
int64_t reserved_moov_pos;
|
int64_t reserved_moov_pos;
|
||||||
|
int max_fragment_duration;
|
||||||
} MOVMuxContext;
|
} MOVMuxContext;
|
||||||
|
|
||||||
#define FF_MOV_FLAG_RTP_HINT 1
|
#define FF_MOV_FLAG_RTP_HINT 1
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user