我目前正在尝试计算 iPhone 扬声器/麦克风往返的频率响应。我在扬声器上播放正弦扫描,通过麦克风录制它,然后尝试从中获取频率响应。最终目标是能够将 FR 乘以任何给定的声音,使其听起来像 iPhone 的扬声器/麦克风。
我的代码如下:
//apply window function
vDSP_vmul(sineSweepMic,1,hammingWindow,1,sineSweepMic,1,n);
vDSP_vmul(sineSweepFile,1,hammingWindow,1,sineSweepFile,1,n);
//put both signals in complex arrays
vDSP_ctoz((DSPComplex *)sineSweepMic, 2, &fftSineSweepMic, 1, nOver2);
vDSP_ctoz((DSPComplex *)sineSweepFile, 2, &fftSineSweepFile, 1, nOver2);
//fft of both file and mic sweeps
vDSP_fft_zrip(fftSetup, &fftSineSweepFile, 1, log2n, FFT_FORWARD);
vDSP_fft_zrip(fftSetup, &fftSineSweepMic, 1, log2n, FFT_FORWARD);
//back to interleaved
vDSP_ztoc(&fftSineSweepFile, 1, (COMPLEX *)sineSweepFile, 2, nOver2);
vDSP_ztoc(&fftSineSweepMic, 1, (COMPLEX *)sineSweepMic, 2, nOver2);
//divide mic-sweep by file-sweep to create frequency response
vDSP_vdiv(sineSweepFile, 1, sineSweepMic, 1, frequencyResponse, 1, n);
到目前为止这个方法是可行的,当我用初始文件扫描乘以FR时,听起来像麦克风扫描。
我的问题: 这只对生成FR的确切文件(扫描)有效。一旦我使用FR修改其他声音,例如音乐,只会听到噪音。
我是这样使用FR的(都在频域内,交错排列,不复杂,甚至长度相同):
vDSP_vmul(soundToModify, 1, frequencyResponse, 1, soundToModify, 1, n);
我的来自文件的正弦扫描在扬声器上播放:
我的记录的正弦扫描(衰减的低频可见):
我在代码中生成的FR与上述内容相乘的文件正弦扫描:
![enter image description here](https://istack.dev59.com/D1ZDy.webp)
我的目标: 据我所知,频率响应是关于每个频率的信息,系统通过多少衰减或放大它(在我的例子中,它无法再现低频)。为了获得这种信息,我会生成包含每个所需频率的声音(正弦扫描),播放它并分析每个频率如何被修改,即通过除以记录到的扫描/文件扫描(代码中的除法)。
将此频率响应(FR)在频率域中与任何声音相乘,是否应该修改频率幅度以模拟在我的系统上播放?
谢谢!
更新: 最终问题出在缺少复杂算术上,正弦扫描和粉色噪声都很好地作为恢复脉冲响应的冲激。
要获得可工作的代码,只需通过初步扫描fft数据进行复杂除法即可。