我正在使用音频单元框架在macosx上开发一个VOIP应用程序。在我的程序中,我设置了一个输入AUHAL,并使用默认的流格式(44.1kHz,32位/通道)从麦克风捕获音频。在这种情况下,我的程序运行良好。
这是代码:
//The default setting in my program
CheckError(AudioUnitGetProperty(m_audCapUnit,
kAudioUnitProperty_StreamFormat,
kAudioUnitScope_Output, //the value is 0
inputBus, //the value is 1
&m_audCapUnitOutputStreamFormat,
&propertySize),
"Couldn't get OutputSample ASBD from input unit") ;
//the inOutputSampleRate is 44100.0
m_audCapUnitOutputStreamFormat.mSampleRate = inOutputSampleRate ;
CheckError(AudioUnitSetProperty(m_audCapUnit,
kAudioUnitProperty_StreamFormat,
kAudioUnitScope_Output,
inputBus,
&m_audCapUnitOutputStreamFormat,
propertySize),
"Couldn't set OutputSample ASBD on input unit");
//
由于我正在开发VOIP应用程序,默认格式(44.1kHz,32位/频道)不适合我的程序,所以我想将采样率更改为8kHz。我写了这段代码来改变我程序中的格式:
//......
inOutputFormat.mSampleRate = 8000. ;
inOutputFormat.mFormatID = kAudioFormatLinearPCM ;
inOutputFormat.mChannelsPerFrame = 2 ;
inOutputFormat.mBitsPerChannel = 16 ;
inOutputFormat.mBytesPerFrame = 2 ;
inOutputFormat.mBytesPerPacket = 2 ;
inOutputFormat.mFramesPerPacket = 1 ;
inOutputFormat.mFormatFlags = kAudioFormatFlagsAudioUnitCanonical ;
inOutputFormat.mReserved = 0 ;
CheckError(AudioUnitSetProperty(m_audCapUnit,
kAudioUnitProperty_StreamFormat,
kAudioUnitScope_Output,
inputBus,
&inOutputFormat,
ui32PropSize),
"Couldn't set AUHAL Unit Output Format") ;
//.......
在这种情况下,程序运行良好,直到我的程序调用回调函数中的AudioUnitRender
;它调用CCD_ 2失败,错误代码CCD_kAudioUnitErr_NoConnection
。错误代码让我很困惑,所以我在谷歌上搜索了一下,但找不到任何有用的信息。有人能告诉我这个错误到底意味着什么吗?
这还没有结束,我又用这个代码更改了格式:
//.....
inOutputFormat.mSampleRate = 8000. ;
inOutputFormat.mFormatID = kAudioFormatLinearPCM ;
inOutputFormat.mChannelsPerFrame = 2 ;
inOutputFormat.mBitsPerChannel = 32 ;
inOutputFormat.mBytesPerFrame = 4 ;
inOutputFormat.mBytesPerPacket = 4 ;
inOutputFormat.mFramesPerPacket = 1 ;
inOutputFormat.mFormatFlags = kAudioFormatFlagsAudioUnitCanonical ;
inOutputFormat.mReserved = 0 ;
CheckError(AudioUnitSetProperty(m_audCapUnit,
kAudioUnitProperty_StreamFormat,
kAudioUnitScope_Output,
inputBus,
&inOutputFormat,
ui32PropSize),
"Couldn't set AUHAL Unit Output Format") ;
//........
在这种情况下,程序未能再次调用AudioUnitRender
,并返回另一个错误代码-10863(kAudioUnitErr_CannotDoInCurrentContext)
。我在谷歌上搜索了一下,但我发现有用的东西。在阅读了那里的信息后,我想我在AUHAL上设置的采样率或格式可能不受硬件支持。
所以我写了一些代码来检查默认输入设备上的可用采样率:
//..........
UInt32 propertySize = sizeof(AudioDeviceID) ;
Boolean isWritable = false ;
CheckError(AudioDeviceGetPropertyInfo(inDeviceID, //the inDeviceID is the default input device
0,
true,
kAudioDevicePropertyAvailableNominalSampleRates,
&propertySize,
&isWritable),
"Get the Available Sample Rate Count Failed") ;
m_valueCount = propertySize / sizeof(AudioValueRange) ;
printf("Available %d Sample Raten",m_valueCount) ;
CheckError(AudioDeviceGetProperty(inDeviceID,
0,
false,
kAudioDevicePropertyAvailableNominalSampleRates,
&propertySize,
m_valueTabe),
"Get the Available Sample Rate Count Failed") ;
for(UInt32 i = 0 ; i < m_valueCount ; ++i)
{
printf("Available Sample Rate value : %ldn",(long)m_valueTabe[i].mMinimum) ;
}
//..............
然后我发现可用的采样率是8000、16000、32000、44100、48000、88200和96000。
8000采样率是我之前设置的,但我通过调用AudioUnitRender
得到了一个错误代码,我只想说,为什么?
我在谷歌上搜索了很多,也读了很多文档,但我找不到答案,有人能解决我遇到的这个问题吗
换句话说;如何更改仅输入AUHAL的采样率或格式
昨天我终于自己解决了这个问题。
这是我的解决方案:
- 首先,我使用
AudioDeviceGetProperty
在我的默认输入设备上获得可用格式列表,然后我发现可用格式列表包含:8khz, 16khz, 32khz, 44.1khz, 48khz, 88.2khz,96khz
(我只是在这里列出采样率字段,但还有其他列表中的字段) - 然后我选择在第一步中获得的可用格式之一。以我的程序为例,我选择格式
(8khz,32bits/Channel)
,并使用AudioDeviceSetProperty
将其设置在默认设备上,而不是AUHAL上,这是我的程序在AUHAL(OutputScope,inputBus)上设置格式后正常工作的关键 - 最后一步,我使用
AudioUnitSetProperty
设置我想要的格式,程序运行良好
通过这个问题和解决方案,我想如果我想在仅输入AUHAL上设置格式,格式必须匹配或可以转换为输入设备正在使用的可用格式。所以我需要做的是首先在输入设备上设置格式,然后在仅输入AUHAL上设置格式。
根据我的经验,使用44.1kHz和16位音频以外的设置会导致各种奇怪的错误。一些可能会让你走上正确道路的通用建议:
- 试试Novocaine(https://github.com/alexbw/novocaine)。它确实减轻了使用AudioUnits的痛苦
- 试着让你的应用程序以44.1kHz的频率工作,然后自己对音频进行下采样
- 您正在设置的比特率可能与所需的采样率不兼容
你的回答对我很有帮助。然而,AudioDeviceGetProperty的使用是折旧的。以下列表可能有助于更新内容。作为示例,采样率被设置为32kHz。
// Get the available sample rates of the default input device.
defaultDeviceProperty.mSelector = kAudioDevicePropertyAvailableNominalSampleRates;
CheckError (AudioObjectGetPropertyDataSize(defaultDevice,
&defaultDeviceProperty,
0,
NULL,
&propertySize),
"Couldn't get sample rate count");
int m_valueCount = propertySize / sizeof(AudioValueRange) ;
printf("Available %d Sample Ratesn",m_valueCount) ;
AudioValueRange m_valueTabe[m_valueCount];
CheckError (AudioObjectGetPropertyData(defaultDevice,
&defaultDeviceProperty,
0,
NULL,
&propertySize,
m_valueTabe),
"Couldn't get available sample rates");
for(UInt32 i = 0 ; i < m_valueCount ; ++i)
{
printf("Available Sample Rate value : %fn", m_valueTabe[i].mMinimum) ;
}
// Set the sample rate to one of the available values.
AudioValueRange inputSampleRate;
inputSampleRate.mMinimum = 32000;
inputSampleRate.mMaximum = 32000;
defaultDeviceProperty.mSelector = kAudioDevicePropertyNominalSampleRate;
CheckError (AudioObjectSetPropertyData(defaultDevice,
&defaultDeviceProperty,
0,
NULL,
sizeof(inputSampleRate),
&inputSampleRate),
"Couldn't get available sample rates");