c++ - 将 WAVEFORMATEXTENSIBLE 与 WAVE_FORMAT_IEEE_FLOAT 一起使用,waveOutOpen 返回 WAVERR_BADFORMAT
问题描述
我一直在使用WAVEFORMATEX
WaveOut 在 Windows 中以 44.1KHz 到 192KHz 的速率播放音频WAVE_FORMAT_IEEE_FLOAT
。该程序用 C++ 编写并在 MinGW 中编译。这一切正常:
现在我正在尝试扩展到四声道多声道输出,这似乎需要WAVEFORMATEXTENSIBLE
. 的超集WAVEFORMATEX
。以下是应用了这些更改的相关代码:
WAVEFORMATEXTENSIBLE wfx;
memset( &wfx, 0, sizeof(wfx) );
wfx.Format.nChannels = want.channels;
wfx.dwChannelMask = (want.channels == 4) ? 0x33 : ((want.channels == 1) ? 0x4 : 0x3);
wfx.Format.nSamplesPerSec = want.freq;
wfx.Format.cbSize = sizeof(wfx) - sizeof(wfx.Format);
MMRESULT wave_out_result = ~MMSYSERR_NOERROR;
if( userdata.HighRes )
{
wfx.Format.wFormatTag = WAVE_FORMAT_IEEE_FLOAT;
wfx.SubFormat = {0x00000003,0x0000,0x0010,0x80,0x00,0x00,0xAA,0x00,0x38,0x9B,0x71}; //KSDATAFORMAT_SUBTYPE_IEEE_FLOAT
wfx.Format.wBitsPerSample = 32;
wfx.Samples.wValidBitsPerSample = wfx.Format.wBitsPerSample;
wfx.Format.nBlockAlign = wfx.Format.nChannels * wfx.Format.wBitsPerSample / 8;
wfx.Format.nAvgBytesPerSec = wfx.Format.nBlockAlign * wfx.Format.nSamplesPerSec;
wave_out_result = waveOutOpen( &WaveOutHandle, WAVE_MAPPER, &(wfx.Format), (DWORD_PTR) &WaveOutCallback, 0, CALLBACK_FUNCTION );
}
if( wave_out_result != MMSYSERR_NOERROR )
{
if( userdata.HighRes )
fprintf( stderr, "waveOutOpen returned %i%s when attempting float output\n", wave_out_result, (wave_out_result == WAVERR_BADFORMAT)?" (WAVERR_BADFORMAT)":"" );
wfx.Format.wFormatTag = WAVE_FORMAT_PCM;
wfx.SubFormat = {0x00000001,0x0000,0x0010,0x80,0x00,0x00,0xAA,0x00,0x38,0x9B,0x71}; //KSDATAFORMAT_SUBTYPE_PCM
wfx.Format.wBitsPerSample = 16;
wfx.Samples.wValidBitsPerSample = wfx.Format.wBitsPerSample;
wfx.Format.nBlockAlign = wfx.Format.nChannels * wfx.Format.wBitsPerSample / 8;
wfx.Format.nAvgBytesPerSec = wfx.Format.nBlockAlign * wfx.Format.nSamplesPerSec;
wave_out_result = waveOutOpen( &WaveOutHandle, WAVE_MAPPER, &(wfx.Format), (DWORD_PTR) &WaveOutCallback, 0, CALLBACK_FUNCTION );
if( wave_out_result == MMSYSERR_NOERROR )
userdata.HighRes = false;
else
fprintf( stderr, "waveOutOpen returned %i%s when attempting int16 output\n", wave_out_result, (wave_out_result == WAVERR_BADFORMAT)?" (WAVERR_BADFORMAT)":"" );
}
如果我设置Format.nChannels = 2; dwChannelMask = 0x3;
为立体声,则waveOutOpen
使用 IEEE-float 格式的第一次尝试失败并返回代码WAVERR_BADFORMAT
,但使用 PCM 格式的第二次尝试成功。
如果我尝试Format.nChannels = 4; dwChannelMask = 0x33;
四声道,IEEE-float 和 PCMwaveOutOpen
尝试都会失败,并显示WAVERR_BADFORMAT
.
但是,如果我设置Format.cbSize = 0;
,那么任何一种格式的 2 个通道都可以正常工作,这是有道理的,因为这基本上是我之前一直在做的事情WAVEFORMATEX
。但这不适用于 4 个通道。
我在这里做错了什么? 我的最终目标是 IEEE 浮点格式的四声道或 5.1 环绕声输出。我特别困惑为什么我什至无法使用立体声 IEEE-float 输出,WAVEFORMATEXTENSIBLE
但它与WAVEFORMATEX
.
解决方案
如果使用的是WAVEFORMATEXTENSIBLE
结构,则必须通过在 "base" 中设置正确的格式标记来表明这一点WAVEFORMATEX
。
wfx.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE;
请参阅WAVEFORMATEXTENSIBLE的 Format 成员的文档。
这使得被调用代码(只有一个指向 a 的指针WAVEFORMATEX
)可以辨别它是在处理 aWAVEFORMATEX
还是WAVEFORMATEXTENSIBLE
结构。
在实际音频格式的情况下,WAVEFORMATEXTENSIBLE
则由SubFormat
成员唯一标识。
推荐阅读
- sql - 在 Postgres SQL 查询中获取大于或等于的数据
- r - 如果一行满足条件,则删除常用值行
- javascript - href 在第二次点击时附加到先前生成的“a”标签中
- typescript - TypeScript 在新闭包中使用时忽略可能未分配的变量
- javascript - 我如何使高度和宽度全屏
- java - 仅当 N 个文件出现在 Apache Camel 的多节点设置中时才开始处理
- javascript - Web3 未按预期返回承诺
- java - 有什么方法可以从本地计算机(本地服务器)读取数据库文件并复制到android中的资产文件夹?
- visual-studio - 无法评估扩展元数据的“Cosmos.CRTCompat.dll”。异常消息:错误的 IL 格式
- r - 如何在内存方面超越 R 的限制