1.avcodec_send_packet
int avcodec_send_packet(AVCodecContext *avctx,const AVPacket *avpkt);
第一个参数:原来解封装的时候有一个上下文,现在做解码有另外一个上下文,这个上下文贯穿于整个解码逻辑。
第二个参数:会把这个AVPacket放到我们的缓存中。
2.avcodec_receive_frame
int avcodec_receive_frame(AVCodecContext *avctx,AVFrame *frame);
第一个参数:同上。第二个参数:我们在做av_read_packet的时候,讲了AVPacket是由调用者分配的。AVFrame会在每次调用的时候把上次调用的空间清掉,然后重新分配。所以这块可以一直穿同一个对象。
总结:第一个接口只做一件事情,就是把packet写到一个解码队列中。
第二个接口做的事是从已经解码成功的数据中取出一个frame。
注意一点:我们第二个接口时立即获取已经解码的数据,而第一个接口取解码数据的时候因为在队列中有可能无法立即获取。所以前几帧可能会失败。send一次只能发一个包,不代表receive只能收到一个包。第二个接口一次可以receive多个包
看如下代码:
///解码视频
//发送packet到解码线程 send传NULL后调用多次receive取出所有缓冲帧
re = avcodec_send_packet(cc, pkt);
//释放,引用计数-1 为0释放空间
av_packet_unref(pkt);
if (re != 0)
{
char buf[1024] = { 0 };
av_strerror(re, buf, sizeof(buf) - 1);
cout << "avcodec_send_packet failed! :" << buf << endl;
continue;
}
for(;;)
{
//从线程中获取解码接口,一次send可能对应多次receive
re = avcodec_receive_frame(cc,frame);
if (re != 0) break;
cout << "recv frame " << frame->format << " " << frame->linesize[0] << endl;
}