首页 > 代码库 > 从ffmpeg filter里出来的数据直接送给avcodec_encode_audio2编码,写文件有错。
从ffmpeg filter里出来的数据直接送给avcodec_encode_audio2编码,写文件有错。
http://hi.baidu.com/mingyuejingque/item/78e71aff57ae9ec5a835a2e4
感谢mingyuejingque
st = avformat_new_stream( m_oc, NULL);
if (!st) {
fprintf(stderr, "Could not allocate stream\n");
exit(1);
}
st->id = m_oc->nb_streams-1;
AVCodecContext *c = st->codec;
c->codec_type = in_stream->codec->codec_type;
/* Some formats want stream headers to be separate. */
if (m_oc->oformat->flags & AVFMT_GLOBALHEADER)
c->flags |= CODEC_FLAG_GLOBAL_HEADER;
switch ((codec)->type) {
case AVMEDIA_TYPE_AUDIO:
c->sample_fmt = FORCE_smaple_fmt;
// c->bit_rate = FORCE_BIT_RATE;
c->sample_rate = FORCE_sample_rate;
c->codec_id = FORCE_CODEC_AUDIO;
c->channels = in_stream->codec->channels;
c->channel_layout = in_stream->codec->channel_layout;
c->time_base = in_stream->codec->time_base;
// c->profile = FORCE_profile;
m_AudioStream = st;
/* open it */
ret = avcodec_open2(c, codec, NULL);
if (ret < 0) {
printf("Could not open audio codec");
exit(1);
}
m_Codec = codec;
m_CodecContext = c; //这是音频编码的,视频没做编码。
break;
}
/* Some formats want stream headers to be separate. */
if (m_oc->oformat->flags & AVFMT_GLOBALHEADER)
c->flags |= CODEC_FLAG_GLOBAL_HEADER;
叼,这样写的代码有个潜规则,那就是:
if (av_buffersrc_add_frame_flags(m_filter_in_param.buffersrc_ctx, in_frame, 0) < 0) {
av_log(NULL, AV_LOG_ERROR, "Error while feeding the audio filtergraph\n");
goto quit;
}
while (1) {
ret = av_buffersink_get_frame(m_filter_in_param.buffersink_ctx, filt_frame);
if(ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
{
ret = -1;
goto quit;
}
if(ret < 0)
goto quit;
ret = avcodec_encode_audio2(m_CodecContext, new_packet, filt_frame, &got_frame);
av_frame_free( &filt_frame );
if ( got_frame )
goto quit;
}
从这里编码出来的packet,让ffmpeg写到文件里去的时候会报错:
[mp4 @ 0119ad00] Malformed AAC bitstream detected: use audio bitstream filter
‘aac_adtstoasc‘ to fix it (‘-bsf:a aac_adtstoasc‘ option with ffmpeg)
小孩子要什么filter来修正啊,叼,试了大半天也不知道哪个合适,并且也不想随便就加filter下去,万一它消耗太多cpu怎么办。
又是看ffmpeg的源码,avcodec_open2 看到返回0是正常的,内部调用了c->codec_init() 我也看到aacenc.c里去了,用反证法:它里面要是说参数又问题,肯定会返回负值,既然返回0就说明它是正常执行的。
AVCodec *codec = avcodec_find_encoder( codecid ); //这个没什么好说,下面又没对codec做什么改变。
下面的AVCodecContext *c = st->codec;你就诡异了,
/* open it */
ret = avcodec_open2(c, codec, NULL);
if (ret < 0) {
printf("Could not open audio codec");
exit(1);
}
严重怀疑c,c就是stream->codec,这个stream是在AVFormatContext *m_oc里的流。 跟到ffmpeg源码里去,发现是:
c->flags |= CODEC_FLAG_GLOBAL_HEADER; 这个标志位没设置,叼,一般这个都是写在函数末尾,这里先提前了,放到avcodec_open2前面去,这肯定是我粗心大意?
/* Some formats want stream headers to be separate. */
if (m_oc->oformat->flags & AVFMT_GLOBALHEADER)
c->flags |= CODEC_FLAG_GLOBAL_HEADER;
avcodec_open2( c, codec, NULL ); 哟西,没事了。假装没事发生。。。。。。。。