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 时被省略了。
我有一个视频文件,我已成功提取平均帧率和持续时间:
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 时被省略了。