我正在尝试使用斯坦福的 STK 进行一些实时波表合成。我正在使用 FMVoices 乐器类https://ccrma.stanford.edu/software/stk/classstk_1_1FMVoices.html 并尝试在下面定义的回调例程中使用它。
int tick( void *outputBuffer, void *inputBuffer, unsigned int nBufferFrames,
double streamTime, RtAudioStreamStatus status, void *dataPointer )
{
FMVoices *FM = (FMVoices *) dataPointer;
register StkFloat *samples = (StkFloat *) outputBuffer;
for ( unsigned int i=0; i<nBufferFrames; i++ )
*samples++ = FM->tick();
return 0;
}
我认为问题在于最后一个参数的类型。我收到运行时错误:“0xC0000005:执行位置 0x000000001 的访问冲突。” 现在,这是应该为其他 STK 乐器(如 Clarinet 甚至 FileLoop 类)编写回调的方式,但 FMVoices 有一些时髦的东西。该对象作为指向 void 的指针传递给 openStream(它处理特定于平台的实时输出)。当系统的音频缓冲区已满时,会自动调用回调。下面显示了实现此功能并适用于其他仪器的代码片段:
int main()
{
// Set the global sample rate before creating class instances.
Stk::setSampleRate( 44100.0 );
RtAudio dac;
Instrmnt * instrument_FM;
int nFrames = 10000;
try {
instrument_FM = new FMVoices;
}
catch ( StkError & ) {
goto cleanup;
}
instrument_FM->setFrequency(440.0);
// Figure out how many bytes in an StkFloat and setup the RtAudio stream.
RtAudio::StreamParameters parameters;
parameters.deviceId = dac.getDefaultOutputDevice();
parameters.nChannels = 1;
RtAudioFormat format = ( sizeof(StkFloat) == 8 ) ? RTAUDIO_FLOAT64 : RTAUDIO_FLOAT32;
unsigned int bufferFrames = RT_BUFFER_SIZE;
try {
dac.openStream( ¶meters, NULL, format, (unsigned int)Stk::sampleRate(), &bufferFrames, &tick, (void *)&instrument_FM);
}
catch ( RtError &error ) {
error.printMessage();
Sleep(1000);
goto cleanup;
}
nFrames 的大小似乎没有影响。在我看来,这些类型的错误通常来自引用指向 void 的指针。