如何用ffmpeg用msec寻找?

时间:2013-05-29 17:13:38

标签: c++ ffmpeg

我试图用ffmpeg在毫秒内搜索视频。我一直在尝试使用this question中的代码,该代码使用avformat_seek_file(我使用-1代表流号和AVSEEK_FLAG_ANY标志)。

在调用之后,我尝试读取下一帧,即:

if (av_read_frame(fmt_ctx, &pkt) >= 0)
{
    int ret = 0;

    if (pkt.stream_index == video_stream_idx) {
        /* decode video frame */
        ret = avcodec_decode_video2(video_dec_ctx, frame, got_frame, &pkt);
        if (ret < 0) {
            fprintf(stderr, "Error decoding video frame\n");
            return ret;
        }
//do something with frame
}

但是,检索到的帧的frame->pts始终保留在搜索之前读取的最后一帧之后的帧的时间。

编辑:尽管frame-&gt; pts形成了完整的序列,但确实会发生搜索。出于某些奇怪的原因,我读到的第一帧是第一帧。事实上,在我运行之后:

   int got_frame = 0;
   do
   if (av_read_frame(fmt_ctx, &pkt) >= 0) {
       decode_packet_ro(&got_frame, 0);
       av_free_packet(&pkt);
   }
   else
   {
       read_cache = true;
       pkt.data = NULL;
       pkt.size = 0;
       break;
   }
   while(!got_frame || this->frame->pts*av_q2d(video_dec_ctx->time_base) * 1000 < tsms);

我读的下一帧始终是第一帧。

1 个答案:

答案 0 :(得分:1)

最后,我能够使用以下代码寻找:

/*!
 * \brief ffmpeg_reader::seekMs seek to millisecond
 * \param tsms timestamp
 * \return success of seeking
 */
bool ffmpeg_reader::seekFrame(int s_frame)
{
    if (!isOk())
        return false;

   printf("\t avformat_seek_file to %d\n",s_frame);
   int flags = AVSEEK_FLAG_FRAME;
   if (s_frame < this->frame->pkt_dts)
   {
       flags |= AVSEEK_FLAG_BACKWARD;
   }

   if(av_seek_frame(fmt_ctx,video_stream_idx,s_frame,flags))
   {
       printf("\nFailed to seek for time %d",s_frame);
      return false;
   }

   avcodec_flush_buffers(video_dec_ctx);
   /*read frame without converting it*/
   int got_frame = 0;
   do
   if (av_read_frame(fmt_ctx, &pkt) == 0) {
       decode_packet(&got_frame, 0, false);
       av_free_packet(&pkt);
   }
   else
   {
       read_cache = true;
       pkt.data = NULL;
       pkt.size = 0;
       break;
   }
   while(!(got_frame && this->frame->pkt_dts >= s_frame));
   return true;
}

我自己并没有想出来,但我(遗憾地)不记得信用到期的地方。