如何解决IMFTransform SetInputType/ SetOutputType失败
我试图在Windows 7上使用WASAPI共享模式和媒体基础IMFSourceReader来播放MP3(和类似的音频文件)。据我了解,我必须在IMFSourceReader解码和WASAPI播放之间使用IMFTransform。除了我在IMFTransform上调用SetInputType()/ SetOutputType()之外,一切似乎都还不错?
相关的代码段为:
MFCreateSourceReaderFromURL(...); // Various test mp3 files
...
sourceReader->GetCurrentMediaType(MF_SOURCE_READER_FIRST_AUDIO_STREAM,&reader.audioType);
//sourceReader->GetNativeMediaType(MF_SOURCE_READER_FIRST_AUDIO_STREAM,&reader.audioType);
...
audioClient->GetMixFormat(&player.mixFormat);
...
MFCreateMediaType(&player.audioType);
MFInitMediaTypeFromWaveFormatEx(player.audioType,player.mixFormat,sizeof(WAVEFORMATEX) + player.mixFormat->cbSize);
...
hr = CoCreateInstance(CLSID_CResamplerMediaObject,NULL,CLSCTX_INPROC_SERVER,IID_IUnknown,(void**)&unknown);
ASSERT(SUCCEEDED(hr));
hr = unknown->QueryInterface(IID_PPV_ARGS(&resampler.transform));
ASSERT(SUCCEEDED(hr));
unknown->Release();
hr = resampler.transform->SetInputType(0,inType,0);
ASSERT(hr != DMO_E_INVALIDSTREAMINDEX);
ASSERT(hr != DMO_E_TYPE_NOT_ACCEPTED);
ASSERT(SUCCEEDED(hr)); // Fails here with hr = 0xc00d36b4
hr = resampler.transform->SetOutputType(0,outType,0);
ASSERT(hr != DMO_E_INVALIDSTREAMINDEX);
ASSERT(hr != DMO_E_TYPE_NOT_ACCEPTED);
ASSERT(SUCCEEDED(hr)); // Fails here with hr = 0xc00d6d60
我怀疑我误会了如何在事物之间协商IMFMediaType的输入/输出,以及如何考虑IMFTransform需要对未压缩的数据进行操作?
对于我来说,输出类型失败似乎很奇怪,但这也许是对输入类型首先失败的影响-如果我尝试先设置输出类型,它也会失败。
解决方法
在Windows的最新版本中,您可能希望利用已经为您提供的库存功能。
在配置Source Reader对象时,IMFSourceReader::SetCurrentMediaType
允许您指定数据要放入的媒体类型。如果设置的媒体类型与WASAPI要求兼容,Source Reader将自动添加转换以为您转换数据。 / p>
但是...
在Windows 8的源阅读器中添加了音频重采样支持。在Windows 8之前的Windows版本中,源阅读器不支持音频重采样。如果您需要在Windows 8之前的Windows版本中对音频进行重新采样,则可以使用Audio Resampler DSP。
...这意味着实际上您可能需要自己管理MFT。 MFT的输入媒体类型应该来自IMFSourceReader::GetCurrentMediaType
。要指示源阅读器使用未压缩的音频,您需要构建一种媒体类型解码器,此类型的流会将音频解码到该类型的流。例如,如果您的文件是MP3,则您将读取通道数,采样率并构建兼容的PCM媒体类型(或使用系统解码器并分别询问输出媒体类型,这甚至是一种更干净的方法)。您可以使用IMFSourceReader::SetCurrentMediaType
设置此未压缩的音频媒体类型。该媒体类型也将是您的音频重采样器MFT的输入媒体类型。这将指示源阅读器添加必要的解码器,而IMFSourceReader::ReadSample
将为您提供转换后的数据。
重新放大器MFT的输出媒体类型将从您从WASAPI获得的音频格式派生,并使用您在代码段顶部提到的API调用进行转换。
要查找错误代码,您可以使用以下代码:
通常,您还应该可以更轻松地使用Media Foundation Media Session API播放音频文件。 Media Session使用相同的原语来构建播放管道并负责格式调整。
嗯,你是说我需要创建一个附加的对象,该对象是解码器,以适合IMFSourceReader和IMFTransform / Resampler之间吗?
不。通过使用适当的媒体类型进行SetCurrentMediaType
,您可以在Source Reader内部添加解码器,以便可以为您提供已解压缩的数据。从Windows 8开始,它还可以在PCM风格之间进行转换,但是在Windows 7中,您需要自己使用Audio Resampler DSP来解决这一问题。
您可以自己管理解码器,但不必这样做,因为Source Reader的解码器可以更可靠地完成解码器。
您可能需要一个单独的解码器,以帮助您猜测将产生哪种PCM媒体类型解码器,以便您从Source Reader请求。 MFTEnumEx
是查找解码器的正确API。
我不确定如何决定或创建合适的解码器对象?我是否需要以某种方式枚举合适的列表,而不是假设特定的列表?
提到的MFTEnum
,MFTEnumEx
API调用可以枚举所有可用的解码器,或按给定条件过滤的解码器。
另一种方法是使用部分媒体类型(请参见此处的相关说明和代码段:Tutorial: Decoding Audio)。部分媒体类型是有关所需格式的信号,要求Media Foundation API提供与该部分类型匹配的原语。有关相关讨论链接,请参见下面的评论。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。