C DTS和PTS不同,尽管没有B帧(ffmpeg)

C DTS和PTS不同,尽管没有B帧(ffmpeg),c,ffmpeg,video-encoding,multiplexing,C,Ffmpeg,Video Encoding,Multiplexing,这是一个关于视频编码/使用ffmpeg对视频文件进行muxing的快速问题。基本上,我有我的muxer功能,我试图让我的包输出正确的PTS/DTS 这是我的代码的一部分,对我的AVFrame进行编码,将其多路复用到输出文件中: int ret; int got_packet = 0; AVPacket pkt = { 0 }; av_init_packet(&pkt); pkt.data = NULL; pkt.size = 0; /* encode the image */ re

这是一个关于视频编码/使用ffmpeg对视频文件进行muxing的快速问题。基本上,我有我的muxer功能,我试图让我的包输出正确的PTS/DTS

这是我的代码的一部分,对我的AVFrame进行编码,将其多路复用到输出文件中:

int ret;
int got_packet = 0;

AVPacket pkt = { 0 };
av_init_packet(&pkt);

pkt.data = NULL;
pkt.size = 0;

/* encode the image */
ret = avcodec_encode_video2(cc, &pkt, frame, &got_packet);
if (ret < 0)
{
    fprintf(stderr, "error encoding video frame: %s\n", av_err2str(ret));
    exit(EXIT_FAILURE);
}

if (got_packet)
{
    av_packet_rescale_ts(&pkt, cc->time_base, st->time_base);

    fprintf(stderr, "\npkt.pts     = %ld\n", pkt.pts);
    fprintf(stderr,   "pkt.dts     = %ld\n", pkt.dts);
    fprintf(stderr, "writing frame\n");

    ret = av_interleaved_write_frame(fmt_ctx, &pkt);
    av_packet_unref(&pkt);
}
else
{
    ret = 0;
}

...
我的目标是让我的PST和DST都具有以下模式:1502303450606508

但第一个DTS值似乎重复了一次,因此与对应的PTS值不同步。还值得一提的是,编解码器上下文被配置为没有b帧,因此这里只存在i帧和p帧

有经验的人对此有什么见解吗

添加:

我在终端中运行了以下命令,以检查我的DTS和PTS值是否与我的打印语句一致:

sudo ./ffprobe -show_packets -print_format json mux_test.ts | less
我得到了以下信息:

pkt.pts     = 0
pkt.dts     = 0
writing frame

pkt.pts     = 1502
pkt.dts     = 0
writing frame

pkt.pts     = 3003
pkt.dts     = 1502
writing frame

pkt.pts     = 4505
pkt.dts     = 3003
writing frame

...
{
    "packets": [
        {
            "codec_type": "video",
            "stream_index": 0,
            "pts": 0,
            "pts_time": "0.000000",
            "dts": -1501,
            "dts_time": "-0.016678",
            "duration": 1501,
            "duration_time": "0.016678",
            "convergence_duration": "N/A",
            "convergence_duration_time": "N/A",
            "size": "55409",
            "pos": "564",
            "flags": "K"
        },
        {
            "codec_type": "video",
            "stream_index": 0,
            "pts": 1502,
            "pts_time": "0.016689",
            "dts": 0,
            "dts_time": "0.000000",
            "duration": 1501,
            "duration_time": "0.016678",
            "convergence_duration": "N/A",
            "convergence_duration_time": "N/A",
            "size": "46574",
            "pos": "60160",
            "flags": "_"
        },
        {
            "codec_type": "video",
            "stream_index": 0,
            "pts": 3003,
            "pts_time": "0.033367",
            "dts": 1502,
            "dts_time": "0.016689",
            "duration": 1501,
            "duration_time": "0.016678",
            "convergence_duration": "N/A",
            "convergence_duration_time": "N/A",
            "size": "2544",
            "pos": "110356",
            "flags": "_"
        },

        ...

它不会重复显示我的第一个DTS值,但会继续显示我的DTS,比我的PTS慢一个周期。

调试完API后,我得出了一个结论

要使DTS值有效,它们必须以单调、一致的速率增加(假设时基和帧速率在muxing过程中没有发生任何变化),因此值本身并不重要

这段代码直接来自ffmpeg库。可在2074-2080行的mpegvideo_enc.c中找到(为清晰起见,重新格式化):

如您所见,只有第一帧会进入if(!s->current\u picture.f->coded\u picture\u number)语句,coded\u picture\u number的值为0。随后的每一帧都将进入else语句,将当前DTS设置为等于先前的PTS值

因此,对于使用MPEG-2编码器的多路复用情况,这种行为似乎是正常的。DTS应落后“1个周期”

...

pkt->pts = s->current_picture.f->pts;
if (!s->low_delay && s->pict_type != AV_PICTURE_TYPE_B) 
{                                      
    if (!s->current_picture.f->coded_picture_number)
    {
        pkt->dts = pkt->pts - s->dts_delta;
    }
    else
    {
        pkt->dts = s->reordered_pts;
    }
    s->reordered_pts = pkt->pts;

...