Media Foundation 获取视频文件的确切帧(样本)计数

Media Foundation get exact frame ( sample ) count of video file

我有一个视频文件,我已成功提取平均帧率和持续时间:

UINT64 frameRate = 0;
mediaType->GetUINT64(MF_MT_FRAME_RATE, &frameRate)

PROPVARIANT prop;
m_pReader->GetPresentationAttribute(MF_SOURCE_READER_MEDIASOURCE,
    MF_PD_DURATION, &prop);
double duration = prop.hVal.QuadPart / 1e7;

现在我只是简单的计算帧数: frameCount = duration*frameRate

然后我通过迭代将其与实际样本数进行比较:

realFrameCount = 0;
while(true)
{
    m_pReader->ReadSample(
            MF_SOURCE_READER_FIRST_VIDEO_STREAM,
            0, &streamIndex, index.  &flags, &llTimeStamp, &videoSample);

        if (flags & MF_SOURCE_READERF_ENDOFSTREAM)
            break;

    realFrameCount++;
}

在某些情况下,我得到了完全匹配,但对于某些文件,我确实得到了很小的差异。

我的问题是: 有没有一种方法可以在不遍历所有样本的情况下获得准确的帧数?

谢谢

您假设您正在处理一个固定帧率文件并且元数据是准确的,然后文件没有 dropped/missing 帧。有时是这种情况,您的乘法会得到精确的值,而在其他时候,假设是不正确的。通常,您从文件元数据中读取了持续时间和帧时间值,它不必与您实际读取的帧的实际编号和参数相匹配。元数据是相当信息化的。

简短的回答是:您必须完成文件读取、跳过数据、计数样本。

Media Foundation 中的标准定位假定使用时间格式和每单位 100 ns 的刻度。 IMFSourceReader::SetCurrentPosition,一次,这样解释:

guidTimeFormat [in]

A GUID that specifies the time format. The time format defines the units for the varPosition parameter. The following value is defined for all media sources:

GUID_NULL - 100-nanosecond units.

Some media sources might support additional values.

也就是说,某些特定于格式的来源可能支持其他时间格式,例如视频帧,在这种情况下,您可以通过序号引用特定帧并获取以帧为单位的持续时间。我不知道是否有任何库存来源实际提供此功能。在前身 API 中,有 TIME_FORMAT_FRAME 用于此目的,我认为它在过渡到 Media Foundation 时被省略了。