|
|
|
@ -1445,6 +1445,93 @@ static void generate_silence(uint8_t* buf, enum AVSampleFormat sample_fmt, size_ |
|
|
|
|
memset(buf, fill_char, size); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void flush_encoders(int ist_index, OutputStream *ost_table, int nb_ostreams) |
|
|
|
|
{ |
|
|
|
|
int i, ret; |
|
|
|
|
|
|
|
|
|
for (i = 0; i < nb_ostreams; i++) { |
|
|
|
|
OutputStream *ost = &ost_table[i]; |
|
|
|
|
|
|
|
|
|
if (ost->source_index == ist_index) { |
|
|
|
|
AVCodecContext *enc = ost->st->codec; |
|
|
|
|
AVFormatContext *os = output_files[ost->file_index].ctx; |
|
|
|
|
|
|
|
|
|
if(ost->st->codec->codec_type == AVMEDIA_TYPE_AUDIO && enc->frame_size <=1) |
|
|
|
|
continue; |
|
|
|
|
if(ost->st->codec->codec_type == AVMEDIA_TYPE_VIDEO && (os->oformat->flags & AVFMT_RAWPICTURE)) |
|
|
|
|
continue; |
|
|
|
|
|
|
|
|
|
if (ost->encoding_needed) { |
|
|
|
|
for(;;) { |
|
|
|
|
AVPacket pkt; |
|
|
|
|
int fifo_bytes; |
|
|
|
|
av_init_packet(&pkt); |
|
|
|
|
pkt.stream_index= ost->index; |
|
|
|
|
|
|
|
|
|
switch (ost->st->codec->codec_type) { |
|
|
|
|
case AVMEDIA_TYPE_AUDIO: |
|
|
|
|
fifo_bytes = av_fifo_size(ost->fifo); |
|
|
|
|
ret = 0; |
|
|
|
|
/* encode any samples remaining in fifo */ |
|
|
|
|
if (fifo_bytes > 0) { |
|
|
|
|
int osize = av_get_bytes_per_sample(enc->sample_fmt); |
|
|
|
|
int fs_tmp = enc->frame_size; |
|
|
|
|
|
|
|
|
|
av_fifo_generic_read(ost->fifo, audio_buf, fifo_bytes, NULL); |
|
|
|
|
if (enc->codec->capabilities & CODEC_CAP_SMALL_LAST_FRAME) { |
|
|
|
|
enc->frame_size = fifo_bytes / (osize * enc->channels); |
|
|
|
|
} else { /* pad */ |
|
|
|
|
int frame_bytes = enc->frame_size*osize*enc->channels; |
|
|
|
|
if (allocated_audio_buf_size < frame_bytes) |
|
|
|
|
exit_program(1); |
|
|
|
|
generate_silence(audio_buf+fifo_bytes, enc->sample_fmt, frame_bytes - fifo_bytes); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
ret = avcodec_encode_audio(enc, bit_buffer, bit_buffer_size, (short *)audio_buf); |
|
|
|
|
pkt.duration = av_rescale((int64_t)enc->frame_size*ost->st->time_base.den, |
|
|
|
|
ost->st->time_base.num, enc->sample_rate); |
|
|
|
|
enc->frame_size = fs_tmp; |
|
|
|
|
} |
|
|
|
|
if (ret <= 0) { |
|
|
|
|
ret = avcodec_encode_audio(enc, bit_buffer, bit_buffer_size, NULL); |
|
|
|
|
} |
|
|
|
|
if (ret < 0) { |
|
|
|
|
fprintf(stderr, "Audio encoding failed\n"); |
|
|
|
|
exit_program(1); |
|
|
|
|
} |
|
|
|
|
audio_size += ret; |
|
|
|
|
pkt.flags |= AV_PKT_FLAG_KEY; |
|
|
|
|
break; |
|
|
|
|
case AVMEDIA_TYPE_VIDEO: |
|
|
|
|
ret = avcodec_encode_video(enc, bit_buffer, bit_buffer_size, NULL); |
|
|
|
|
if (ret < 0) { |
|
|
|
|
fprintf(stderr, "Video encoding failed\n"); |
|
|
|
|
exit_program(1); |
|
|
|
|
} |
|
|
|
|
video_size += ret; |
|
|
|
|
if(enc->coded_frame && enc->coded_frame->key_frame) |
|
|
|
|
pkt.flags |= AV_PKT_FLAG_KEY; |
|
|
|
|
if (ost->logfile && enc->stats_out) { |
|
|
|
|
fprintf(ost->logfile, "%s", enc->stats_out); |
|
|
|
|
} |
|
|
|
|
break; |
|
|
|
|
default: |
|
|
|
|
ret=-1; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (ret <= 0) |
|
|
|
|
break; |
|
|
|
|
pkt.data = bit_buffer; |
|
|
|
|
pkt.size = ret; |
|
|
|
|
if (enc->coded_frame && enc->coded_frame->pts != AV_NOPTS_VALUE) |
|
|
|
|
pkt.pts= av_rescale_q(enc->coded_frame->pts, enc->time_base, ost->st->time_base); |
|
|
|
|
write_frame(os, &pkt, ost->st->codec, ost->bitstream_filters); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* pkt = NULL means EOF (needed to flush decoder buffers) */ |
|
|
|
|
static int output_packet(InputStream *ist, int ist_index, |
|
|
|
|
OutputStream *ost_table, int nb_ostreams, |
|
|
|
@ -1779,87 +1866,7 @@ static int output_packet(InputStream *ist, int ist_index, |
|
|
|
|
discard_packet: |
|
|
|
|
if (pkt == NULL) { |
|
|
|
|
/* EOF handling */ |
|
|
|
|
|
|
|
|
|
for(i=0;i<nb_ostreams;i++) { |
|
|
|
|
ost = &ost_table[i]; |
|
|
|
|
if (ost->source_index == ist_index) { |
|
|
|
|
AVCodecContext *enc= ost->st->codec; |
|
|
|
|
os = output_files[ost->file_index].ctx; |
|
|
|
|
|
|
|
|
|
if(ost->st->codec->codec_type == AVMEDIA_TYPE_AUDIO && enc->frame_size <=1) |
|
|
|
|
continue; |
|
|
|
|
if(ost->st->codec->codec_type == AVMEDIA_TYPE_VIDEO && (os->oformat->flags & AVFMT_RAWPICTURE)) |
|
|
|
|
continue; |
|
|
|
|
|
|
|
|
|
if (ost->encoding_needed) { |
|
|
|
|
for(;;) { |
|
|
|
|
AVPacket pkt; |
|
|
|
|
int fifo_bytes; |
|
|
|
|
av_init_packet(&pkt); |
|
|
|
|
pkt.stream_index= ost->index; |
|
|
|
|
|
|
|
|
|
switch(ost->st->codec->codec_type) { |
|
|
|
|
case AVMEDIA_TYPE_AUDIO: |
|
|
|
|
fifo_bytes = av_fifo_size(ost->fifo); |
|
|
|
|
ret = 0; |
|
|
|
|
/* encode any samples remaining in fifo */ |
|
|
|
|
if (fifo_bytes > 0) { |
|
|
|
|
int osize = av_get_bytes_per_sample(enc->sample_fmt); |
|
|
|
|
int fs_tmp = enc->frame_size; |
|
|
|
|
|
|
|
|
|
av_fifo_generic_read(ost->fifo, audio_buf, fifo_bytes, NULL); |
|
|
|
|
if (enc->codec->capabilities & CODEC_CAP_SMALL_LAST_FRAME) { |
|
|
|
|
enc->frame_size = fifo_bytes / (osize * enc->channels); |
|
|
|
|
} else { /* pad */ |
|
|
|
|
int frame_bytes = enc->frame_size*osize*enc->channels; |
|
|
|
|
if (allocated_audio_buf_size < frame_bytes) |
|
|
|
|
exit_program(1); |
|
|
|
|
generate_silence(audio_buf+fifo_bytes, enc->sample_fmt, frame_bytes - fifo_bytes); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
ret = avcodec_encode_audio(enc, bit_buffer, bit_buffer_size, (short *)audio_buf); |
|
|
|
|
pkt.duration = av_rescale((int64_t)enc->frame_size*ost->st->time_base.den, |
|
|
|
|
ost->st->time_base.num, enc->sample_rate); |
|
|
|
|
enc->frame_size = fs_tmp; |
|
|
|
|
} |
|
|
|
|
if(ret <= 0) { |
|
|
|
|
ret = avcodec_encode_audio(enc, bit_buffer, bit_buffer_size, NULL); |
|
|
|
|
} |
|
|
|
|
if (ret < 0) { |
|
|
|
|
fprintf(stderr, "Audio encoding failed\n"); |
|
|
|
|
exit_program(1); |
|
|
|
|
} |
|
|
|
|
audio_size += ret; |
|
|
|
|
pkt.flags |= AV_PKT_FLAG_KEY; |
|
|
|
|
break; |
|
|
|
|
case AVMEDIA_TYPE_VIDEO: |
|
|
|
|
ret = avcodec_encode_video(enc, bit_buffer, bit_buffer_size, NULL); |
|
|
|
|
if (ret < 0) { |
|
|
|
|
fprintf(stderr, "Video encoding failed\n"); |
|
|
|
|
exit_program(1); |
|
|
|
|
} |
|
|
|
|
video_size += ret; |
|
|
|
|
if(enc->coded_frame && enc->coded_frame->key_frame) |
|
|
|
|
pkt.flags |= AV_PKT_FLAG_KEY; |
|
|
|
|
if (ost->logfile && enc->stats_out) { |
|
|
|
|
fprintf(ost->logfile, "%s", enc->stats_out); |
|
|
|
|
} |
|
|
|
|
break; |
|
|
|
|
default: |
|
|
|
|
ret=-1; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if(ret<=0) |
|
|
|
|
break; |
|
|
|
|
pkt.data= bit_buffer; |
|
|
|
|
pkt.size= ret; |
|
|
|
|
if(enc->coded_frame && enc->coded_frame->pts != AV_NOPTS_VALUE) |
|
|
|
|
pkt.pts= av_rescale_q(enc->coded_frame->pts, enc->time_base, ost->st->time_base); |
|
|
|
|
write_frame(os, &pkt, ost->st->codec, ost->bitstream_filters); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
flush_encoders(ist_index, ost_table, nb_ostreams); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return 0; |
|
|
|
|