我试图在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, 0, &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将自动添加转换以为您转换数据。
然而...

...这意味着您确实可能需要自己管理MFT。 MFT的输入媒体类型应该来自IMFSourceReader::GetCurrentMediaType。要指示源阅读器使用未压缩的音频,您需要构建一种媒体类型解码器,此类型的流将音频解码为该类型的流。例如,如果您的文件是MP3,则您将读取通道数,采样率并构建兼容的PCM媒体类型(或使用系统解码器并分别询问输出媒体类型,这甚至是一种更干净的方法)。您可以使用IMFSourceReader::SetCurrentMediaType设置此未压缩的音频媒体类型。该媒体类型也将是您的音频重采样器MFT的输入媒体类型。这将指示源阅读器添加必要的解码器,并且IMFSourceReader::ReadSample将为您提供转换后的数据。
重新放大器MFT的输出媒体类型将从您从WASAPI获得的音频格式派生,并使用您在代码段顶部提到的API调用进行转换。
要查找错误代码,可以使用以下命令:

  • https://www.magnumdb.com/search?q=0xc00d36b4
  • https://www.magnumdb.com/search?q=0xc00d6d60

  • 而且,通常,您应该可以使用Media Foundation Media Session API花费较小的精力来播放音频文件。 Media Session使用相同的原语来构建播放管道并负责格式调整。

    否。通过使用适当的媒体类型来执行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提供与该部分类型匹配的原语。有关相关讨论链接,请参见下面的评论。

    关于audio - IMFTransform SetInputType()/SetOutputType()失败,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/64593597/

    10-10 11:36